xref: /dflybsd-src/usr.bin/tip/unidialer.c (revision 0720b42f944927d00240ac621ea5eb26e8844bc6)
1be09fc23SSascha Wildner /*
2be09fc23SSascha Wildner  * Copyright (c) 1986, 1993
3be09fc23SSascha Wildner  *	The Regents of the University of California.  All rights reserved.
4be09fc23SSascha Wildner  *
5be09fc23SSascha Wildner  * Redistribution and use in source and binary forms, with or without
6be09fc23SSascha Wildner  * modification, are permitted provided that the following conditions
7be09fc23SSascha Wildner  * are met:
8be09fc23SSascha Wildner  * 1. Redistributions of source code must retain the above copyright
9be09fc23SSascha Wildner  *    notice, this list of conditions and the following disclaimer.
10be09fc23SSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
11be09fc23SSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
12be09fc23SSascha Wildner  *    documentation and/or other materials provided with the distribution.
13*0720b42fSzrj  * 3. Neither the name of the University nor the names of its contributors
14be09fc23SSascha Wildner  *    may be used to endorse or promote products derived from this software
15be09fc23SSascha Wildner  *    without specific prior written permission.
16be09fc23SSascha Wildner  *
17be09fc23SSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18be09fc23SSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19be09fc23SSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20be09fc23SSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21be09fc23SSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22be09fc23SSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23be09fc23SSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24be09fc23SSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25be09fc23SSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26be09fc23SSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27be09fc23SSascha Wildner  * SUCH DAMAGE.
28be09fc23SSascha Wildner  *
29be09fc23SSascha Wildner  * @(#)unidialer.c	8.1 (Berkeley) 6/6/93
30be09fc23SSascha Wildner  * $FreeBSD: src/usr.bin/tip/libacu/unidialer.c,v 1.7 1999/08/28 01:06:30 peter Exp $
31be09fc23SSascha Wildner  */
32be09fc23SSascha Wildner 
33be09fc23SSascha Wildner /*
34be09fc23SSascha Wildner  * Generalized routines for calling up on a Hayes AT command set based modem.
35be09fc23SSascha Wildner  * Control variables are pulled out of a modem caps-style database to
36be09fc23SSascha Wildner  * configure the driver for a particular modem.
37be09fc23SSascha Wildner  */
38be09fc23SSascha Wildner #include "tip.h"
39be09fc23SSascha Wildner #include "pathnames.h"
40be09fc23SSascha Wildner 
41be09fc23SSascha Wildner #include <sys/times.h>
42be09fc23SSascha Wildner #include <assert.h>
43be09fc23SSascha Wildner #include <err.h>
44be09fc23SSascha Wildner #include <stdio.h>
45be09fc23SSascha Wildner #include <stdlib.h>
46be09fc23SSascha Wildner 
47be09fc23SSascha Wildner #include "acucommon.h"
48be09fc23SSascha Wildner #include "tod.h"
49be09fc23SSascha Wildner 
50be09fc23SSascha Wildner /* #define DEBUG */
51be09fc23SSascha Wildner #define	MAXRETRY	5
52be09fc23SSascha Wildner 
53be09fc23SSascha Wildner typedef enum
54be09fc23SSascha Wildner {
55be09fc23SSascha Wildner 	mpt_notype, mpt_string, mpt_number, mpt_boolean
56be09fc23SSascha Wildner } modem_parm_type_t;
57be09fc23SSascha Wildner 
58be09fc23SSascha Wildner typedef struct {
59be09fc23SSascha Wildner 	modem_parm_type_t modem_parm_type;
60be09fc23SSascha Wildner 	const char *name;
61be09fc23SSascha Wildner 	union {
62be09fc23SSascha Wildner 		char **string;
63be09fc23SSascha Wildner 		unsigned int *number;
64be09fc23SSascha Wildner 	} value;
65be09fc23SSascha Wildner 	union {
66be09fc23SSascha Wildner 		char *string;
67be09fc23SSascha Wildner 		unsigned int number;
68be09fc23SSascha Wildner 	} default_value;
69be09fc23SSascha Wildner } modem_parm_t;
70be09fc23SSascha Wildner 
71be09fc23SSascha Wildner /*
72be09fc23SSascha Wildner 	Configuration
73be09fc23SSascha Wildner */
74be09fc23SSascha Wildner static char modem_name [80];
75be09fc23SSascha Wildner static char *dial_command;
76be09fc23SSascha Wildner static char *hangup_command;
77be09fc23SSascha Wildner static char *echo_off_command;
78be09fc23SSascha Wildner static char *reset_command;
79be09fc23SSascha Wildner static char *init_string;
80be09fc23SSascha Wildner static char *escape_sequence;
81be09fc23SSascha Wildner static int hw_flow_control;
82be09fc23SSascha Wildner static int lock_baud;
83be09fc23SSascha Wildner static unsigned int intercharacter_delay;
84be09fc23SSascha Wildner static unsigned int intercommand_delay;
85be09fc23SSascha Wildner static unsigned int escape_guard_time;
86be09fc23SSascha Wildner static unsigned int reset_delay;
87be09fc23SSascha Wildner 
88be09fc23SSascha Wildner static int unidialer_dialer(char *num, char *acu);
89be09fc23SSascha Wildner static void unidialer_disconnect(void);
90be09fc23SSascha Wildner static void unidialer_abort(void);
91be09fc23SSascha Wildner static int unidialer_connect(void);
92be09fc23SSascha Wildner static int unidialer_swallow(char *);
93be09fc23SSascha Wildner #ifdef DEBUG
94be09fc23SSascha Wildner static void unidialer_verbose_read (void);
95be09fc23SSascha Wildner #endif
966dd3947fSSascha Wildner static void unidialer_modem_cmd(int fd, const char *cmd);
976dd3947fSSascha Wildner static void unidialer_write(int fd, const char *cp, int n);
986dd3947fSSascha Wildner static void unidialer_write_str(int fd, const char *cp);
99be09fc23SSascha Wildner static void sigALRM(int);
100be09fc23SSascha Wildner static int unidialersync(void);
101be09fc23SSascha Wildner 
102be09fc23SSascha Wildner static acu_t unidialer =
103be09fc23SSascha Wildner {
104be09fc23SSascha Wildner 	modem_name,
105be09fc23SSascha Wildner 	unidialer_dialer,
106be09fc23SSascha Wildner 	unidialer_disconnect,
107be09fc23SSascha Wildner 	unidialer_abort
108be09fc23SSascha Wildner };
109be09fc23SSascha Wildner 
110be09fc23SSascha Wildner /*
111be09fc23SSascha Wildner 	Table of parameters kept in modem database
112be09fc23SSascha Wildner */
113be09fc23SSascha Wildner modem_parm_t modem_parms [] = {
114be09fc23SSascha Wildner 	{ mpt_string, "dial_command", { &dial_command }, { "ATDT%s\r" } },
115be09fc23SSascha Wildner 	{ mpt_string, "hangup_command", { &hangup_command }, { "ATH\r", } },
116be09fc23SSascha Wildner 	{ mpt_string, "echo_off_command", { &echo_off_command }, { "ATE0\r" } },
117be09fc23SSascha Wildner 	{ mpt_string, "reset_command", { &reset_command }, { "ATZ\r" } },
118be09fc23SSascha Wildner 	{ mpt_string, "init_string", { &init_string }, { "AT{ &F\r", } },
119be09fc23SSascha Wildner 	{ mpt_string, "escape_sequence", { &escape_sequence }, { "+++" } },
120be09fc23SSascha Wildner 	{ mpt_boolean, "hw_flow_control", { (char **)&hw_flow_control }, { NULL } },
121be09fc23SSascha Wildner 	{ mpt_boolean, "lock_baud", { (char **)&lock_baud }, { NULL } },
122be09fc23SSascha Wildner 	{ mpt_number, "intercharacter_delay", { (char **)&intercharacter_delay }, { (char *)50 } },
123be09fc23SSascha Wildner 	{ mpt_number, "intercommand_delay", { (char **)&intercommand_delay }, { (char *)300 } },
124be09fc23SSascha Wildner 	{ mpt_number, "escape_guard_time", { (char **)&escape_guard_time }, { (char *)300 } },
125be09fc23SSascha Wildner 	{ mpt_number, "reset_delay", { (char **)&reset_delay }, { (char *)3000 } },
126be09fc23SSascha Wildner 	{ mpt_notype, NULL, { NULL }, { NULL } }
127be09fc23SSascha Wildner };
128be09fc23SSascha Wildner 
129be09fc23SSascha Wildner /*
130be09fc23SSascha Wildner 	Global vars
131be09fc23SSascha Wildner */
132be09fc23SSascha Wildner static int timeout = 0;
133be09fc23SSascha Wildner static int connected = 0;
134be09fc23SSascha Wildner static jmp_buf timeoutbuf;
135be09fc23SSascha Wildner 
136be09fc23SSascha Wildner #define cgetflag(f)	(cgetcap(bp, f, ':') != NULL)
137be09fc23SSascha Wildner 
138be09fc23SSascha Wildner #ifdef DEBUG
139be09fc23SSascha Wildner 
140be09fc23SSascha Wildner #define print_str(x) printf (#x " = %s\n", x)
141be09fc23SSascha Wildner #define print_num(x) printf (#x " = %d\n", x)
142be09fc23SSascha Wildner 
dumpmodemparms(char * modem)143be09fc23SSascha Wildner void dumpmodemparms (char *modem)
144be09fc23SSascha Wildner {
145be09fc23SSascha Wildner 		printf ("modem parms for %s\n", modem);
146be09fc23SSascha Wildner 		print_str (dial_command);
147be09fc23SSascha Wildner 		print_str (hangup_command);
148be09fc23SSascha Wildner 		print_str (echo_off_command);
149be09fc23SSascha Wildner 		print_str (reset_command);
150be09fc23SSascha Wildner 		print_str (init_string);
151be09fc23SSascha Wildner 		print_str (escape_sequence);
152be09fc23SSascha Wildner 		print_num (lock_baud);
153be09fc23SSascha Wildner 		print_num (intercharacter_delay);
154be09fc23SSascha Wildner 		print_num (intercommand_delay);
155be09fc23SSascha Wildner 		print_num (escape_guard_time);
156be09fc23SSascha Wildner 		print_num (reset_delay);
157be09fc23SSascha Wildner 		printf ("\n");
158be09fc23SSascha Wildner }
159be09fc23SSascha Wildner #endif
160be09fc23SSascha Wildner 
getmodemparms(const char * modem)161be09fc23SSascha Wildner static int getmodemparms (const char *modem)
162be09fc23SSascha Wildner {
163be09fc23SSascha Wildner 	char *bp, *db_array [3], *modempath;
164be09fc23SSascha Wildner 	int ndx, stat;
165be09fc23SSascha Wildner 	modem_parm_t *mpp;
166be09fc23SSascha Wildner 
167be09fc23SSascha Wildner 	modempath = getenv ("MODEMS");
168be09fc23SSascha Wildner 
169be09fc23SSascha Wildner 	ndx = 0;
170be09fc23SSascha Wildner 
171be09fc23SSascha Wildner 	if (modempath != NULL)
172be09fc23SSascha Wildner 		db_array [ndx++] = modempath;
173be09fc23SSascha Wildner 
174be09fc23SSascha Wildner 	db_array [ndx++] = _PATH_MODEMS;
175be09fc23SSascha Wildner 	db_array [ndx] = NULL;
176be09fc23SSascha Wildner 
177be09fc23SSascha Wildner 	if ((stat = cgetent (&bp, db_array, (char *)modem)) < 0) {
178be09fc23SSascha Wildner 		switch (stat) {
179be09fc23SSascha Wildner 		case -1:
180be09fc23SSascha Wildner 			warnx ("unknown modem %s", modem);
181be09fc23SSascha Wildner 			break;
182be09fc23SSascha Wildner 		case -2:
183be09fc23SSascha Wildner 			warnx ("can't open modem description file");
184be09fc23SSascha Wildner 			break;
185be09fc23SSascha Wildner 		case -3:
186be09fc23SSascha Wildner 			warnx ("possible reference loop in modem description file");
187be09fc23SSascha Wildner 			break;
188be09fc23SSascha Wildner 		}
189be09fc23SSascha Wildner 		return 0;
190be09fc23SSascha Wildner 	}
191be09fc23SSascha Wildner 	for (mpp = modem_parms; mpp->name; mpp++)
192be09fc23SSascha Wildner 	{
193be09fc23SSascha Wildner 		switch (mpp->modem_parm_type)
194be09fc23SSascha Wildner 		{
195be09fc23SSascha Wildner 		case mpt_string:
196be09fc23SSascha Wildner 			if (cgetstr (bp, (char *)mpp->name, mpp->value.string) == -1)
197be09fc23SSascha Wildner 				*mpp->value.string = mpp->default_value.string;
198be09fc23SSascha Wildner 			break;
199be09fc23SSascha Wildner 		case mpt_number:
200be09fc23SSascha Wildner 		{
201be09fc23SSascha Wildner 			long l;
202be09fc23SSascha Wildner 			if (cgetnum (bp, (char *)mpp->name, &l) == -1)
203be09fc23SSascha Wildner 				*mpp->value.number = mpp->default_value.number;
204be09fc23SSascha Wildner 			else
205be09fc23SSascha Wildner 				*mpp->value.number = (unsigned int)l;
206be09fc23SSascha Wildner 		}
207be09fc23SSascha Wildner 			break;
208be09fc23SSascha Wildner 		case mpt_boolean:
209be09fc23SSascha Wildner 			*mpp->value.number = cgetflag ((char *)mpp->name);
210be09fc23SSascha Wildner 			break;
211be09fc23SSascha Wildner 		default:
212be09fc23SSascha Wildner 			break;
213be09fc23SSascha Wildner 		}
214be09fc23SSascha Wildner 	}
215be09fc23SSascha Wildner 	strncpy (modem_name, modem, sizeof (modem_name) - 1);
216be09fc23SSascha Wildner 	modem_name [sizeof (modem_name) - 1] = '\0';
217be09fc23SSascha Wildner 	return 1;
218be09fc23SSascha Wildner }
219be09fc23SSascha Wildner 
220be09fc23SSascha Wildner /*
221be09fc23SSascha Wildner */
222be09fc23SSascha Wildner acu_t *
unidialer_getmodem(const char * modem_name)223be09fc23SSascha Wildner unidialer_getmodem(const char *modem_name)
224be09fc23SSascha Wildner {
225be09fc23SSascha Wildner 	acu_t* rc = NULL;
226be09fc23SSascha Wildner 	if (getmodemparms (modem_name))
227be09fc23SSascha Wildner 		rc = &unidialer;
228be09fc23SSascha Wildner 	return rc;
229be09fc23SSascha Wildner }
230be09fc23SSascha Wildner 
231be09fc23SSascha Wildner static int
unidialer_modem_ready(void)232be09fc23SSascha Wildner unidialer_modem_ready(void)
233be09fc23SSascha Wildner {
234be09fc23SSascha Wildner 	int state;
235be09fc23SSascha Wildner 	ioctl (FD, TIOCMGET, &state);
236be09fc23SSascha Wildner 	return (state & TIOCM_DSR) ? 1 : 0;
237be09fc23SSascha Wildner }
238be09fc23SSascha Wildner 
unidialer_waitfor_modem_ready(int ms)239be09fc23SSascha Wildner static int unidialer_waitfor_modem_ready (int ms)
240be09fc23SSascha Wildner {
241be09fc23SSascha Wildner 	int count;
242be09fc23SSascha Wildner 	for (count = 0; count < ms; count += 100)
243be09fc23SSascha Wildner 	{
244be09fc23SSascha Wildner 		if (unidialer_modem_ready ())
245be09fc23SSascha Wildner 		{
246be09fc23SSascha Wildner #ifdef DEBUG
247be09fc23SSascha Wildner 			printf ("unidialer_waitfor_modem_ready: modem ready.\n");
248be09fc23SSascha Wildner #endif
249be09fc23SSascha Wildner 			break;
250be09fc23SSascha Wildner 		}
251be09fc23SSascha Wildner 		acu_nap (100);
252be09fc23SSascha Wildner 	}
253be09fc23SSascha Wildner 	return (count < ms);
254be09fc23SSascha Wildner }
255be09fc23SSascha Wildner 
256be09fc23SSascha Wildner static void
unidialer_tty_clocal(int flag)257be09fc23SSascha Wildner unidialer_tty_clocal(int flag)
258be09fc23SSascha Wildner {
259be09fc23SSascha Wildner 	struct termios t;
260be09fc23SSascha Wildner 	tcgetattr (FD, &t);
261be09fc23SSascha Wildner 	if (flag)
262be09fc23SSascha Wildner 		t.c_cflag |= CLOCAL;
263be09fc23SSascha Wildner 	else
264be09fc23SSascha Wildner 		t.c_cflag &= ~CLOCAL;
265be09fc23SSascha Wildner 	tcsetattr (FD, TCSANOW, &t);
266be09fc23SSascha Wildner }
267be09fc23SSascha Wildner 
268be09fc23SSascha Wildner static int
unidialer_get_modem_response(char * buf,int bufsz,int response_timeout)269be09fc23SSascha Wildner unidialer_get_modem_response(char *buf, int bufsz, int response_timeout)
270be09fc23SSascha Wildner {
271be09fc23SSascha Wildner 	sig_t f;
272be09fc23SSascha Wildner 	char c, *p = buf, *lid = buf + bufsz - 1;
273be09fc23SSascha Wildner 	int state;
274be09fc23SSascha Wildner 
275be09fc23SSascha Wildner 	assert (bufsz > 0);
276be09fc23SSascha Wildner 
277be09fc23SSascha Wildner 	f = signal (SIGALRM, sigALRM);
278be09fc23SSascha Wildner 
279be09fc23SSascha Wildner 	timeout = 0;
280be09fc23SSascha Wildner 
281be09fc23SSascha Wildner 	if (setjmp (timeoutbuf)) {
282be09fc23SSascha Wildner 		signal (SIGALRM, f);
283be09fc23SSascha Wildner 		*p = '\0';
284be09fc23SSascha Wildner #ifdef DEBUG
2858350c6bdSSascha Wildner 		printf ("get_response: timeout buf=%s\n", buf);
286be09fc23SSascha Wildner #endif
287be09fc23SSascha Wildner 		return (0);
288be09fc23SSascha Wildner 	}
289be09fc23SSascha Wildner 
290be09fc23SSascha Wildner 	ualarm (response_timeout * 1000, 0);
291be09fc23SSascha Wildner 
292be09fc23SSascha Wildner 	state = 0;
293be09fc23SSascha Wildner 
294be09fc23SSascha Wildner 	while (1)
295be09fc23SSascha Wildner 	{
296be09fc23SSascha Wildner 		switch (state)
297be09fc23SSascha Wildner 		{
298be09fc23SSascha Wildner 			case 0:
299be09fc23SSascha Wildner 				if (read (FD, &c, 1) == 1)
300be09fc23SSascha Wildner 				{
301be09fc23SSascha Wildner 					if (c == '\r')
302be09fc23SSascha Wildner 					{
303be09fc23SSascha Wildner 						++state;
304be09fc23SSascha Wildner 					}
305be09fc23SSascha Wildner 					else
306be09fc23SSascha Wildner 					{
307be09fc23SSascha Wildner #ifdef DEBUG
308be09fc23SSascha Wildner 						printf ("get_response: unexpected char %s.\n", ctrl (c));
309be09fc23SSascha Wildner #endif
310be09fc23SSascha Wildner 					}
311be09fc23SSascha Wildner 				}
312be09fc23SSascha Wildner 				break;
313be09fc23SSascha Wildner 
314be09fc23SSascha Wildner 			case 1:
315be09fc23SSascha Wildner 				if (read (FD, &c, 1) == 1)
316be09fc23SSascha Wildner 				{
317be09fc23SSascha Wildner 					if (c == '\n')
318be09fc23SSascha Wildner 					{
319be09fc23SSascha Wildner #ifdef DEBUG
320be09fc23SSascha Wildner 						printf ("get_response: <CRLF> encountered.\n");
321be09fc23SSascha Wildner #endif
322be09fc23SSascha Wildner 						++state;
323be09fc23SSascha Wildner 					}
324be09fc23SSascha Wildner 					else
325be09fc23SSascha Wildner 					{
326be09fc23SSascha Wildner 							state = 0;
327be09fc23SSascha Wildner #ifdef DEBUG
328be09fc23SSascha Wildner 							printf ("get_response: unexpected char %s.\n", ctrl (c));
329be09fc23SSascha Wildner #endif
330be09fc23SSascha Wildner 					}
331be09fc23SSascha Wildner 				}
332be09fc23SSascha Wildner 				break;
333be09fc23SSascha Wildner 
334be09fc23SSascha Wildner 			case 2:
335be09fc23SSascha Wildner 				if (read (FD, &c, 1) == 1)
336be09fc23SSascha Wildner 				{
337be09fc23SSascha Wildner 					if (c == '\r')
338be09fc23SSascha Wildner 						++state;
339be09fc23SSascha Wildner 					else if (c >= ' ' && p < lid)
340be09fc23SSascha Wildner 						*p++ = c;
341be09fc23SSascha Wildner 				}
342be09fc23SSascha Wildner 				break;
343be09fc23SSascha Wildner 
344be09fc23SSascha Wildner 			case 3:
345be09fc23SSascha Wildner 				if (read (FD, &c, 1) == 1)
346be09fc23SSascha Wildner 				{
347be09fc23SSascha Wildner 					if (c == '\n')
348be09fc23SSascha Wildner 					{
349be09fc23SSascha Wildner 						signal (SIGALRM, f);
350be09fc23SSascha Wildner 						/* ualarm (0, 0); */
351be09fc23SSascha Wildner 						alarm (0);
352be09fc23SSascha Wildner 						*p = '\0';
353be09fc23SSascha Wildner #ifdef DEBUG
354be09fc23SSascha Wildner 						printf ("get_response: %s\n", buf);
355be09fc23SSascha Wildner #endif
356be09fc23SSascha Wildner 						return (1);
357be09fc23SSascha Wildner 					}
358be09fc23SSascha Wildner 					else
359be09fc23SSascha Wildner 					{
360be09fc23SSascha Wildner 						state = 0;
361be09fc23SSascha Wildner 						p = buf;
362be09fc23SSascha Wildner 					}
363be09fc23SSascha Wildner 				}
364be09fc23SSascha Wildner 				break;
365be09fc23SSascha Wildner 		}
366be09fc23SSascha Wildner 	}
367be09fc23SSascha Wildner }
368be09fc23SSascha Wildner 
369be09fc23SSascha Wildner static int
unidialer_get_okay(int ms)370be09fc23SSascha Wildner unidialer_get_okay(int ms)
371be09fc23SSascha Wildner {
372be09fc23SSascha Wildner 	int okay;
373be09fc23SSascha Wildner 	char buf [BUFSIZ];
374be09fc23SSascha Wildner 	okay = unidialer_get_modem_response (buf, sizeof (buf), ms) &&
375be09fc23SSascha Wildner 		strcmp (buf, "OK") == 0;
376be09fc23SSascha Wildner 	return okay;
377be09fc23SSascha Wildner }
378be09fc23SSascha Wildner 
379be09fc23SSascha Wildner static int
unidialer_dialer(char * num,char * acu)380be09fc23SSascha Wildner unidialer_dialer(char *num, char *acu)
381be09fc23SSascha Wildner {
382be09fc23SSascha Wildner 	char *cp;
383be09fc23SSascha Wildner 	char dial_string [80];
384be09fc23SSascha Wildner 	char line [80];
385be09fc23SSascha Wildner 
386be09fc23SSascha Wildner #ifdef DEBUG
387be09fc23SSascha Wildner 	dumpmodemparms (modem_name);
388be09fc23SSascha Wildner #endif
389be09fc23SSascha Wildner 
390be09fc23SSascha Wildner 	if (lock_baud) {
391be09fc23SSascha Wildner 		int i;
392be09fc23SSascha Wildner 		if ((i = speed(number(value(BAUDRATE)))) == 0)
393be09fc23SSascha Wildner 			return 0;
394be09fc23SSascha Wildner 		ttysetup (i);
395be09fc23SSascha Wildner 	}
396be09fc23SSascha Wildner 
397be09fc23SSascha Wildner 	if (boolean(value(VERBOSE)))
398be09fc23SSascha Wildner 		printf("Using \"%s\"\n", acu);
399be09fc23SSascha Wildner 
400be09fc23SSascha Wildner 	acu_hupcl ();
401be09fc23SSascha Wildner 
402be09fc23SSascha Wildner 	/*
403be09fc23SSascha Wildner 	 * Get in synch.
404be09fc23SSascha Wildner 	 */
405be09fc23SSascha Wildner 	if (!unidialersync()) {
406be09fc23SSascha Wildner badsynch:
407be09fc23SSascha Wildner 		printf("tip: can't synchronize with %s\n", modem_name);
408be09fc23SSascha Wildner 		logent(value(HOST), num, modem_name, "can't synch up");
409be09fc23SSascha Wildner 		return (0);
410be09fc23SSascha Wildner 	}
411be09fc23SSascha Wildner 
412be09fc23SSascha Wildner 	unidialer_modem_cmd (FD, echo_off_command);	/* turn off echoing */
413be09fc23SSascha Wildner 
414be09fc23SSascha Wildner 	sleep(1);
415be09fc23SSascha Wildner 
416be09fc23SSascha Wildner #ifdef DEBUG
417be09fc23SSascha Wildner 	if (boolean(value(VERBOSE)))
418be09fc23SSascha Wildner 		unidialer_verbose_read();
419be09fc23SSascha Wildner #endif
420be09fc23SSascha Wildner 
421be09fc23SSascha Wildner 	acu_flush (); /* flush any clutter */
422be09fc23SSascha Wildner 
423be09fc23SSascha Wildner 	unidialer_modem_cmd (FD, init_string);
424be09fc23SSascha Wildner 
425be09fc23SSascha Wildner 	if (!unidialer_get_okay (reset_delay))
426be09fc23SSascha Wildner 		goto badsynch;
427be09fc23SSascha Wildner 
428be09fc23SSascha Wildner 	fflush (stdout);
429be09fc23SSascha Wildner 
430be09fc23SSascha Wildner 	for (cp = num; *cp; cp++)
431be09fc23SSascha Wildner 		if (*cp == '=')
432be09fc23SSascha Wildner 			*cp = ',';
433be09fc23SSascha Wildner 
434be09fc23SSascha Wildner 	(void) sprintf (dial_string, dial_command, num);
435be09fc23SSascha Wildner 
436be09fc23SSascha Wildner 	unidialer_modem_cmd (FD, dial_string);
437be09fc23SSascha Wildner 
438be09fc23SSascha Wildner 	connected = unidialer_connect ();
439be09fc23SSascha Wildner 
440be09fc23SSascha Wildner 	if (connected && hw_flow_control) {
441be09fc23SSascha Wildner 		acu_hw_flow_control (hw_flow_control);
442be09fc23SSascha Wildner 	}
443be09fc23SSascha Wildner 
444be09fc23SSascha Wildner 	if (timeout) {
445be09fc23SSascha Wildner 		sprintf(line, "%d second dial timeout",
446be09fc23SSascha Wildner 			number(value(DIALTIMEOUT)));
447be09fc23SSascha Wildner 		logent(value(HOST), num, modem_name, line);
448be09fc23SSascha Wildner 	}
449be09fc23SSascha Wildner 
450be09fc23SSascha Wildner 	if (timeout)
451be09fc23SSascha Wildner 		unidialer_disconnect ();
452be09fc23SSascha Wildner 
453be09fc23SSascha Wildner 	return (connected);
454be09fc23SSascha Wildner }
455be09fc23SSascha Wildner 
456be09fc23SSascha Wildner static void
unidialer_disconnect(void)457be09fc23SSascha Wildner unidialer_disconnect(void)
458be09fc23SSascha Wildner {
459be09fc23SSascha Wildner 	int okay, retries;
460be09fc23SSascha Wildner 
461be09fc23SSascha Wildner 	acu_flush (); /* flush any clutter */
462be09fc23SSascha Wildner 
463be09fc23SSascha Wildner 	unidialer_tty_clocal (TRUE);
464be09fc23SSascha Wildner 
465be09fc23SSascha Wildner  	/* first hang up the modem*/
466be09fc23SSascha Wildner 	ioctl (FD, TIOCCDTR, 0);
467be09fc23SSascha Wildner 	acu_nap (250);
468be09fc23SSascha Wildner 	ioctl (FD, TIOCSDTR, 0);
469be09fc23SSascha Wildner 
470be09fc23SSascha Wildner 	/*
471be09fc23SSascha Wildner 	 * If AT&D2, then dropping DTR *should* just hangup the modem. But
472be09fc23SSascha Wildner 	 * some modems reset anyway; also, the modem may be programmed to reset
473be09fc23SSascha Wildner 	 * anyway with AT&D3. Play it safe and wait for the full reset time before
474be09fc23SSascha Wildner 	 * proceeding.
475be09fc23SSascha Wildner 	 */
476be09fc23SSascha Wildner 	acu_nap (reset_delay);
477be09fc23SSascha Wildner 
478be09fc23SSascha Wildner 	if (!unidialer_waitfor_modem_ready (reset_delay))
479be09fc23SSascha Wildner 	{
480be09fc23SSascha Wildner #ifdef DEBUG
481be09fc23SSascha Wildner 			printf ("unidialer_disconnect: warning CTS low.\r\n");
482be09fc23SSascha Wildner #endif
483be09fc23SSascha Wildner 	}
484be09fc23SSascha Wildner 
485be09fc23SSascha Wildner 	/*
486be09fc23SSascha Wildner 	 * If not strapped for DTR control, try to get command mode.
487be09fc23SSascha Wildner 	 */
488be09fc23SSascha Wildner 	for (retries = okay = 0; retries < MAXRETRY && !okay; retries++)
489be09fc23SSascha Wildner 	{
490be09fc23SSascha Wildner 		int timeout_value;
491be09fc23SSascha Wildner 		/* flush any clutter */
492be09fc23SSascha Wildner 		if (!acu_flush ())
493be09fc23SSascha Wildner 		{
494be09fc23SSascha Wildner #ifdef DEBUG
495be09fc23SSascha Wildner 			printf ("unidialer_disconnect: warning flush failed.\r\n");
496be09fc23SSascha Wildner #endif
497be09fc23SSascha Wildner 		}
498be09fc23SSascha Wildner 		timeout_value = escape_guard_time;
499be09fc23SSascha Wildner 		timeout_value += (timeout_value * retries / MAXRETRY);
500be09fc23SSascha Wildner 		acu_nap (timeout_value);
501be09fc23SSascha Wildner 		acu_flush (); /* flush any clutter */
502be09fc23SSascha Wildner 		unidialer_modem_cmd (FD, escape_sequence);
503be09fc23SSascha Wildner 		acu_nap (timeout_value);
504be09fc23SSascha Wildner 		unidialer_modem_cmd (FD, hangup_command);
505be09fc23SSascha Wildner 		okay = unidialer_get_okay (reset_delay);
506be09fc23SSascha Wildner 	}
507be09fc23SSascha Wildner 	if (!okay)
508be09fc23SSascha Wildner 	{
509be09fc23SSascha Wildner 		logent(value(HOST), "", modem_name, "can't hang up modem");
510be09fc23SSascha Wildner 		if (boolean(value(VERBOSE)))
511be09fc23SSascha Wildner 			printf("hang up failed\n");
512be09fc23SSascha Wildner 	}
513be09fc23SSascha Wildner 	(void) acu_flush ();
514be09fc23SSascha Wildner 	close (FD);
515be09fc23SSascha Wildner }
516be09fc23SSascha Wildner 
517be09fc23SSascha Wildner static void
unidialer_abort(void)518be09fc23SSascha Wildner unidialer_abort(void)
519be09fc23SSascha Wildner {
520be09fc23SSascha Wildner 	unidialer_write_str (FD, "\r");	/* send anything to abort the call */
521be09fc23SSascha Wildner 	unidialer_disconnect ();
522be09fc23SSascha Wildner }
523be09fc23SSascha Wildner 
524be09fc23SSascha Wildner static void
sigALRM(int signo)525be09fc23SSascha Wildner sigALRM(int signo)
526be09fc23SSascha Wildner {
527be09fc23SSascha Wildner 	(void) printf("\07timeout waiting for reply\n");
528be09fc23SSascha Wildner 	timeout = 1;
529be09fc23SSascha Wildner 	longjmp(timeoutbuf, 1);
530be09fc23SSascha Wildner }
531be09fc23SSascha Wildner 
unidialer_swallow(char * match)532be09fc23SSascha Wildner static int unidialer_swallow (char *match)
533be09fc23SSascha Wildner {
534be09fc23SSascha Wildner 	sig_t f;
535be09fc23SSascha Wildner 	char c;
536be09fc23SSascha Wildner 
537be09fc23SSascha Wildner 	f = signal(SIGALRM, sigALRM);
538be09fc23SSascha Wildner 
539be09fc23SSascha Wildner 	timeout = 0;
540be09fc23SSascha Wildner 
541be09fc23SSascha Wildner 	if (setjmp(timeoutbuf)) {
542be09fc23SSascha Wildner 		signal(SIGALRM, f);
543be09fc23SSascha Wildner 		return (0);
544be09fc23SSascha Wildner 	}
545be09fc23SSascha Wildner 
546be09fc23SSascha Wildner 	alarm(number(value(DIALTIMEOUT)));
547be09fc23SSascha Wildner 
548be09fc23SSascha Wildner 	do {
549be09fc23SSascha Wildner 		if (*match =='\0') {
550be09fc23SSascha Wildner 			signal(SIGALRM, f);
551be09fc23SSascha Wildner 			alarm (0);
552be09fc23SSascha Wildner 			return (1);
553be09fc23SSascha Wildner 		}
554be09fc23SSascha Wildner 		do {
555be09fc23SSascha Wildner 			read (FD, &c, 1);
556be09fc23SSascha Wildner 		} while (c == '\0');
557be09fc23SSascha Wildner 		c &= 0177;
558be09fc23SSascha Wildner #ifdef DEBUG
559be09fc23SSascha Wildner 		if (boolean(value(VERBOSE)))
560be09fc23SSascha Wildner 		{
561be09fc23SSascha Wildner 			/* putchar(c); */
562be09fc23SSascha Wildner 			printf (ctrl (c));
563be09fc23SSascha Wildner 		}
564be09fc23SSascha Wildner #endif
565be09fc23SSascha Wildner 	} while (c == *match++);
566be09fc23SSascha Wildner 	signal(SIGALRM, SIG_DFL);
567be09fc23SSascha Wildner 	alarm(0);
568be09fc23SSascha Wildner #ifdef DEBUG
569be09fc23SSascha Wildner 	if (boolean(value(VERBOSE)))
570be09fc23SSascha Wildner 		fflush (stdout);
571be09fc23SSascha Wildner #endif
572be09fc23SSascha Wildner 	return (0);
573be09fc23SSascha Wildner }
574be09fc23SSascha Wildner 
575be09fc23SSascha Wildner static struct baud_msg {
576be09fc23SSascha Wildner 	char *msg;
577be09fc23SSascha Wildner 	int baud;
578be09fc23SSascha Wildner } baud_msg[] = {
579be09fc23SSascha Wildner 	{ "",		B300 },
580be09fc23SSascha Wildner 	{ " 1200",	B1200 },
581be09fc23SSascha Wildner 	{ " 2400",	B2400 },
582be09fc23SSascha Wildner 	{ " 9600",	B9600 },
583be09fc23SSascha Wildner 	{ " 9600/ARQ",	B9600 },
584be09fc23SSascha Wildner 	{ 0,		0 },
585be09fc23SSascha Wildner };
586be09fc23SSascha Wildner 
587be09fc23SSascha Wildner static int
unidialer_connect(void)588be09fc23SSascha Wildner unidialer_connect(void)
589be09fc23SSascha Wildner {
590be09fc23SSascha Wildner 	char c;
591be09fc23SSascha Wildner 	int nc, nl, n;
592be09fc23SSascha Wildner 	char dialer_buf[64];
593be09fc23SSascha Wildner 	struct baud_msg *bm;
594be09fc23SSascha Wildner 	sig_t f;
595be09fc23SSascha Wildner 
596be09fc23SSascha Wildner 	if (unidialer_swallow("\r\n") == 0)
597be09fc23SSascha Wildner 		return (0);
598be09fc23SSascha Wildner 	f = signal(SIGALRM, sigALRM);
599be09fc23SSascha Wildner again:
600be09fc23SSascha Wildner 	nc = 0; nl = sizeof(dialer_buf)-1;
601be09fc23SSascha Wildner 	bzero(dialer_buf, sizeof(dialer_buf));
602be09fc23SSascha Wildner 	timeout = 0;
603be09fc23SSascha Wildner 	for (nc = 0, nl = sizeof(dialer_buf)-1 ; nl > 0 ; nc++, nl--) {
604be09fc23SSascha Wildner 		if (setjmp(timeoutbuf))
605be09fc23SSascha Wildner 			break;
606be09fc23SSascha Wildner 		alarm(number(value(DIALTIMEOUT)));
607be09fc23SSascha Wildner 		n = read(FD, &c, 1);
608be09fc23SSascha Wildner 		alarm(0);
609be09fc23SSascha Wildner 		if (n <= 0)
610be09fc23SSascha Wildner 			break;
611be09fc23SSascha Wildner 		c &= 0x7f;
612be09fc23SSascha Wildner 		if (c == '\r') {
613be09fc23SSascha Wildner 			if (unidialer_swallow("\n") == 0)
614be09fc23SSascha Wildner 				break;
615be09fc23SSascha Wildner 			if (!dialer_buf[0])
616be09fc23SSascha Wildner 				goto again;
617be09fc23SSascha Wildner 			if (strcmp(dialer_buf, "RINGING") == 0 &&
618be09fc23SSascha Wildner 			    boolean(value(VERBOSE))) {
619be09fc23SSascha Wildner #ifdef DEBUG
620be09fc23SSascha Wildner 				printf("%s\r\n", dialer_buf);
621be09fc23SSascha Wildner #endif
622be09fc23SSascha Wildner 				goto again;
623be09fc23SSascha Wildner 			}
624be09fc23SSascha Wildner 			if (strncmp(dialer_buf, "CONNECT",
625be09fc23SSascha Wildner 				    sizeof("CONNECT")-1) != 0)
626be09fc23SSascha Wildner 				break;
627be09fc23SSascha Wildner 			if (lock_baud) {
628be09fc23SSascha Wildner 				signal(SIGALRM, f);
629be09fc23SSascha Wildner #ifdef DEBUG
630be09fc23SSascha Wildner 				if (boolean(value(VERBOSE)))
631be09fc23SSascha Wildner 					printf("%s\r\n", dialer_buf);
632be09fc23SSascha Wildner #endif
633be09fc23SSascha Wildner 				return (1);
634be09fc23SSascha Wildner 			}
635be09fc23SSascha Wildner 			for (bm = baud_msg ; bm->msg ; bm++)
636be09fc23SSascha Wildner 				if (strcmp(bm->msg, dialer_buf+sizeof("CONNECT")-1) == 0) {
637be09fc23SSascha Wildner 					if (!acu_setspeed (bm->baud))
638be09fc23SSascha Wildner 						goto error;
639be09fc23SSascha Wildner 					signal(SIGALRM, f);
640be09fc23SSascha Wildner #ifdef DEBUG
641be09fc23SSascha Wildner 					if (boolean(value(VERBOSE)))
642be09fc23SSascha Wildner 						printf("%s\r\n", dialer_buf);
643be09fc23SSascha Wildner #endif
644be09fc23SSascha Wildner 					return (1);
645be09fc23SSascha Wildner 				}
646be09fc23SSascha Wildner 			break;
647be09fc23SSascha Wildner 		}
648be09fc23SSascha Wildner 		dialer_buf[nc] = c;
649be09fc23SSascha Wildner 	}
650be09fc23SSascha Wildner 	printf("%s\r\n", dialer_buf);
651be09fc23SSascha Wildner error:
652be09fc23SSascha Wildner 	signal(SIGALRM, f);
653be09fc23SSascha Wildner 	return (0);
654be09fc23SSascha Wildner }
655be09fc23SSascha Wildner 
656be09fc23SSascha Wildner /*
657be09fc23SSascha Wildner  * This convoluted piece of code attempts to get
658be09fc23SSascha Wildner  * the unidialer in sync.
659be09fc23SSascha Wildner  */
660be09fc23SSascha Wildner static int
unidialersync(void)661be09fc23SSascha Wildner unidialersync(void)
662be09fc23SSascha Wildner {
663be09fc23SSascha Wildner 	int already = 0;
664be09fc23SSascha Wildner 	int len;
665be09fc23SSascha Wildner 	char buf[40];
666be09fc23SSascha Wildner 
667be09fc23SSascha Wildner 	while (already++ < MAXRETRY) {
668be09fc23SSascha Wildner 		acu_nap (intercommand_delay);
669be09fc23SSascha Wildner 		acu_flush (); /* flush any clutter */
670be09fc23SSascha Wildner 		unidialer_write_str (FD, reset_command); /* reset modem */
671be09fc23SSascha Wildner 		bzero(buf, sizeof(buf));
672be09fc23SSascha Wildner 		acu_nap (reset_delay);
673be09fc23SSascha Wildner 		ioctl (FD, FIONREAD, &len);
674be09fc23SSascha Wildner 		if (len) {
675be09fc23SSascha Wildner 			len = read(FD, buf, sizeof(buf));
676be09fc23SSascha Wildner #ifdef DEBUG
677be09fc23SSascha Wildner 			buf [len] = '\0';
678be09fc23SSascha Wildner 			printf("unidialersync (%s): (\"%s\")\n\r", modem_name, buf);
679be09fc23SSascha Wildner #endif
680be09fc23SSascha Wildner 			if (index(buf, '0') ||
681be09fc23SSascha Wildner 		   	   (index(buf, 'O') && index(buf, 'K')))
682be09fc23SSascha Wildner 				return(1);
683be09fc23SSascha Wildner 		}
684be09fc23SSascha Wildner 		/*
685be09fc23SSascha Wildner 		 * If not strapped for DTR control,
686be09fc23SSascha Wildner 		 * try to get command mode.
687be09fc23SSascha Wildner 		 */
688be09fc23SSascha Wildner 		acu_nap (escape_guard_time);
689be09fc23SSascha Wildner 		unidialer_write_str (FD, escape_sequence);
690be09fc23SSascha Wildner 		acu_nap (escape_guard_time);
691be09fc23SSascha Wildner 		unidialer_write_str (FD, hangup_command);
692be09fc23SSascha Wildner 		/*
693be09fc23SSascha Wildner 		 * Toggle DTR to force anyone off that might have left
694be09fc23SSascha Wildner 		 * the modem connected.
695be09fc23SSascha Wildner 		 */
696be09fc23SSascha Wildner 		acu_nap (escape_guard_time);
697be09fc23SSascha Wildner 		ioctl (FD, TIOCCDTR, 0);
698be09fc23SSascha Wildner 		acu_nap (1000);
699be09fc23SSascha Wildner 		ioctl (FD, TIOCSDTR, 0);
700be09fc23SSascha Wildner 	}
701be09fc23SSascha Wildner 	acu_nap (intercommand_delay);
702be09fc23SSascha Wildner 	unidialer_write_str (FD, reset_command);
703be09fc23SSascha Wildner 	return (0);
704be09fc23SSascha Wildner }
705be09fc23SSascha Wildner 
706be09fc23SSascha Wildner /*
707be09fc23SSascha Wildner 	Send commands to modem; impose delay between commands.
708be09fc23SSascha Wildner */
unidialer_modem_cmd(int fd,const char * cmd)709be09fc23SSascha Wildner static void unidialer_modem_cmd (int fd, const char *cmd)
710be09fc23SSascha Wildner {
711be09fc23SSascha Wildner 	static struct timeval oldt = { 0, 0 };
712be09fc23SSascha Wildner 	struct timeval newt;
713be09fc23SSascha Wildner 	tod_gettime (&newt);
714be09fc23SSascha Wildner 	if (tod_lt (&newt, &oldt))
715be09fc23SSascha Wildner 	{
716be09fc23SSascha Wildner 		unsigned int naptime;
717be09fc23SSascha Wildner 		tod_subfrom (&oldt, newt);
718be09fc23SSascha Wildner 		naptime = oldt.tv_sec * 1000 + oldt.tv_usec / 1000;
719be09fc23SSascha Wildner 		if (naptime > intercommand_delay)
720be09fc23SSascha Wildner 		{
721be09fc23SSascha Wildner #ifdef DEBUG
722be09fc23SSascha Wildner 		printf ("unidialer_modem_cmd: suspicious naptime (%u ms)\r\n", naptime);
723be09fc23SSascha Wildner #endif
724be09fc23SSascha Wildner 			naptime = intercommand_delay;
725be09fc23SSascha Wildner 		}
726be09fc23SSascha Wildner #ifdef DEBUG
727be09fc23SSascha Wildner 		printf ("unidialer_modem_cmd: delaying %u ms\r\n", naptime);
728be09fc23SSascha Wildner #endif
729be09fc23SSascha Wildner 		acu_nap (naptime);
730be09fc23SSascha Wildner 	}
731be09fc23SSascha Wildner 	unidialer_write_str (fd, cmd);
732be09fc23SSascha Wildner 	tod_gettime (&oldt);
733be09fc23SSascha Wildner 	newt.tv_sec = 0;
734be09fc23SSascha Wildner 	newt.tv_usec = intercommand_delay;
735be09fc23SSascha Wildner 	tod_addto (&oldt, &newt);
736be09fc23SSascha Wildner }
737be09fc23SSascha Wildner 
unidialer_write_str(int fd,const char * cp)738be09fc23SSascha Wildner static void unidialer_write_str (int fd, const char *cp)
739be09fc23SSascha Wildner {
740be09fc23SSascha Wildner #ifdef DEBUG
741be09fc23SSascha Wildner 	printf ("unidialer (%s): sending %s\n", modem_name, cp);
742be09fc23SSascha Wildner #endif
743be09fc23SSascha Wildner 	unidialer_write (fd, cp, strlen (cp));
744be09fc23SSascha Wildner }
745be09fc23SSascha Wildner 
unidialer_write(int fd,const char * cp,int n)746be09fc23SSascha Wildner static void unidialer_write (int fd, const char *cp, int n)
747be09fc23SSascha Wildner {
748be09fc23SSascha Wildner 	acu_nap (intercharacter_delay);
749be09fc23SSascha Wildner 	for ( ; n-- ; cp++) {
750be09fc23SSascha Wildner 		write (fd, cp, 1);
751be09fc23SSascha Wildner 		acu_nap (intercharacter_delay);
752be09fc23SSascha Wildner 	}
753be09fc23SSascha Wildner }
754be09fc23SSascha Wildner 
755be09fc23SSascha Wildner #ifdef DEBUG
unidialer_verbose_read(void)756be09fc23SSascha Wildner static void unidialer_verbose_read(void)
757be09fc23SSascha Wildner {
758be09fc23SSascha Wildner 	int n = 0;
759be09fc23SSascha Wildner 	char buf[BUFSIZ];
760be09fc23SSascha Wildner 
761be09fc23SSascha Wildner 	if (ioctl(FD, FIONREAD, &n) < 0)
762be09fc23SSascha Wildner 		return;
763be09fc23SSascha Wildner 	if (n <= 0)
764be09fc23SSascha Wildner 		return;
765be09fc23SSascha Wildner 	if (read(FD, buf, n) != n)
766be09fc23SSascha Wildner 		return;
767be09fc23SSascha Wildner 	write(1, buf, n);
768be09fc23SSascha Wildner }
769be09fc23SSascha Wildner #endif
770be09fc23SSascha Wildner 
771be09fc23SSascha Wildner /* end of unidialer.c */
772