1
2
3
4
5
6
7
8
9
10#include "libbb.h"
11
12
13#define DEFAULT_CHAT_TIMEOUT 45*1000
14
15
16#define MAX_ABORT_LEN 50
17
18
19enum {
20 ERR_OK = 0,
21 ERR_MEM,
22 ERR_IO,
23 ERR_TIMEOUT,
24 ERR_ABORT,
25
26
27};
28
29
30#define exitcode bb_got_signal
31
32
33static void signal_handler(UNUSED_PARAM int signo)
34{
35
36 exitcode = ERR_IO;
37}
38
39#if !ENABLE_FEATURE_CHAT_IMPLICIT_CR
40#define unescape(s, nocr) unescape(s)
41#endif
42static size_t unescape(char *s, int *nocr)
43{
44 char *start = s;
45 char *p = s;
46
47 while (*s) {
48 char c = *s;
49
50
51
52 if ('\\' == c) {
53 c = *++s;
54 if (c) {
55#if ENABLE_FEATURE_CHAT_IMPLICIT_CR
56 if ('c' == c) {
57 *nocr = 1;
58 goto next;
59 }
60#endif
61 if ('N' == c) {
62 c = '\0';
63 } else if ('s' == c) {
64 c = ' ';
65#if ENABLE_FEATURE_CHAT_NOFAIL
66
67
68 } else if ('-' == c && (start + 1 == s)) {
69
70#endif
71 } else {
72 c = bb_process_escape_sequence((const char **)&s);
73 s--;
74 }
75 }
76
77 } else if ('^' == c) {
78 c = *++s-'@';
79 }
80
81 *p++ = c;
82#if ENABLE_FEATURE_CHAT_IMPLICIT_CR
83 next:
84#endif
85
86 s++;
87 }
88 *p = '\0';
89
90 return p - start;
91}
92
93int chat_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
94int chat_main(int argc UNUSED_PARAM, char **argv)
95{
96 int record_fd = -1;
97 bool echo = 0;
98
99 llist_t *aborts = NULL;
100
101 int timeout = DEFAULT_CHAT_TIMEOUT;
102
103#if ENABLE_FEATURE_CHAT_VAR_ABORT_LEN
104 size_t max_abort_len = 0;
105#else
106#define max_abort_len MAX_ABORT_LEN
107#endif
108#if ENABLE_FEATURE_CHAT_TTY_HIFI
109 struct termios tio0, tio;
110#endif
111
112 enum {
113 DIR_HANGUP = 0,
114 DIR_ABORT,
115#if ENABLE_FEATURE_CHAT_CLR_ABORT
116 DIR_CLR_ABORT,
117#endif
118 DIR_TIMEOUT,
119 DIR_ECHO,
120 DIR_SAY,
121 DIR_RECORD,
122 };
123
124
125 xfunc_error_retval = ERR_IO;
126
127
128 bb_signals(0
129 + (1 << SIGHUP)
130 + (1 << SIGINT)
131 + (1 << SIGTERM)
132 + (1 << SIGPIPE)
133 , signal_handler);
134
135#if ENABLE_FEATURE_CHAT_TTY_HIFI
136 tcgetattr(STDIN_FILENO, &tio);
137 tio0 = tio;
138 cfmakeraw(&tio);
139 tcsetattr(STDIN_FILENO, TCSAFLUSH, &tio);
140#endif
141
142#if ENABLE_FEATURE_CHAT_SWALLOW_OPTS
143 getopt32(argv, "vVsSE");
144 argv += optind;
145#else
146 argv++;
147#endif
148
149 while (*argv) {
150
151 int key = index_in_strings(
152 "HANGUP\0" "ABORT\0"
153#if ENABLE_FEATURE_CHAT_CLR_ABORT
154 "CLR_ABORT\0"
155#endif
156 "TIMEOUT\0" "ECHO\0" "SAY\0" "RECORD\0"
157 , *argv
158 );
159 if (key >= 0) {
160
161 char *arg = *++argv;
162
163 bool onoff = (0 != strcmp("OFF", arg));
164
165 if (DIR_HANGUP == key) {
166
167 signal(SIGHUP, onoff ? signal_handler : SIG_IGN);
168 } else if (DIR_ABORT == key) {
169
170#if ENABLE_FEATURE_CHAT_VAR_ABORT_LEN
171 size_t len = strlen(arg);
172 if (len > max_abort_len)
173 max_abort_len = len;
174#endif
175 llist_add_to_end(&aborts, arg);
176#if ENABLE_FEATURE_CHAT_CLR_ABORT
177 } else if (DIR_CLR_ABORT == key) {
178
179
180#if ENABLE_FEATURE_CHAT_VAR_ABORT_LEN
181 max_abort_len = 0;
182#endif
183 for (llist_t *l = aborts; l; l = l->link) {
184#if ENABLE_FEATURE_CHAT_VAR_ABORT_LEN
185 size_t len = strlen(l->data);
186#endif
187 if (!strcmp(arg, l->data)) {
188 llist_unlink(&aborts, l);
189 continue;
190 }
191#if ENABLE_FEATURE_CHAT_VAR_ABORT_LEN
192 if (len > max_abort_len)
193 max_abort_len = len;
194#endif
195 }
196#endif
197 } else if (DIR_TIMEOUT == key) {
198
199
200 timeout = atoi(arg) * 1000;
201
202
203 if (!timeout)
204 timeout = DEFAULT_CHAT_TIMEOUT;
205 } else if (DIR_ECHO == key) {
206
207
208 echo = onoff;
209 } else if (DIR_RECORD == key) {
210
211
212
213 if (record_fd > 0)
214 close(record_fd);
215
216 record_fd = (onoff) ? xopen(arg, O_WRONLY|O_CREAT|O_TRUNC) : -1;
217 } else if (DIR_SAY == key) {
218
219
220 bb_error_msg("%s", arg);
221 }
222
223 argv++;
224
225 } else {
226
227
228
229 int expect_len;
230 size_t buf_len = 0;
231 size_t max_len = max_abort_len;
232
233 struct pollfd pfd;
234#if ENABLE_FEATURE_CHAT_NOFAIL
235 int nofail = 0;
236#endif
237 char *expect = *argv++;
238
239
240 if (!expect)
241 goto expect_done;
242
243#if ENABLE_FEATURE_CHAT_NOFAIL
244
245 if ('-' == *expect) {
246
247 expect++;
248
249 nofail++;
250 }
251#endif
252
253#ifdef ___TEST___BUF___
254# undef COMMON_BUFSIZE
255# define COMMON_BUFSIZE 6
256#endif
257
258 expect_len = unescape(expect, &expect_len );
259 if (expect_len > max_len)
260 max_len = expect_len;
261
262
263
264 if (!expect_len)
265 goto expect_done;
266 if (max_len >= COMMON_BUFSIZE) {
267 exitcode = ERR_MEM;
268 goto expect_done;
269 }
270
271
272 pfd.fd = STDIN_FILENO;
273 pfd.events = POLLIN;
274 while (!exitcode
275 && poll(&pfd, 1, timeout) > 0
276 && (pfd.revents & POLLIN)
277 ) {
278#define buf bb_common_bufsiz1
279 llist_t *l;
280 ssize_t delta;
281
282
283 if (safe_read(STDIN_FILENO, buf+buf_len, 1) > 0) {
284
285 if (record_fd > 0) {
286 full_write(record_fd, buf+buf_len, 1);
287 }
288
289 if (echo > 0) {
290
291
292
293
294 full_write(STDERR_FILENO, buf+buf_len, 1);
295 }
296 buf_len++;
297
298 if (buf_len > COMMON_BUFSIZE) {
299 memmove(buf, buf+buf_len-max_len, max_len);
300 buf_len = max_len;
301 }
302 }
303
304
305
306
307
308
309 for (l = aborts, exitcode = ERR_ABORT; l; l = l->link, ++exitcode) {
310 size_t len = strlen(l->data);
311 delta = buf_len-len;
312 if (delta >= 0 && !memcmp(buf+delta, l->data, len))
313 goto expect_done;
314 }
315 exitcode = ERR_OK;
316
317
318 delta = buf_len - expect_len;
319 if (delta >= 0 && !memcmp(buf+delta, expect, expect_len))
320 goto expect_done;
321#undef buf
322 }
323
324
325 exitcode = ERR_TIMEOUT;
326 expect_done:
327#if ENABLE_FEATURE_CHAT_NOFAIL
328
329
330 if (nofail) {
331 if (!exitcode) {
332
333 while (*argv && argv[1] && '-' == argv[1][0])
334 argv += 2;
335
336
337 if (!*argv++ || !*argv++)
338 break;
339 }
340
341 if (ERR_IO != exitcode)
342 exitcode = ERR_OK;
343 }
344#endif
345
346 if (exitcode)
347 break;
348
349
350
351
352 if (*argv) {
353#if ENABLE_FEATURE_CHAT_IMPLICIT_CR
354 int nocr = 0;
355#endif
356 char *loaded = NULL;
357 size_t len;
358 char *buf = *argv++;
359
360
361
362 if ('@' == *buf) {
363
364 trim(++buf);
365 buf = loaded = xmalloc_xopen_read_close(buf, NULL);
366 }
367
368 len = unescape(buf, &nocr);
369
370
371 alarm(timeout);
372 pfd.fd = STDOUT_FILENO;
373 pfd.events = POLLOUT;
374 while (len && !exitcode
375 && poll(&pfd, 1, -1) > 0
376 && (pfd.revents & POLLOUT)
377 ) {
378#if ENABLE_FEATURE_CHAT_SEND_ESCAPES
379
380
381 char c = *buf;
382 if ('\\' == c) {
383 c = *++buf;
384 if ('d' == c) {
385 sleep(1);
386 len--;
387 continue;
388 }
389 if ('p' == c) {
390 usleep(10000);
391 len--;
392 continue;
393 }
394 if ('K' == c) {
395 tcsendbreak(STDOUT_FILENO, 0);
396 len--;
397 continue;
398 }
399 buf--;
400 }
401 if (safe_write(STDOUT_FILENO, buf, 1) != 1)
402 break;
403 len--;
404 buf++;
405#else
406 len -= full_write(STDOUT_FILENO, buf, len);
407#endif
408 }
409 alarm(0);
410
411
412 if (len)
413 exitcode = ERR_IO;
414
415
416 if (loaded)
417 free(loaded);
418#if ENABLE_FEATURE_CHAT_IMPLICIT_CR
419
420 else if (!nocr)
421 xwrite(STDOUT_FILENO, "\r", 1);
422#endif
423
424 if (exitcode)
425 break;
426 }
427 }
428 }
429
430#if ENABLE_FEATURE_CHAT_TTY_HIFI
431 tcsetattr(STDIN_FILENO, TCSAFLUSH, &tio0);
432#endif
433
434 return exitcode;
435}
436