]> git.gir.st - sendHID.git/blob - main.c
added modified version of pass, the license for it, and a small demo for sending...
[sendHID.git] / main.c
1 /*
2 description: sends a sequence of keystrokes to the hid device.
3 parameters:
4 device file (e.g. /dev/hidg0)
5 keyboard layout (1=en_us, 2=de_at, 3=de_at-nodeadkeys)
6 unicode method: 1=gtk_holddown, 2=gtk_spaceend, 3=windows
7 the string to send (as whitespace is important, the `echo` way of concatenating all parameters is not supported. if your string has white space in it and you are in an interactive session, quote your string.)
8 */
9 #include <stdio.h>
10 #include <stdlib.h>
11 #include <string.h>
12 #include "scancodes.h"
13
14 enum params {//argv-indices:
15 P_EXE, //executable name
16 P_DEV, //device file
17 P_LAY, //layout
18 P_UNI, //unicode method
19 P_STR, //string to type
20 NUM_P //number of parameters
21 };
22 enum kbdl { //keyboard layouts:
23 na_NA, //reserved
24 en_US,
25 de_AT,
26 de_ND //de_AT-nodeadkeys
27 };
28 enum uni_m {//unicode methods:
29 SKIP, //ignore any keys not on the layout
30 GTK_HOLD, //hold ctrl and shift while entering hex values
31 GTK_SPACE, //end hex sequence with spacebar
32 WINDOWS //use alt+numpad
33 };
34 enum errors {
35 ERR_SUCCESS, //no error
36 ERR_ARGCOUNT, //wrong number of arguments
37 ERR_SYMBOL, //symbol not in look up table
38 ERR_LAYOUT, //parameter P_LAY does not contain a correct keyboard layout
39 ERR_LAZY //i haven't done this
40 };
41
42 void send_key (FILE* hid_dev, unsigned short key, unsigned short mod);
43 struct layout* tolay (struct keysym* s, enum kbdl layout);
44 enum errors send_unicode (FILE* hid_dev, unsigned int unicode, enum uni_m method, enum kbdl layout);
45
46 int main (int argc, char** argv) {
47 if (argc != NUM_P) {
48 fprintf (stderr, "Usage: %s <device file> <layout> <unicode> \"<string>\"\n", argv[P_EXE]);
49 return ERR_ARGCOUNT;
50 }
51 FILE* hid_dev = fopen ("/dev/hidg0", "w");
52 for (int i = 0; i < strlen (argv[P_STR]); i++) {
53
54 char tmp[UTF8_MAX_LENGTH] = {argv[P_STR][i], argv[P_STR][i+1], argv[P_STR][i+2], '\0'};
55 //TODO: replace by something less stupid
56 if (argv[P_STR][i] < 128) { // not multi byte
57 tmp[1] = '\0';
58 } else { // is multi byte
59 if (argv[P_STR][i] < 0xe0) {
60 i++; //skip next thing
61 tmp[2] = 0;
62 } else {
63 i+=2; //WARN: fails on utf8 > 3 byte
64 }
65 }
66
67 struct keysym* s = toscan (tmp);
68 if (s == NULL) {
69 fprintf (stderr, "Key Symbol not found.\n");
70 fclose (hid_dev);
71 return ERR_SYMBOL;
72 }
73 struct layout* l = tolay (s, atoi (argv[P_LAY]));
74 if (l == NULL) {
75 fprintf (stderr, "Unrecognised keyboard layout.\n");
76 fclose (hid_dev);
77 return ERR_LAYOUT;
78 }
79 if (l->key != 0x00) {
80 send_key(hid_dev, l->key, l->mod);
81 send_key(hid_dev, '\0', '\0'); //release all keys
82 if (l->is_dead) {
83 //dead keys need to be pressed twice to show up
84 send_key(hid_dev, l->key, l->mod);
85 send_key(hid_dev, '\0', '\0'); //release all keys
86 }
87 } else {
88 //key does not exist in this layout, use unicode method
89 //fprintf (stderr, "Warning: Key '%s'(0x%x) not in this layout!\n", s->sym, s->unicode);
90 send_unicode (hid_dev, s->unicode, atoi (argv[P_UNI]), atoi(argv[P_LAY]));
91 }
92 }
93 fclose (hid_dev);
94
95 return ERR_SUCCESS;
96 }
97
98 void send_key (FILE* hid_dev, unsigned short key, unsigned short mod) {
99 fprintf (hid_dev, "%c%c%c%c%c%c%c%c", mod, '\0', key, '\0', '\0', '\0', '\0', '\0');
100 }
101
102 struct layout* tolay (struct keysym* s, enum kbdl layout) {
103 switch (layout) {
104 case en_US: return &(s->en_us);
105 case de_AT: return &(s->de_at);
106 case de_ND: return &(s->de_nd);
107 default: return NULL;
108 }
109 }
110
111 enum errors send_unicode (FILE* hid_dev, unsigned int unicode, enum uni_m method, enum kbdl layout) {
112 char buf[10];
113 struct keysym* s;
114 struct layout* l;
115
116 if (unicode == 0x00) {
117 fprintf (stderr, "Symbol not in lookup table!\n");
118 return ERR_SYMBOL;
119 }
120
121 switch (method) {
122 case SKIP:
123 break;
124 case GTK_HOLD:
125 sprintf (buf, "%x", unicode);
126 s = toscan ("u");
127 l = tolay (s, layout);
128 send_key (hid_dev, l->key, MOD_LCTRL | MOD_LSHIFT);
129 for (int i = 0; i < strlen (buf); i++) {
130 s = toscan ((char[2]){buf[i], '\0'});
131 l = tolay (s, layout);
132 send_key (hid_dev, l->key, MOD_LCTRL | MOD_LSHIFT);
133 }
134 send_key (hid_dev, '\0', '\0');
135 break;
136 case GTK_SPACE:
137 sprintf (buf, "%x ", unicode);
138 s = toscan ("u");
139 l = tolay (s, layout);
140 send_key (hid_dev, l->key, MOD_LCTRL | MOD_LSHIFT);
141 for (int i = 0; i < strlen (buf); i++) {
142 s = toscan ((char[2]){buf[i], '\0'});
143 l = tolay (s, layout);
144 send_key (hid_dev, l->key, MOD_NONE);
145 }
146 send_key (hid_dev, '\0', '\0');
147 break;
148 case WINDOWS:
149 fprintf (stderr, "windows method not implemented!\n");
150 return ERR_LAZY;
151 default:
152 fprintf (stderr, "unknown unicode method!\n");
153 return ERR_LAYOUT; //TODO: better error code
154 }
155 return ERR_SUCCESS;
156 }
Imprint / Impressum