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