如何(可靠地)读取嵌入式(无头)USB 条码扫描仪 Linux?
How to (reliably) read USB barcode scanner in embedded (headless) Linux?
我需要将通用 USB 扫描仪(不是单个定义的模型)与嵌入式 Linux 设备(mips32/MT7620 运行 Linux 3.18,如果重要的话)。
所有扫描仪都在 "keyboard emulation mode" 中运行,果然,如果插入桌面 Linux 将它们的数据作为键盘输入直接发送到控制台。
这确实不会发生在设备上(即:我没有看到串行调试控制台的任何字符,这是有道理的,因为它错过了所有 X 输入子系统)。
所有扫描仪都在 /dev/input/event0
上显示为输入设备。
我目前的尝试是使用python evdev
;代码非常简单:
import asyncio
import evdev
edev = evdev.InputDevice('/dev/input/event0')
async for event in edev.async_read_loop():
if event.type == evdev.ecodes.EV_KEY and event.value == 1:
handle_event(event)
注意:这只是一个摘录片段;如果需要,我可以 post 一个可运行的示例。
这本质上是 (我正在使用 asyncio
,这会是个问题吗?)并且它 似乎 可以工作,但实际上会丢失事件。
如果条形码扫描器连续发送事件,我似乎很早就开始丢失事件(大约 16 个事件 <= 8 个字符后!)。
如果我可以在字符之间插入延迟(~1 毫秒就足够了),那么一切都会按预期工作,但这在许多扫描仪上都不是一个选项。
我错过了什么?
致有同样困境的人:
我找不到 Pure Python working 解决方案。
我使出了一个非常简单的"C"程序(大部分都是表格):
#include <stdio.h>
#include <fcntl.h>
#include <linux/input.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
void INThandler() {
exit(0);
}
char ttab[] = {
0, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b', /* Backspace */
'\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']','\n', /* Enter key */
0, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';','\'', '`', 0, /* Left shift */
'\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
char ntab[] = {
0, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b', /* Backspace */
'\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']','\n', /* Enter key */
0, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';','\'', '`', 0, /* Left shift */
'\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
char stab[] = {
0, 27, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '_', '+', 0, /* Backspace */
0, 'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', 0, /* Enter key */
0, 'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', ':', '"', 0,'\n', /* Left shift */
0, 'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', '?', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
int main() {
char devname[] = "/dev/input/event0";
int device = open(devname, O_RDONLY);
struct input_event ev;
int shift = 0;
char line[4096], *p = line;
signal(SIGINT, INThandler);
fputs("starting\n", stdout);
//fputs("starting\n", stderr);
while (1) {
read(device, &ev, sizeof(ev));
if (ev.type == 1) {
if (ev.code == 42)
shift = ev.value;
else if (ev.value) {
//printf("Key: %i State: %i\n", ev.code, ev.value);
char *t = shift? stab: ntab;
char ch = t[ev.code];
//printf("Key: %02d State: %d [%c]\n", ev.code, ev.value, ch);
if (ch == '\n') {
*p = '[=10=]';
fputs(line, stdout); fputc('\n', stdout); fflush(stdout);
//fputs(line, stderr); fputc('\n', stderr); fflush(stderr);
p = line;
} else
*p++ = ch;
}
}
}
}
... 由 Python:
驱动
async def handle_events(self):
log.debug("handle_events(%s): st start", self.dev)
while True:
if self.hid is None:
log.debug("handle_events(%s): checking...", self.dev)
self.hid = await asyncio.create_subprocess_exec(
'/usr/local/bin/readEvents',
stdout=asyncio.subprocess.PIPE
)
try:
out = await asyncio.wait_for(self.hid.stdout.readline(), 10)
except asyncio.TimeoutError:
log.debug("handle_events(%s): timeout on startup message", self.dev)
self.hid.terminate()
self.hid = None
else:
out = out.strip()
if out != b'starting':
log.debug("handle_events(%s): bad startup message: %s", self.dev, out)
self.hid.terminate()
self.hid = None
else:
log.debug("handle_events(%s): startup msg received", self.dev)
else:
if self.hid.stdout.at_eof():
log.info("handle_events(%s): killing subprocess", self.dev)
rc = await self.hid.wait()
log.info("handle_events(%s): subprocess exited with status %s", self.dev, rc)
self.hid = None
await asyncio.sleep(10)
if self.hid is not None:
line = await self.hid.stdout.readline()
if not line:
log.error('handle_events: readline() ERROR:')
self.hid.terminate()
self.hid = None
else:
line = line.strip()
log.info('handle_events: received line: "%s"', line)
await self.process_line(line)
我相信 Python 驱动程序可以简化。这是我的(工作)应用程序的逐字记录。
嗨!
我需要将通用 USB 扫描仪(不是单个定义的模型)与嵌入式 Linux 设备(mips32/MT7620 运行 Linux 3.18,如果重要的话)。
所有扫描仪都在 "keyboard emulation mode" 中运行,果然,如果插入桌面 Linux 将它们的数据作为键盘输入直接发送到控制台。
这确实不会发生在设备上(即:我没有看到串行调试控制台的任何字符,这是有道理的,因为它错过了所有 X 输入子系统)。
所有扫描仪都在 /dev/input/event0
上显示为输入设备。
我目前的尝试是使用python evdev
;代码非常简单:
import asyncio
import evdev
edev = evdev.InputDevice('/dev/input/event0')
async for event in edev.async_read_loop():
if event.type == evdev.ecodes.EV_KEY and event.value == 1:
handle_event(event)
注意:这只是一个摘录片段;如果需要,我可以 post 一个可运行的示例。
这本质上是 asyncio
,这会是个问题吗?)并且它 似乎 可以工作,但实际上会丢失事件。
如果条形码扫描器连续发送事件,我似乎很早就开始丢失事件(大约 16 个事件 <= 8 个字符后!)。
如果我可以在字符之间插入延迟(~1 毫秒就足够了),那么一切都会按预期工作,但这在许多扫描仪上都不是一个选项。
我错过了什么?
致有同样困境的人: 我找不到 Pure Python working 解决方案。
我使出了一个非常简单的"C"程序(大部分都是表格):
#include <stdio.h>
#include <fcntl.h>
#include <linux/input.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
void INThandler() {
exit(0);
}
char ttab[] = {
0, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b', /* Backspace */
'\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']','\n', /* Enter key */
0, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';','\'', '`', 0, /* Left shift */
'\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
char ntab[] = {
0, 27, '1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '-', '=', '\b', /* Backspace */
'\t', 'q', 'w', 'e', 'r', 't', 'y', 'u', 'i', 'o', 'p', '[', ']','\n', /* Enter key */
0, 'a', 's', 'd', 'f', 'g', 'h', 'j', 'k', 'l', ';','\'', '`', 0, /* Left shift */
'\', 'z', 'x', 'c', 'v', 'b', 'n', 'm', ',', '.', '/', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
char stab[] = {
0, 27, '!', '@', '#', '$', '%', '^', '&', '*', '(', ')', '_', '+', 0, /* Backspace */
0, 'Q', 'W', 'E', 'R', 'T', 'Y', 'U', 'I', 'O', 'P', '{', '}', 0, /* Enter key */
0, 'A', 'S', 'D', 'F', 'G', 'H', 'J', 'K', 'L', ':', '"', 0,'\n', /* Left shift */
0, 'Z', 'X', 'C', 'V', 'B', 'N', 'M', '<', '>', '?', 0, /* Right shift */
'*',
0, /* Alt */
' ', /* Space bar */
0, /* Caps lock */
0, /* 59 - F1 key ... > */
0, 0, 0, 0, 0, 0, 0, 0,
0, /* < ... F10 */
0, /* 69 - Num lock*/
0, /* Scroll Lock */
0, /* Home key */
0, /* Up Arrow */
0, /* Page Up */
'-',
0, /* Left Arrow */
0,
0, /* Right Arrow */
'+',
0, /* 79 - End key*/
0, /* Down Arrow */
0, /* Page Down */
0, /* Insert Key */
0, /* Delete Key */
0, 0, 0,
0, /* F11 Key */
0, /* F12 Key */
0, /* All other keys are undefined */
};
int main() {
char devname[] = "/dev/input/event0";
int device = open(devname, O_RDONLY);
struct input_event ev;
int shift = 0;
char line[4096], *p = line;
signal(SIGINT, INThandler);
fputs("starting\n", stdout);
//fputs("starting\n", stderr);
while (1) {
read(device, &ev, sizeof(ev));
if (ev.type == 1) {
if (ev.code == 42)
shift = ev.value;
else if (ev.value) {
//printf("Key: %i State: %i\n", ev.code, ev.value);
char *t = shift? stab: ntab;
char ch = t[ev.code];
//printf("Key: %02d State: %d [%c]\n", ev.code, ev.value, ch);
if (ch == '\n') {
*p = '[=10=]';
fputs(line, stdout); fputc('\n', stdout); fflush(stdout);
//fputs(line, stderr); fputc('\n', stderr); fflush(stderr);
p = line;
} else
*p++ = ch;
}
}
}
}
... 由 Python:
驱动async def handle_events(self):
log.debug("handle_events(%s): st start", self.dev)
while True:
if self.hid is None:
log.debug("handle_events(%s): checking...", self.dev)
self.hid = await asyncio.create_subprocess_exec(
'/usr/local/bin/readEvents',
stdout=asyncio.subprocess.PIPE
)
try:
out = await asyncio.wait_for(self.hid.stdout.readline(), 10)
except asyncio.TimeoutError:
log.debug("handle_events(%s): timeout on startup message", self.dev)
self.hid.terminate()
self.hid = None
else:
out = out.strip()
if out != b'starting':
log.debug("handle_events(%s): bad startup message: %s", self.dev, out)
self.hid.terminate()
self.hid = None
else:
log.debug("handle_events(%s): startup msg received", self.dev)
else:
if self.hid.stdout.at_eof():
log.info("handle_events(%s): killing subprocess", self.dev)
rc = await self.hid.wait()
log.info("handle_events(%s): subprocess exited with status %s", self.dev, rc)
self.hid = None
await asyncio.sleep(10)
if self.hid is not None:
line = await self.hid.stdout.readline()
if not line:
log.error('handle_events: readline() ERROR:')
self.hid.terminate()
self.hid = None
else:
line = line.strip()
log.info('handle_events: received line: "%s"', line)
await self.process_line(line)
我相信 Python 驱动程序可以简化。这是我的(工作)应用程序的逐字记录。
嗨!