ThinkPad 键盘上的第一行键现在默认在不按下Fn键时执行多媒体按键功能,按下Fn时才能执行F1-F12的功能,对于像我这种 Vim 和 htop 等的用户来说颇为不便。还好,BIOS 选项里可以改回来。
然后我遇到了 X250,发现Insert和End键怎么也受那个选项的影响了!也就是,如果F1-F12设置得方便了,那么End键就需要Fn键配合。而Home、End这种光标移动键虽然很少用,但毕竟还是要用到的,比如在 htop、weechat、mutt 以及不支持自定义编辑键的 Qt 程序里的时候。
所以呢,我在~/.Xmodmap
里把这两个键交换了:
keysym End = Insert keysym Insert = End
这下子用笔记本上的键盘是没问题了。可是我用外接键盘的话,这两个键就又反过来了 Orz……
当然网上会有 udev 规则,在插上外设时跑个脚本什么的。可不管怎么映射,总有个键盘的按键是反的啊!
仔细询问 Google 之后,在 Gentoo 的论坛里终于发现这么一条线索:
To alter keymap of a particular keyboard you need to issue EVIOCSKEYCODE ioctl on corresponding /dev/input/eventX node.
所以,找到相应的 event 设备文件之后,只需要 ioctl 一下就可以了?可是EVIOCSKEYCODE
是个什么鬼啊……
找过 manpages、头文件、内核文档、Google 之后,我不得不相信这个东西真的没文档!于是只好看源码了……还好有 LXR,不用在本地近一个G的源码里搜索。
所以,翻完文档又试验,最终有了这个程序:
#include<sys/ioctl.h> #include<sys/types.h> #include<sys/stat.h> #include<fcntl.h> #include<stdio.h> #include<stdlib.h> #include<linux/input.h> #define EVENT_FILE "/dev/input/by-path/platform-i8042-serio-0-event-kbd" int main(int argc, char **argv){ unsigned int codes[][2] = { {0xd2, 107}, //Insert -> End {0xcf, 110}, //End -> Insert {0, 0}, }; int fd = open(EVENT_FILE, O_RDONLY); if(fd < 0) { perror("open " EVENT_FILE); exit(1); } unsigned int (*p)[2]; for(p=codes; *p[0]; p++){ if(ioctl(fd, EVIOCSKEYCODE, *p)) { perror("ioctl EVIOCSKEYCODE"); exit(1); } } return 0; }
那个EVENT_FILE
当然就是笔记本键盘的节点啦,在/dev/input/by-path
下很容易识别的。
编译之后,每次启动系统后执行一次就可以了(大概)。
对了,顺便说一下,找那些代码可以用 showkey 和 getkeycodes 之类的命令。当然我们有 setkeycodes,但是它不能为指定键盘单独设置。传进去的参数,第一个整数是 scancode,就是硬件上报的编码,第二个是 keycode,内核给键的编码,不同硬件的不同 scancode 可以对应同一个 keycode(比如本文所做的)。然后文本终端还有 keymap、X Window 还有键盘布局和 xmodmap,大概是把 keycode 映射到可读的键名。