Refactored keyboard thread by moving the keypad scan logic to a dedicated function inside input.h

pull/87/head
Silvano Seva 2022-06-14 07:34:53 +02:00
rodzic f1b1dc2034
commit 3cc5852ef3
5 zmienionych plików z 140 dodań i 120 usunięć

Wyświetl plik

@ -20,12 +20,49 @@
#ifndef INPUT_H
#define INPUT_H
#include <interfaces/keyboard.h>
#include <inttypes.h>
#include <stdbool.h>
/**
* Time interval in milliseconds after which a keypress is considered a long-press
*/
static const uint16_t input_longPressTimeout = 700;
/**
* Structure that represents a keyboard event payload
* The maximum size of an event payload is 30 bits
* For a keyboard event we use 1 bit to signal a short or long press
* And the remaining 29 bits to communicate currently pressed keys.
*/
typedef union
{
struct
{
uint32_t long_press : 1,
keys : 29,
_padding : 2;
};
uint32_t value;
}
kbd_msg_t;
/**
* Scan all the keyboard buttons to detect possible keypresses filling a
* keyboard event data structure. The function returns true if a keyboard event
* has been detected.
*
* @param msg: keyboard event message to be filled.
* @return true if a keyboard event has been detected, false otherwise.
*/
bool input_scanKeyboard(kbd_msg_t *msg);
/**
* This function returns true if at least one number is pressed on the
* keyboard.
*
* @param msg: the keyboard queue message
* @return true if at least a number is pressed on the keyboard
*/
@ -34,6 +71,7 @@ bool input_isNumberPressed(kbd_msg_t msg);
/**
* This function returns the smallest number that is pressed on the keyboard,
* 0 if none is pressed.
*
* @param msg: the keyboard queue message
* @return the smalled pressed number on the keyboard
*/

Wyświetl plik

@ -64,36 +64,13 @@ enum key
/**
* Number of supported keys
*/
static const uint8_t kbd_num_keys = 29;
/**
* Time interval in ticks after which a keypress is considered a long-press
*/
static const uint16_t kbd_long_interval = TICK_FREQ * 0.7;
#define KBD_NUM_KEYS 29
/**
* Mask for the numeric keys in a key map
* Numeric keys: bit0->bit11 = 0xFFF
*/
static const uint32_t kbd_num_mask = 0xFFF;
/**
* Structure that represents a keyboard event payload
* The maximum size of an event payload is 30 bits
* For a keyboard event we use 1 bit to signal a short or long press
* And the remaining 29 bits to communicate currently pressed keys.
*/
typedef union
{
struct
{
uint32_t long_press : 1,
keys : 29,
_padding : 2;
};
uint32_t value;
} kbd_msg_t;
#define KBD_NUM_MASK 0x0FFF
/**
* We encode the status of all the keys with a uint32_t value

Wyświetl plik

@ -17,20 +17,80 @@
* along with this program; if not, see <http://www.gnu.org/licenses/> *
***************************************************************************/
#include <interfaces/keyboard.h>
#include <interfaces/delays.h>
#include <inttypes.h>
#include <stdbool.h>
#include <input.h>
static long long keyTs[KBD_NUM_KEYS]; // Timestamp of each keypress
static uint32_t longPressSent; // Flags to manage long-press events
static keyboard_t prevKeys = 0; // Previous keyboard status
bool input_scanKeyboard(kbd_msg_t *msg)
{
msg->value = 0;
bool kbd_event = false;
keyboard_t keys = kbd_getKeys();
long long now = getTick();
// The key status has changed
if(keys != prevKeys)
{
// Find newly pressed keys
keyboard_t newKeys = (keys ^ prevKeys) & keys;
// Save timestamp
for(uint8_t k = 0; k < KBD_NUM_KEYS; k++)
{
keyboard_t mask = 1 << k;
if((newKeys & mask) != 0)
{
keyTs[k] = now;
longPressSent &= ~mask;
}
}
// New keyboard event
msg->keys = keys;
kbd_event = true;
}
// Some key is kept pressed
else if(keys != 0)
{
// Check for saved timestamp to trigger long-presses
for(uint8_t k = 0; k < KBD_NUM_KEYS; k++)
{
keyboard_t mask = 1 << k;
// The key is pressed and its long-press timer is over
if(((keys & mask) != 0) &&
((longPressSent & mask) == 0) &&
((now - keyTs[k]) >= input_longPressTimeout))
{
msg->long_press = 1;
msg->keys = keys;
kbd_event = true;
longPressSent |= mask;
}
}
}
prevKeys = keys;
return kbd_event;
}
bool input_isNumberPressed(kbd_msg_t msg)
{
return msg.keys & kbd_num_mask;
return msg.keys & KBD_NUM_MASK;
}
uint8_t input_getPressedNumber(kbd_msg_t msg)
{
uint32_t masked_input = msg.keys & kbd_num_mask;
uint32_t masked_input = msg.keys & KBD_NUM_MASK;
if (masked_input == 0)
return 0;
return __builtin_ctz(msg.keys & kbd_num_mask);
return __builtin_ctz(msg.keys & KBD_NUM_MASK);
}

Wyświetl plik

@ -25,7 +25,6 @@
#include <state.h>
#include <threads.h>
#include <battery.h>
#include <interfaces/keyboard.h>
#include <interfaces/graphics.h>
#include <interfaces/platform.h>
#include <interfaces/delays.h>
@ -36,6 +35,7 @@
#include <minmea.h>
#include <string.h>
#include <utils.h>
#include <input.h>
#ifdef HAS_GPS
#include <interfaces/gps.h>
#include <gps.h>
@ -144,69 +144,16 @@ void *kbd_task(void *arg)
// Initialize keyboard driver
kbd_init();
// Allocate timestamp array
long long key_ts[kbd_num_keys];
long long now;
// Allocate bool array to send only one long-press event
bool long_press_sent[kbd_num_keys];
// Variable for saving previous and current keyboard status
keyboard_t prev_keys = 0;
keyboard_t keys = 0;
bool long_press = false;
bool send_event = false;
while(1)
{
// Reset flags and get current time
long_press = false;
send_event = false;
// Lock display mutex and read keyboard status
pthread_mutex_lock(&display_mutex);
keys = kbd_getKeys();
pthread_mutex_unlock(&display_mutex);
now = getTick();
// The key status has changed
if(keys != prev_keys)
{
for(uint8_t k=0; k < kbd_num_keys; k++)
{
// Key has been pressed
if(!(prev_keys & (1 << k)) && (keys & (1 << k)))
{
// Save timestamp
key_ts[k] = now;
send_event = true;
long_press_sent[k] = false;
}
// Key has been released
else if((prev_keys & (1 << k)) && !(keys & (1 << k)))
{
send_event = true;
}
}
}
// Some key is kept pressed
else if(keys != 0)
{
// Check for saved timestamp to trigger long-presses
for(uint8_t k=0; k < kbd_num_keys; k++)
{
// The key is pressed and its long-press timer is over
if(keys & (1 << k) && !long_press_sent[k] && (now - key_ts[k]) >= kbd_long_interval)
{
long_press = true;
send_event = true;
long_press_sent[k] = true;
}
}
}
if(send_event)
{
kbd_msg_t msg;
msg.long_press = long_press;
msg.keys = keys;
pthread_mutex_lock(&display_mutex);
bool event = input_scanKeyboard(&msg);
pthread_mutex_unlock(&display_mutex);
if(event)
{
// Send event_t as void * message to use with OSQPost
event_t event;
event.type = EVENT_KBD;
@ -214,8 +161,6 @@ void *kbd_task(void *arg)
// Send keyboard status in queue
(void) queue_post(&ui_queue, event.value);
}
// Save current keyboard state as previous
prev_keys = keys;
// Read keyboard state at 40Hz
sleepFor(0u, 25u);

Wyświetl plik

@ -886,7 +886,7 @@ void _ui_textInputKeypad(char *buf, uint8_t max_len, kbd_msg_t msg, bool callsig
if(ui_state.last_keypress != 0)
{
// Same key pressed and timeout not expired: cycle over chars of current key
if((ui_state.input_number == num_key) && ((now - ui_state.last_keypress) < kbd_long_interval))
if((ui_state.input_number == num_key) && ((now - ui_state.last_keypress) < input_longPressTimeout))
{
ui_state.input_set = (ui_state.input_set + 1) % num_symbols;
}