1 /* 2 * Copyright (c) 1995 Ugen J.S.Antsilevich 3 * 4 * Redistribution and use in source forms, with and without modification, 5 * are permitted provided that this entire comment appears intact. 6 * 7 * Redistribution in binary form may occur without any restrictions. 8 * Obviously, it would be nice if you gave credit where credit is due 9 * but requiring it would be too onerous. 10 * 11 * This software is provided ``AS IS'' without any warranties of any kind. 12 * 13 * Snoop stuff. 14 */ 15 16 #ifndef lint 17 static const char rcsid[] = 18 "$FreeBSD$"; 19 #endif /* not lint */ 20 21 #include <sys/types.h> 22 #include <sys/fcntl.h> 23 #include <sys/filio.h> 24 #include <sys/snoop.h> 25 #include <sys/stat.h> 26 27 #include <err.h> 28 #include <locale.h> 29 #include <paths.h> 30 #include <signal.h> 31 #include <stdio.h> 32 #include <stdlib.h> 33 #include <string.h> 34 #include <sysexits.h> 35 #include <termcap.h> 36 #include <termios.h> 37 #include <unistd.h> 38 39 #define MSG_INIT "Snoop started." 40 #define MSG_OFLOW "Snoop stopped due to overflow. Reconnecting." 41 #define MSG_CLOSED "Snoop stopped due to tty close. Reconnecting." 42 #define MSG_CHANGE "Snoop device change by user request." 43 #define MSG_NOWRITE "Snoop device change due to write failure." 44 45 46 #define DEV_NAME_LEN 1024 /* for /dev/ttyXX++ */ 47 #define MIN_SIZE 256 48 49 #define CHR_SWITCH 24 /* Ctrl+X */ 50 #define CHR_CLEAR 23 /* Ctrl+V */ 51 52 53 int opt_reconn_close = 0; 54 int opt_reconn_oflow = 0; 55 int opt_interactive = 1; 56 int opt_timestamp = 0; 57 int opt_write = 0; 58 int opt_no_switch = 0; 59 60 char dev_name[DEV_NAME_LEN]; 61 int snp_io; 62 dev_t snp_tty; 63 int std_in = 0, std_out = 1; 64 65 66 int clear_ok = 0; 67 struct termios otty; 68 char tbuf[1024], buf[1024]; 69 70 71 void 72 clear() 73 { 74 if (clear_ok) 75 tputs(buf, 1, putchar); 76 fflush(stdout); 77 } 78 79 void 80 timestamp(buf) 81 char *buf; 82 { 83 time_t t; 84 char btmp[1024]; 85 clear(); 86 printf("\n---------------------------------------------\n"); 87 t = time(NULL); 88 strftime(btmp, 1024, "Time: %d %b %H:%M", localtime(&t)); 89 printf("%s\n", btmp); 90 printf("%s\n", buf); 91 printf("---------------------------------------------\n"); 92 fflush(stdout); 93 } 94 95 void 96 set_tty() 97 { 98 struct termios ntty; 99 100 tcgetattr (std_in, &otty); 101 ntty = otty; 102 ntty.c_lflag &= ~ICANON; /* disable canonical operation */ 103 ntty.c_lflag &= ~ECHO; 104 #ifdef FLUSHO 105 ntty.c_lflag &= ~FLUSHO; 106 #endif 107 #ifdef PENDIN 108 ntty.c_lflag &= ~PENDIN; 109 #endif 110 #ifdef IEXTEN 111 ntty.c_lflag &= ~IEXTEN; 112 #endif 113 ntty.c_cc[VMIN] = 1; /* minimum of one character */ 114 ntty.c_cc[VTIME] = 0; /* timeout value */ 115 116 ntty.c_cc[VINTR] = 07; /* ^G */ 117 ntty.c_cc[VQUIT] = 07; /* ^G */ 118 tcsetattr (std_in, TCSANOW, &ntty); 119 } 120 121 void 122 unset_tty() 123 { 124 tcsetattr (std_in, TCSANOW, &otty); 125 } 126 127 128 void 129 fatal(err, buf) 130 unsigned int err; 131 char *buf; 132 { 133 unset_tty(); 134 if (buf) 135 errx(err, "fatal: %s", buf); 136 else 137 exit(err); 138 } 139 140 int 141 open_snp() 142 { 143 char snp[] = {_PATH_DEV "snpX"}; 144 char c; 145 int f, mode; 146 147 if (opt_write) 148 mode = O_RDWR; 149 else 150 mode = O_RDONLY; 151 152 for (c = '0'; c <= '9'; c++) { 153 snp[8] = c; 154 if ((f = open(snp, mode)) < 0) 155 continue; 156 return f; 157 } 158 fatal(EX_OSFILE, "cannot open snoop device"); 159 return (0); 160 } 161 162 163 void 164 cleanup() 165 { 166 if (opt_timestamp) 167 timestamp("Logging Exited."); 168 close(snp_io); 169 unset_tty(); 170 exit(EX_OK); 171 } 172 173 174 static void 175 usage() 176 { 177 fprintf(stderr, "usage: watch [-ciotnW] [tty name]\n"); 178 exit(EX_USAGE); 179 } 180 181 void 182 setup_scr() 183 { 184 char *cbuf = buf, *term; 185 if (!opt_interactive) 186 return; 187 if ((term = getenv("TERM"))) 188 if (tgetent(tbuf, term) == 1) 189 if (tgetstr("cl", &cbuf)) 190 clear_ok = 1; 191 set_tty(); 192 clear(); 193 } 194 195 196 int 197 ctoh(c) 198 char c; 199 { 200 if (c >= '0' && c <= '9') 201 return (int) (c - '0'); 202 203 if (c >= 'a' && c <= 'f') 204 return (int) (c - 'a' + 10); 205 206 fatal(EX_DATAERR, "bad tty number"); 207 return (0); 208 } 209 210 211 void 212 detach_snp() 213 { 214 dev_t dev; 215 216 dev = -1; 217 ioctl(snp_io, SNPSTTY, &dev); 218 } 219 220 void 221 attach_snp() 222 { 223 if (ioctl(snp_io, SNPSTTY, &snp_tty) != 0) 224 fatal(EX_UNAVAILABLE, "cannot attach to tty"); 225 if (opt_timestamp) 226 timestamp("Logging Started."); 227 } 228 229 230 void 231 set_dev(name) 232 char *name; 233 { 234 char buf[DEV_NAME_LEN]; 235 struct stat sb; 236 237 if (strlen(name) > 5 && !strncmp(name, _PATH_DEV, sizeof _PATH_DEV - 1)) { 238 snprintf(buf, sizeof buf, "%s", name); 239 } 240 else { 241 if (strlen(name) == 2) 242 sprintf(buf, "%s%s", _PATH_TTY, name); 243 else 244 sprintf(buf, "%s%s", _PATH_DEV, name); 245 } 246 247 if (*name == '\0' || stat(buf, &sb) < 0) 248 fatal(EX_DATAERR, "bad device name"); 249 250 if ((sb.st_mode & S_IFMT) != S_IFCHR) 251 fatal(EX_DATAERR, "must be a character device"); 252 253 snp_tty = sb.st_rdev; 254 attach_snp(); 255 } 256 257 void 258 ask_dev(dev_name, msg) 259 char *dev_name, *msg; 260 { 261 char buf[DEV_NAME_LEN]; 262 int len; 263 264 clear(); 265 unset_tty(); 266 267 if (msg) 268 printf("%s\n", msg); 269 if (dev_name) 270 printf("Enter device name [%s]:", dev_name); 271 else 272 printf("Enter device name:"); 273 274 if (fgets(buf, DEV_NAME_LEN - 1, stdin)) { 275 len = strlen(buf); 276 if (buf[len - 1] == '\n') 277 buf[len - 1] = '\0'; 278 if (buf[0] != '\0' && buf[0] != ' ') 279 strcpy(dev_name, buf); 280 } 281 set_tty(); 282 } 283 284 #define READB_LEN 5 285 286 int 287 main(ac, av) 288 int ac; 289 char **av; 290 { 291 int res, nread, b_size = MIN_SIZE; 292 char ch, *buf, chb[READB_LEN]; 293 fd_set fd_s; 294 295 (void) setlocale(LC_TIME, ""); 296 297 if (isatty(std_out)) 298 opt_interactive = 1; 299 else 300 opt_interactive = 0; 301 302 303 while ((ch = getopt(ac, av, "Wciotn")) != -1) 304 switch (ch) { 305 case 'W': 306 opt_write = 1; 307 break; 308 case 'c': 309 opt_reconn_close = 1; 310 break; 311 case 'i': 312 opt_interactive = 1; 313 break; 314 case 'o': 315 opt_reconn_oflow = 1; 316 break; 317 case 't': 318 opt_timestamp = 1; 319 break; 320 case 'n': 321 opt_no_switch = 1; 322 break; 323 case '?': 324 default: 325 usage(); 326 } 327 328 signal(SIGINT, cleanup); 329 330 setup_scr(); 331 snp_io = open_snp(); 332 333 if (*(av += optind) == NULL) { 334 if (opt_interactive && !opt_no_switch) 335 ask_dev(dev_name, MSG_INIT); 336 else 337 fatal(EX_DATAERR, "no device name given"); 338 } else 339 strncpy(dev_name, *av, DEV_NAME_LEN); 340 341 set_dev(dev_name); 342 343 if (!(buf = (char *) malloc(b_size))) 344 fatal(EX_UNAVAILABLE, "malloc failed"); 345 346 FD_ZERO(&fd_s); 347 348 while (1) { 349 if (opt_interactive) 350 FD_SET(std_in, &fd_s); 351 FD_SET(snp_io, &fd_s); 352 res = select(snp_io + 1, &fd_s, NULL, NULL, NULL); 353 if (opt_interactive && FD_ISSET(std_in, &fd_s)) { 354 355 if ((res = ioctl(std_in, FIONREAD, &nread)) != 0) 356 fatal(EX_OSERR, "ioctl(FIONREAD)"); 357 if (nread > READB_LEN) 358 nread = READB_LEN; 359 if (read(std_in,chb,nread)!=nread) 360 fatal(EX_IOERR, "read (stdin) failed"); 361 362 switch (chb[0]) { 363 case CHR_CLEAR: 364 clear(); 365 break; 366 case CHR_SWITCH: 367 if (opt_no_switch) 368 break; 369 detach_snp(); 370 ask_dev(dev_name, MSG_CHANGE); 371 set_dev(dev_name); 372 break; 373 default: 374 if (opt_write) { 375 if (write(snp_io,chb,nread) != nread) { 376 detach_snp(); 377 if (opt_no_switch) 378 fatal(EX_IOERR, "write failed"); 379 ask_dev(dev_name, MSG_NOWRITE); 380 set_dev(dev_name); 381 } 382 } 383 384 } 385 } 386 if (!FD_ISSET(snp_io, &fd_s)) 387 continue; 388 389 if ((res = ioctl(snp_io, FIONREAD, &nread)) != 0) 390 fatal(EX_OSERR, "ioctl(FIONREAD)"); 391 392 switch (nread) { 393 case SNP_OFLOW: 394 if (opt_reconn_oflow) 395 attach_snp(); 396 else if (opt_interactive && !opt_no_switch) { 397 ask_dev(dev_name, MSG_OFLOW); 398 set_dev(dev_name); 399 } else 400 cleanup(); 401 case SNP_DETACH: 402 case SNP_TTYCLOSE: 403 if (opt_reconn_close) 404 attach_snp(); 405 else if (opt_interactive && !opt_no_switch) { 406 ask_dev(dev_name, MSG_CLOSED); 407 set_dev(dev_name); 408 } else 409 cleanup(); 410 default: 411 if (nread < (b_size / 2) && (b_size / 2) > MIN_SIZE) { 412 free(buf); 413 if (!(buf = (char *) malloc(b_size / 2))) 414 fatal(EX_UNAVAILABLE, "malloc failed"); 415 b_size = b_size / 2; 416 } 417 if (nread > b_size) { 418 b_size = (nread % 2) ? (nread + 1) : (nread); 419 free(buf); 420 if (!(buf = (char *) malloc(b_size))) 421 fatal(EX_UNAVAILABLE, "malloc failed"); 422 } 423 if (read(snp_io, buf, nread) < nread) 424 fatal(EX_IOERR, "read failed"); 425 if (write(std_out, buf, nread) < nread) 426 fatal(EX_IOERR, "write failed"); 427 } 428 } /* While */ 429 return(0); 430 } 431 432