Triggerhappy:值得推荐的小型C语言开源项目

Triggerhappy 是什么?

简单地说,Triggerhappy 是一个监测输入事件的守护进程。

它会根据配置文件的内容,监测 /dev/input/event* 设备,然后执行用户预先配置的命令。

例如你的板子上有一个红外接收器,你可以很轻松地使用 Triggerhappy 来实现响应红外遥控器的按键事件。

Triggerhappy 非常的小巧,大约只有 80K,非常适合嵌入式场景。

学习 Triggerhappy 能学到什么?

1、模块化

2、数据结构

3、文本解析

4、单例程序的实现

5、信号的用法

6、IO 多路复用

7、输入设备的访问

8、进程的创建

9、本地 socket 的用法

Triggerhappy 的代码特点

Triggerhappy 的整体代码风格和 Linux 设备驱动的代码风格很像,会用一个结构体来描述对象,然后用一个链表将多个对象管理起来。

功能划分很清晰,大约有 7、8个功能模块,每一个模块的实现都非常的干净利索,代码不超过 300 行,而且风格统一。

综合来看,符合 UNIX 的设计哲学:简单实用,且内容丰富,是一个非常适合练习 Linux 应用编程的开源项目。

Triggerhappy 怎么用?

Triggerhappy 的用法很简单。

$ thd --dump /dev/input/event2
EV_KEY KEY_A 1 /dev/input/event2
EV_KEY KEY_A 0 /dev/input/event2
EV_KEY KEY_B 1 /dev/input/event2
EV_KEY KEY_B 0 /dev/input/event2

上面的命令会从 /dev/input/event2 中读取输入事件,并打印出来。

另外,还可以通过配置文件设置事件处理器:

$ cat /etc/triggerhappy.d/triggerhappy.conf
#   
KEY_VOLUMEUP 1  /usr/bin/amixer set Master 5% 
KEY_VOLUMEUP 2  /usr/bin/amixer set Master 5% 
KEY_VOLUMEDOWN 1  /usr/bin/amixer set Master 5%-
KEY_VOLUMEDOWN 2  /usr/bin/amixer set Master 5%-

1 表示 pressing,2 表示 holding。

$ thd --triggers /etc/triggerhappy.d/triggerhappy.conf /dev/input/event*

这样运行后,我们就可以通过 KEY_VOLUMEUP、KEY_VOLUMEDOWN 控制音量了。

最后,Triggerhappy 还支持通过 socket 通讯进行命令行控制:

$ thd --socket /var/run/triggerhappy.socket --triggers /etc/triggerhappy.d/ /dev/input/event*

# 动态添加或移除设备
$ th-cmd --socket /var/run/triggerhappy.socket --add /dev/input/event0
th-cmd --socket /var/run/triggerhappy.socket --remove /dev/input/event3

这是一种很实用的设计,工作里,很多时候我发现自己都需要在不重新启动程序的情况动态地通知程序做某些事情。

Triggerhappy 怎么实现?

Triggerhappy 大致做了下面这些事情。

1、解析配置文件。

根据配置文件创建相应的 trigger,一个 1 trigger 代表一条事件触发规则,trigger 都挂在链表 trigger_list 上:

typedef struct trigger {
 int type;   // 类型
 int code;   // 键码
 int value;  // 键值
    [...]
 char *action; // 事件发生时应执行的命令
 struct trigger *next;
} trigger;

解析过程大致是使用 getline() 读取一行数据,然后用 strdup() 复制出来,最后再用 strtok_r 切割出按键、键值、执行命令。
嵌入式物联网需要学的东西真的非常多,千万不要学错了路线和内容,导致工资要不上去!

Triggerhappy:值得推荐的小型C语言开源项目

2、创建 socket 通讯、信号处理等辅助设施。

这些都是非核心功能,只是让 Triggerhappy 变得更易用。

int bind_cmdsocket(char *name);
struct command *read_command(int cmd_fd);
int send_command(int cmd_fd, enum command_type type, char *param, int passfd, int exclusive, char *tag);
int connect_cmdsocket(char *name);

Triggherhappy 包含 2 个可执行程序:后台服务程序 thd,命令行工具 thd-cmd。

thd 会调用 bind_cmdsocket() 创建一个 local socket,thd-cmd 通过 connect_cmdsocket() 连接它。接着 thd-cmd 可以通过 send_command() 发送命令,thd 则通过 read_command() 接收命令。

3、根据命令行参数创建 devices。

1 个 device 代表 1 个输入设备,所有的 device 都会挂在链表 device_list 上。

typedef struct device {
 int fd;
 char *devname;
 struct device *next;
} device;
void add_device(char *dev, int fd, int excl, char *tag);
int remove_device(char *dev);
void clear_devices(void);

add_device() 会构造一个 device,并且 open() 设备以得到文件描述符 fd。

4、不断循环从所有 devices 中读取数据。

Triggerhappy:值得推荐的小型C语言开源项目

这个步骤是 Triggerhappy 的核心功能,是一个基于 select() 的eventloop。

这个 eventloop 很简单,就是用 select() 检测所有 devices,当有数据可读时,对所有 devices 都调用 read_event() 读取数据:

Triggerhappy:值得推荐的小型C语言开源项目

点击查看大图

读到数据后,会根据键值找到相应的 trigger,并执行 trigger 对应的命令,实现方法就是 fork() system():

Triggerhappy:值得推荐的小型C语言开源项目

点击查看大图

到此,Triggerhappy 的核心流程就分析完毕了。

总结

Triggerhappy 功能划分很清晰,大约有 7、8个功能模块,每一个模块的实现都非常的干净利索,代码都不超过 300 行,且风格和 Linux 设备驱动类似。

综合来看,Triggerhappy 符合 UNIX 的设计哲学。它简单实用,内容丰富,代码清晰,是一个非常嵌入式开发人员练习 Linux 应用编程的开源项目。

内容出处:,

声明:本网站所收集的部分公开资料来源于互联网,转载的目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。如果您发现网站上有侵犯您的知识产权的作品,请与我们取得联系,我们会及时修改或删除。文章链接:http://www.yixao.com/procedure/30193.html

发表评论

登录后才能评论