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