#include "ckcsym.h" #ifndef NOLOCAL #ifndef NODIAL char *dialv = "Dial Command, 5A(063) 4 Oct 94"; /* C K U D I A -- Module for automatic modem dialing. */ /* Copyright (C) 1985, 1994, Trustees of Columbia University in the City of New York. The C-Kermit software may not be, in whole or in part, licensed or sold for profit as a software product itself, nor may it be included in or distributed with commercial products or otherwise distributed by commercial concerns to their clients or customers without written permission of the Office of Kermit Development and Distribution, Columbia University. This copyright notice must not be removed, altered, or obscured. Original (version 1, 1985) author: Herm Fischer, Encino, CA. Contributed to Columbia University in 1985 for inclusion in C-Kermit 4.0. Author and maintainer since 1985: Frank da Cruz, Columbia University, fdc@columbia.edu. Contributions by many others throughout the years, including: Mark Berryman, Fernando Cabral, John Chmielewski, Joe Doupnik, Richard Hill, Larry Jacobs, Eric Jones, Tom Kloos, Bob Larson, Peter Mauzey, Joe Orost, Kevin O'Gorman, Kai Uwe Rommel, Dan Schullman, Warren Tucker, and others too numerous to list here (but see acknowledgements in ckcmai.c). This module calls externally defined system-dependent functions for communications i/o, as defined in CKCPLM.DOC, the C-Kermit Program Logic Manual, and thus should be portable to all systems that implement those functions, and where alarm() and signal() work as they do in UNIX. */ /* To add support for another modem, do the following, all in this module: 1. Define a modem-type number symbol (n_XXX) for it. 2. Adjust MAX_MDM to the new number of modem types. 3. Create a MDMINF structure for it. 4. Add the address of the MDMINF structure to the ptrtab[] array, according to the numerical value of the modem-type number. 5. Add the user-visible (SET MODEM) name and corresponding modem number to the mdmtab[] array, in alphabetical order by modem-name string. 6. Read through the code and add modem-specific sections as necessary. NOTE: The MINIDIAL symbol is used to build this module to include support for only a minimum number of standard and/or generally useful modem types, namely Hayes, CCITT V.25bis, "Unknown", and None. When adding support for a new modem type, keep it outside of the MINIDIAL sections. */ #include "ckcdeb.h" #ifndef MAC #include #endif /* MAC */ #include "ckcasc.h" #include "ckcker.h" #include "ckucmd.h" #include "ckcnet.h" #ifndef ZILOG #include /* Longjumps */ #else #include #endif /* ZILOG */ #ifdef MAC #define signal msignal #define SIGTYP long #define alarm malarm #define SIG_IGN 0 #define SIGALRM 1 #define SIGINT 2 SIGTYP (*msignal(int type, SIGTYP (*func)(int)))(int); #endif /* MAC */ #ifdef AMIGA #define signal asignal #define alarm aalarm #define SIGALRM (_NUMSIG+1) #define SIGTYP void SIGTYP (*asignal(int type, SIGTYP (*func)(int)))(int); unsigned aalarm(unsigned); #endif /* AMIGA */ #ifdef STRATUS /* VOS doesn't have alarm(), but it does have some things we can work with. */ /* However, we have to catch all the signals in one place to do this, so */ /* we intercept the signal() routine and call it from our own replacement. */ #define signal vsignal #define alarm valarm SIGTYP (*vsignal(int type, SIGTYP (*func)(int)))(int); int valarm(int interval); #ifdef putchar #undef putchar #endif /* putchar */ #define putchar(x) conoc(x) #ifdef getchar #undef getchar #endif /* getchar */ #define getchar(x) coninc(0) #endif /* STRATUS */ int /* SET DIAL parameters */ dialhng = 1, /* DIAL HANGUP, default is ON */ dialdpy = 0, /* DIAL DISPLAY, default is OFF */ mdmspd = 1, /* DIAL SPEED-MATCHING (1 = ON) */ dialtmo = 0, /* DIAL TIMEOUT */ dialksp = 0, /* DIAL KERMIT-SPOOF, 0 = OFF */ dialmnp = 0, /* DIAL MNP-ENABLE, 0 = OFF */ #ifdef NOMDMHUP dialmhu = 0; /* DIAL MODEM-HANGUP, 0 = OFF */ #else dialmhu = 1; /* DIAL MODEM-HANGUP */ #endif /* NOMDMHUP */ int dialsta = DIA_UNK; /* Detailed return code (ckuusr.h) */ char *dialdir = NULL; /* DIAL DIRECTORY, default none */ char *dialini = NULL; /* DIAL INIT-STRING, default none */ char *dialcmd = NULL; /* DIAL DIAL-COMMAND, default none */ char *dialnpr = NULL; /* DIAL NUMBER-PREFIX, ditto */ /* char *hupcmd = NULL; */ /* Modem hangup command */ FILE * dialfd = NULL; /* File descriptor of dial directory */ #ifndef MINIDIAL /* Telebit model codes: ATI Model Numbers Examples --- ------------- -------- 123 Telebit in "total Hayes-1200" emulation mode 960 Telebit in Conventional Command (Hayes) mode 961 RA12C IBM PC internal original Trailblazer 962 RA12E External original Trailblazer 963 RM12C Rackmount original Trailblazer 964 T18PC IBM PC internal Trailblazer-Plus (TB+) 965 T18SA, T2SAA, T2SAS External TB+, T1600, T2000, T3000, WB, and later 966 T18RMM Rackmount TB+ 967 T2MC IBM PS/2 internal TB+ 968 T1000 External T1000 969 ? Qblazer 970 Qblazer Plus 971 T2500 External T2500 972 T2500 Rackmount T2500 */ /* Telebit model codes */ #define TB_UNK 0 /* Unknown Telebit model */ #define TB_BLAZ 1 /* Original TrailBlazer */ #define TB_PLUS 2 /* TrailBlazer Plus */ #define TB_1000 3 /* T1000 */ #define TB_1500 4 /* T1500 */ #define TB_1600 5 /* T1600 */ #define TB_2000 6 /* T2000 */ #define TB_2500 7 /* T2500 */ #define TB_3000 8 /* T3000 */ #define TB_QBLA 9 /* Qblazer */ #define TB_WBLA 10 /* WorldBlazer */ #define TB__MAX 10 /* Highest number */ char *tb_name[] = { /* Array of model names */ "Unknown", /* TB_UNK */ "TrailBlazer", /* TB_BLAZ */ "TrailBlazer-Plus", /* TB_PLUS */ "T1000", /* TB_1000 */ "T1500", /* TB_1500 */ "T1600", /* TB_1600 */ "T2000", /* TB_2000 */ "T2500", /* TB_2500 */ "T3000", /* TB_3000 */ "Qblazer", /* TB_QBLA */ "WorldBlazer", /* TB_WBLA */ "" }; #endif /* MINIDIAL */ extern int flow, local, mdmtyp, quiet, backgrd, parity, seslog, network; extern int carrier, duplex; #ifdef NETCONN extern int ttnproto; #endif /* NETCONN */ extern CHAR stchr; extern long speed; extern char ttname[], sesfil[]; /* Failure codes */ #define F_TIME 1 /* timeout */ #define F_INT 2 /* interrupt */ #define F_MODEM 3 /* modem-detected failure */ #define F_MINIT 4 /* cannot initialize modem */ static int fail_code = 0; /* Default failure reason. */ #define DW_NOTHING 0 /* What we are doing */ #define DW_INIT 1 #define DW_DIAL 2 static int dial_what = DW_NOTHING; /* Nothing at first. */ static int mymdmtyp; /* Local copy of modem type. */ _PROTOTYP (static int ddinc, (int) ); _PROTOTYP (int dialhup, (void) ); _PROTOTYP (static int getok, (int,int) ); _PROTOTYP (char * getdws, (int) ); _PROTOTYP (char * ck_time, (void) ); _PROTOTYP (static VOID ttslow, (char *, int) ); #ifdef COMMENT _PROTOTYP (static VOID xcpy, (char *, char *, unsigned int) ); #endif /* COMMENT */ _PROTOTYP (static VOID waitfor, (char *) ); _PROTOTYP (static VOID dialoc, (char) ); _PROTOTYP (static int didweget, (char *, char *) ); _PROTOTYP (static VOID spdchg, (long) ); _PROTOTYP (static VOID tbati3, (int) ); _PROTOTYP (static int dialfail, (int) ); #define MDMINF struct mdminf MDMINF /* structure for modem-specific information */ { int dial_time; /* time modem allows for dialing (secs) */ char *pause_chars; /* character(s) to tell modem to pause */ int pause_time; /* time associated with pause chars (secs) */ char *wake_str; /* string to wakeup modem & put in cmd mode */ int wake_rate; /* delay between wake_str characters (msecs) */ char *wake_prompt; /* string prompt after wake_str */ char *dmode_str; /* string to put modem in dialing mode */ char *dmode_prompt; /* string prompt for dialing mode */ char *dial_str; /* dialing string, with "%s" for number */ int dial_rate; /* delay between dialing characters (msecs) */ int esc_time; /* guard time on escape sequence (msecs) */ char *esc_str; /* escape sequence */ char *hup_str; /* hangup string */ _PROTOTYP( int (*ok_fn), (int,int) ); /* func to read response string */ }; /* * Define symbolic modem numbers. * * The numbers MUST correspond to the ordering of entries * within the ptrtab array, and start at one (1). * * It is assumed that there are relatively few of these * values, and that the high(er) bytes of the value may * be used for modem-specific mode information. * * REMEMBER that only the first eight characters of these * names are guaranteed to be unique. */ #ifdef MINIDIAL /* Minimum dialer support */ /* Only for CCITT, HAYES, and UNK */ #define n_CCITT 1 #define n_HAYES 2 #define n_UNKNOWN 3 #define MAX_MDM 3 /* Number of modem types */ #else /* Full-blown dialer support */ #define n_ATTDTDM 1 #define n_ATTISN 2 #define n_ATTMODEM 3 #define n_CCITT 4 #define n_CERMETEK 5 #define n_DF03 6 #define n_DF100 7 #define n_DF200 8 #define n_GDC 9 #define n_HAYES 10 #define n_PENRIL 11 #define n_RACAL 12 #define n_UNKNOWN 13 #define n_USROBOT 14 #define n_VENTEL 15 #define n_CONCORD 16 #define n_ATTUPC 17 /* aka UNIX PC and ATT7300 */ #define n_ROLM 18 /* Rolm CBX DCM */ #define n_MICROCOM 19 #define n_HST 20 #define n_TELEBIT 21 /* Telebits of all kinds */ #define n_DIGITEL 22 /* Digitel DT-22 (CCITT variant) */ #define MAX_MDM 22 /* Number of modem types */ #endif /* MINIDIAL */ /* * Declare modem "variant" numbers for any of the above for which it is * necessary to note various operational modes, using the second byte * of a modem number. * * It is assumed that such modem modes share the same modem-specific * information (see MDMINF structure) but may differ in some of the actions * that are performed. */ /* Warning - this is starting to get kind of hokey... */ #define DIAL_NV 256 #define n_HAYESNV ( n_HAYES | DIAL_NV ) #ifndef MINIDIAL #define DIAL_PEP 512 #define DIAL_V32 1024 #define DIAL_V42 2048 #define DIAL_SLO 4096 #define n_TBPEP ( n_TELEBIT | DIAL_PEP ) #define n_TB3 ( n_TELEBIT | DIAL_V32 ) #define n_TBNV ( n_TELEBIT | DIAL_NV ) #define n_TBPNV ( n_TELEBIT | DIAL_NV | DIAL_PEP ) #define n_TB3NV ( n_TELEBIT | DIAL_NV | DIAL_V32 ) #define n_TB4 ( n_TELEBIT | DIAL_V42 ) #define n_TBS ( n_TELEBIT | DIAL_SLO ) #define n_TB4NV ( n_TELEBIT | DIAL_NV | DIAL_V42 ) #define n_TBSNV ( n_TELEBIT | DIAL_NV | DIAL_SLO ) #endif /* MINIDIAL */ /* * Declare structures containing modem-specific information. * * REMEMBER that only the first SEVEN characters of these * names are guaranteed to be unique. */ #ifndef MINIDIAL static MDMINF ATTISN = /* AT&T ISN Network */ { 30, /* Dial time */ "", /* Pause characters */ 0, /* Pause time */ "\015\015\015\015", /* Wake string */ 900, /* Wake rate */ "DIAL", /* Wake prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF ATTMODEM = /* information for AT&T switched-network modems */ /* "Number" following "dial" can include: p's and * t's to indicate pulse or tone (default) dialing, * + for wait for dial tone, , for pause, r for * last number dialed, and, except for 2224B, some * comma-delimited options like o12=y, before number. * "Important" options for the modems: * * All: Except for 2224B, enable option 12 for "transparent * data," o12=y. If a computer port used for both * incoming and outgoing calls is connected to the * modem, disable "enter interactive mode on carriage * return," EICR. The Kermit "dial" command can * function with EIA leads standard, EIAS. * * 2212C: Internal hardware switches at their default * positions (four rockers down away from numbers) * unless EICR is not wanted (rocker down at the 4). * For EIAS, rocker down at the 1. * * 2224B: Front-panel switch position 1 must be up (at the 1, * closed). Disable EICR with position 2 down. * For EIAS, position 4 down. * All switches on the back panel down. * * 2224CEO: All front-panel switches down except either 5 or 6. * Enable interactive flow control with o16=y. * Select normal asynchronous mode with o34=0 (zero). * Disable EICR with position 3 up. For EIAS, 1 up. * Reset the modem after changing switches. * * 2296A: If option 00 (zeros) is present, use o00=0. * Enable interactive flow control with o16=y. * Select normal asynchronous mode with o34=0 (zero). * (available in Microcom Networking version, but * not necessarily other models of the 2296A). * Enable modem-port flow control (if available) with * o42=y. Enable asynchronous operation with o50=y. * Disable EICR with o69=n. For EIAS, o66=n, using * front panel. */ { 20, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ "+", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "at%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF ATTDTDM = /* information for AT&T Digital Terminal Data Module * For dialing: KYBD switch down, others usually up. */ { 20, /* dial_time */ "", /* pause_chars */ 0, /* pause_time */ "", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "%s\015", /* dial_str */ /* not used */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; #endif /* MINIDIAL */ static MDMINF CCITT = /* CCITT V.25bis autodialer */ /* According to V.25bis: . Even parity is required for giving commands to the modem. . Commands might or might not echo. . Responses ("Indications") from the modem are terminated by CR and LF. . Call setup is accomplished by: - DTE raises DTR (V.24 circuit 108) [ttopen() does this] - Modem raises CTS (V.24 circuit 106) [C-Kermit ignores this] - DTE issues a call request command ("CRN") - Modem responds with "VAL" ("command accepted") - If the call is completed: modem responds with "CNX" ("call connected"); modem turns CTS (106) OFF; modem turns DSR (107) ON; else: modem responds with "CFI " ("call failure indication"). . To clear a call, the DTE turns DTR (108) OFF. . There is no mention of the Carrier Detect circuit (109) in the standard. . There is no provision for "escaping back" to the modem's command mode. It is not known whether there exists in real life a pure V.25bis modem. If there is, this code has never been tested on it. See the Digitel entry. */ { 40, /* dial_time -- programmable -- */ ",:", /* pause_chars -- "," waits for programmable time */ /* ":" waits for dial tone */ 10, /* pause_time (seconds, just a guess) */ "", /* wake_str (none) */ 200, /* wake_rate (msec) */ "VAL", /* wake_prompt */ "", /* dmode_str (none) */ "", /* dmode_prompt (none) */ "CRN%s\015", /* dial_str */ 200, /* dial_rate (msec) */ 0, /* No esc_time */ "", /* No esc_str */ "", /* No hup_str */ NULL /* No ok_fn */ }; #ifndef MINIDIAL /* Don't include the following if -DMINIDIAL ... */ static MDMINF CERMETEK = /* Information for "Cermetek Info-Mate 212 A" modem */ { 20, /* dial_time */ "BbPpTt", /* pause_chars */ 0, /* pause_time */ /** unknown -- DS **/ " XY\016R\015", /* wake_str */ 200, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "\016D '%s'\015", /* dial_str */ 200, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF DF03 = /* information for "DEC DF03-AC" modem */ { 27, /* dial_time */ "=", /* pause_chars */ /* wait for second dial tone */ 15, /* pause_time */ "\001\002", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "%s", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF DF100 = /* information for "DEC DF100-series" modem */ /* * The telephone "number" can include "P"s and/or "T"s * within it to indicate that subsequent digits are * to be dialed using pulse or tone dialing. The * modem defaults to pulse dialing. You may modify * the dial string below to explicitly default all * dialing to pulse or tone, but doing so prevents * the use of phone numbers that you may have stored * in the modem's memory. */ { 30, /* dial_time */ "=", /* pause_chars */ /* wait for second dial tone */ 15, /* pause_time */ "\001", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "%s#", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF DF200 = /* information for "DEC DF200-series" modem */ /* * The telephone "number" can include "P"s and/or "T"s * within it to indicate that subsequent digits are * to be dialed using pulse or tone dialing. The * modem defaults to pulse dialing. You may modify * the dial string below to explicitly default all * dialing to pulse or tone, but doing so prevents * the use of phone numbers that you may have stored * in the modem's memory. */ { 30, /* dial_time */ "=W", /* pause_chars */ /* =: second tone; W: 5 secs */ 15, /* pause_time */ /* worst case */ "\002", /* wake_str */ /* allow stored number usage */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ #ifdef COMMENT "%s!", /* dial_str */ #else " d %s\015", #endif /* COMMENT */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF DIGITEL = /* Digitel DT-22 CCITT variant used in Brazil */ /* Attempts to adhere strictly to the V.25bis specification do not produce good results in real life. The modem for which this code was developed: (a) ignores parity; (b) sometimes terminates responses with LF CR instead of CR LF; (c) has a Hayes-like escape sequence; (d) supports a hangup ("HUP") command. Information from Fernando Cabral in Brasilia. */ { 40, /* dial_time -- programmable -- */ ",:", /* pause_chars -- "," waits for programmable time */ /* ":" waits for dial tone */ 10, /* pause_time (seconds, just a guess) */ "HUP\015", /* wake_str (Not Standard CCITT) */ 200, /* wake_rate (msec) */ "VAL", /* wake_prompt */ "", /* dmode_str (none) */ "", /* dmode_prompt (none) */ "CRN%s\015", /* dial_str */ 200, /* dial_rate (msec) */ 1100, /* esc_time (Not Standard CCITT) */ "+++", /* esc_str (Not Standard CCITT) */ "HUP\015", /* hup_str (Not Standard CCITT) */ getok /* ok_fn */ }; static MDMINF GDC = /* information for "GeneralDataComm 212A/ED" modem */ { 32, /* dial_time */ "%", /* pause_chars */ 3, /* pause_time */ "\015\015", /* wake_str */ 500, /* wake_rate */ "$", /* wake_prompt */ "D\015", /* dmode_str */ ":", /* dmode_prompt */ "T%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; #endif /* MINIDIAL */ static MDMINF HAYES = /* Information for Hayes and Hayes-like modems */ { 35, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ "ATQ0S2=43\015", /* wake_str */ /* Note: Other wake_str's are possible here. For a Hayes 2400 that is to be used for both in and out calls, AT&F&D3 might be best. For out calls only, maybe AT&F&D2. See Hayes 2400 manual. */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "ATD%s\015", /* dial_str, note: user can supply D or T */ 0, /* dial_rate */ 1100, /* esc_time */ "+++", /* esc_str */ "ATQ0H0\015", /* hup_str */ getok /* ok_fn */ }; #ifndef MINIDIAL static MDMINF PENRIL = /* information for "Penril" modem */ { 50, /* dial_time */ "", /* pause_chars */ /** unknown -- HF **/ 0, /* pause_time */ "\015\015", /* wake_str */ 300, /* wake_rate */ ">", /* wake_prompt */ "k\015", /* dmode_str */ ":", /* dmode_prompt */ "%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF RACAL = /* information for "Racal Vadic" modem, e.g. VA4492E */ { 35, /* dial_time (manual says modem is hardwired to 60) */ "Kk", /* pause_chars */ 5, /* pause_time */ "\005\015", /* wake_str, ^E^M */ 50, /* wake_rate */ "*", /* wake_prompt */ "D\015", /* dmode_str */ "?", /* dmode_prompt */ "%s\015", /* dial_str */ 0, /* dial_rate */ 1100, /* esc_time */ "\003\004", /* esc_str, ^C^D (this actually hangs up) */ "\005", /* hup_str, ^E (this goes back to command mode) */ NULL /* ok_fn */ }; #endif /* MINIDIAL */ /* The intent of the "unknown" modem is to allow KERMIT to support unknown modems by having the user type the entire autodial sequence (possibly including control characters, etc.) as the "phone number". The protocol and other characteristics of this modem are unknown, with some "reasonable" values being chosen for some of them. The only way to detect if a connection is made is to look for carrier. */ static MDMINF UNKNOWN = /* information for "Unknown" modem */ { 30, /* dial_time */ "", /* pause_chars */ 0, /* pause_time */ "", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; #ifndef MINIDIAL static MDMINF USROBOT = /* information for "US Robotics 212A" modem */ { 30, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ "ATQ0S2=43\015", /* wake_str */ 0, /* wake_rate */ "OK\015", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "ATD%s\015", /* dial_str */ 0, /* dial_rate */ 1100, /* esc_time */ "+++", /* esc_str */ "ATQ0H0\015", /* hup_str */ getok /* ok_fn */ }; #ifdef COMMENT /* Reportedly this does not work at all. */ static MDMINF VENTEL = /* information for "Ventel" modem */ { 20, /* dial_time */ "%", /* pause_chars */ 5, /* pause_time */ "\015\015\015", /* wake_str */ 300, /* wake_rate */ "$", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "", /* dial_str (was "") */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; #else /* and this does. */ static MDMINF VENTEL = /* information for "Ventel" modem */ { 20, /* dial_time */ "%", /* pause_chars */ 5, /* pause_time */ "\015\015\015", /* wake_str */ 300, /* wake_rate */ "$", /* wake_prompt */ "K\015", /* dmode_str (was "") */ "Number to call: ", /* dmode_prompt (was NULL) */ "%s\015", /* dial_str (was "") */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; #endif /* COMMENT */ static MDMINF CONCORD = /* Info for Condor CDS 220 2400b modem */ { 35, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ "\015\015", /* wake_str */ 20, /* wake_rate */ "CDS >", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF ATTUPC = /* dummy information for "ATT7300/Unix PC" internal modem */ { 30, /* dial_time */ "", /* pause_chars */ 0, /* pause_time */ "", /* wake_str */ 0, /* wake_rate */ "", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF ROLM = /* IBM / Siemens / Rolm 8000, 9000, 9751 CBX */ { 60, /* dial_time */ "", /* pause_chars */ 0, /* pause_time */ "\015\015", /* wake_str */ 5, /* wake_rate */ "MODIFY?", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "CALL %s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF MICROCOM = /* information for "Microcom" modems in native mode */ /* (long answer only) */ { 35, /* dial_time */ ",!@", /* pause_chars (! and @ aren't pure pauses) */ 3, /* pause_time */ "\015", /* wake_str */ 100, /* wake_rate */ "!", /* wake_prompt */ "", /* dmode_str */ NULL, /* dmode_prompt */ "d%s\015", /* dial_str */ 0, /* dial_rate */ 0, /* esc_time */ "", /* esc_str */ "", /* hup_str */ NULL /* ok_fn */ }; static MDMINF HST = /* information for USR Courier and Sportster modems */ { 35, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ "ATQ0S2=43X4&M4\015", /* wake_str (X7 not supported on Sportster) */ 0, /* wake_rate */ "OK\015", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "ATD%s\015", /* dial_str */ 0, /* dial_rate */ 1100, /* esc_time */ "+++", /* esc_str */ "ATQ0H0\015", /* hup_str */ getok /* ok_fn */ }; static MDMINF TELEBIT = /* information for Telebits */ { 60, /* dial_time */ ",", /* pause_chars */ 2, /* pause_time */ /* NOTE: The wake_string MUST contain the I command (model query), and otherwise must contain commands that work on ALL Telebit models. Here we ensure that result codes are returned (Q0), and ask for extended result codes (X1), and ensure that the escape sequence is +++ and it is enabled. And also, make sure the final character is not a digit (whose echo might be mistaken for a result code). The Ctrl-Q and multiple A's are recommended by Telebit. */ "\021AAAAATQ0X1S12=50 S2=43 I\015", /* wake_str. */ 100, /* wake_rate = 100 msec */ "OK\015", /* wake_prompt */ "", /* dmode_str */ "", /* dmode_prompt */ "ATD%s\015", /* dial_str, Note: no T or P */ 80, /* dial_rate */ 1100, /* esc_time (guard time) */ "+++", /* esc_str */ "ATQ0H0\015", /* hup_str */ getok /* ok_fn */ }; #endif /* MINIDIAL */ /* * Declare table for converting modem numbers to information pointers. * * The entries MUST be in ascending order by modem number, without any * "gaps" in the numbers, and starting from one (1). * * This table should NOT include entries for the "variant" modem numbers, * since they share the same information as the normal value. */ static MDMINF *ptrtab[] = { #ifdef MINIDIAL &CCITT, &HAYES, &UNKNOWN #else &ATTDTDM, &ATTISN, &ATTMODEM, &CCITT, &CERMETEK, &DF03, &DF100, &DF200, &GDC, &HAYES, &PENRIL, &RACAL, &UNKNOWN, &USROBOT, &VENTEL, &CONCORD, &ATTUPC, &ROLM, &MICROCOM, &HST, &TELEBIT, &DIGITEL #endif /* MINIDIAL */ }; /* * Declare modem names and associated numbers for command parsing, * and also for doing number-to-name translation. * * The entries must be in alphabetical order by modem name. */ struct keytab mdmtab[] = { #ifndef MINIDIAL "attdtdm", n_ATTDTDM, 0, "attisn", n_ATTISN, 0, "attmodem", n_ATTMODEM, 0, "att7300", n_ATTUPC, 0, #endif /* MINIDIAL */ "ccitt-v25bis", n_CCITT, 0, #ifndef MINIDIAL "cermetek", n_CERMETEK, 0, "concord", n_CONCORD, 0, "courier", n_HST, 0, "df03-ac", n_DF03, 0, "df100-series", n_DF100, 0, "df200-series", n_DF200, 0, "digitel-dt22", n_DIGITEL, 0, #endif /* MINIDIAL */ "direct", 0, CM_INV, /* Synonym for NONE */ #ifndef MINIDIAL "gdc-212a/ed", n_GDC, 0, "gendatacomm", n_GDC, CM_INV, /* Synonym for GDC */ #endif /* MINIDIAL */ "hayes", n_HAYES, 0, #ifndef MINIDIAL "hst-courier", n_HST, CM_INV, /* Synonym for COURIER */ "microcom", n_MICROCOM, 0, #endif /* MINIDIAL */ "none", 0, 0, #ifndef MINIDIAL "penril", n_PENRIL, 0, "pep-telebit", n_TBPEP, 0, "racalvadic", n_RACAL, 0, "rolm", n_ROLM, 0, "slow-telebit", n_TBS, 0, "sportster", n_HST, 0, "telebit", n_TELEBIT, 0, #endif /* MINIDIAL */ "unknown", n_UNKNOWN, 0, #ifndef MINIDIAL "usrobotics-212a", n_USROBOT, 0, "v32-telebit", n_TB3, 0, "v42-telebit", n_TB4, 0, "ventel", n_VENTEL, 0 #endif /* MINIDIAL */ }; int nmdm = (sizeof(mdmtab) / sizeof(struct keytab)); /* Number of modems */ #define CONNECTED 1 /* For completion status */ #define FAILED 2 static int tries = 0; static int mdmecho = 0; /* assume modem does not echo */ static int augmdmtyp; /* "augmented" modem type, to handle modem modes */ static char *p; /* For command strings & messages */ #ifdef DYNAMIC #define LBUFL 256 static char *lbuf = NULL; #else #define LBUFL 100 static char lbuf[LBUFL]; #endif /* DYNAMIC */ #ifdef DYNAMIC #define RBUFL 256 static char *rbuf = NULL; #else #define RBUFL 63 static char rbuf[RBUFL+1]; #endif /* DYNAMIC */ #ifdef DYNAMIC #define FULLNUML 256 char *fbuf = NULL; /* For full (prefixed) phone number */ #else #define FULLNUML 100 char fbuf[FULLNUML]; #endif /* DYNAMIC */ #ifdef CK_POSIX_SIG static sigjmp_buf sjbuf; #else static jmp_buf sjbuf; #endif /* CK_POSIX_SIG */ static SIGTYP (*savalrm)(); /* For saving alarm handler */ static SIGTYP (*savint)(); /* For saving interrupt handler */ #ifndef MINIDIAL int tbmodel = 0; /* Telebit modem model */ char * gtbmodel() { /* Function to return name of Telebit model */ if (tbmodel < 0 || tbmodel > TB__MAX) tbmodel = TB_UNK; return(tb_name[tbmodel]); } #ifdef COMMENT static VOID xcpy(to,from,len) /* Copy the given number of bytes */ register char *to, *from; register unsigned int len; { while (len--) *to++ = *from++; } #endif /* COMMENT */ #endif /* MINIDIAL */ static SIGTYP dialtime(foo) int foo; { /* Timer interrupt handler */ fail_code = F_TIME; /* Failure reason = timeout */ debug(F100,"dialtime caught SIGALRM","",0); #ifdef __EMX__ signal(SIGALRM, SIG_ACK); /* Needed for OS/2 */ #endif /* __EMX__ */ #ifdef OSK /* OS-9 */ /* We are in an intercept routine but do not perform a F$RTE (done implicitly by RTS), so we have to decrement the sigmask as F$RTE does. Warning: longjump only restores the CPU registers, NOT the FPU registers. So, don't use FPU at all or at least don't use common FPU (double or float) register variables. */ sigmask(-1); #endif /* OSK */ #ifdef CK_POSIX_SIG siglongjmp(sjbuf,1); #else longjmp(sjbuf,1); #endif /* CK_POSIX_SIG */ } static SIGTYP dialint(foo) int foo; { /* Keyboard interrupt handler */ fail_code = F_INT; debug(F100,"dialint caught SIGINT","",0); #ifdef __EMX__ signal(SIGINT, SIG_ACK); /* Needed for OS/2 */ #endif /* __EMX__ */ #ifdef OSK /* OS-9, see comment in dialtime() */ sigmask(-1); #endif /* OSK */ #ifdef CK_POSIX_SIG siglongjmp(sjbuf,1); #else longjmp(sjbuf,1); #endif /* CK_POSIX_SIG */ } /* Routine to read a character from communication device, handling TELNET protocol negotiations in case we're connected to the modem through a TCP/IP TELNET modem server. */ static int ddinc(n) int n; { int c; #ifdef TNCODE int done = 0; debug(F101,"ddinc entry n","",n); while (!done) { c = ttinc(n); debug(F000,"ddinc","",c); if (c < 0) return(c); if (c == IAC && network && ttnproto == NP_TELNET) { switch (tn_doop((CHAR)(c & 0xff),duplex,ttinc)) { case 2: duplex = 0; continue; case 1: duplex = 1; default: continue; } } else done = 1; } return(c & 0xff); #else debug(F101,"ddinc entry n","",n); return(ttinc(n)); #endif /* TNCODE */ } static VOID ttslow(s,millisec) char *s; int millisec; { /* Output s-l-o-w-l-y */ #ifdef TCPSOCKET extern int tn_nlm; #endif /* TCPSOCKET */ if (dialdpy && duplex) /* Echo the command in case modem */ printf("%s\n",s); /* isn't echoing commands. */ for (; *s; s++) { ttoc(*s); #ifdef TCPSOCKET if (*s == CR && network && ttnproto == NP_TELNET && tn_nlm != TNL_CR) ttoc((char)((tn_nlm == TNL_CRLF) ? LF : NUL)); #endif /* TCPSOCKET */ msleep(millisec); } } /* * Wait for a string of characters. * * The characters are waited for individually, and other characters may * be received "in between". This merely guarantees that the characters * ARE received, and in the order specified. */ static VOID waitfor(s) char *s; { CHAR c, x; while ( c = *s++ ) { /* while more characters remain... */ do { /* wait for the character */ x = ddinc(0) & 0177; debug(F000,"dial waitfor got","",x); if (dialdpy) { if (x != LF) conoc(x); if (x == CR) conoc(LF); } } while ( x != c); } } static int didweget(s,r) char *s, *r; { /* Looks in string s for response r */ int lr = (int)strlen(r); /* 0 means not found, 1 means found it */ int i; debug(F110,"didweget",r,0); debug(F110," in",s,0); for (i = (int)strlen(s)-lr; i >= 0; i--) if ( s[i] == r[0] ) if ( !strncmp(s+i,r,lr) ) return( 1 ); return( 0 ); } /* R E S E T -- Reset alarms, etc. on exit. */ static VOID dreset() { alarm(0); signal(SIGALRM,savalrm); /* restore alarm handler */ signal(SIGINT,savint); /* restore interrupt handler */ } /* Call this routine when the modem reports that it has connected at a certain speed, giving that speed as the argument. If the connection speed is not the same as Kermit's current communication speed, AND the modem interface speed is not locked (i.e. DIAL SPEED-MATCHING is not ON), then change the device speed to the one given. */ static VOID #ifdef CK_ANSIC spdchg(long s) #else spdchg(s) long s; #endif /* CK_ANSIC */ /* spdchg */ { int s2; if (!mdmspd) /* If modem interface speed locked, */ return; /* don't do this. */ if (speed != s) { /* Speeds differ? */ s2 = s / 10L; /* Convert to cps expressed as int */ if (ttsspd(s2) < 0) { /* Change speed. */ printf(" Warning: speed change to %ld failed.\r\n",s); } else { printf(" Speed changed to %ld.\r\n",s); speed = s; /* Update global speed variable */ } } } /* Display all characters received from modem dialer through this routine, for consistent handling of carriage returns and linefeeds. */ static VOID #ifdef CK_ANSIC dialoc(char c) #else dialoc(c) char c; #endif /* CK_ANSIC */ { /* dialoc */ /* Dial Output Character */ if (dialdpy) { if (c != LF) conoc(c); /* Don't echo LF */ if (c == CR) conoc(LF); /* Echo CR as CRLF */ } } #ifndef MINIDIAL /* tbati3() -- Routine to find out Telebit model when ATI reports "965" or "971". This routine sends another query, ATI3, to get further info to narrow down the model number. Argument is ATI response as integer. Result: sets tbmodel variable to Telebit model. */ static VOID tbati3(n) int n; { int status; ttflui(); /* Flush input buffer */ ttslow("ATI3\015",100); /* Send ATI3 */ status = getok(5,0); /* Get OK response, nonstrict */ if (status < 1) { /* ERROR or timeout */ tbmodel = TB_UNK; debug(F111,"tbati3 fails",rbuf,status); return; } debug(F110,"tbati3 rbuf",rbuf,0); /* Got a good response, check the model info */ if (n == 965) { /* "965" - various models. */ if (didweget(rbuf,"T1600")) { tbmodel = TB_1600; /* T1600 */ } else if (didweget(rbuf,"T3000")) { tbmodel = TB_3000; /* T3000 */ } else if (didweget(rbuf,"World")) { tbmodel = TB_WBLA; /* WorldBlazer */ } else if (didweget(rbuf,"Version B") || /* TrailBlazer-Plus models */ didweget(rbuf,"TBSA") || didweget(rbuf,"TBRM") || didweget(rbuf,"DC")) { /* Ven-Tel EC18K */ tbmodel = TB_PLUS; } else tbmodel = TB_UNK; /* Others: Unknown */ } else if (n == 971) { /* "971" could be T1500 or T1600. */ if (didweget(rbuf,"T1500")) tbmodel = TB_1500; else tbmodel = TB_2500; } /* Other, don't change tbmodel. */ } #endif /* MINIDIAL */ static int dialfail(x) int x; { char * s; fail_code = x; debug(F101,"ckudial dialfail","",x); alarm(0); /* Disable timeouts */ dreset(); /* Reset alarm signals */ printf("DIAL Failure: "); if (dialdpy) { /* If showing progress */ p = ck_time(); /* get current time; */ if (*p) printf("%s: ",p); } switch (fail_code) { /* Type of failure */ case F_TIME: /* Timeout */ if (dial_what == DW_INIT) printf ("Timed out while trying to initialize modem.\n"); else if (dial_what == DW_DIAL) printf ("DIAL TIMEOUT interval expired.\n"); else printf("Timeout.\n"); if (mymdmtyp == n_HAYES #ifndef MINIDIAL || mymdmtyp == n_TELEBIT || mymdmtyp == n_HST #endif /* MINIDIAL */ ) ttoc('\015'); /* Send CR to interrupt dialing */ /* Some Hayes modems don't fail with BUSY on busy lines */ dialsta = DIA_TIMO; debug(F110,"dial","timeout",0); break; case F_INT: /* Dialing interrupted */ printf ("Interrupted.\n"); debug(F110,"dial","interrupted",0); if (mymdmtyp == n_HAYES #ifndef MINIDIAL || mymdmtyp == n_TELEBIT || mymdmtyp == n_HST #endif /* MINIDIAL */ ) ttoc('\015'); /* Send CR to interrupt dialing */ dialsta = DIA_INTR; break; case F_MODEM: /* Modem detected a failure */ printf ("Call not completed."); if (*lbuf) { printf(" Modem message: \""); for (s = lbuf; *s; s++) if (isprint(*s)) putchar(*s); /* Display printable reason */ printf ("\""); } printf("\n"); debug(F110,"dial",lbuf,0); if (dialsta < 0) dialsta = DIA_UNSP; break; case F_MINIT: /* Failure to initialize modem */ printf ("Error initializing modem.\n"); debug(F110,"dial","modem init",0); dialsta = DIA_NOIN; break; } #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ if (dialsta < 0) dialsta = DIA_UERR; /* Set failure code */ return(0); /* Return zero (important) */ } /* C K D I A L -- Dial up the remote system */ /* Returns 1 if call completed, 0 otherwise */ static int waitct, mdmwait, mdmstat = 0; int ckdial(telnbr) char *telnbr; { char c, c2; #define ERMSGL 50 char errmsg[ERMSGL], *erp; /* for error messages */ MDMINF *pmdminf; /* pointer to modem-specific info */ int x, m, n = F_TIME; char *s, *ws; char *mmsg = "Sorry, DIAL memory buffer can't be allocated\n"; long conspd; char *cptr; mymdmtyp = mdmtyp; /* Make local copy of modem type */ dialsta = DIA_UNK; /* Start with return code = unknown */ dial_what = DW_NOTHING; /* Doing nothing at first. */ #ifndef MINIDIAL tbmodel = TB_UNK; /* Initialize Telebit model */ #endif /* MINIDIAL */ if (mymdmtyp < 1) { if (network) printf("Please SET HOST first, and then SET MODEM\n"); else printf("Sorry, you must SET MODEM first\n"); dialsta = DIA_NOMO; return(0); } if (!local) { printf("Sorry, you must SET LINE or SET HOST first\n"); dialsta = DIA_NOLI; return(0); } if (!network && (speed < 0L) #ifdef UNIX && (strcmp(ttname,"/dev/null")) #endif /* UNIX */ ) { printf("Sorry, you must SET SPEED first\n"); dialsta = DIA_NOSP; return(0); } debug(F110,"dial number",telnbr,0); debug(F110,"dial prefix",(dialnpr ? dialnpr : ""), 0); #ifdef DYNAMIC if (!(lbuf = malloc(LBUFL+1))) { /* Allocate input line buffer */ printf("%s", mmsg); dialsta = DIA_IE; return(0); } *lbuf = NUL; debug(F101,"DIAL lbuf malloc ok","",LBUFL+1); if (!rbuf) { /* This one might already have been allocated by getok() */ if (!(rbuf = malloc(RBUFL+1))) { /* Allocate input line buffer */ printf("%s", mmsg); dialsta = DIA_IE; if (lbuf) free(lbuf); lbuf = NULL; return(0); } else debug(F101,"DIAL rbuf malloc ok","",RBUFL+1); } if (!(fbuf = malloc(FULLNUML+1))) { /* Allocate input line buffer */ printf("%s", mmsg); dialsta = DIA_IE; if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; return(0); } debug(F101,"DIAL fbuf malloc ok","",FULLNUML+1); #endif /* DYNAMIC */ /* Add prefix to phone number */ if (dialnpr && *dialnpr) { if (((int)strlen(dialnpr) + (int)strlen(dialnpr)) > FULLNUML) { printf("DIAL prefix + phone number too long!\n"); #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(0); } sprintf(fbuf,"%s%s",(dialnpr ? dialnpr : ""),telnbr); telnbr = fbuf; } debug(F110,"prefixed number", telnbr, 0); if (ttopen(ttname,&local,mymdmtyp,0) < 0) { /* Open, no carrier wait */ erp = errmsg; if ((int)strlen(ttname) < (ERMSGL - 18)) sprintf(erp,"Sorry, can't open %s",ttname); else sprintf(erp,"Sorry, can't open device"); perror(errmsg); dialsta = DIA_OPEN; #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(0); } /* Condition console terminal and communication line */ /* Place line into "clocal" dialing state, */ /* important mainly for System V UNIX. */ if (ttpkt(speed,FLO_DIAL,parity) < 0) { ttclos(0); /* If ttpkt fails do all this... */ if (ttopen(ttname,&local,mymdmtyp,0) < 0) { erp = errmsg; if ((int)strlen(ttname) < (ERMSGL - 18)) sprintf(erp,"Sorry, can't reopen %s",ttname); else sprintf(erp,"Sorry, can't reopen device"); perror(errmsg); dialsta = DIA_OPEN; #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(0); } /* And try again. */ if ((ttpkt(speed,FLO_DIAL,parity) < 0) #ifdef UNIX && (strcmp(ttname,"/dev/null")) #endif /* UNIX */ ) { printf("Sorry, Can't condition communication line\n"); printf("Try 'set line %s' again\n",ttname); dialsta = DIA_OPEN; #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(0); } } msleep(500); pmdminf = ptrtab[ (mymdmtyp & 0xff) -1 ]; /* set pointer to modem info */ augmdmtyp = mymdmtyp; /* initialize "augmented" modem type */ mymdmtyp &= 0xff; /* major modem type */ /* Interdigit waits for tone dial */ if (dialtmo < 1) { /* Automatic computation. */ waitct = 1 * (int)strlen(telnbr) ; /* Compute worst case dial time */ waitct += pmdminf->dial_time; /* dialtone + completion wait times */ for (s = telnbr; *s; s++) { /* add in pause characters time */ for (p=pmdminf->pause_chars; *p; p++) if (*s == *p) { waitct += pmdminf->pause_time; break; } } #ifndef MINIDIAL if (augmdmtyp == n_TBPEP || augmdmtyp == n_TBPNV) { waitct += 30; /* Longer connect wait for PEP call */ } #endif /* MINIDIAL */ } else { /* User-specified timeout */ waitct = dialtmo; } /* waitct is our alarm() timer. mdmwait is how long we tell the modem to wait for carrier. We set mdmwait to be 1 second less than waitct, to increase the chance that we get a response from the modem before timing out. */ if (waitct < 0) waitct = 0; mdmwait = (waitct > 5) ? waitct - 5 : waitct; for (m = 0; m < nmdm; m++) { /* Look up modem type. */ if (mdmtab[m].kwval == mymdmtyp) { break; } } if (!quiet && !backgrd) { /* Print information messages. */ printf(" Number: %s\n", telnbr); if (network) { printf(" Via modem server: %s, modem-dialer: %s\n", ttname, (m >= nmdm ? "(unknown)" : mdmtab[m].kwd) ); } else { printf(" Device: %s, modem-dialer: %s", ttname, (m >= nmdm ? "(unknown)" : mdmtab[m].kwd) ); if (speed > -1L) printf(", speed: %ld\n", speed); else printf(", speed: (unknown)\n"); } printf(" Dial timeout: %d seconds\n",waitct); printf( #ifdef MAC " Type Command-. to cancel dialing.\n" #else #ifdef UNIX " To cancel: type your interrupt character (normally Ctrl-C).\n" #else " To cancel: type Ctrl-C.\n" #endif /* UNIX */ #endif /* MAC */ ); } debug(F111,"ckdial",ttname,(int) (speed / 10L)); debug(F101,"ckdial timeout","",waitct); /* Set timer and interrupt handlers. */ if ( #ifdef CK_POSIX_SIG sigsetjmp(sjbuf,1) #else setjmp(sjbuf) #endif /* CK_POSIX_SIG */ ) { /* Handle failures */ return(dialfail(fail_code)); } else { alarm(0); /* No alarms yet. */ savalrm = signal(SIGALRM,dialtime); /* Enable alarm() handler */ #ifdef MAC savint = signal(SIGINT, dialint); /* And terminal interrupt handler. */ #else /* MAC */ #ifdef OSK if ((savint = signal(SIGINT,SIG_IGN)) != (SIGTYP (*)()) SIG_IGN) signal(SIGINT,dialint); #else if ((savint = signal(SIGINT,SIG_IGN)) != SIG_IGN ) signal(SIGINT,dialint); #endif /* OS-9 */ #endif /* MAC */ } /* Hang up the modem (in case it wasn't "on hook") */ /* But only if SET DIAL HANGUP ON... */ if (dialhup() < 0) { debug(F100,"ckdial dialhup failed","",0); #ifndef MINIDIAL if (mymdmtyp == n_TELEBIT) /* Telebit might need a BREAK */ ttsndb(); /* first. */ #endif /* MINIDIAL */ if (dialhng) { /* If it failed, */ ttclos(0); /* close and reopen the device. */ if (ttopen(ttname,&local,mymdmtyp,0) < 0) { printf("Sorry, Can't hang up communication device\n"); printf("Try 'set line %s' again\n",ttname); dialsta = DIA_HANG; #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ dreset(); return(0); } } } #ifndef MINIDIAL if (augmdmtyp == n_ROLM) /* Don't start talking to Rolm */ msleep(500); /* too soon... */ #endif /* MINIDIAL */ /* Put modem in command mode. */ if ( #ifdef MINIDIAL 1 #else augmdmtyp != n_ATTUPC #endif /* MINIDIAL */ ) { fail_code = F_MINIT; /* Default failure code */ dial_what = DW_INIT; /* What I'm Doing Now */ if (dialdpy) { /* If showing progress, */ p = ck_time(); /* display timestamp. */ if (*p) printf(" Initializing: %s...\n",p); } } switch (augmdmtyp) { /* Send wakeup string, if any. */ #ifndef MINIDIAL #ifdef ATT7300 case n_ATTUPC: { /* For ATT7300/Unix PC's with their special internal modem. Whole dialing process is handled right here, an exception to the normal structure. Timeout and user interrupts are enabled during dialing. attdial() is in file ckutio.c. - jrd */ _PROTOTYP( int attdial, (char *, long, char *) ); fail_code = F_MODEM; /* Default failure code */ dial_what = DW_DIAL; if (dialdpy) { /* If showing progress */ p = ck_time(); /* get current time; */ if (*p) printf(" Dialing: %s...\n",p); } alarm(waitct); /* do alarm properly */ if (attdial(ttname,speed,telnbr)) { /* dial internal modem */ dreset(); /* reset alarms, etc. */ printf(" Call failed.\r\n"); dialhup(); /* Hangup the call */ #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ dialsta = DIA_UERR; return(0); /* return failure */ } dreset(); /* reset alarms, etc. */ ttpkt(speed,FLO_DIAX,parity); /* cancel dialing ioctl */ if (!quiet && !backgrd) { if (dialdpy) printf("\n"); printf(" Call complete.\07\r\n"); } dialsta = DIA_OK; #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(1); /* no conversation with modem to complete dialing */ } #endif /* ATT7300 */ #endif /* MINIDIAL */ /* For Hayes modem command language, figure out if modem is giving verbose or digit result codes. */ case n_HAYES: case n_HAYESNV: #ifndef MINIDIAL case n_HST: if (augmdmtyp == n_HST) m = 255; else #endif /* MINIDIAL */ m = 60; /* Maximum value for S7 */ if (dialini) /* Get wakeup string */ ws = dialini; else ws = #ifndef MINIDIAL (augmdmtyp == n_HST) ? HST.wake_str : #endif /* MINIDIAL */ HAYES.wake_str; for (tries = 4; tries > 0; tries--) { ttslow(ws,pmdminf->wake_rate); mdmstat = getok(4,1); if (mdmstat > 0) break; if (dialdpy && tries > 1) printf(" No response from modem, retrying%s...\n", (tries < 4) ? " again" : ""); /* sleep(1); */ /* Wait before retrying */ } debug(F101,"ckdial wake_str mdmstat","",mdmstat); if (mdmstat < 1) { /* Initialized OK? */ return(dialfail(F_MINIT)); /* No, fail. */ } else { /* Yes. */ char hbuf[16]; /* Now tell the modem about our dial timeout. For Hayes 1200, the maximum is 60 seconds. For Hayes 2400 (the manual says) it is 30 seconds, but I'm not sure I believe this (I don't have one handy to see for myself). If you give the modem a value larger than its maximum, it sets the timeout to its maximum. The manual does not say what happens when the value is 0, but experimentation shows that it allows the modem to wait forever, in which case Kermit will time out at the desired time. */ /* Note by Kai Uwe Rommel: This is not acceptable for general use of the hayes modem type with other compatible modems. Most other modems allow a range of 1..255 while 0 is invalid on several modems. Let it be the user's responsibility to make sure a valid value is used. Higher values are desirable for users with rotary dialing and with high speed modems, where protocol negotiation takes quite a long time. */ #ifdef COMMENT if (mdmwait > m) /* If larger than maximum, */ mdmwait = 0; /* make it infinite. */ #else if (mdmwait > 255) /* If larger than maximum, */ mdmwait = 255; /* make it maximum. */ #endif /* COMMENT */ sprintf(hbuf,"ATS7=%d%c",mdmwait,13); /* S7 = Carrier wait time */ ttslow(hbuf,pmdminf->wake_rate); /* Set it. */ mdmstat = getok(4,1); /* Get response from modem */ debug(F101,"ckdial S7 mdmstat","",mdmstat); break; /* Errors here are not fatal. */ } #ifndef MINIDIAL /* Telebit modems fall into two basic groups: old and new. The functions and command sets are different between the two groups, and also vary by specific models within each group, and even by firmware ROM revision number. Read ckcker.bwr for details. Commands used by C-Kermit include: Old New Meaning ------- -------- ---------------------------------------- Q0 Q0 Enable result codes. X1 X1 Extended result codes. X1 X1 Extended result codes + BUSY, NO DIALTONE, etc. I I Model number inquiry. I3 I3 Additional model information inquiry. S12=50 S12=50 Escape sequence guard time (1 sec). S2=43 S2=43 Escape character is '+'. S7=xx S7=xx DIAL TIMEOUT, calculated or SET by user. S48=0 S48=0 7-bit data (Kermit's PARITY is not NONE). S48=1 S48=1 8-bit data (Kermit's PARITY is NONE). S50=0 S50=0 Automatic speed & protocol determination. S50=3 S50=3 2400/1200/300 bps. S50=6 S50=6 V.32 (9600 bps). S50=255 S50=255 PEP mode. S110=1 S190=1 S191=7 Allow compression in PEP mode. S51=? S51=? DTE interface speed (left alone by Kermit). S54=3 S61=0 S63=0 Pass BREAK signal (always). S58=2 S58=2 RTS/CTS flow control if Kermit's FLOW is RTS/CTS. S58=? S58=? S58 unchanged if Kermit's FLOW is not RTS/CTS. S68=255 S68=255 Use flow control specified by S58 (always). S95=0 S180=0 MNP disabled (SET DIAL MNP-ENABLE OFF) S95=2 S180=3 MNP, fallback to direct (also as V.42 fallback) S97=1 S180=2 Enable V.42 (LAPM) error correction S98=3 Enable compression in both directions S106=1 V.42bis compression enable For Kermit Spoof (same commands for all models that support it): S111=0 No Kermit spoofing S111=10 Kermit with no parity S111=11 Kermit with odd parity S111=12 Kermit with even parity S111=13 Kermit with mark parity S111=14 Kermit with space parity S112=?? Kermit's start-of-packet character (stchr). */ case n_TELEBIT: /* Telebits... */ case n_TBPEP: case n_TB3: case n_TB4: case n_TBS: case n_TBNV: case n_TBPNV: case n_TB3NV: case n_TB4NV: case n_TBSNV: { int S111; /* Telebit Kermit spoof register */ char tbcmdbuf[64]; /* Telebit modem command buffer */ char *mnpstr = ""; /* Pointer to MNP-enable string */ char *dprstr = ""; /* Pointer to dial protocol string */ /* If user defined a DIAL INIT-STRING, send that now, otherwise send built-in Telebit string. Try up to 4 times to get OK or 0 response from modem. NOTE: The default init string *must* be independent of Telebit model. */ ws = dialini ? dialini : TELEBIT.wake_str; debug(F110,"ckdial telebit init string",ws,0); for (tries = 4; tries > 0; tries--) { ttsndb(); /* Begin by sending BREAK */ ttslow(ws,pmdminf->wake_rate); /* Send wakeup string */ mdmstat = getok(5,0); /* Get modem's response */ if (mdmstat) break; /* If response OK, done */ if (dialdpy && tries > 1) printf(" No response from modem, retrying%s...\n", (tries < 4) ? " again" : ""); msleep(300); /* Otherwise, sleep 1/3 second */ dialhup(); /* Hang up */ ttflui(); /* Flush input buffer and try again */ } if (mdmstat < 1) /* If we didn't get a response, */ return(dialfail(F_MINIT)); /* fail. */ if (!dialini) { /* If using built-in init strings... */ /* Try to get the model number. It should be in the getok() response buffer, rbuf[], because the Telebit init string asks for it with the "I" command. If the model number is 965, we have to make another query to narrow it down. */ if (didweget(rbuf,"962") || /* Check model number */ didweget(rbuf,"961") || didweget(rbuf,"963")) { tbmodel = TB_BLAZ; /* Trailblazer */ } else if (didweget(rbuf,"972")) { tbmodel = TB_2500; /* T2500 */ } else if (didweget(rbuf,"968")) { tbmodel = TB_1000; /* T1000 */ } else if (didweget(rbuf,"966") || didweget(rbuf,"967") || didweget(rbuf,"964")) { tbmodel = TB_PLUS; /* Trailblazer-Plus */ } else if (didweget(rbuf,"969")) { tbmodel = TB_QBLA; /* Qblazer */ } else if (didweget(rbuf,"970")) { tbmodel = TB_QBLA; /* Qblazer Plus */ } else if (didweget(rbuf,"965")) { /* Most new models */ tbati3(965); /* Go find out */ } else if (didweget(rbuf,"971")) { /* T1500 or T2500 */ tbati3(971); /* Go find out */ } else if (didweget(rbuf,"123") || didweget(rbuf,"960")) { tbmodel = TB_UNK; /* Telebit in Hayes mode */ } debug(F111,"Telebit model",tb_name[tbmodel],tbmodel); if (dialdpy) printf("Telebit model: %s\n",tb_name[tbmodel]); ttflui(); /* Dial timeout. S7 is set to the DIAL TIMEOUT value, or else to 255 if the dial timeout is greater than 255, which is the maximum value for Telebits. S7 can't be set to 0 on Telebits -- it gives an error. */ if (mdmwait > 255) /* If dial timeout too big */ mdmwait = 255; /* make it as big as possible. */ /* Flow control. If C-Kermit's FLOW-CONTROL is RTS/CTS, then we set this on the modem too. Unfortunately, many versions of UNIX only allow RTS/CTS to be set outside of Kermit (e.g. by selecting a special device name). In that case, Kermit doesn't know that it should set RTS/CTS on the modem, in which case the user can add the appropriate Telebit command with SET DIAL DIAL-COMMAND. */ if (flow == FLO_RTSC) { /* RTS/CTS active in Kermit */ sprintf(tbcmdbuf, "ATS7=%d S48=%d S50=0 S58=2 S68=255\015", mdmwait, parity ? 0 : 1); } else sprintf(tbcmdbuf, /* Otherwise, don't touch modem's */ "ATS7=%d S48=%d S50=0 S68=255\015", /* flow control */ mdmwait, parity ? 0 : 1); /* setting. */ s = tbcmdbuf; debug(F110,"ckdial Telebit init step 2",s,0); for (tries = 4; tries > 0; tries--) { ttslow(s,pmdminf->wake_rate); mdmstat = getok(5,1); if (mdmstat) break; if (dialdpy && tries > 1) printf(" No response from modem, retrying%s...\n", (tries < 4) ? " again" : ""); msleep(500); ttflui(); } if (mdmstat < 1) return(dialfail(F_MINIT)); /* Model-dependent items, but constant per model. */ switch(tbmodel) { case TB_BLAZ: case TB_PLUS: /* TrailBlazer-Plus */ case TB_1000: /* T1000 */ case TB_2000: /* T2000 */ case TB_2500: /* T2500 */ #ifdef COMMENT /* Code from edit 183 told modem to follow RS-232 wrt CD and DTR */ /* DTR, CD, follow RS-232, pass BREAK */ sprintf(tbcmdbuf,"ATS52=1 S53=4 S54=3\015"); #else /* But everybody agreed we should not touch modem's CD and DTR settings. */ /* Just pass BREAK */ sprintf(tbcmdbuf,"ATS54=3\015"); #endif /* COMMENT */ break; case TB_1600: /* T1600 */ case TB_3000: /* T3000 */ case TB_WBLA: /* WorldBlazer */ case TB_QBLA: /* Qblazer */ #ifdef COMMENT /* Code from edit 183 */ /* Follow RS-232, No CONNECT suffix, pass BREAK */ sprintf(tbcmdbuf,"AT&C1&D2&Q0 S59=0 S61=0 S63=0\015"); #else /* Everybody agrees we should not touch modem's CD and DTR settings. */ /* Also no more &Q0, no more S59=0 (doesn't matter, so don't touch). */ /* So this section now deals only with treatment of BREAK. */ /* Here we also raise the result code from X1 to X2, which allows */ /* the T1600, T3000, and WB to supply NO DIALTONE, BUSY, RRING, and DIALING. */ /* X2 means something else on the other models. */ /* Transmit BREAK in sequence, raise result code. */ sprintf(tbcmdbuf,"ATX2 S61=0 S63=0\015"); #endif /* COMMENT */ break; default: /* Others, do nothing */ tbcmdbuf[0] = NUL; break; } s = tbcmdbuf; if (*s) { debug(F110,"ckdial Telebit init step 3",s,0); for (tries = 4; tries > 0; tries--) { ttslow(s,pmdminf->wake_rate); mdmstat = getok(5,1); if (mdmstat) break; if (dialdpy && tries > 1) printf(" No response from modem, retrying%s...\n", (tries < 4) ? " again" : ""); msleep(500); ttflui(); } if (mdmstat < 1) dialfail(F_MINIT); } else debug(F100,"ckdial Telebit init step 3 skipped","",0); /* Error correction, MNP or V.42 */ if (augmdmtyp & DIAL_V42) { /* User wants V.42 */ switch (tbmodel) { /* which implies fallback to MNP. */ case TB_PLUS: /* BC7.00 and up firmware */ case TB_2000: /* now really the same as TB+ ? */ case TB_2500: /* LAPM+compress->MNP->direct */ mnpstr = "S50=0 S95=2 S97=1 S98=3 S106=1"; break; case TB_1600: case TB_3000: case TB_WBLA: case TB_QBLA: #ifdef COMMENT mnpstr = "S180=2 S181=0"; /* V.42, fallback = lock speed */ #else /* Better not to mess with S181, let it be used however user has it set. */ /* S180=2 allows fallback to MNP, S180=1 disallows fallback to MNP. */ mnpstr = "S180=2"; /* V.42 */ #endif /* COMMENT */ break; default: if (dialdpy) printf("V.42 not supported by this Telebit model\n"); } } else { /* Handle DIAL MNP-ENABLE setting */ switch (tbmodel) { case TB_BLAZ: /* TrailBlazer */ case TB_PLUS: /* TrailBlazer-Plus */ case TB_1000: /* T1000 */ case TB_2000: /* T2000 */ case TB_2500: /* T2500 */ mnpstr = dialmnp ? "S95=2" : "S95=0"; /* ON, OFF */ break; case TB_1600: /* T1600 */ case TB_3000: /* T3000 */ case TB_WBLA: /* WorldBlazer */ case TB_QBLA: /* Qblazer */ mnpstr = dialmnp ? "S180=3" : "S180=0"; /* ON, OFF */ /* (Leave S181 fallback method alone) */ break; default: mnpstr = ""; } } /* Dialing protocol */ dprstr = ""; /* Initialize dialing protocol string */ p = ""; /* and message string */ switch (augmdmtyp) { case n_TELEBIT: /* Start at highest and work down */ case n_TBNV: p = "standard"; switch (tbmodel) { /* First group starts with PEP */ case TB_BLAZ: /* TrailBlazer */ case TB_PLUS: /* TrailBlazer-Plus */ case TB_1000: /* T1000 */ case TB_2000: /* T2000 */ case TB_2500: /* T2500 */ dprstr = "S50=0 S110=1"; /* PEP, compression allowed. */ break; case TB_WBLA: /* WorldBlazer has PEP */ dprstr = "S50=0 S190=1 S191=7"; /* PEP, */ break; /* compression allowed. */ case TB_1600: /* T1600 doesn't have PEP */ case TB_3000: /* T3000 doesn't */ case TB_QBLA: /* Qblazer doesn't*/ default: dprstr = "S50=0"; /* No PEP available */ break; } break; case n_TBS: /* Telebit up to 2400 Baud */ case n_TBSNV: /* i.e. "Slow mode". */ p = "300/1200/2400 Baud"; /* Leave S90 alone assuming it is */ dprstr = "S50=3"; /* already set for V.22 vs 212A */ break; case n_TB3: /* Telebit V.32 */ case n_TB3NV: if (tbmodel == TB_3000 || tbmodel == TB_1600 || tbmodel == TB_2500 || tbmodel == TB_WBLA) { p = "V.32"; /* Note: we don't touch S51 (interface speed) here. */ /* We're already talking to the modem, and the modem */ /* SHOULD be able to make a V.32 call no matter what */ /* its interface speed is. (In practice, however, */ /* that is not always true.) */ dprstr = "S50=6"; } else if (dialdpy) printf("V.32 not supported by this Telebit model.\n"); break; case n_TBPEP: /* Force PEP Protocol */ case n_TBPNV: if (tbmodel != TB_1600 && /* Models that don't support PEP */ tbmodel != TB_3000 && tbmodel != TB_QBLA) { p = "PEP"; if (tbmodel == TB_WBLA) /* WorldBlazer */ dprstr = "S50=255 S190=1 S191=7"; else if (tbmodel != TB_1000) dprstr = "S50=255 S110=1"; /* TrailBlazer, etc. */ else dprstr = "S50=255"; /* T1000, no compression */ } else if (dialdpy) printf("PEP not supported by this Telebit model.\n"); break; } /* Kermit Spoof */ if (dialksp) { p = "Kermit Spoof"; switch (parity) { /* S111 value depends on parity */ case 'e': S111 = 12; break; case 'm': S111 = 13; break; case 'o': S111 = 11; break; case 's': S111 = 14; break; case 0: default: S111 = 10; break; } if (tbmodel != TB_QBLA) sprintf(tbcmdbuf,"AT%s %s S111=%d S112=%d\015", mnpstr,dprstr,S111,stchr); else { /* Qblazer has no Kermit spoof */ sprintf(tbcmdbuf,"AT%s %s\015", mnpstr,dprstr); p = "No Kermit Spoof"; if (dialdpy) printf("Kermit Spoof not supported by Qblazer\n"); } } else { /* KERMIT-SPOOF OFF */ p = "No Kermit Spoof"; sprintf(tbcmdbuf,"AT%s %s %s\015", mnpstr, dprstr, (tbmodel == TB_QBLA) ? "" : "S111=0"); } s = tbcmdbuf; debug(F111,"ckdial Telebit config",p,speed); debug(F110,"ckdial Telebit init step 4",s,0); if (*s) { for (tries = 4; tries > 0; tries--) { ttslow(s,pmdminf->wake_rate); mdmstat = getok(5,1); if (mdmstat) break; if (dialdpy && tries > 1) printf(" No response from modem, retrying%s...\n", (tries < 4) ? " again" : ""); msleep(500); ttflui(); } debug(F101,"ckdial telebit init mdmstat","",mdmstat); if (mdmstat < 1) dialfail(F_MINIT); } } /* Done with Telebit protocols, remove bits from modem type */ /* Except nonverbal bit */ augmdmtyp &= ~(DIAL_PEP|DIAL_V32|DIAL_V42|DIAL_SLO); debug(F101,"ckdial Telebit augmdmtyp","",augmdmtyp); break; } case n_MICROCOM: /* Interdigit waits for tone dial */ #ifdef COMMENT { jmp_buf savejmp; alarm(0); savalrm = signal(SIGALRM,dialtime); /* Set alarm handler. */ xcpy((char *)savejmp, (char *)sjbuf, sizeof savejmp); if (setjmp(sjbuf)) { /* try the autobaud sequence */ xcpy((char *)sjbuf, (char *)savejmp, sizeof savejmp); alarm(5); ttslow("44445", MICROCOM.wake_rate); waitfor(MICROCOM.wake_str); } else { alarm(2); ws = dialini ? dialini : MICROCOM.wake_str; ttslow(ws, MICROCOM.wake_rate); waitfor(ws); /* Kermit spoof for Microcom modems. Untested. This is reportedly the way to do it for QX/3296c, QX/4232hs, QX/4232bis, and QX/9624c. The effect on other models is unknown. */ ws = dialksp ? "APM1\015" : "APM0\015"; /* Kermit spoof */ ttslow(ws, MICROCOM.wake_rate); waitfor(ws); alarm(0); signal(SIGALRM,savalrm); /* Set alarm handler. */ xcpy((char *)sjbuf, (char *)savejmp, sizeof savejmp); } } #else ws = dialini ? dialini : MICROCOM.wake_str; debug(F110,"ckdial microcom ttslow",ws,0); ttslow(ws, MICROCOM.wake_rate); alarm(3); debug(F110,"ckdial microcom waitfor",MICROCOM.wake_prompt,0); waitfor(MICROCOM.wake_prompt); debug(F110,"ckdial microcom waitfor done","",0); alarm(0); /* Kermit spoof for Microcom modems. Untested. This is reportedly the way to do it for QX/3296c, QX/4232hs, QX/4232bis, and QX/9624c. The effect on other models is unknown. */ ws = dialksp ? "APM1\015" : "APM0\015"; /* Kermit spoof */ ttslow(ws, MICROCOM.wake_rate); alarm(3); waitfor(ws); alarm(0); #endif /* COMMENT */ break; case n_ATTDTDM: /* DTDM requires BREAK to wake up */ ttsndb(); /* Send BREAK */ break; /* ttsndb() defined in ckutio.c */ #endif /* MINIDIAL */ default: /* Place modem into command mode */ ws = dialini ? dialini : pmdminf->wake_str; if (ws && (int)strlen(ws) > 0) { debug(F111,"ckdial default, wake string", ws, pmdminf->wake_rate); ttslow(ws, pmdminf->wake_rate); } else debug(F100,"ckdial no wake_str","",0); if (pmdminf->wake_prompt && (int)strlen(pmdminf->wake_prompt) > 0) { debug(F110,"ckdial default, waiting for wake_prompt", pmdminf->wake_prompt,0); alarm(10); waitfor(pmdminf->wake_prompt); } else debug(F100,"ckdial no wake_prompt","",0); break; } alarm(0); /* Turn off alarm */ debug(F100,"ckdial got wake prompt","",0); msleep(500); /* Allow settling time */ #ifndef MINIDIAL /* Enable/disable MNP (Telebit already done above) */ switch (augmdmtyp) { case n_HST: if (dialmnp) /* EC & compression enabled */ ttslow("AT&M4&K1\015",pmdminf->wake_rate); else /* EC & compression disabled */ ttslow("AT&M0&K0\015",pmdminf->wake_rate); getok(5,1); /* Get response */ break; /* Add others here ... */ default: break; } #endif /* MINIDIAL */ /* Put modem into dialing mode, if the modem requires it. */ if (pmdminf->dmode_str && *(pmdminf->dmode_str)) { ttslow(pmdminf->dmode_str, pmdminf->dial_rate); savalrm = signal(SIGALRM,dialtime); alarm(10); /* Wait for prompt, if any expected */ if (pmdminf->dmode_prompt && *(pmdminf->dmode_prompt)) { waitfor(pmdminf->dmode_prompt); msleep(300); } alarm(0); /* Turn off alarm on dialing prompts */ signal(SIGALRM,savalrm); /* Restore alarm */ ttflui(); /* Clear out stuff from waking modem up */ } /* Dial the number. First form the dialing string. */ #ifdef DYNAMIC if (!lbuf) { /* If, for some reason, this is NULL */ if (!(lbuf = malloc(LBUFL+1))) { /* allocate it... */ dialsta = DIA_IE; return(0); } } #endif /* DYNAMIC */ sprintf(lbuf, dialcmd ? dialcmd : pmdminf->dial_str, telnbr); if ((int)strlen(lbuf) > LBUFL) { printf("DIAL command + phone number too long!\n"); dreset(); #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ return(1); /* No conversation with modem to complete dialing */ } debug(F110,"dialing",lbuf,0); ttslow(lbuf,pmdminf->dial_rate); /* Send the dialing string */ fail_code = F_MODEM; /* New default failure code changes */ dial_what = DW_DIAL; /* and our state, too. */ if (dialdpy) { /* If showing progress */ p = ck_time(); /* get current time; */ if (*p) printf(" Dialing: %s...\n",p); } x = alarm(waitct); /* This much time allowed. */ debug(F101,"ckdial old alarm","",x); debug(F101,"ckdial waitct","",waitct); #ifndef MINIDIAL switch (augmdmtyp) { case n_RACAL: /* Acknowledge dialing string */ sleep(3); ttflui(); ttoc('\015'); break; case n_VENTEL: waitfor("\012\012"); /* Ignore the first two strings */ break; default: break; } #endif /* MINIDIAL */ /* Check for connection */ mdmstat = 0; /* No status yet */ strcpy(lbuf,"No Connection"); /* Default reason for failure */ debug(F101,"dial awaiting response, augmdmtyp","",augmdmtyp); while (mdmstat == 0) { switch (augmdmtyp) { default: for (n = -1; n < LBUFL-1; ) { /* Accumulate response */ c2 = ddinc(0); /* Read a character, blocking */ if (c2 == 0 || c2 == -1) /* Ignore NULs and errors */ continue; /* (Timeout will handle errors) */ else /* Real character, keep it */ lbuf[++n] = c2 & 0177; dialoc(lbuf[n]); /* Maybe echo it */ if (augmdmtyp == n_CCITT) { /* V.25 bis dialing... */ /* This assumes that V.25bis indications are all at least 3 characters long and are terminated by either CRLF or LFCR. */ if (n < 3) continue; if ((lbuf[n] == CR) && (lbuf[n-1] == LF)) break; if ((lbuf[n] == LF) && (lbuf[n-1] == CR)) break; } #ifndef MINIDIAL else if (augmdmtyp == n_DIGITEL) { if (((lbuf[n] == CR) && (lbuf[n-1] == LF)) || ((lbuf[n] == LF) && (lbuf[n-1] == CR))) break; else continue; } #endif /* MINIDIAL */ else { /* All others, break on CR or LF */ if ( lbuf[n] == CR || lbuf[n] == LF ) break; } } lbuf[++n] = '\0'; /* Terminate response from modem */ debug(F111,"ckdial modem response",lbuf,n); if (n) { /* If one or more characters present */ switch (augmdmtyp) { /* check for modem response message. */ #ifndef MINIDIAL case n_ATTMODEM: /* Careful - "Connected" / "Not Connected" */ if (didweget(lbuf,"Busy")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"Not connected") || didweget(lbuf,"Not Connected")) { mdmstat = FAILED; dialsta = DIA_NOCA; } else if (didweget(lbuf,"No dial tone") || didweget(lbuf,"No Dial Tone")) { mdmstat = FAILED; dialsta = DIA_NODT; } else if (didweget(lbuf,"No answer") || didweget(lbuf,"No Answer")) { mdmstat = FAILED; dialsta = DIA_NOAN; } else if (didweget(lbuf,"Answered") || didweget(lbuf,"Connected")) { mdmstat = CONNECTED; dialsta = DIA_OK; } break; case n_ATTISN: if (didweget(lbuf,"ANSWERED")) { mdmstat = CONNECTED; dialsta = DIA_OK; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"DISCONNECT")) { mdmstat = FAILED; dialsta = DIA_DISC; } else if (didweget(lbuf,"NO ANSWER")) { mdmstat = FAILED; dialsta = DIA_NOAN; } else if (didweget(lbuf,"WRONG ADDRESS")) { mdmstat = FAILED; dialsta = DIA_NOAC; } break; case n_ATTDTDM: if (didweget(lbuf,"ANSWERED")) { mdmstat = CONNECTED; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"CHECK OPTIONS")) { mdmstat = FAILED; dialsta = DIA_ERR; } else if (didweget(lbuf,"DISCONNECTED")) { mdmstat = FAILED; dialsta = DIA_DISC; } else if (didweget(lbuf,"DENIED")) { mdmstat = FAILED; dialsta = DIA_NOAC; } #ifdef DEBUG #ifdef ATT6300 else if (deblog && didweget(lbuf,"~~")) mdmstat = CONNECTED; #endif /* ATT6300 */ #endif /* DEBUG */ break; #endif /* MINIDIAL */ case n_CCITT: /* CCITT V.25bis */ #ifndef MINIDIAL case n_DIGITEL: /* or Digitel variant */ #endif /* MINIDIAL */ if (didweget(lbuf,"VAL")) { /* Dial command confirmation */ #ifndef MINIDIAL if (augmdmtyp == n_CCITT) #endif /* MINIDIAL */ continue; /* Go back and read more */ #ifndef MINIDIAL /* Digitel doesn't give an explicit connect confirmation message */ else { int n; for (n = -1; n < LBUFL-1; ) { lbuf[++n] = c2 = ddinc(0) & 0177; dialoc(lbuf[n]); if (((lbuf[n] == CR) && (lbuf[n-1] == LF)) || ((lbuf[n] == LF) && (lbuf[n-1] == CR))) break; } mdmstat = CONNECTED; /* Assume we're connected */ if (dialdpy && carrier != CAR_OFF) { sleep(1); /* Wait a second */ n = ttgmdm(); /* Try to read modem signals */ if ((n > -1) && ((n & BM_DCD) == 0)) printf("Warning: No Carrier\n"); } } #endif /* MINIDIAL */ } else if (didweget(lbuf,"CNX")) { /* Connected */ mdmstat = CONNECTED; } else if (didweget(lbuf, "INV")) { mdmstat = FAILED; /* Command error */ dialsta = DIA_ERR; strcpy(lbuf,"INV"); } else if (didweget(lbuf,"CFI")) { /* Call Failure */ #ifdef COMMENT /* V.25 bis says that the failure reason comes on the same line, so we don't need to read any more characters here. */ for (n = 0; n < LBUFL-1; n++) { /* Read reason */ lbuf[n] = c2 = (ddinc(0) & 0177); if (c2 == LF) /* Modem answers LF CR */ continue; dialoc(lbuf[n]); if (lbuf[n] == CR || lbuf[n] == LF) break; } #endif /* COMMENT */ if (didweget(lbuf,"AB")) { /* Interpret reason code */ strcpy(lbuf,"AB: Timed out"); dialsta = DIA_TIMO; } else if (didweget(lbuf,"CB")) { strcpy(lbuf,"CB: Local DCE Busy"); dialsta = DIA_NRDY; } else if (didweget(lbuf,"ET")) { strcpy(lbuf,"ET: Busy"); dialsta = DIA_BUSY; } else if (didweget(lbuf, "NS")) { strcpy(lbuf,"NS: Number not stored"); dialsta = DIA_ERR; } else if (didweget(lbuf,"NT")) { strcpy(lbuf,"NT: No answer"); dialsta = DIA_NOAN; } else if (didweget(lbuf,"RT")) { strcpy(lbuf,"RT: Ring tone"); dialsta = DIA_RING; } else if (didweget(lbuf,"PV")) { strcpy(lbuf,"PV: Parameter value error"); dialsta = DIA_ERR; } else if (didweget(lbuf,"PS")) { strcpy(lbuf,"PS: Parameter syntax error"); dialsta = DIA_ERR; } else if (didweget(lbuf,"MS")) { strcpy(lbuf,"MS: Message syntax error"); dialsta = DIA_ERR; } else if (didweget(lbuf,"CU")) { strcpy(lbuf,"CU: Command unknown"); dialsta = DIA_ERR; } else if (didweget(lbuf,"FC")) { strcpy(lbuf,"FC: Forbidden call"); dialsta = DIA_NOAC; } mdmstat = FAILED; } else if (didweget(lbuf,"INC")) { /* Incoming Call */ strcpy(lbuf,"INC: Incoming call"); dialsta = DIA_RING; mdmstat = FAILED; } else if (didweget(lbuf,"DLC")) { /* Delayed Call */ strcpy(lbuf,"DLC: Delayed call"); dialsta = DIA_NOAN; mdmstat = FAILED; } else /* Response was probably an echo. */ #ifndef MINIDIAL if (augmdmtyp == n_CCITT) #endif /* MINIDIAL */ continue; #ifndef MINIDIAL else /* Digitel: If no error, then connect. */ mdmstat = CONNECTED; #endif /* MINIDIAL */ break; #ifndef MINIDIAL case n_CERMETEK: if (didweget(lbuf,"\016A")) { mdmstat = CONNECTED; ttslow("\016U 1\015",200); /* Make transparent*/ } break; case n_DF100: /* DF100 has short response codes */ if (strcmp(lbuf,"A") == 0) { mdmstat = CONNECTED; /* Attached */ dialsta = DIA_OK; } else if (strcmp(lbuf,"N") == 0) { mdmstat = FAILED; dialsta = DIA_NOAN; /* No answer or no dialtone */ } else if (strcmp(lbuf,"E") == 0 || /* Error */ strcmp(lbuf,"R") == 0) { /* "Ready" (?) */ mdmstat = FAILED; dialsta = DIA_ERR; /* Command error */ } /* otherwise fall thru... */ case n_DF200: if (didweget(lbuf,"Attached")) { mdmstat = CONNECTED; dialsta = DIA_OK; /* * The DF100 will respond with "Attached" even if DTR * and/or carrier are not present. Another reason to * (also) wait for carrier? */ } else if (didweget(lbuf,"Busy")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"Disconnected")) { mdmstat = FAILED; dialsta = DIA_DISC; } else if (didweget(lbuf,"Error")) { mdmstat = FAILED; dialsta = DIA_ERR; } else if (didweget(lbuf,"No answer")) { mdmstat = FAILED; dialsta = DIA_NOAN; } else if (didweget(lbuf,"No dial tone")) { mdmstat = FAILED; dialsta = DIA_NODT; } else if (didweget(lbuf,"Speed:)")) { mdmstat = FAILED; dialsta = DIA_ERR; } /* * It appears that the "Speed:..." response comes after an * "Attached" response, so this is never seen. HOWEVER, * it would be very handy to detect this and temporarily * reset the speed, since it's a nuisance otherwise. * If we wait for some more input from the modem, how do * we know if it's from the remote host or the modem? * Carrier reportedly doesn't get set until after the * "Speed:..." response (if any) is sent. Another reason * to (also) wait for carrier. */ break; case n_GDC: if (didweget(lbuf,"ON LINE")) mdmstat = CONNECTED; else if (didweget(lbuf,"NO CONNECT")) mdmstat = FAILED; break; case n_USROBOT: case n_HST: case n_TELEBIT: #endif /* MINIDIAL */ case n_HAYES: if (mdmspd && !network) { s = lbuf; while (*s != '\0' && *s != 'C') s++; cptr = (*s == 'C') ? s : NULL; conspd = 0L; if ((cptr != NULL) && !strncmp(cptr,"CONNECT ",8)) { if ((int)strlen(cptr) < 9) /* Just CONNECT, */ conspd = 300L; /* use 300 bps */ else if (isdigit(*(cptr+8))) /* not CONNECT FAST */ conspd = atol(cptr + 8); /* CONNECT nnnn */ if (conspd != speed) { if ((conspd / 10L) > 0) { if (ttsspd((int) (conspd / 10L)) < 0) { printf(" Can't change speed to %ld\r\n", conspd); } else { speed = conspd; mdmstat = CONNECTED; if ( !quiet && !backgrd ) printf(" Speed changed to %ld\r\n", conspd); } } } /* Expanded to handle any conceivable speed */ } } #ifndef MINIDIAL if (mymdmtyp == n_TELEBIT) { if (didweget(lbuf,"CONNECT FAST/KERM")) { mdmstat = CONNECTED; if (!quiet && !backgrd) printf("FAST/KERM "); break; } } #endif /* MINIDIAL */ if (didweget(lbuf,"RRING") || didweget(lbuf,"RINGING") || didweget(lbuf,"DIALING")) { mdmstat = 0; } else if (didweget(lbuf,"CONNECT")) { mdmstat = CONNECTED; } else if (didweget(lbuf,"NO CARRIER")) { mdmstat = FAILED; dialsta = DIA_NOCA; } else if (didweget(lbuf,"NO DIALTONE")) { mdmstat = FAILED; dialsta = DIA_NODT; } else if (didweget(lbuf,"NO DIAL TONE")) { mdmstat = FAILED; dialsta = DIA_NODT; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"NO ANSWER")) { mdmstat = FAILED; dialsta = DIA_NOAN; } else if (didweget(lbuf,"VOICE")) { mdmstat = FAILED; dialsta = DIA_VOIC; } else if (didweget(lbuf,"RING")) { mdmstat = FAILED; dialsta = DIA_RING; } else if (didweget(lbuf,"ERROR")) { mdmstat = FAILED; dialsta = DIA_ERR; } break; #ifndef MINIDIAL case n_PENRIL: if (didweget(lbuf,"OK")) { mdmstat = CONNECTED; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"NO RING")) { mdmstat = FAILED; dialsta = DIA_NOCA; } break; case n_RACAL: if (didweget(lbuf,"ON LINE")) mdmstat = CONNECTED; else if (didweget(lbuf,"FAILED CALL")) mdmstat = FAILED; break; case n_ROLM: if (didweget(lbuf,"CALLING")) mdmstat = 0; else if (didweget(lbuf,"COMPLETE")) mdmstat = CONNECTED; else if (didweget(lbuf,"FAILED") || didweget(lbuf,"ABANDONDED")) { mdmstat = FAILED; dialsta = DIA_NOCA; } else if (didweget(lbuf,"NOT AVAILABLE") || didweget(lbuf,"LACKS PERMISSION") || didweget(lbuf,"NOT A DATALINE") || didweget(lbuf,"INVALID DATA LINE NUMBER") || didweget(lbuf,"INVALID GROUP NAME")) { mdmstat = FAILED; dialsta = DIA_NOAC; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"DOES NOT ANSWER")) { mdmstat = FAILED; dialsta = DIA_NOAN; } break; case n_VENTEL: if (didweget(lbuf,"ONLINE!") || didweget(lbuf,"Online!")) { mdmstat = CONNECTED; } else if (didweget(lbuf,"BUSY") || didweget(lbuf,"Busy")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"DEAD PHONE")) { mdmstat = FAILED; dialsta = DIA_DISC; } break; case n_CONCORD: if (didweget(lbuf,"INITIATING")) mdmstat = CONNECTED; else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"CALL FAILED")) { mdmstat = FAILED; dialsta = DIA_NOCA; } break; case n_MICROCOM: /* "RINGBACK" means phone line ringing, continue */ if (didweget(lbuf,"NO CONNECT")) { mdmstat = FAILED; dialsta = DIA_NOCA; } else if (didweget(lbuf,"BUSY")) { mdmstat = FAILED; dialsta = DIA_BUSY; } else if (didweget(lbuf,"NO DIALTONE")) { mdmstat = FAILED; dialsta = DIA_NODT; } else if (didweget(lbuf,"COMMAND ERROR")) { mdmstat = FAILED; dialsta = DIA_ERR; } else if (didweget(lbuf,"IN USE")) { mdmstat = FAILED; dialsta = DIA_NOAC; } else if (didweget(lbuf,"CONNECT")) { mdmstat = CONNECTED; /* trailing speed ignored */ } break; #endif /* MINIDIAL */ } } break; #ifndef MINIDIAL case n_DF03: /* because response lacks CR or NL */ c = ddinc(0) & 0177; dialoc(c); debug(F000,"dial df03 got","",c); if ( c == 'A' ) mdmstat = CONNECTED; if ( c == 'B' ) mdmstat = FAILED; break; case n_TBNV: /* Hayeslike modems in digit */ case n_TB3NV /* response mode... */: case n_TBPNV: case n_TB4NV: case n_TBSNV: #endif /* MINIDIAL */ case n_HAYESNV: /* The method for reading Hayes numeric result codes has been totally redone as of 5A(174) to account for all of the following. Not all have been tested, and others probably need to be added. Hayes numeric result codes (Hayes 1200 and higher): 0 = OK 1 = CONNECT at 300 bps (or 1200 bps on Hayes 1200 with basic code set) 2 = RING 3 = NO CARRIER 4 = ERROR (in command line) 5 = CONNECT 1200 (extended code set) Hayes 2400 and higher: 6 = NO DIALTONE 7 = BUSY 8 = NO ANSWER 9 = (there is no 9) 10 = CONNECT 2400 Reportedly, the codes for Hayes V.32 modems are: 1x = CONNECT 5x = CONNECT 1200 9x = CONNECT 2400 11x = CONNECT 4800 12x = CONNECT 9600 Where: x: suffix: R = RELIABLE RC = RELIABLE COMPRESSED L = LAPM LC = LAPM COMPRESSED And for Telebits, all the above, except no suffix in numeric mode, plus: 11 = CONNECT 4800 12 = CONNECT 9600 13 = CONNECT 14400 14 = CONNECT 19200 15 = CONNECT 38400 16 = CONNECT 57600 20 = CONNECT 300/REL (= MNP) 22 = CONNECT 1200/REL (= MNP) 23 = CONNECT 2400/REL (= MNP) 46 = CONNECT 7512 (i.e. 75/1200) 47 = CONNECT 1275 (i.e. 1200/75) 48 = CONNECT 7200 49 = CONNECT 12000 50 = CONNECT FAST (not on T1600/3000) 52 = RRING 53 = DIALING 54 = NO PROMPTTONE 61 = CONNECT FAST/KERM (Kermit spoof) 70 = CONNECT FAST/COMP (PEP + compression) 71 = CONNECT FAST/KERM/COMP (PEP + compression + Kermit spoof) */ #define NBUFL 8 { /* Nonverbal response code handler */ char nbuf[NBUFL+1]; /* Response buffer */ int i, j; /* Buffer pointers */ debug(F101,"RESPONSE mdmecho","",mdmecho); if (mdmecho) { /* Sponge up dialing string echo. */ while (1) { c = ddinc(0) & 0x7f; debug(F000,"SPONGE","",c); dialoc(c); if (c == CR) break; } } while (mdmstat == 0) { /* Read response */ for (i = 0; i < NBUFL; i++) /* Clear the buffer */ nbuf[i] = '\0'; i = 0; /* Reset the buffer pointer. */ c = ddinc(0) & 0177; /* Get first digit of response. */ /* using an untimed, blocking read. */ debug(F000,"RESPONSE-A","",c); dialoc(c); /* Echo it if requested. */ if (!isdigit(c)) /* If not a digit, keep looking. */ continue; nbuf[i++] = c; /* Got first digit, save it. */ while (c != CR && i < 8) { /* Now read characters up to CR */ x = ddinc(0) & 0177; /* Get a character. */ c = (char) x; /* Got it OK. */ debug(F000,"RESPONSE-C","",c); if (c != CR) /* If it's not a carriage return, */ nbuf[i++] = c; /* save it. */ dialoc(c); /* Echo it. */ } nbuf[i] = '\0'; /* Done, terminate the buffer. */ debug(F111,"dial hayesnv lbuf",lbuf,n); debug(F111,"dial hayesnv got",nbuf,i); /* Separate any non-numeric suffix from the numeric result code with a null. */ for (j = i-1; (j > -1) && !isdigit(nbuf[j]); j--) nbuf[j+1] = nbuf[j]; j++; nbuf[j++] = '\0'; debug(F110,"dial hayesnv numeric",nbuf,0); debug(F111,"dial hayesnv suffix ",nbuf+j,j); if ((int)strlen(nbuf) > 3) /* Probably phone number echoing. */ continue; /* Now read and interpret the results... */ i = atoi(nbuf); /* Convert to integer */ switch (i) { case 1: /* CONNECT */ mdmstat = CONNECTED; /* Could be any speed */ break; case 2: /* RING */ if (dialdpy) printf("\r\n Local phone is ringing!\r\n"); mdmstat = FAILED; dialsta = DIA_RING; break; case 3: /* NO CARRIER */ if (dialdpy) printf("\r\n No Carrier.\r\n"); mdmstat = FAILED; dialsta = DIA_NOCA; break; case 4: /* ERROR */ if (dialdpy) printf("\r\n Modem Command Error.\r\n"); mdmstat = FAILED; dialsta = DIA_ERR; break; case 5: /* CONNECT 1200 */ spdchg(1200L); /* Change speed if necessary. */ mdmstat = CONNECTED; break; case 6: /* NO DIALTONE */ if (dialdpy) printf("\r\n No Dialtone.\r\n"); mdmstat = FAILED; dialsta = DIA_NODT; break; case 7: /* BUSY */ if (dialdpy) printf("\r\n Busy.\r\n"); mdmstat = FAILED; dialsta = DIA_BUSY; break; case 8: /* NO ANSWER */ if (dialdpy) printf("\r\n No Answer.\r\n"); mdmstat = FAILED; dialsta = DIA_NOAN; break; case 9: /* CONNECT 2400 */ case 10: spdchg(2400L); /* Change speed if necessary. */ mdmstat = CONNECTED; break; case 11: /* CONNECT 4800 */ spdchg(4800L); mdmstat = CONNECTED; break; case 12: /* CONNECT 9600 */ spdchg(9600L); mdmstat = CONNECTED; break; case 13: /* CONNECT 14400 */ spdchg(14400L); mdmstat = CONNECTED; break; case 14: spdchg(19200L); /* CONNECT 19200 */ mdmstat = CONNECTED; break; case 15: /* CONNECT 34800 */ spdchg(38400L); mdmstat = CONNECTED; break; case 16: /* CONNECT 57600 */ spdchg(57600L); mdmstat = CONNECTED; break; case 20: /* CONNECT 300/REL */ spdchg(300L); mdmstat = CONNECTED; break; case 22: /* CONNECT 1200/REL */ spdchg(1200L); mdmstat = CONNECTED; break; case 23: /* CONNECT 2400/REL */ spdchg(2400L); mdmstat = CONNECTED; break; case 46: /* CONNECT 7512 */ spdchg(8880L); /* 75/1200 split speed */ mdmstat = CONNECTED; /* (special C-Kermit code) */ break; case 47: /* CONNECT 1200/75 */ mdmstat = CONNECTED; /* Speed not supported by Kermit */ printf("CONNECT 1200/75 - Not support by C-Kermit\r\n"); break; case 48: /* CONNECT 7200 */ spdchg(7200L); mdmstat = CONNECTED; break; case 49: /* CONNECT 12000 */ spdchg(12000L); mdmstat = CONNECTED; break; #ifndef MINIDIAL case 50: /* CONNECT FAST */ if (mymdmtyp == n_TELEBIT) /* Early models only */ mdmstat = CONNECTED; break; case 52: /* RRING */ if (mymdmtyp == n_TELEBIT) if (dialdpy) printf(" Ringing...\r\n"); break; case 53: /* DIALING */ if (mymdmtyp == n_TELEBIT) if (dialdpy) printf(" Dialing...\r\n"); break; case 54: /* NO PROMPTTONE */ if (mymdmtyp == n_TELEBIT) { if (dialdpy) printf("\r\n No Prompttone.\r\n"); mdmstat = FAILED; dialsta = DIA_NODT; } break; case 61: /* Various Telebit PEP modes */ case 62: case 63: case 70: case 71: case 72: case 73: if (mymdmtyp == n_TELEBIT) /* Early models only */ mdmstat = CONNECTED; break; #endif /* MINIDIAL */ default: break; } } if (mdmstat == CONNECTED && nbuf[j] != '\0') { if (dialdpy) { printf("\r\n"); if (nbuf[j] == 'R') printf("RELIABLE"); if (nbuf[j] == 'L') printf("LAPM"); if (nbuf[j+1] == 'C') printf(" COMPRESSED"); printf("\r\n"); } } } break; case n_UNKNOWN: { int x, y = waitct; mdmstat = FAILED; /* Assume failure. */ while (y-- > -1) { x = ttchk(); if (x > 0) { if (x > LBUFL) x = LBUFL; x = ttxin(x,(CHAR *)lbuf); if ((x > 0) && dialdpy) conol(lbuf); } x = ttgmdm(); /* Try to read modem signals */ if (x < 0) break; /* Can't, fail. */ if (x & BM_DCD) { /* Got signals OK. Carrier present? */ mdmstat = CONNECTED; /* Yes, done. */ break; } /* No, keep waiting. */ sleep(1); } break; } } /* switch (augmdmtyp) */ } /* while (mdmstat == 0) */ x = alarm(0); /* Turn off alarm. */ debug(F101,"ckdial alarm off","",x); if ( mdmstat != CONNECTED ) /* Failure detected by modem */ return(dialfail(F_MODEM)); msleep(1000); /* In case DTR blinks */ debug(F100,"dial succeeded","",0); if ( #ifndef MINIDIAL augmdmtyp != n_ROLM /* Rolm has weird modem signaling */ #else 1 #endif /* MINIDIAL */ ) { alarm(3); /* In case ttpkt() gets stuck... */ ttpkt(speed,FLO_DIAX,parity); /* Cancel dialing state ioctl */ } dreset(); /* Reset alarms and signals. */ if (!quiet && !backgrd) if (dialdpy) { /* If DIAL DISPLAY ON, */ p = ck_time(); /* include timestamp. */ if (*p) printf(" Call complete: %s.\07\n",p); } else printf (" Call complete.\07\n"); #ifdef DYNAMIC if (lbuf) free(lbuf); lbuf = NULL; if (rbuf) free(rbuf); rbuf = NULL; if (fbuf) free(fbuf); fbuf = NULL; #endif /* DYNAMIC */ dialsta = DIA_OK; return(1); /* Return successfully */ } /* getok() - wait up to n seconds for OK (0) or ERROR (4) response from modem. Use with Hayeslike or CCITT modems for reading the reply to a nondialing command. Second argument says whether to be strict about numeric result codes, i.e. to require they be preceded by CR or else be the first character in the response, e.g. to prevent the ATH0 echo from looking like a valid response. Strict == 0 is needed for ATI on Telebit, which can return the model number concatenated with the numeric response code, e.g. "9620" ("962" is the model number, "0" is the response code). getok() Returns: 0 if it timed out, 1 if it succeeded, -1 on modem command, i/o, or other error. */ #ifdef CK_POSIX_SIG /* Jump-buf for getok(). */ static sigjmp_buf okbuf; #else static jmp_buf okbuf; #endif /* CK_POSIX_SIG */ static SIGTYP oktimo(foo) int foo; { /* Alarm handler for getok(). */ #ifdef OSK /* OS-9, see comment in dialtime(). */ sigmask(-1); #endif /* OSK */ #ifdef CK_POSIX_SIG siglongjmp(okbuf,1); #else longjmp(okbuf,1); #endif /* CK_POSIX_SIG */ } static int getok(n, strict) int n, strict; { CHAR c; int i, x, status, oldalarm; SIGTYP (*saval)(); /* For saving alarm handler locally */ debug(F101,"getok entry n","",n); #ifdef DYNAMIC if (!rbuf) { if (!(rbuf = malloc(RBUFL+1))) { /* Allocate input line buffer */ dialsta = DIA_IE; return(-1); } debug(F101,"GETOK rbuf malloc ok","",RBUFL+1); } #endif /* DYNAMIC */ mdmecho = 0; /* Assume no echoing of commands */ saval = signal(SIGALRM,oktimo); /* Set response timer, */ oldalarm = alarm(n); /* saving old one. */ debug(F101,"getok alarm ok","",oldalarm); if ( #ifdef CK_POSIX_SIG sigsetjmp(okbuf,1) #else setjmp(okbuf) #endif /* CK_POSIX_SIG */ ) { /* Timed out. */ alarm(oldalarm); /* Restore old alarm */ if (saval) signal(SIGALRM,saval); /* and alarm handler */ debug(F100,"getok timeout","",0); ttflui(); /* Flush input buffer */ return(0); /* and return timeout indication */ } else if (augmdmtyp == n_CCITT /* CCITT, easy... */ #ifndef MINIDIAL || augmdmtyp == n_DIGITEL /* Digitel, ditto. */ #endif /* MINIDIAL */ ) { waitfor("VAL"); return(1); } else { /* Hayes & friends, start here... */ status = 0; /* No status yet. */ for (x = 0; x < RBUFL; x++) /* Initialize response buffer */ rbuf[x] = SP; /* to all spaces */ rbuf[RBUFL] = NUL; /* and terminate with NUL. */ debug(F100,"getok rbuf init ok","",0); while (status == 0) { /* While no status... */ x = ddinc(n); /* Read a character */ if (x < 0) { /* I/O error */ alarm(oldalarm); /* Turn off alarm */ if (saval) signal(SIGALRM,saval); /* and restore handler. */ return(-1); /* Return error code. */ } debug(F101,"getok ddinc","",x); /* Got a character. */ c = x & 0x7f; /* Get low order 7 bits */ if (!c) /* Don't deposit NULs */ continue; /* or else didweget() won't work */ if (dialdpy) conoc((char)c); /* Echo it if requested */ for (i = 0; i < RBUFL-1; i++) /* Rotate buffer */ rbuf[i] = rbuf[i+1]; rbuf[RBUFL-1] = c; /* Deposit character at end */ debug(F000,"getok:",rbuf,(int) c); /* Log it */ switch (c) { /* Interpret it. */ case CR: /* Got a carriage return. */ switch(rbuf[RBUFL-2]) { /* Look at character before it. */ case '0': /* 0 = OK numeric response */ if (!strict || rbuf[RBUFL-3] == CR || rbuf[RBUFL-3] == SP) { augmdmtyp |= DIAL_NV; /* OR in the "nonverbal" bit. */ status = 1; /* Good response */ } break; case '4': /* 4 = ERROR numeric response */ #ifndef MINIDIAL /* Or Telebit model number 964! */ if (mymdmtyp == n_TELEBIT && isdigit(rbuf[RBUFL-3]) && isdigit(rbuf[RBUFL-4])) break; else #endif /* MINIDIAL */ if (!strict || rbuf[RBUFL-3] == CR || rbuf[RBUFL-3] == SP) { augmdmtyp |= DIAL_NV; /* OR in the nonverbal bit. */ status = -1; /* Bad command */ } break; } if (dialdpy && (augmdmtyp & DIAL_NV)) /* If numeric results, */ conoc(LF); /* echo a linefeed too. */ break; case LF: /* Got a linefeed. */ /* Note use of explicit octal codes in the string for CR and LF. We want real CR and LF here, not whatever the compiler happens to replace \r and \n with... */ if (!strcmp(rbuf+RBUFL-4,"OK\015\012")) /* Good response */ status = 1; else if (!strcmp(rbuf+RBUFL-7,"ERROR\015\012")) /* Error */ status = -1; augmdmtyp &= ~(DIAL_NV); /* Turn off the nonverbal bit */ break; /* Check whether modem echoes its commands... */ case 't': /* Got little t */ if (!strcmp(rbuf+RBUFL-3,"\015at") || /* See if it's "at" */ !strcmp(rbuf+RBUFL-3," at")) mdmecho = 1; debug(F111,"MDMECHO-t",rbuf+RBUFL-2,mdmecho); break; case 'T': /* Got Big T */ if (!strcmp(rbuf+RBUFL-3,"\015AT") || /* See if it's "AT" */ !strcmp(rbuf+RBUFL-3," AT")) mdmecho = 1; debug(F111,"MDMECHO-T",rbuf+RBUFL-3,mdmecho); break; default: /* Other characters, accumulate. */ status = 0; break; } } debug(F101,"getok returns","",status); alarm(oldalarm); /* Restore previous alarm */ if (saval) signal(SIGALRM,saval); /* and handler */ ttflui(); /* Flush input buffer */ return(status); /* Return status */ } } /* Maybe hang up the phone, depending on various SET DIAL parameters. */ int dialhup() { int x = 0; if (dialhng) { /* DIAL HANGUP ON? */ x = mdmhup(); /* Try modem-specific method first */ debug(F101,"dialhup mdmhup","",x); if (x > 0) { /* If it worked, */ if (dialdpy) printf(" Modem hangup OK\r\n"); /* fine. */ } else if (network) { /* If we're telnetted to */ if (dialdpy) /* a modem server, just print a msg */ printf(" Warning: modem hangup failed\r\n"); /* don't hangup! */ return(0); } else { /* Otherwise */ x = tthang(); /* Tell the OS to turn off DTR. */ if (dialdpy) { /* DIAL DISPLAY ON? */ if (x > 0) /* Yes, tell results from tthang() */ printf(" Hangup OK\r\n"); else if (x == 0) printf(" Hangup skipped\r\n"); else perror(" Hangup error"); } } } else if (dialdpy) printf(" Hangup skipped\r\n"); /* DIAL HANGUP OFF */ return(x); } /* M D M H U P -- Sends escape sequence to modem, then sends its hangup command. Returns: 0: If modem type is 0 (direct serial connection), or if modem type is < 0 (network connection), or if no action taken because DIAL MODEM-HANGUP is OFF) or because no hangup string for current modem type, or C-Kermit is in remote mode, or if action taken but there was no positive response from modem; 1: Success: modem is in command state and acknowledged the hangup command; -1: On modem command error. */ int mdmhup() { #ifdef MDMHUP MDMINF *p; /* Modem info structure pointer */ int m, x = 0; int xparity; if (dialmhu == 0 || local == 0) /* If DIAL MODEM-HANGUP is OFF, */ return(0); /* or not in local mode, fail. */ #ifdef OS2 /* In OS/2, if CARRIER is OFF, and there is indeed no carrier signal, any attempt to do i/o at this point can hang the program. This might be true for other operating systems too. */ if (!network) { /* Not a network connection */ m = ttgmdm(); /* Get modem signals */ if ((m > -1) && (m & BM_DCD == 0)) /* Check for carrier */ return(0); /* No carrier, skip the rest */ } #endif /* OS2 */ m = mdmtyp & 0xff; /* Get basic modem type (no bits!). */ if ((m < 1) || (m > MAX_MDM)) /* If modem type not in range, */ return(0); /* fail. */ p = ptrtab[m-1]; /* Get modem info pointer */ if (!(p->hup_str) || !*(p->hup_str)) { /* No hangup string? */ debug(F100,"mdmhup no hup_str","",0); /* No, */ return(0); /* fail. */ } else { debug(F110,"mdmhup hup_str",p->hup_str,0); /* Yes. */ if (p->esc_str && *(p->esc_str)) { /* Have escape sequence? */ debug(F110,"mdmhup esc_str",p->esc_str,0); debug(F101,"mdmhup esc_time","",p->esc_time); xparity = parity; /* Set PARITY to NONE temporarily */ parity = 0; if (ttpkt(speed,FLO_DIAL,parity) < 0) { /* Condition line */ parity = xparity; return(-1); /* for dialing. */ } if (p->esc_time) /* If we have a guard time */ msleep(p->esc_time); /* Pause for guard time */ debug(F100,"mdmhup pause 1 OK","",0); #ifdef NETCONN /* Send modem's escape sequence */ if (network) { /* Must catch errors here. */ if (ttol((CHAR *)(p->esc_str),(int)strlen(p->esc_str)) < 0) { parity = xparity; return(-1); } } else { ttslow(p->esc_str,p->wake_rate); /* Send escape sequence */ debug(F110,"mdmhup ttslow ok",p->esc_str,0); } #else ttslow(p->esc_str,p->wake_rate); /* Send escape sequence */ debug(F110,"mdmhup ttslow ok",p->esc_str,0); #endif /* NETCONN */ if (p->esc_time) /* Pause for guard time again */ msleep(p->esc_time); msleep(500); /* Wait half a sec for echoes. */ debug(F100,"mdmhup pause 1 OK","",0); ttflui(); /* Flush response or echo, if any */ debug(F100,"mdmhup ttflui OK","",0); } ttslow(p->hup_str,p->wake_rate); /* Now Send hangup string */ debug(F110,"mdmhup ttslow ok",p->hup_str,0); if (p->ok_fn) { /* Look for OK response */ debug(F100,"mdmhup calling response function","",0); x = (*(p->ok_fn))(3,1); /* Give it 3 seconds, be strict. */ debug(F101,"mdmhup hangup response","",x); } else { /* No OK function, */ x = 1; /* so assume it worked */ debug(F101,"mdmhup no ok_fn","",x); } parity = xparity; /* Restore prevailing parity */ return(x); /* Return OK function's return code. */ } #else /* MDMHUP not defined. */ return(0); /* Always fail. */ #endif /* MDMHUP */ } char * /* Let external routines ask */ getdws(mdmtyp) int mdmtyp; { /* about dial init-string. */ MDMINF * p; if (dialini) return(dialini); if ((mdmtyp & 0xff) < 1 || (mdmtyp & 0xff) > MAX_MDM) return(""); p = ptrtab[(mdmtyp & 0xff) -1]; return(p->wake_str); } char * getdcs(mdmtyp) int mdmtyp; { /* Same deal for dial-command */ MDMINF * p; if (dialcmd) return(dialcmd); if ((mdmtyp & 0xff) < 1 || (mdmtyp & 0xff) > MAX_MDM) return(""); p = ptrtab[(mdmtyp & 0xff) -1]; return(p->dial_str); } #else /* NODIAL */ char *dialv = "Dial Command Disabled"; int /* To allow NODIAL versions to */ mdmhup() { /* call mdmhup(), so calls to */ return(0); /* mdmhup() need not be within */ } /* #ifndef NODIAL conditionals */ #endif /* NODIAL */ #endif /* NOLOCAL */