4 * This file contains Unix specific code for setting terminal modes,
5 * very little is specific to ZMODEM or YMODEM per se (that code is in
6 * sz.c and rz.c). The CRC-16 routines used by XMODEM, YMODEM, and ZMODEM
7 * are also in this file, a fast table driven macro version
9 * V7/BSD HACKERS: SEE NOTES UNDER mode(2) !!!
11 * This file is #included so the main file can set parameters such as HOWMANY.
12 * See the main files (rz.c/sz.c) for compile instructions.
16 #include <sys/types.h>
21 long Locmode
; /* Saved "local mode" for 4.x BSD "new driver" */
22 long Locbit
= LLITOUT
; /* Bit SUPPOSED to disable output translations */
28 #include <sys/types.h>
41 #include <sys/types.h>
44 #include <sys/ioctl.h>
45 #define OS "SYS III/V"
52 static unsigned getspeed(int code
);
55 Howmany must be
255 or less
59 * return 1 iff stdout and stderr are different devices
60 * indicating this program operating with a modem on a
63 int Fromcu
; /* Were called from cu or yam */
69 fstat(1, &a
); fstat(2, &b
);
70 Fromcu
= a
.st_rdev
!= b
.st_rdev
;
77 fprintf(stderr
,"Please read the manual page BUGS chapter!\r\n");
101 int Twostop
; /* Use two stop bits */
108 * Return non 0 iff something to read from io descriptor f
114 ioctl(f
, FIONREAD
, &lf
);
122 char checked
= '\0' ;
124 * Nonblocking I/O is a bit different in System V, Release 2
130 savestat
= fcntl(f
, F_GETFL
) ;
131 fcntl(f
, F_SETFL
, savestat
| O_NDELAY
) ;
132 lf
= read(f
, &checked
, 1) ;
133 fcntl(f
, F_SETFL
, savestat
) ;
145 for (n
=0; speeds
[n
].baudr
; ++n
)
146 if (speeds
[n
].speedcode
== code
)
147 return speeds
[n
].baudr
;
148 return 38400; /* Assume fifo if ioctl failed */
154 struct termios oldtty
, tty
;
157 struct termio oldtty
, tty
;
159 struct sgttyb oldtty
, tty
;
160 struct tchars oldtch
, tch
;
164 int iofd
= 0; /* File descriptor for ioctls & reads */
168 * 3: save old tty stat, set raw mode with flow control
169 * 2: set XON/XOFF for sb/sz with ZMODEM or YMODEM-g
170 * 1: save old tty stat, set raw mode
171 * 0: restore original tty mode
176 static int did0
= FALSE
;
181 case 2: /* Un-raw mode used by sz, sb when -g detected */
183 (void) tcgetattr(iofd
, &oldtty
);
186 tty
.c_iflag
= BRKINT
|IXON
;
188 tty
.c_oflag
= 0; /* Transparent output */
190 tty
.c_cflag
&= ~PARENB
; /* Disable parity */
191 tty
.c_cflag
|= CS8
; /* Set character size = 8 */
193 tty
.c_cflag
|= CSTOPB
; /* Set two stop bits */
197 tty
.c_cc
[VINTR
] = Zmodem
? 03:030; /* Interrupt char */
198 tty
.c_cc
[VQUIT
] = -1; /* Quit char */
199 tty
.c_cc
[VMIN
] = 3; /* This many chars satisfies reads */
200 tty
.c_cc
[VTIME
] = 1; /* or in this many tenths of seconds */
202 (void) tcsetattr(iofd
, TCSANOW
, &tty
);
208 (void) tcgetattr(iofd
, &oldtty
);
211 tty
.c_iflag
= n
==3 ? (IGNBRK
|IXOFF
) : IGNBRK
;
213 /* No echo, crlf mapping, INTR, QUIT, delays, no erase/kill */
214 tty
.c_lflag
&= ~(ECHO
| ECHONL
| ICANON
| ISIG
| IEXTEN
);
216 tty
.c_oflag
= 0; /* Transparent output */
218 tty
.c_cflag
&= ~PARENB
; /* Same baud rate, disable parity */
219 tty
.c_cflag
|= CS8
; /* Set character size = 8 */
221 tty
.c_cflag
|= CSTOPB
; /* Set two stop bits */
222 tty
.c_cc
[VMIN
] = HOWMANY
; /* This many chars satisfies reads */
223 tty
.c_cc
[VTIME
] = 1; /* or in this many tenths of seconds */
224 (void) tcsetattr(iofd
, TCSANOW
, &tty
);
226 Baudrate
= cfgetospeed(&tty
);
230 case 2: /* Un-raw mode used by sz, sb when -g detected */
232 (void) ioctl(iofd
, TCGETA
, &oldtty
);
235 tty
.c_iflag
= BRKINT
|IXON
;
237 tty
.c_oflag
= 0; /* Transparent output */
239 tty
.c_cflag
&= ~PARENB
; /* Disable parity */
240 tty
.c_cflag
|= CS8
; /* Set character size = 8 */
242 tty
.c_cflag
|= CSTOPB
; /* Set two stop bits */
246 tty
.c_lflag
= Zmodem
? 0 : ISIG
;
247 tty
.c_cc
[VINTR
] = Zmodem
? -1:030; /* Interrupt char */
250 tty
.c_cc
[VINTR
] = Zmodem
? 03:030; /* Interrupt char */
252 tty
.c_cc
[VQUIT
] = -1; /* Quit char */
256 tty
.c_cc
[VMIN
] = 3; /* This many chars satisfies reads */
258 tty
.c_cc
[VTIME
] = 1; /* or in this many tenths of seconds */
260 (void) ioctl(iofd
, TCSETAW
, &tty
);
266 (void) ioctl(iofd
, TCGETA
, &oldtty
);
269 tty
.c_iflag
= n
==3 ? (IGNBRK
|IXOFF
) : IGNBRK
;
271 /* No echo, crlf mapping, INTR, QUIT, delays, no erase/kill */
272 tty
.c_lflag
&= ~(ECHO
| ICANON
| ISIG
);
274 tty
.c_oflag
= 0; /* Transparent output */
276 tty
.c_cflag
&= ~PARENB
; /* Same baud rate, disable parity */
277 tty
.c_cflag
|= CS8
; /* Set character size = 8 */
279 tty
.c_cflag
|= CSTOPB
; /* Set two stop bits */
281 tty
.c_cc
[VMIN
] = 1; /* This many chars satisfies reads */
283 tty
.c_cc
[VMIN
] = HOWMANY
; /* This many chars satisfies reads */
285 tty
.c_cc
[VTIME
] = 1; /* or in this many tenths of seconds */
286 (void) ioctl(iofd
, TCSETAW
, &tty
);
288 Baudrate
= getspeed(tty
.c_cflag
& CBAUD
);
293 * NOTE: this should transmit all 8 bits and at the same time
294 * respond to XOFF/XON flow control. If no FIONREAD or other
295 * READCHECK alternative, also must respond to INTRRUPT char
296 * This doesn't work with V7. It should work with LLITOUT,
297 * but LLITOUT was broken on the machine I tried it on.
299 case 2: /* Un-raw mode used by sz, sb when -g detected */
302 ioctl(iofd
, TIOCEXCL
, 0);
304 ioctl(iofd
, TIOCGETP
, &oldtty
);
305 ioctl(iofd
, TIOCGETC
, (struct sgttyb
*) &oldtch
);
307 ioctl(iofd
, TIOCLGET
, &Locmode
);
313 tch
.t_intrc
= Zmodem
? -1:030; /* Interrupt char */
315 tch
.t_intrc
= Zmodem
? 03:030; /* Interrupt char */
318 tty
.sg_flags
|= ODDP
;
321 tty
.sg_flags
|= EVENP
;
324 tty
.sg_flags
|= CBREAK
;
327 tty
.sg_flags
&= ~ALLDELAY
;
330 tty
.sg_flags
&= ~CRMOD
;
333 tty
.sg_flags
&= ~ECHO
;
336 tty
.sg_flags
&= ~LCASE
;
339 ioctl(iofd
, TIOCSETP
, &tty
);
340 ioctl(iofd
, TIOCSETC
, (struct sgttyb
*) &tch
);
342 ioctl(iofd
, TIOCLBIS
, &Locbit
);
344 bibi(99); /* un-raw doesn't work w/o lit out */
351 ioctl(iofd
, TIOCEXCL
, 0);
353 ioctl(iofd
, TIOCGETP
, &oldtty
);
354 ioctl(iofd
, TIOCGETC
, (struct sgttyb
*) &oldtch
);
356 ioctl(iofd
, TIOCLGET
, &Locmode
);
361 tty
.sg_flags
&= ~ECHO
;
362 ioctl(iofd
, TIOCSETP
, &tty
);
364 Baudrate
= getspeed(tty
.sg_ospeed
);
371 /* Wait for output to drain, flush input queue, restore
372 * modes and restart output.
374 (void) tcsetattr(iofd
, TCSAFLUSH
, &oldtty
);
375 (void) tcflow(iofd
, TCOON
);
378 (void) ioctl(iofd
, TCSBRK
, 1); /* Wait for output to drain */
379 (void) ioctl(iofd
, TCFLSH
, 1); /* Flush input queue */
380 (void) ioctl(iofd
, TCSETAW
, &oldtty
); /* Restore modes */
381 (void) ioctl(iofd
, TCXONC
,1); /* Restart output */
384 ioctl(iofd
, TIOCSETP
, &oldtty
);
385 ioctl(iofd
, TIOCSETC
, (struct sgttyb
*) &oldtch
);
387 ioctl(iofd
, TIOCNXCL
, 0);
390 ioctl(iofd
, TIOCLSET
, &Locmode
);
403 tcsendbreak(iofd
, 1);
409 ioctl(iofd
, TIOCSBRK
, 0);
411 ioctl(iofd
, TIOCCBRK
, 0);
416 ioctl(iofd
, TCSBRK
, 0);