1 Star 0 Fork 107

林杰强/key_board

forked from wei/key_board 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
key_board.c 18.08 KB
一键复制 编辑 原始数据 按行查看 历史
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644
#include "key_board/key_board.h"
#if (KEY_BOARD_SCAN_TIME <= 0)
#error "the KEY_BOARD_SCAN_TIME must be greater than 0."
#endif
#if (KEY_BOARD_MAX_NUM <= 0)
#error "the KEY_BOARD_MAX_NUM must be greater than 0."
#endif
#if (KEY_MAX_NUM <= 0)
#error "the KEY_MAX_NUM must be greater than 0."
#endif
#if (KEY_BOARD_MAX_COMBINE <= 1)
#error "the KEY_BOARD_MAX_COMBINE must be greater than 1."
#endif
#if (KEY_DEFAULT_LONG_TRRIGER_TIME < KEY_BOARD_SCAN_TIME)
#error "the KEY_DEFAULT_LONG_TRRIGER_TIME must be greater than KEY_BOARD_SCAN_TIME."
#endif
#if (KEY_DEFAULT_MULTI_INTERVAL_TIME < KEY_BOARD_SCAN_TIME)
#error "the KEY_DEFAULT_MULTI_INTERVAL_TIME must be greater than KEY_BOARD_SCAN_TIME."
#endif
#if (KEY_DEFAULT_CONTINUOUS_INIT_TRRIGER_TIME < KEY_BOARD_SCAN_TIME)
#error "the KEY_DEFAULT_CONTINUOUS_INIT_TRRIGER_TIME must be greater than KEY_BOARD_SCAN_TIME."
#endif
#if (KEY_DEFAULT_CONTINUOUS_PERIOD_TRRIGER_TIME < KEY_BOARD_SCAN_TIME)
#error "the KEY_DEFAULT_CONTINUOUS_PERIOD_TRRIGER_TIME must be greater than KEY_BOARD_SCAN_TIME."
#endif
#if (KEY_DEFAULT_CONTINUOUS_INIT_TRRIGER_TIME < KEY_DEFAULT_CONTINUOUS_PERIOD_TRRIGER_TIME)
#error "KEY_DEFAULT_CONTINUOUS_INIT_TRRIGER_TIME must be greater than KEY_DEFAULT_CONTINUOUS_PERIOD_TRRIGER_TIME."
#endif
#if (KEY_DEFAULT_COMBINE_INTERVAL_TIME < KEY_BOARD_SCAN_TIME)
#error "the KEY_DEFAULT_COMBINE_INTERVAL_TIME must be greater than KEY_BOARD_SCAN_TIME."
#endif
#if ((KEY_LONG_SUPPORT != KEY_ENABLE) && (KEY_LONG_SUPPORT != KEY_DISABLE))
#error "KEY_LONG_SUPPORT can be only set to KEY_ENABLE or KEY_DISABLE."
#endif
#if ((KEY_MULTI_SUPPORT != KEY_ENABLE) && (KEY_MULTI_SUPPORT != KEY_DISABLE))
#error "KEY_MULTI_SUPPORT can be only set to KEY_ENABLE or KEY_DISABLE."
#endif
#if ((KEY_CONTINUOUS_SUPPORT != KEY_ENABLE) && (KEY_CONTINUOUS_SUPPORT != KEY_DISABLE))
#error "KEY_CONTINUOUS_SUPPORT can be only set to KEY_ENABLE or KEY_DISABLE."
#endif
#if ((KEY_COMBINE_SUPPORT != KEY_ENABLE) && (KEY_COMBINE_SUPPORT != KEY_DISABLE))
#error "KEY_COMBINE_SUPPORT can be only set to KEY_ENABLE or KEY_DISABLE."
#endif
#undef TIME_TO_TIMES
#define TIME_TO_TIMES(x) (x / KEY_BOARD_SCAN_TIME)
struct key_private_t {
struct key_public_sig_t *property;
enum key_state_t state; //按键状态
bool last_level; //按键上次电平(有效/无效)
bool this_level; //按键本次电平(有效/无效)
#if (KEY_LONG_SUPPORT == KEY_ENABLE)
struct {
unsigned int trriger_time; //触发时间
unsigned int press_time; //已按下时间
volatile bool trriger_flag; //触发标志
}long_press;
#endif
#if (KEY_MULTI_SUPPORT == KEY_ENABLE)
struct {
unsigned int count; //多击次数
unsigned int timecnt; //记录距上次点击的时间
unsigned int interval; //连续两次点击的最大允许间隔时间
}multi_click_press, multi_click_release;
#endif
#if (KEY_CONTINUOUS_SUPPORT == KEY_ENABLE)
struct {
unsigned int trriger_time; //触发时间
unsigned int trriger_period; //触发周期
unsigned int press_time; //已按下时间
}continuous;
#endif
};
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
struct {
unsigned int timecnt; //记录距上次触发的时间
unsigned int interval; //连续两次触发的最大允许间隔时间
struct match_list_t {
struct key_combine_t *match; //需要匹配的组合状态
unsigned int n; //记录组合状态的位数
unsigned int match_count; //已匹配成功计数
struct match_list_t *next; //用于链接下一个注册的组合状态
}*head; //注册的组合状态的链表头
}static combine;
#endif
struct key_board_t {
enum key_board_type_t type; //键盘类型
unsigned int sig_num; //信号线的数量
struct key_private_t *sig; //信号线的相关信息
//仅用于矩阵键盘
unsigned int ctrl_num; //控制线的数量
unsigned int sig_per_ctrl; //每根控制线上信号线的数量
struct key_public_ctrl_t *ctrl; //控制线的相关信息
};
static struct key_board_t *key_board[KEY_BOARD_MAX_NUM];
static unsigned int key_hash_map[KEY_MAX_NUM]; //用于根据id快速查找按键的对象(线性探测法)
static unsigned int key_press_count; //用于记录当前按下键的计数
static print_debug_callback internal_print_debug; //调试信息输出
static inline void debug(const char *func, int line, const char *message)
{
#include <stdio.h>
char buff[128];
sprintf(buff, "func:%s, line:%d, %s\n", func, line, message);
if(internal_print_debug)
{
internal_print_debug(buff);
}
}
int key_board_init(void)
{
memset(key_board, 0, sizeof(key_board));
for(unsigned int i= 0;i < ARRAY_SIZE(key_hash_map);i++)
{
key_hash_map[i] = (unsigned int)-1;
}
key_press_count = 0;
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
memset(&combine, 0, sizeof(combine));
combine.head = malloc(sizeof(struct match_list_t));
if(!combine.head)
{
debug(__FUNCTION__, __LINE__, "malloc failed");
return -1;
}
combine.head->next = NULL;
#endif
return 0;
}
struct key_board_t *key_board_register(enum key_board_type_t type, const struct key_public_sig_t sig[], unsigned int key_sig_n, const struct key_public_ctrl_t ctrl[], unsigned int key_ctrl_n)
{
struct key_board_t *obj;
unsigned int handle_no_use;
unsigned int i;
for(handle_no_use = 0;handle_no_use < ARRAY_SIZE(key_board) && key_board[handle_no_use];handle_no_use++);
if(handle_no_use >= ARRAY_SIZE(key_board))
{
debug(__FUNCTION__, __LINE__, "exceed the max configure value [KEY_BOARD_MAX_NUM]");
return NULL;
}
obj = malloc(sizeof(struct key_board_t));
if(!obj)
{
debug(__FUNCTION__, __LINE__, "malloc failed");
return NULL;
}
memset(obj, 0, sizeof(struct key_board_t));
key_board[handle_no_use] = obj;
obj->type = type;
obj->sig_num = key_sig_n;
obj->sig = malloc(key_sig_n * sizeof(struct key_private_t));
if(!obj->sig)
{
debug(__FUNCTION__, __LINE__, "malloc failed");
return NULL;
}
memset(obj->sig, 0, key_sig_n * sizeof(struct key_private_t));
for(i = 0;i < key_sig_n;i++)
{
obj->sig[i].property = (struct key_public_sig_t *)&sig[i];
#if (KEY_LONG_SUPPORT == KEY_ENABLE)
obj->sig[i].long_press.trriger_time = TIME_TO_TIMES(KEY_DEFAULT_LONG_TRRIGER_TIME);
#endif
#if (KEY_MULTI_SUPPORT == KEY_ENABLE)
obj->sig[i].multi_click_press.interval = TIME_TO_TIMES(KEY_DEFAULT_MULTI_INTERVAL_TIME);
obj->sig[i].multi_click_release.interval = TIME_TO_TIMES(KEY_DEFAULT_MULTI_INTERVAL_TIME);
#endif
#if (KEY_CONTINUOUS_SUPPORT == KEY_ENABLE)
obj->sig[i].continuous.trriger_time = TIME_TO_TIMES(KEY_DEFAULT_CONTINUOUS_INIT_TRRIGER_TIME);
obj->sig[i].continuous.trriger_period = TIME_TO_TIMES(KEY_DEFAULT_CONTINUOUS_PERIOD_TRRIGER_TIME);
#endif
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
combine.interval = TIME_TO_TIMES(KEY_DEFAULT_COMBINE_INTERVAL_TIME);
#endif
//通过取模得到插入位置
unsigned int idx, backup;
backup = idx = obj->sig[i].property->id % ARRAY_SIZE(key_hash_map);
while(key_hash_map[idx] != (unsigned int)-1)
{
if(obj->sig[key_hash_map[idx]].property->id == obj->sig[i].property->id)
{
debug(__FUNCTION__, __LINE__, "the key id is already register");
//id 已存在
return NULL;
}
idx = (idx + 1) % ARRAY_SIZE(key_hash_map);
if(idx == backup)
{
debug(__FUNCTION__, __LINE__, "exceed the max configure value [KEY_MAX_NUM]");
//无空位置
return NULL;
}
}
key_hash_map[idx] = i;
}
if(obj->type == KEY_BOARD_MATRIX)
{
obj->ctrl_num = key_ctrl_n;
obj->sig_per_ctrl = key_sig_n / key_ctrl_n;
obj->ctrl = (struct key_public_ctrl_t *)ctrl;
for(i = 0;i < key_ctrl_n;i++)
{
//将控制线拉到无效电平
ctrl[i].set(ctrl[i].pin_desc, false);
}
}
return obj;
}
void key_board_unregister(struct key_board_t *obj)
{
unsigned int i;
for(i = 0;i < ARRAY_SIZE(key_board) && key_board[i];i++)
{
if(obj == key_board[i])
{
free(obj->sig);
obj->sig = NULL;
free(obj);
key_board[i] = NULL;
}
}
}
void key_board_destroy(void)
{
struct key_board_t *obj;
unsigned int i;
for(i = 0;i < ARRAY_SIZE(key_board) && key_board[i];i++)
{
obj = key_board[i];
free(obj->sig);
obj->sig = NULL;
free(obj);
key_board[i] = NULL;
}
}
unsigned int key_check_state(unsigned int id, enum key_state_t state)
{
struct key_board_t *obj;
unsigned int i;
unsigned int idx, backup;
for(i = 0;i < ARRAY_SIZE(key_board) && key_board[i];i++)
{
obj = key_board[i];
//如果要查找的id对应在hash表中向后查找已经有空位了那这个id肯定不存在
backup = idx = id % ARRAY_SIZE(key_hash_map);
while(key_hash_map[idx] != (unsigned int)-1)
{
if(obj->sig[key_hash_map[idx]].property->id == id)
{
switch(obj->sig[key_hash_map[idx]].state & state)
{
#if (KEY_MULTI_SUPPORT == KEY_ENABLE)
case KEY_PRESS_MULTI:
return obj->sig[key_hash_map[idx]].multi_click_press.count;
case KEY_RELEASE_MULTI:
return obj->sig[key_hash_map[idx]].multi_click_release.count;
#endif
default:
return !!(obj->sig[key_hash_map[idx]].state & state);
}
}
idx = (idx + 1) % ARRAY_SIZE(key_hash_map);
if(idx == backup)
{
//未找到
break;
}
}
}
return KEY_NONE;
}
unsigned int key_press_count_get(void)
{
return key_press_count;
}
static inline void key_press_count_stats(struct key_private_t *sig)
{
if(sig->state & KEY_PRESS)
{
++key_press_count;
}
else if(sig->state & KEY_RELEASE)
{
--key_press_count;
}
}
unsigned int key_combine_register(const struct key_combine_t c[], unsigned int n)
{
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
struct match_list_t *item;
item = malloc(sizeof(struct match_list_t));
if(!item)
{
debug(__FUNCTION__, __LINE__, "malloc failed");
return 0;
}
item->match = malloc(n * sizeof(struct key_combine_t));
if(!item->match)
{
debug(__FUNCTION__, __LINE__, "malloc failed");
return 0;
}
memcpy(item->match, c, n * sizeof(struct key_combine_t));
item->n = n;
item->match_count = 0;
item->next = combine.head->next;
combine.head->next = item;
return (unsigned int)item;
#else
return 0;
#endif
}
void key_combine_unregister(unsigned int id)
{
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
struct match_list_t *item;
struct match_list_t *del;
item = combine.head;
while(item->next && (item->next != (struct match_list_t *)id))
{
item = item->next;
}
if(item->next)
{
del = item->next;
item->next = item->next->next;
free(del);
}
#endif
}
unsigned int key_check_combine_state(unsigned int id)
{
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
struct match_list_t *item;
item = (struct match_list_t *)id;
return (item->match_count == item->n);
#else
return 0;
#endif
}
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
static inline void key_check_combine_pre(void)
{
struct match_list_t *item;
++combine.timecnt;
if(combine.timecnt >= combine.interval)
{
combine.timecnt = 0;
//两次触发间隔超时
item = combine.head->next;
while(item)
{
item->match_count = 0;
item = item->next;
}
}
item = combine.head->next;
while(item)
{
if(item->match_count == item->n)
{
//自动清除状态
item->match_count = 0;
}
item = item->next;
}
}
static inline void key_check_combine(struct key_private_t *sig)
{
struct match_list_t *item;
bool flag;
item = combine.head->next;
while(item)
{
if(item->match[item->match_count].state & sig->state)
{
//重新计时
combine.timecnt = 0;
flag = (item->match[item->match_count].id == sig->property->id);
if(!flag && item->match_count)
{
//本次匹配失败,但是之前已经有成功匹配的,则重新匹配
item->match_count = 0;
flag = (item->match[item->match_count].id == sig->property->id) &&
(item->match[item->match_count].state & sig->state);
if(flag)
{
++item->match_count;
}
}
else if(flag)
{
//匹配成功
++item->match_count;
}
}
item = item->next;
}
}
#endif
#if (KEY_LONG_SUPPORT == KEY_ENABLE)
static inline void key_check_long(struct key_private_t *sig)
{
if(sig->state & KEY_PRESS)
{
sig->long_press.trriger_flag = false;
sig->long_press.press_time = 0;
}
if(sig->state & KEY_PRESSING)
{
++sig->long_press.press_time;
if(sig->long_press.press_time >= sig->long_press.trriger_time)
{
//仅触发一次
if(!sig->long_press.trriger_flag)
{
sig->state |= KEY_PRESS_LONG;
}
sig->long_press.trriger_flag = true;
}
}
if(sig->state & KEY_RELEASE)
{
if(sig->long_press.trriger_flag)
{
sig->state |= KEY_RELEASE_LONG;
}
}
}
#endif
#if (KEY_MULTI_SUPPORT == KEY_ENABLE)
static inline void key_check_multi(struct key_private_t *sig)
{
if((sig->property->function & KEY_FLAG_PRESS_MULTI) && (sig->state & KEY_PRESS))
{
//标记按下次数
++sig->multi_click_press.count;
if(sig->multi_click_press.count >= 2)
{
sig->state |= KEY_PRESS_MULTI;
}
sig->multi_click_press.timecnt = 0;
}
if(sig->multi_click_press.count)
{
++sig->multi_click_press.timecnt;
if(sig->multi_click_press.timecnt >= sig->multi_click_press.interval)
{
sig->multi_click_press.count = 0;
}
}
if((sig->property->function & KEY_FLAG_RELEASE_MULTI) && (sig->state & KEY_RELEASE))
{
//标记弹起次数
++sig->multi_click_release.count;
if(sig->multi_click_release.count >= 2)
{
sig->state |= KEY_RELEASE_MULTI;
}
sig->multi_click_release.timecnt = 0;
}
if(sig->multi_click_release.count)
{
++sig->multi_click_release.timecnt;
if(sig->multi_click_release.timecnt >= sig->multi_click_release.interval)
{
sig->multi_click_release.count = 0;
}
}
}
#endif
#if (KEY_CONTINUOUS_SUPPORT == KEY_ENABLE)
static inline void key_check_continuous(struct key_private_t *sig)
{
if(sig->state & KEY_PRESS)
{
sig->continuous.press_time = 0;
}
if(sig->state & KEY_PRESSING)
{
++sig->continuous.press_time;
if(sig->continuous.press_time >= sig->continuous.trriger_time)
{
sig->state |= KEY_PRESS_CONTINUOUS;
sig->continuous.press_time = sig->continuous.trriger_time - sig->continuous.trriger_period;
}
}
}
#endif
void key_check(void)
{
struct key_board_t *obj;
unsigned int i, j;
unsigned int last, next;
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
key_check_combine_pre();
#endif
for(i = 0;i < ARRAY_SIZE(key_board) && key_board[i];i++)
{
obj = key_board[i];
last = obj->ctrl_num - 1;
next = 0;
for(j = 0;j < obj->sig_num;j++)
{
if((obj->type == KEY_BOARD_MATRIX) && ((j % obj->sig_per_ctrl) == 0))
{
//上次扫描的控制线拉到无效电平
obj->ctrl[last].set(obj->ctrl[last].pin_desc, false);
last = next = j / obj->sig_per_ctrl;
//即将扫描的控制线拉到有效电平
obj->ctrl[next].set(obj->ctrl[next].pin_desc, true);
}
obj->sig[j].this_level = obj->sig[j].property->get(obj->sig[j].property->pin_desc);
switch((obj->sig[j].this_level << 4) | (obj->sig[j].last_level))
{
case 0x00:
obj->sig[j].state = KEY_NONE;
break;
case 0x01:
obj->sig[j].state = KEY_RELEASE;
break;
case 0x10:
obj->sig[j].state = KEY_PRESS;
break;
case 0x11:
obj->sig[j].state = KEY_PRESSING;
break;
}
obj->sig[j].last_level = obj->sig[j].this_level;
key_press_count_stats(&obj->sig[j]);
#if (KEY_LONG_SUPPORT == KEY_ENABLE)
key_check_long(&obj->sig[j]);
#endif
#if (KEY_MULTI_SUPPORT == KEY_ENABLE)
key_check_multi(&obj->sig[j]);
#endif
#if (KEY_CONTINUOUS_SUPPORT == KEY_ENABLE)
key_check_continuous(&obj->sig[j]);
#endif
#if (KEY_COMBINE_SUPPORT == KEY_ENABLE)
key_check_combine(&obj->sig[j]);
#endif
}
}
}
void key_board_debug_register(print_debug_callback print_debug)
{
internal_print_debug = print_debug;
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
C
1
https://gitee.com/linjieqiang/key_board.git
git@gitee.com:linjieqiang/key_board.git
linjieqiang
key_board
key_board
master

搜索帮助