xref: /freebsd/usr.bin/tip/libacu/biz31.c (revision 87569f75a91f298c52a71823c04d41cf53c88889)
1 /*	$OpenBSD: biz31.c,v 1.6 2001/10/24 18:38:58 millert Exp $	*/
2 /*	$NetBSD: biz31.c,v 1.5 1997/02/11 09:24:14 mrg Exp $	*/
3 
4 /*
5  * Copyright (c) 1983, 1993
6  *	The Regents of the University of California.  All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  * 3. All advertising materials mentioning features or use of this software
17  *    must display the following acknowledgement:
18  *	This product includes software developed by the University of
19  *	California, Berkeley and its contributors.
20  * 4. Neither the name of the University nor the names of its contributors
21  *    may be used to endorse or promote products derived from this software
22  *    without specific prior written permission.
23  *
24  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
25  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
26  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
27  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
28  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
29  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
30  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
31  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
32  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
33  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
34  * SUCH DAMAGE.
35  */
36 
37 #include <sys/cdefs.h>
38 __FBSDID("$FreeBSD$");
39 
40 #ifndef lint
41 #if 0
42 static char sccsid[] = "@(#)biz31.c	8.1 (Berkeley) 6/6/93";
43 static char rcsid[] = "$OpenBSD: biz31.c,v 1.6 2001/10/24 18:38:58 millert Exp $";
44 #endif
45 #endif /* not lint */
46 
47 #include "tip.h"
48 
49 #define MAXRETRY	3		/* sync up retry count */
50 #define DISCONNECT_CMD	"\21\25\11\24"	/* disconnection string */
51 
52 static	void sigALRM();
53 static	int timeout = 0;
54 static	jmp_buf timeoutbuf;
55 
56 /*
57  * Dial up on a BIZCOMP Model 1031 with either
58  * 	tone dialing (mod = "f")
59  *	pulse dialing (mod = "w")
60  */
61 static int
62 biz_dialer(num, mod)
63 	char *num, *mod;
64 {
65 	int connected = 0;
66 
67 	if (!bizsync(FD)) {
68 		logent(value(HOST), "", "biz", "out of sync");
69 		printf("bizcomp out of sync\n");
70 		delock(uucplock);
71 		exit(0);
72 	}
73 	if (boolean(value(VERBOSE)))
74 		printf("\nstarting call...");
75 	echo("#\rk$\r$\n");			/* disable auto-answer */
76 	echo("$>$.$ #\r");			/* tone/pulse dialing */
77 	echo(mod);
78 	echo("$\r$\n");
79 	echo("$>$.$ #\re$ ");			/* disconnection sequence */
80 	echo(DISCONNECT_CMD);
81 	echo("\r$\n$\r$\n");
82 	echo("$>$.$ #\rr$ ");			/* repeat dial */
83 	echo(num);
84 	echo("\r$\n");
85 	if (boolean(value(VERBOSE)))
86 		printf("ringing...");
87 	/*
88 	 * The reply from the BIZCOMP should be:
89 	 *	`^G NO CONNECTION\r\n^G\r\n'	failure
90 	 *	` CONNECTION\r\n^G'		success
91 	 */
92 	connected = detect(" ");
93 #ifdef ACULOG
94 	if (timeout) {
95 		char line[80];
96 
97 		(void)sprintf(line, "%ld second dial timeout",
98 			number(value(DIALTIMEOUT)));
99 		logent(value(HOST), num, "biz", line);
100 	}
101 #endif
102 	if (!connected)
103 		flush(" NO CONNECTION\r\n\07\r\n");
104 	else
105 		flush("CONNECTION\r\n\07");
106 	if (timeout)
107 		biz31_disconnect();	/* insurance */
108 	return (connected);
109 }
110 
111 biz31w_dialer(num, acu)
112 	char *num, *acu;
113 {
114 
115 	return (biz_dialer(num, "w"));
116 }
117 
118 biz31f_dialer(num, acu)
119 	char *num, *acu;
120 {
121 
122 	return (biz_dialer(num, "f"));
123 }
124 
125 biz31_disconnect()
126 {
127 
128 	write(FD, DISCONNECT_CMD, 4);
129 	sleep(2);
130 	tcflush(FD, TCIOFLUSH);
131 }
132 
133 biz31_abort()
134 {
135 
136 	write(FD, "\33", 1);
137 }
138 
139 static int
140 echo(s)
141 	char *s;
142 {
143 	char c;
144 
145 	while (c = *s++) switch (c) {
146 
147 	case '$':
148 		read(FD, &c, 1);
149 		s++;
150 		break;
151 
152 	case '#':
153 		c = *s++;
154 		write(FD, &c, 1);
155 		break;
156 
157 	default:
158 		write(FD, &c, 1);
159 		read(FD, &c, 1);
160 	}
161 }
162 
163 static void
164 sigALRM()
165 {
166 
167 	timeout = 1;
168 	longjmp(timeoutbuf, 1);
169 }
170 
171 static int
172 detect(s)
173 	char *s;
174 {
175 	sig_t f;
176 	char c;
177 
178 	f = signal(SIGALRM, sigALRM);
179 	timeout = 0;
180 	while (*s) {
181 		if (setjmp(timeoutbuf)) {
182 			printf("\07timeout waiting for reply\n");
183 			biz31_abort();
184 			break;
185 		}
186 		alarm(number(value(DIALTIMEOUT)));
187 		read(FD, &c, 1);
188 		alarm(0);
189 		if (c != *s++)
190 			break;
191 	}
192 	signal(SIGALRM, f);
193 	return (timeout == 0);
194 }
195 
196 static int
197 flush(s)
198 	char *s;
199 {
200 	sig_t f;
201 	char c;
202 
203 	f = signal(SIGALRM, sigALRM);
204 	while (*s++) {
205 		if (setjmp(timeoutbuf))
206 			break;
207 		alarm(10);
208 		read(FD, &c, 1);
209 		alarm(0);
210 	}
211 	signal(SIGALRM, f);
212 	timeout = 0;			/* guard against disconnection */
213 }
214 
215 /*
216  * This convoluted piece of code attempts to get
217  *  the bizcomp in sync.  If you don't have the capacity or nread
218  *  call there are gory ways to simulate this.
219  */
220 static int
221 bizsync(fd)
222 {
223 #ifdef FIOCAPACITY
224 	struct capacity b;
225 #	define chars(b)	((b).cp_nbytes)
226 #	define IOCTL	FIOCAPACITY
227 #endif
228 #ifdef FIONREAD
229 	long b;
230 #	define chars(b)	(b)
231 #	define IOCTL	FIONREAD
232 #endif
233 	int already = 0;
234 	char buf[10];
235 
236 retry:
237 	if (ioctl(fd, IOCTL, (caddr_t)&b) >= 0 && chars(b) > 0)
238 		tcflush(FD, TCIOFLUSH);
239 	write(fd, "\rp>\r", 4);
240 	sleep(1);
241 	if (ioctl(fd, IOCTL, (caddr_t)&b) >= 0) {
242 		if (chars(b) != 10) {
243 	nono:
244 			if (already > MAXRETRY)
245 				return (0);
246 			write(fd, DISCONNECT_CMD, 4);
247 			sleep(2);
248 			already++;
249 			goto retry;
250 		} else {
251 			read(fd, buf, 10);
252 			if (strncmp(buf, "p >\r\n\r\n>", 8))
253 				goto nono;
254 		}
255 	}
256 	return (1);
257 }
258