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