xref: /freebsd/usr.bin/tip/libacu/hayes.c (revision 5e3934b15a2741b2de6b217e77dc9d798d740804)
1fa92e883SRuslan Ermilov /*	$OpenBSD: hayes.c,v 1.13 2006/03/17 19:17:13 moritz Exp $	*/
20f3bdf5dSMark Murray /*	$NetBSD: hayes.c,v 1.6 1997/02/11 09:24:17 mrg Exp $	*/
30f3bdf5dSMark Murray 
4*8a16b7a1SPedro F. Giffuni /*-
5*8a16b7a1SPedro F. Giffuni  * SPDX-License-Identifier: BSD-3-Clause
6*8a16b7a1SPedro F. Giffuni  *
718cc36fbSJordan K. Hubbard  * Copyright (c) 1983, 1993
818cc36fbSJordan K. Hubbard  *	The Regents of the University of California.  All rights reserved.
918cc36fbSJordan K. Hubbard  *
1018cc36fbSJordan K. Hubbard  * Redistribution and use in source and binary forms, with or without
1118cc36fbSJordan K. Hubbard  * modification, are permitted provided that the following conditions
1218cc36fbSJordan K. Hubbard  * are met:
1318cc36fbSJordan K. Hubbard  * 1. Redistributions of source code must retain the above copyright
1418cc36fbSJordan K. Hubbard  *    notice, this list of conditions and the following disclaimer.
1518cc36fbSJordan K. Hubbard  * 2. Redistributions in binary form must reproduce the above copyright
1618cc36fbSJordan K. Hubbard  *    notice, this list of conditions and the following disclaimer in the
1718cc36fbSJordan K. Hubbard  *    documentation and/or other materials provided with the distribution.
18fa92e883SRuslan Ermilov  * 3. Neither the name of the University nor the names of its contributors
1918cc36fbSJordan K. Hubbard  *    may be used to endorse or promote products derived from this software
2018cc36fbSJordan K. Hubbard  *    without specific prior written permission.
2118cc36fbSJordan K. Hubbard  *
2218cc36fbSJordan K. Hubbard  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2318cc36fbSJordan K. Hubbard  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2418cc36fbSJordan K. Hubbard  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2518cc36fbSJordan K. Hubbard  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2618cc36fbSJordan K. Hubbard  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2718cc36fbSJordan K. Hubbard  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2818cc36fbSJordan K. Hubbard  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2918cc36fbSJordan K. Hubbard  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3018cc36fbSJordan K. Hubbard  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3118cc36fbSJordan K. Hubbard  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3218cc36fbSJordan K. Hubbard  * SUCH DAMAGE.
3318cc36fbSJordan K. Hubbard  */
3418cc36fbSJordan K. Hubbard 
3518cc36fbSJordan K. Hubbard /*
3618cc36fbSJordan K. Hubbard  * Routines for calling up on a Hayes Modem
3718cc36fbSJordan K. Hubbard  * (based on the old VenTel driver).
3818cc36fbSJordan K. Hubbard  * The modem is expected to be strapped for "echo".
3918cc36fbSJordan K. Hubbard  * Also, the switches enabling the DTR and CD lines
4018cc36fbSJordan K. Hubbard  * must be set correctly.
4118cc36fbSJordan K. Hubbard  * NOTICE:
4218cc36fbSJordan K. Hubbard  * The easy way to hang up a modem is always simply to
4318cc36fbSJordan K. Hubbard  * clear the DTR signal. However, if the +++ sequence
4418cc36fbSJordan K. Hubbard  * (which switches the modem back to local mode) is sent
4518cc36fbSJordan K. Hubbard  * before modem is hung up, removal of the DTR signal
4618cc36fbSJordan K. Hubbard  * has no effect (except that it prevents the modem from
4718cc36fbSJordan K. Hubbard  * recognizing commands).
4818cc36fbSJordan K. Hubbard  * (by Helge Skrivervik, Calma Company, Sunnyvale, CA. 1984)
4918cc36fbSJordan K. Hubbard  */
5018cc36fbSJordan K. Hubbard /*
5118cc36fbSJordan K. Hubbard  * TODO:
5218cc36fbSJordan K. Hubbard  * It is probably not a good idea to switch the modem
5318cc36fbSJordan K. Hubbard  * state between 'verbose' and terse (status messages).
5418cc36fbSJordan K. Hubbard  * This should be kicked out and we should use verbose
5518cc36fbSJordan K. Hubbard  * mode only. This would make it consistent with normal
5618cc36fbSJordan K. Hubbard  * interactive use thru the command 'tip dialer'.
5718cc36fbSJordan K. Hubbard  */
5818cc36fbSJordan K. Hubbard #include "tip.h"
5918cc36fbSJordan K. Hubbard 
600f3bdf5dSMark Murray #include <termios.h>
610f3bdf5dSMark Murray #include <sys/ioctl.h>
620f3bdf5dSMark Murray 
6318cc36fbSJordan K. Hubbard #define	min(a,b)	((a < b) ? a : b)
6418cc36fbSJordan K. Hubbard 
65fa92e883SRuslan Ermilov static	int dialtimeout = 0;
6618cc36fbSJordan K. Hubbard static	jmp_buf timeoutbuf;
67fa92e883SRuslan Ermilov 
6818cc36fbSJordan K. Hubbard #define DUMBUFLEN	40
6918cc36fbSJordan K. Hubbard static char dumbuf[DUMBUFLEN];
7018cc36fbSJordan K. Hubbard 
7118cc36fbSJordan K. Hubbard #define	DIALING		1
7218cc36fbSJordan K. Hubbard #define IDLE		2
7318cc36fbSJordan K. Hubbard #define CONNECTED	3
7418cc36fbSJordan K. Hubbard #define	FAILED		4
7518cc36fbSJordan K. Hubbard static	int state = IDLE;
7618cc36fbSJordan K. Hubbard 
77fa92e883SRuslan Ermilov static void	sigALRM(int);
78fa92e883SRuslan Ermilov static char	gobble(char *);
79fa92e883SRuslan Ermilov static void	error_rep(char);
80fa92e883SRuslan Ermilov static void	goodbye(void);
81fa92e883SRuslan Ermilov static int	hay_sync(void);
82fa92e883SRuslan Ermilov 
830f3bdf5dSMark Murray int
hay_dialer(char * num,char * acu)84fa92e883SRuslan Ermilov hay_dialer(char *num, char *acu)
8518cc36fbSJordan K. Hubbard {
860f3bdf5dSMark Murray 	char *cp;
870f3bdf5dSMark Murray 	int connected = 0;
8818cc36fbSJordan K. Hubbard 	char dummy;
890f3bdf5dSMark Murray 	struct termios cntrl;
900f3bdf5dSMark Murray #ifdef ACULOG
9118cc36fbSJordan K. Hubbard 	char line[80];
9218cc36fbSJordan K. Hubbard #endif
9318cc36fbSJordan K. Hubbard 	if (hay_sync() == 0)		/* make sure we can talk to the modem */
9418cc36fbSJordan K. Hubbard 		return(0);
9518cc36fbSJordan K. Hubbard 	if (boolean(value(VERBOSE)))
9618cc36fbSJordan K. Hubbard 		printf("\ndialing...");
9718cc36fbSJordan K. Hubbard 	fflush(stdout);
980f3bdf5dSMark Murray 	tcgetattr(FD, &cntrl);
990f3bdf5dSMark Murray 	cntrl.c_cflag |= HUPCL;
1000f3bdf5dSMark Murray 	tcsetattr(FD, TCSANOW, &cntrl);
1010f3bdf5dSMark Murray 	tcflush(FD, TCIOFLUSH);
10218cc36fbSJordan K. Hubbard 	write(FD, "ATv0\r", 5);	/* tell modem to use short status codes */
10318cc36fbSJordan K. Hubbard 	gobble("\r");
10418cc36fbSJordan K. Hubbard 	gobble("\r");
10518cc36fbSJordan K. Hubbard 	write(FD, "ATTD", 4);	/* send dial command */
1060f3bdf5dSMark Murray 	for (cp = num; *cp; cp++)
1070f3bdf5dSMark Murray 		if (*cp == '=')
1080f3bdf5dSMark Murray 			*cp = ',';
10918cc36fbSJordan K. Hubbard 	write(FD, num, strlen(num));
11018cc36fbSJordan K. Hubbard 	state = DIALING;
11118cc36fbSJordan K. Hubbard 	write(FD, "\r", 1);
11218cc36fbSJordan K. Hubbard 	connected = 0;
11318cc36fbSJordan K. Hubbard 	if (gobble("\r")) {
11418cc36fbSJordan K. Hubbard 		if ((dummy = gobble("01234")) != '1')
11518cc36fbSJordan K. Hubbard 			error_rep(dummy);
11618cc36fbSJordan K. Hubbard 		else
11718cc36fbSJordan K. Hubbard 			connected = 1;
11818cc36fbSJordan K. Hubbard 	}
11918cc36fbSJordan K. Hubbard 	if (connected)
12018cc36fbSJordan K. Hubbard 		state = CONNECTED;
12118cc36fbSJordan K. Hubbard 	else {
12218cc36fbSJordan K. Hubbard 		state = FAILED;
12318cc36fbSJordan K. Hubbard 		return (connected);	/* lets get out of here.. */
12418cc36fbSJordan K. Hubbard 	}
1250f3bdf5dSMark Murray 	tcflush(FD, TCIOFLUSH);
1260f3bdf5dSMark Murray #ifdef ACULOG
127fa92e883SRuslan Ermilov 	if (dialtimeout) {
128fa92e883SRuslan Ermilov 		(void)snprintf(line, sizeof line, "%ld second dial timeout",
12918cc36fbSJordan K. Hubbard 			number(value(DIALTIMEOUT)));
13018cc36fbSJordan K. Hubbard 		logent(value(HOST), num, "hayes", line);
13118cc36fbSJordan K. Hubbard 	}
13218cc36fbSJordan K. Hubbard #endif
133fa92e883SRuslan Ermilov 	if (dialtimeout)
13418cc36fbSJordan K. Hubbard 		hay_disconnect();	/* insurance */
13518cc36fbSJordan K. Hubbard 	return (connected);
13618cc36fbSJordan K. Hubbard }
13718cc36fbSJordan K. Hubbard 
1380f3bdf5dSMark Murray void
hay_disconnect(void)139fa92e883SRuslan Ermilov hay_disconnect(void)
14018cc36fbSJordan K. Hubbard {
14118cc36fbSJordan K. Hubbard 	/* first hang up the modem*/
14218cc36fbSJordan K. Hubbard #ifdef DEBUG
14318cc36fbSJordan K. Hubbard 	printf("\rdisconnecting modem....\n\r");
14418cc36fbSJordan K. Hubbard #endif
14518cc36fbSJordan K. Hubbard 	ioctl(FD, TIOCCDTR, 0);
14618cc36fbSJordan K. Hubbard 	sleep(1);
14718cc36fbSJordan K. Hubbard 	ioctl(FD, TIOCSDTR, 0);
14818cc36fbSJordan K. Hubbard 	goodbye();
14918cc36fbSJordan K. Hubbard }
15018cc36fbSJordan K. Hubbard 
1510f3bdf5dSMark Murray void
hay_abort(void)152fa92e883SRuslan Ermilov hay_abort(void)
15318cc36fbSJordan K. Hubbard {
15418cc36fbSJordan K. Hubbard 	write(FD, "\r", 1);	/* send anything to abort the call */
15518cc36fbSJordan K. Hubbard 	hay_disconnect();
15618cc36fbSJordan K. Hubbard }
15718cc36fbSJordan K. Hubbard 
158fa92e883SRuslan Ermilov /*ARGSUSED*/
15918cc36fbSJordan K. Hubbard static void
sigALRM(int signo)160fa92e883SRuslan Ermilov sigALRM(int signo)
16118cc36fbSJordan K. Hubbard {
16218cc36fbSJordan K. Hubbard 	printf("\07timeout waiting for reply\n\r");
163fa92e883SRuslan Ermilov 	dialtimeout = 1;
16418cc36fbSJordan K. Hubbard 	longjmp(timeoutbuf, 1);
16518cc36fbSJordan K. Hubbard }
16618cc36fbSJordan K. Hubbard 
16718cc36fbSJordan K. Hubbard static char
gobble(char * match)168fa92e883SRuslan Ermilov gobble(char *match)
16918cc36fbSJordan K. Hubbard {
17018cc36fbSJordan K. Hubbard 	char c;
17118cc36fbSJordan K. Hubbard 	sig_t f;
17207bb01e3SRuslan Ermilov 	size_t i;
17307bb01e3SRuslan Ermilov 	int status = 0;
17418cc36fbSJordan K. Hubbard 
17518cc36fbSJordan K. Hubbard 	f = signal(SIGALRM, sigALRM);
176fa92e883SRuslan Ermilov 	dialtimeout = 0;
17718cc36fbSJordan K. Hubbard #ifdef DEBUG
17818cc36fbSJordan K. Hubbard 	printf("\ngobble: waiting for %s\n", match);
17918cc36fbSJordan K. Hubbard #endif
18018cc36fbSJordan K. Hubbard 	do {
18118cc36fbSJordan K. Hubbard 		if (setjmp(timeoutbuf)) {
18218cc36fbSJordan K. Hubbard 			signal(SIGALRM, f);
18318cc36fbSJordan K. Hubbard 			return (0);
18418cc36fbSJordan K. Hubbard 		}
18518cc36fbSJordan K. Hubbard 		alarm(number(value(DIALTIMEOUT)));
18618cc36fbSJordan K. Hubbard 		read(FD, &c, 1);
18718cc36fbSJordan K. Hubbard 		alarm(0);
18818cc36fbSJordan K. Hubbard 		c &= 0177;
18918cc36fbSJordan K. Hubbard #ifdef DEBUG
19018cc36fbSJordan K. Hubbard 		printf("%c 0x%x ", c, c);
19118cc36fbSJordan K. Hubbard #endif
19218cc36fbSJordan K. Hubbard 		for (i = 0; i < strlen(match); i++)
19318cc36fbSJordan K. Hubbard 			if (c == match[i])
19418cc36fbSJordan K. Hubbard 				status = c;
19518cc36fbSJordan K. Hubbard 	} while (status == 0);
19618cc36fbSJordan K. Hubbard 	signal(SIGALRM, SIG_DFL);
19718cc36fbSJordan K. Hubbard #ifdef DEBUG
19818cc36fbSJordan K. Hubbard 	printf("\n");
19918cc36fbSJordan K. Hubbard #endif
20018cc36fbSJordan K. Hubbard 	return (status);
20118cc36fbSJordan K. Hubbard }
20218cc36fbSJordan K. Hubbard 
2030f3bdf5dSMark Murray static void
error_rep(char c)204fa92e883SRuslan Ermilov error_rep(char c)
20518cc36fbSJordan K. Hubbard {
20618cc36fbSJordan K. Hubbard 	printf("\n\r");
20718cc36fbSJordan K. Hubbard 	switch (c) {
20818cc36fbSJordan K. Hubbard 
20918cc36fbSJordan K. Hubbard 	case '0':
21018cc36fbSJordan K. Hubbard 		printf("OK");
21118cc36fbSJordan K. Hubbard 		break;
21218cc36fbSJordan K. Hubbard 
21318cc36fbSJordan K. Hubbard 	case '1':
21418cc36fbSJordan K. Hubbard 		printf("CONNECT");
21518cc36fbSJordan K. Hubbard 		break;
21618cc36fbSJordan K. Hubbard 
21718cc36fbSJordan K. Hubbard 	case '2':
21818cc36fbSJordan K. Hubbard 		printf("RING");
21918cc36fbSJordan K. Hubbard 		break;
22018cc36fbSJordan K. Hubbard 
22118cc36fbSJordan K. Hubbard 	case '3':
22218cc36fbSJordan K. Hubbard 		printf("NO CARRIER");
22318cc36fbSJordan K. Hubbard 		break;
22418cc36fbSJordan K. Hubbard 
22518cc36fbSJordan K. Hubbard 	case '4':
22618cc36fbSJordan K. Hubbard 		printf("ERROR in input");
22718cc36fbSJordan K. Hubbard 		break;
22818cc36fbSJordan K. Hubbard 
22918cc36fbSJordan K. Hubbard 	case '5':
23018cc36fbSJordan K. Hubbard 		printf("CONNECT 1200");
23118cc36fbSJordan K. Hubbard 		break;
23218cc36fbSJordan K. Hubbard 
23318cc36fbSJordan K. Hubbard 	default:
23418cc36fbSJordan K. Hubbard 		printf("Unknown Modem error: %c (0x%x)", c, c);
23518cc36fbSJordan K. Hubbard 	}
23618cc36fbSJordan K. Hubbard 	printf("\n\r");
23718cc36fbSJordan K. Hubbard 	return;
23818cc36fbSJordan K. Hubbard }
23918cc36fbSJordan K. Hubbard 
24018cc36fbSJordan K. Hubbard /*
24118cc36fbSJordan K. Hubbard  * set modem back to normal verbose status codes.
24218cc36fbSJordan K. Hubbard  */
243229c0017SWarner Losh static void
goodbye(void)244229c0017SWarner Losh goodbye(void)
24518cc36fbSJordan K. Hubbard {
2460f3bdf5dSMark Murray 	int len;
24718cc36fbSJordan K. Hubbard 	char c;
24818cc36fbSJordan K. Hubbard 
2490f3bdf5dSMark Murray 	tcflush(FD, TCIOFLUSH);
25018cc36fbSJordan K. Hubbard 	if (hay_sync()) {
25118cc36fbSJordan K. Hubbard 		sleep(1);
25218cc36fbSJordan K. Hubbard #ifndef DEBUG
2530f3bdf5dSMark Murray 		tcflush(FD, TCIOFLUSH);
25418cc36fbSJordan K. Hubbard #endif
25518cc36fbSJordan K. Hubbard 		write(FD, "ATH0\r", 5);		/* insurance */
25618cc36fbSJordan K. Hubbard #ifndef DEBUG
25718cc36fbSJordan K. Hubbard 		c = gobble("03");
25818cc36fbSJordan K. Hubbard 		if (c != '0' && c != '3') {
25918cc36fbSJordan K. Hubbard 			printf("cannot hang up modem\n\r");
26018cc36fbSJordan K. Hubbard 			printf("please use 'tip dialer' to make sure the line is hung up\n\r");
26118cc36fbSJordan K. Hubbard 		}
26218cc36fbSJordan K. Hubbard #endif
26318cc36fbSJordan K. Hubbard 		sleep(1);
26418cc36fbSJordan K. Hubbard 		ioctl(FD, FIONREAD, &len);
26518cc36fbSJordan K. Hubbard #ifdef DEBUG
26618cc36fbSJordan K. Hubbard 		printf("goodbye1: len=%d -- ", len);
26718cc36fbSJordan K. Hubbard 		rlen = read(FD, dumbuf, min(len, DUMBUFLEN));
26818cc36fbSJordan K. Hubbard 		dumbuf[rlen] = '\0';
26918cc36fbSJordan K. Hubbard 		printf("read (%d): %s\r\n", rlen, dumbuf);
27018cc36fbSJordan K. Hubbard #endif
27118cc36fbSJordan K. Hubbard 		write(FD, "ATv1\r", 5);
27218cc36fbSJordan K. Hubbard 		sleep(1);
27318cc36fbSJordan K. Hubbard #ifdef DEBUG
27418cc36fbSJordan K. Hubbard 		ioctl(FD, FIONREAD, &len);
27518cc36fbSJordan K. Hubbard 		printf("goodbye2: len=%d -- ", len);
27618cc36fbSJordan K. Hubbard 		rlen = read(FD, dumbuf, min(len, DUMBUFLEN));
27718cc36fbSJordan K. Hubbard 		dumbuf[rlen] = '\0';
27818cc36fbSJordan K. Hubbard 		printf("read (%d): %s\r\n", rlen, dumbuf);
27918cc36fbSJordan K. Hubbard #endif
28018cc36fbSJordan K. Hubbard 	}
2810f3bdf5dSMark Murray 	tcflush(FD, TCIOFLUSH);
28218cc36fbSJordan K. Hubbard 	ioctl(FD, TIOCCDTR, 0);		/* clear DTR (insurance) */
28318cc36fbSJordan K. Hubbard 	close(FD);
28418cc36fbSJordan K. Hubbard }
28518cc36fbSJordan K. Hubbard 
28618cc36fbSJordan K. Hubbard #define MAXRETRY	5
28718cc36fbSJordan K. Hubbard 
288fa92e883SRuslan Ermilov static int
hay_sync(void)289fa92e883SRuslan Ermilov hay_sync(void)
29018cc36fbSJordan K. Hubbard {
29118cc36fbSJordan K. Hubbard 	int len, retry = 0;
29218cc36fbSJordan K. Hubbard 
29318cc36fbSJordan K. Hubbard 	while (retry++ <= MAXRETRY) {
29418cc36fbSJordan K. Hubbard 		write(FD, "AT\r", 3);
29518cc36fbSJordan K. Hubbard 		sleep(1);
29618cc36fbSJordan K. Hubbard 		ioctl(FD, FIONREAD, &len);
29718cc36fbSJordan K. Hubbard 		if (len) {
29818cc36fbSJordan K. Hubbard 			len = read(FD, dumbuf, min(len, DUMBUFLEN));
2990f3bdf5dSMark Murray 			if (strchr(dumbuf, '0') ||
3000f3bdf5dSMark Murray 		   	(strchr(dumbuf, 'O') && strchr(dumbuf, 'K')))
30118cc36fbSJordan K. Hubbard 				return(1);
30218cc36fbSJordan K. Hubbard #ifdef DEBUG
30318cc36fbSJordan K. Hubbard 			dumbuf[len] = '\0';
30418cc36fbSJordan K. Hubbard 			printf("hay_sync: (\"%s\") %d\n\r", dumbuf, retry);
30518cc36fbSJordan K. Hubbard #endif
30618cc36fbSJordan K. Hubbard 		}
30718cc36fbSJordan K. Hubbard 		ioctl(FD, TIOCCDTR, 0);
30818cc36fbSJordan K. Hubbard 		ioctl(FD, TIOCSDTR, 0);
30918cc36fbSJordan K. Hubbard 	}
31018cc36fbSJordan K. Hubbard 	printf("Cannot synchronize with hayes...\n\r");
31118cc36fbSJordan K. Hubbard 	return(0);
31218cc36fbSJordan K. Hubbard }
313