windows-nt/Source/XPSP1/NT/printscan/fax/provider/t30/main/t30.c

2444 lines
97 KiB
C
Raw Normal View History

2020-09-26 03:20:57 -05:00
/***************************************************************************
Name : T30.C
Comment : Contains the main T30 routine, which drives this whole baby.
It calls the user-supplied protocol function to make all the
protocol decision functions. Thus in a sense this merely
a hollow shell.
This file should be read together with the appropriate
protocol function that is in use, and the T30 flowchart
(the enhanced one which includes ECM that is supplied in
T.30 Appendix-A). Ideally the (paper) copy of the chart
which I've annotated to chow which nodes are implemented
in the protocol callback function.
The other routines contained here implement the T.30
flowchart "subroutines" labelled "Response Received"
"Command Received", "RR Response Received" and "CTC
Response Received". All of which are called (only)
from ET30MainBody().
Most of teh real work is farmed out to HDLC.C (and macros
in HDLC.H), so the T30 routine is reasonably lucid.
It's organized as a block of statements with gotos between
them to closely mirror the T30 flowchart. (It's actually
uncannily close!)
Functions: (see Prototypes just below)
Revision Log
Date Name Description
-------- ----- ---------------------------------------------------------
06/14/92 arulm Created it in the new incarnation for the first time. The
T30Main() function is re-written to call WhatNext() at *all*
decision points. Some parts are simplified. It bears some
resemblance to the original. Command/ResponseReceived is
completely re-written.
06/15/92 arulm Added ECM. Still havn't tried compiling it.
06/16/92 arulm First successful compile.
***************************************************************************/
#include "prep.h"
#include "efaxcb.h"
#include "t30.h"
#include "hdlc.h"
#include "..\comm\modemint.h"
#include "..\comm\fcomint.h"
#include "..\comm\mdmcmds.h"
#include "debug.h"
#include "tiff.h"
#include "glbproto.h"
#include "t30gl.h"
#define faxTlog(m) DEBUGMSG(ZONE_T30, m)
#define faxT2log(m) DEBUGMSG(ZONE_FRAMES, m)
#define FILEID FILEID_T30
#ifdef MDDI // zero-stuffing
# define SetStuffZERO(x) T30SetStuffZERO(x)
extern void T30SetStuffZERO(PThrdGlbl pTG, USHORT cbLineMin);
extern BOOL FilterSendMem(PThrdGlbl pTG, HMODEM h, LPBYTE lpb, USHORT uCount);
#else
# define SetStuffZERO(pTG, x) FComSetStuffZERO(pTG, x)
# define FilterSendMem(pTG, h, lpb, uCnt) ModemSendMem(pTG, h, lpb, uCnt, 0)
#endif
#ifdef DEBUG
# define TIMESTAMP(str) \
faxTlog((SZMOD "TIMESTAMP %lu %s--\r\n", (unsigned long) GetTickCount(), (LPSTR) (str)));
#else // !DEBUG
# define TIMESTAMP(str)
#endif // !DEBUG
/***************************************************************************
Name : ET30MainBody
Purpose : This is the main T30. It's main purpose is to faithfully
reproduce the flowchart on pages 100-103 of Fascicle VII.3
of the CCITT blue book, Recomendation T30.
It is to be called from the protocol module after a call has
been successfully placed or answered & the modem is in HDLC
receive or send mode respectively. It conducts the entire call,
making callbacks at all decision points to a protocol-supplied
callback function. It also calls protocol-supplied callback
functions to get data to send and to unload received data.
Returns : When it returns, the phone is on hook & the modem reset.
It returns TRUE on successful call completion, and FALSE on
error. In all cases, GetLastError() will return the completion
status.
Arguments:
Comment : This function is supposed to mirror the flowchart
so it is structured as a series of statement blocks and a rats
nest of Gotos. Yes I hate GOTOs too. Havn't used one in years,
but try to do this yourself sometime in a "structured" way.
The Labels used are the same as those used in the Flowchart, and
the blocks are arranged in approximately teh same order as in
the chart, so reading both together will be a pleasant surprise.
Make sure you use the chart in the __APPENDIX-A__ of the
**1988** (or Blue Book) CCITT specs. The chart in teh main body
of teh spec is (a) deceptively similar (b) out-of-date
Calls : All of HDLC.C, some of MODEM.C, and a little of FCOM.C.
all the rest of the functions in this file.
Called By:
Revision Log
Date Name Description
-------- ----- ---------------------------------------------------------
06/15/92 arulm Adding ECM. Making it compilable
***************************************************************************/
#ifdef MDDI
# // Ricohs protocol tester complains if this exceeds 40secs
# define T1_TIMEOUT 35000L // use exact 35s nominal value
#else
# define T1_TIMEOUT 40000L // 35s + 5s. On PCs be more lax
#endif
// #define T2_TIMEOUT 7000 // 6s + 1s
#define T2_TIMEOUT 6000 // 6s
#define T3_TIMEOUT 15000L // 10s + 5s
// #define T4_TIMEOUT 3450 // 3s + 15%
#define T4_TIMEOUT 3000 // 3s
// #define T4_TIMEOUT 2550 // 3s - 15%
// if our DCS-TCF collides with the recvrs 2nd NSF-DIS or NSC-DTC
// then if the lengths are just right, we can end up in lock step
// after the first try, increase the TCF-response timeout so we
// get out of lock step! This is bug#6847
#define TCFRESPONSE_TIMEOUT_SLACK 500 // 0.5 secs
/****************** begin prototypes from t30.c *****************/
ET30ACTION PhaseNodeA(PThrdGlbl pTG);
ET30ACTION PhaseNodeT(PThrdGlbl pTG);
ET30ACTION PhaseNodeD(PThrdGlbl pTG, BOOL);
ET30ACTION NonECMPhaseC(PThrdGlbl pTG );
ET30ACTION NonECMPhaseD(PThrdGlbl pTG );
ET30ACTION RecvPhaseB(PThrdGlbl pTG, ET30ACTION action);
ET30ACTION PhaseNodeF(PThrdGlbl pTG, BOOL, BOOL);
ET30ACTION PhaseRecvCmd(PThrdGlbl pTG );
ET30ACTION PhaseGetTCF(PThrdGlbl pTG, IFR, BOOL);
ET30ACTION NonECMRecvPhaseC(PThrdGlbl pTG);
ET30ACTION NonECMRecvPhaseD(PThrdGlbl pTG);
/***************** end of prototypes from t30.c *****************/
USHORT T30MainBody(PThrdGlbl pTG, BOOL fCaller, ET30ACTION actionInitial, HLINE hLine, HMODEM hModem )
{
ET30ACTION action;
USHORT uRet;
BOOL fEnteredHalfway;
uRet = T30_CALLFAIL;
FComCriticalNeg(pTG, TRUE);
(MyDebugPrint(pTG, LOG_ALL, "\r\n\r\n\r\n\r\n\r\n======================= Entering MainBody ==============================\r\n\r\n\r\n\r\n"));
// zero-init the 3 global structs T30, ECM and Params. After this some
// fields are inited to non-zero below. Rest are set approp elsewhere
// in the code (hopefully before use :-)
_fmemset(&pTG->Params, 0, sizeof(pTG->Params));
_fmemset(&pTG->T30, 0, sizeof(pTG->T30));
_fmemset(&pTG->ECM, 0, sizeof(pTG->ECM));
_fmemset(&pTG->EchoProtect, 0, sizeof(pTG->EchoProtect));
pTG->Params.hModem = hModem;
pTG->Params.hLine = hLine;
pTG->Params.lpfnWhatNext = ProtGetWhatNext(pTG);
// INI file settings related stuff
pTG->T30.uSkippedDIS = 0;
// Initialize this global. Very Important!! See HDLC.C for usage.
pTG->T30.fReceivedDIS = FALSE;
pTG->T30.fReceivedDTC = FALSE;
pTG->T30.fReceivedEOM = FALSE;
pTG->T30.uTrainCount = 0;
pTG->T30.uRecvTCFMod = 0xFFFF;
pTG->T30.ifrCommand = pTG->T30.ifrResp = pTG->T30.ifrSend = 0;
pTG->ECM.ifrPrevResponse = 0;
pTG->ECM.SendPageCount = 0;
pTG->ECM.fEndOfPage = pTG->ECM.fRecvEndOfPage = TRUE;
pTG->T30.fAtEndOfRecvPage = FALSE;
pTG->ECM.fSentCTC = FALSE;
pTG->ECM.fRecvdCTC = FALSE;
#ifdef SMM
pTG->T30.lpfs = (LPFRAMESPACE)pTG->bStaticRecvdFrameSpace;
#else
#error NYI ERROR: Code Not Complete--Have to pick GMEM_ flags (FIXED? SHARE?)
...also have to call IFProcSetResFlags(hinstMe).....
pTG->T30.lpfs = (LPFRAMESPACE)IFMemAlloc(0,TOTALRECVDFRAMESPACE, &uwJunk);
#endif
pTG->T30.Nframes = 0;
fEnteredHalfway = FALSE;
if(actionInitial != actionNULL)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Got Initial Action = %d\r\n", actionInitial));
action = actionInitial;
fEnteredHalfway = TRUE;
BG_CHK(action==actionGONODE_F || action==actionGONODE_D);
}
else if(fCaller) // choose the right entry point
action = actionGONODE_T;
else
{
action = actionGONODE_R1;
pTG->T30.fSendAfterSend = TRUE; // CED--NSF/DIS
}
// fall through into PhaseLoop
/******** Phase loop ********/
//PhaseLoop:
for(;;)
{
/******************************************************************************
T = Start of Phase be for transmitter
R1 = Start of Phase B for callee
R2 = start of Phase B for poller
A = Poll/Send decision point
D = Start of DCS/TCF
F = Recv Command loop
RecvCmd = Interpretation of pre-page cmds
RecvPhaseC = Start of Rx PhaseC (ECM and Non-pTG->ECM. New Page in ECM mode)
I = Start of Tx PhaseC (ECM & Non-pTG->ECM. New page in ECM mode)
II = Start of Tx Non-ECM PhaseD
III = Start of Rx Non-ECM PhaseD
IV = Start of Tx ECM PhaseC (Same page, new block)
V = Start of Tx ECM PhaseD (end of partial page)
VII = Start of Rx ECM PhaseD
RecvPRIQ = Handling of recvd PRI-Q commands
E = Handling of received PIP/PIN responses
*******************************************************************************/
switch(action)
{
case actionGONODE_T: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_T\r\n"));
action = PhaseNodeT(pTG);
break;
case actionGONODE_D: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_D\r\n"));
action = PhaseNodeD(pTG, fEnteredHalfway);
fEnteredHalfway = FALSE;
break;
case actionGONODE_A: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_A\r\n"));
action = PhaseNodeA(pTG);
break;
case actionGONODE_R1: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_R1\r\n"));
action = RecvPhaseB(pTG, action);
break;
case actionGONODE_R2: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_R2\r\n"));
action = RecvPhaseB(pTG, action);
break;
case actionNODEF_SUCCESS:
(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionNODEF_SUCCESS\r\n"));
action = PhaseNodeF(pTG, TRUE, FALSE);
break;
case actionGONODE_F: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_F\r\n"));
action = PhaseNodeF(pTG, FALSE, fEnteredHalfway);
fEnteredHalfway = FALSE;
break;
case actionGONODE_RECVCMD:
(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_RECVCMD\r\n"));
action = PhaseRecvCmd(pTG);
break;
case actionGONODE_I: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_I\r\n"));
action = NonECMPhaseC(pTG);
break;
case actionGONODE_II: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_II\r\n"));
action = NonECMPhaseD(pTG);
break;
case actionGONODE_III: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_III\r\n"));
action = NonECMRecvPhaseD(pTG);
break;
case actionGONODE_IV: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_IV_New\r\n"));
action = ECMPhaseC(pTG, FALSE); // ReTx
break;
case actionGONODE_ECMRETRANSMIT:
(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_ECMRETRANSMIT\r\n"));
action = ECMPhaseC(pTG, TRUE); // ReTx
break;
case actionGONODE_V: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_V\r\n"));
action = ECMPhaseD(pTG);
break;
case actionSENDEOR_EOP: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionSENDEOR_EOP\r\n"));
action = ECMSendEOR_EOP(pTG);
break;
case actionGONODE_VII: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_VII\r\n"));
action = ECMRecvPhaseD(pTG);
break;
case actionGONODE_RECVPHASEC:
// (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_RECVPHASEC\r\n"));
/**/
if(ProtReceivingECM(pTG))
action = ECMRecvPhaseC(pTG, FALSE);
else
/**/
action = NonECMRecvPhaseC(pTG);
break;
case actionGONODE_RECVECMRETRANSMIT:
// (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_RECVECMRETRANSMIT\r\n"));
BG_CHK(ProtReceivingECM(pTG));
action = ECMRecvPhaseC(pTG, TRUE);
break;
case actionNULL: BG_CHK(FALSE);
goto error;
case actionDCN_SUCCESS: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionDCN_SUCCESS\r\n"));
uRet = T30_CALLDONE;
goto NodeC; // successful end of send call
case actionHANGUP_SUCCESS:
(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionHANGUP_SUCCESS\r\n"));
ICommGotDisconnect(pTG);
uRet = T30_CALLDONE;
goto NodeB; // successful end of recv call
case actionDCN: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionDCN\r\n"));
goto NodeC;
case actionHANGUP: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionHANGUP\r\n"));
// This should only be called on a _successful_ completion,
// otherwise we get either (a) a DCN that MSGHNDLR does not
// want OR (b) A fake EOJ posted to MSGHNDLR. See bug#4019
////// ICommGotDisconnect(); /////
goto NodeB;
case actionERROR: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionERROR\r\n"));
// ICommStatus(T30STAT_ERROR, 0, 0, 0);
goto error;
default: BG_CHK(FALSE);
goto error;
# ifdef PRI
case actionGONODE_RECVPRIQ:
(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_RECVPRIQ\r\n"));
goto RecvPriQ;
case actionGONODE_E:(MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGONODE_E\r\n"));
goto NodeE;
case actionGOVOICE: (MyDebugPrint(pTG, LOG_ALL, "EndPhase: Got actionGOVOICE\r\n"));
goto VoiceLine;
# endif
}
}
/******** PRI-Q stuff (Rx side) ********/
#ifdef PRI
// no fall through here
BG_CHK(FALSE);
RecvPRIQ:
switch(action = pTG->Params.lpfnWhatNext(pTG, eventGOTPRIQ, (WORD)pTG->T30.ifrCommand))
{
case actionSENDPIP: pTG->T30.ifrSend=ifrPIP; goto VoiceLine;
case actionSENDPIN: pTG->T30.ifrSend=ifrPIN; goto VoiceLine;
case actionGONODE_F: CLEAR_MISSED_TCFS(); break;
case actionGO_RECVPOSTPAGE:
pTG->T30.ifrCommand = pTG->T30.ifrCommand-ifrPRI_MPS+ifrMPS;
action = GONODE_III;
break;
case actionHANGUP: break;
case actionERROR: break; // goto PhaseLoop & exit
//// default: return BadAction(action);
}
goto PhaseLoop;
/******** PIP/PIN stuff (Tx side) ********/
// no fall through here
BG_CHK(FALSE);
NodeE:
switch(action = pTG->Params.lpfnWhatNext(pTG, eventGOTPIPPIN,
(WORD)pTG->T30.ifrResp, (DWORD)pTG->T30.ifrSend))
{
case actionGOVOICE: goto VoiceLine;
case actionGONODE_A: BG_CHK(FALSE); // dunno what to do here
case actionDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionDCN from eventGOTPIPPIN\r\n"));
action = actionDCN;
break;
case actionERROR: break; // goto PhaseLoop & exit
//// default: return BadAction(action);
}
goto PhaseLoop;
// no fall through here
VoiceLine:
SendSingleFrame(pTG, pTG->T30.ifrSend, 0, 0, 1); // re-send PRI-Q or send PIP/PIN
switch(action = pTG->Params.lpfnWhatNext(pTG, eventVOICELINE))
{
case actionHANGUP: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionHANGUP from eventVOICELINE\r\n"));
action = actionHANGUP;
break;
case actionGONODE_T:
case actionGONODE_R1:
case actionGONODE_R2: break;
case actionERROR: break; // goto PhaseLoop & exit
//// default: return BadAction(action);
}
goto PhaseLoop;
#endif
error:
(MyDebugPrint(pTG, LOG_ERR, "=======******* USER ABORT or TRANSPORT FATAL ERROR *******========\r\n"));
// must call this always, because it's resets the Modem Driver's
// global state (e.g. shuts down SW framing & filters if open,
// resets the state variables etc (EndMode()))
// Must call it before the SendDCN() because SendDCN causes
// BG_CHKs if in strange state.
ModemSync(pTG, pTG->Params.hModem, RESYNC_TIMEOUT1);
if(pTG->T30.fReceivedDIS || pTG->T30.fReceivedDTC)
SendDCN(pTG);
// hangup in T30Main
// NCULink(pTG->Params.hLine, 0, 0, NCULINK_HANGUP);
// already hungup?
#ifndef SMM
IFMemFree(pTG, pTG->T30.lpfs);
#endif
FComCriticalNeg(pTG, FALSE);
uRet = T30_CALLFAIL;
goto done;
NodeC:
#ifdef MDDI
ModemSync(pTG, pTG->Params.hModem, RESYNC_TIMEOUT1);
#else // !MDDI
// +++ 4/12/95 Win95 hack -- to prevent ModemSync from sending
// an AT here.
ModemSyncEx(pTG, pTG->Params.hModem, RESYNC_TIMEOUT1, fMDMSYNC_DCN);
#endif // !MDDI
SendDCN(pTG);
// falls through here
NodeB:
// call this here also??
// ModemSync(pTG->Params.hModem, RESYNC_TIMEOUT1);
// hangup in T30Main
// NCULink(pTG->Params.hLine, 0, 0, NCULINK_HANGUP);
#ifndef SMM
IFMemFree(pTG, pTG->T30.lpfs);
#endif
FComCriticalNeg(pTG, FALSE);
// fall through to done:
done:
return uRet;
}
// End of T30 routine!!!!!!!!!
ET30ACTION PhaseNodeA(PThrdGlbl pTG)
{
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseNodeA \n" );
if(pTG->T30.ifrCommand == ifrDIS)
pTG->T30.fReceivedDIS = TRUE;
else if(pTG->T30.ifrCommand == ifrDTC)
pTG->T30.fReceivedDTC = TRUE;
return (pTG->Params.lpfnWhatNext) (pTG, eventNODE_A, (WORD)pTG->T30.ifrCommand);
}
ET30ACTION PhaseNodeT(PThrdGlbl pTG)
{
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseNodeT \n" );
/******** Transmitter Phase B. Fig A-7/T.30 (sheet 1) ********/
// NodeT:
// Have to redo this "DIS bit" everytime through PhaseB
pTG->T30.fReceivedDIS = FALSE;
// also the Received EOM stuff
pTG->T30.fReceivedEOM = FALSE;
// and teh received-DTC stuff
pTG->T30.fReceivedDTC = FALSE;
// INI file settings related stuff
if(pTG->ProtParams.SendT1Timer)
TstartTimeOut(pTG, &pTG->T30.toT1, pTG->ProtParams.SendT1Timer);
else
TstartTimeOut(pTG, &pTG->T30.toT1, T1_TIMEOUT);
do
{
if(pTG->Params.lpfnWhatNext(pTG, eventNODE_T) == actionERROR)
break;
// no need for echo protection. We havnt transmitted anything!
switch(pTG->T30.ifrCommand=GetCommand(pTG, ifrPHASEBcommand))
{
// ifrTIMEOUT means no flags before T2
// ifrNULL means timeout, or loss of carrier, or no flags
// or no frame. ifrBAD means *only* bad frames recvd.
case ifrBAD: SendCRP(pTG); // and fall thru to NULL/TIMEOUT
case ifrTIMEOUT:
case ifrNULL: break;
case ifrDIS: // INI file settings related stuff
if(pTG->ProtParams.IgnoreDIS && ((int)pTG->T30.uSkippedDIS < pTG->ProtParams.IgnoreDIS))
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Ignoring First DIS\r\n"));
pTG->T30.uSkippedDIS++;
break; // continue with T1 loop
}
case ifrDTC:
// case ifrNSF:
// case ifrNSC:
return actionGONODE_A; // got a valid frame
default: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Caller T1: Got random ifr=%d\r\n", pTG->T30.ifrCommand));
break; // continue with T1 loop
}
}
while(TcheckTimeOut(pTG, &pTG->T30.toT1));
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> T1 timeout at Caller\r\n"));
ICommFailureCode(pTG, T30FAILS_T1);
return actionHANGUP;
}
ET30ACTION PhaseNodeD(PThrdGlbl pTG, BOOL fEnteredHere)
{
LPLPFR lplpfr;
USHORT N;
ET30ACTION action;
USHORT uWhichDCS; // 0=first, 1=after NoReply 2=afterFTT
DWORD TiffConvertThreadId;
/******** Transmitter Phase B2. Fig A-7/T.30 (sheet 1) ********/
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseNodeD \n" );
if (pTG->Inst.SendParams.Fax.Encoding == MR_DATA) {
pTG->TiffConvertThreadParams.tiffCompression = TIFF_COMPRESSION_MR;
}
else {
pTG->TiffConvertThreadParams.tiffCompression = TIFF_COMPRESSION_MH;
}
if (pTG->Inst.SendParams.Fax.AwRes & ( AWRES_mm080_077 | AWRES_200_200 ) ) {
pTG->TiffConvertThreadParams.HiRes = 1;
}
else {
pTG->TiffConvertThreadParams.HiRes = 0;
// use LoRes TIFF file prepared by FaxSvc
// pTG->lpwFileName[ wcslen(pTG->lpwFileName) - 1] = (unsigned short) ('$');
}
_fmemcpy (pTG->TiffConvertThreadParams.lpszLineID, pTG->lpszPermanentLineID, 8);
pTG->TiffConvertThreadParams.lpszLineID[8] = 0;
uWhichDCS = 0;
NodeD:
N = 0;
lplpfr = 0;
action = pTG->Params.lpfnWhatNext(pTG, eventSENDDCS,(WORD)uWhichDCS,
(ULONG_PTR)((LPUWORD)&N),(ULONG_PTR)((LPLPLPFR)&lplpfr));
switch(action)
{
case actionDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionDCN from eventSENDDCS(uWhichDCS=%d)\r\n", uWhichDCS));
return actionDCN;
case actionSENDDCSTCF: break;
case actionSKIPTCF: break; // Ricoh hook
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
NodeDprime: // used only by TCF--no reply
ICommStatus(pTG, T30STATS_TRAIN, ProtGetSendMod(pTG), (USHORT)(pTG->T30.uTrainCount+1), 0);
SendManyFrames(pTG, lplpfr, N);
if(action != actionSKIPTCF) // Ricoh hook
{
if (!pTG->fTiffThreadCreated) {
(MyDebugPrint(pTG, LOG_ALL, "Creating TIFF helper thread \r\n"));
pTG->hThread = CreateThread(
NULL,
0,
(LPTHREAD_START_ROUTINE) TiffConvertThreadSafe,
(LPVOID) pTG,
0,
&TiffConvertThreadId
);
if (!pTG->hThread) {
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> TiffConvertThread create FAILED\r\n"));
return actionDCN;
}
pTG->fTiffThreadCreated = 1;
pTG->AckTerminate = 0;
pTG->fOkToResetAbortReqEvent = 0;
if ( (pTG->RecoveryIndex >=0 ) && (pTG->RecoveryIndex < MAX_T30_CONNECT) ) {
T30Recovery[pTG->RecoveryIndex].TiffThreadId = TiffConvertThreadId;
T30Recovery[pTG->RecoveryIndex].CkSum = ComputeCheckSum(
(LPDWORD) &T30Recovery[pTG->RecoveryIndex].fAvail,
sizeof ( T30_RECOVERY_GLOB ) / sizeof (DWORD) - 1 );
}
}
(MyDebugPrint(pTG, LOG_ALL, "SENDING TCF\r\n"));
SendTCF(pTG);
pTG->T30.uTrainCount++;
(MyDebugPrint(pTG, LOG_ALL, "TCF Send Done\r\n"));
}
// no need for echo protection? Wouldnt know what to do anyway!
pTG->T30.ifrResp = GetResponse(pTG, ifrTCFresponse);
#ifdef RICOHAI
if(fEnteredHere)
{
if(pTG->T30.ifrResp != ifrCFR && pTG->T30.ifrResp != ifrFTT)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Send-side AI protocol failed\r\n"));
fUsingOEMProt = 0;
return actionGONODE_T;
}
}
#else
BG_CHK(!fEnteredHere);
#endif
switch(pTG->T30.ifrResp)
{
case ifrDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got ifrDCN in GetResponse after sending TCF\r\n"));
ICommFailureCode(pTG, T30FAILS_TCF_DCN);
return actionHANGUP; // got DCN. Must hangup
case ifrBAD: // ifrBAD means *only* bad frames recvd. Treat like NULL
case ifrNULL: // timeout. May try again
if(pTG->T30.uTrainCount >= 3)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got no reply after 3 TCFs\r\n"));
ICommFailureCode(pTG, T30FAILS_3TCFS_NOREPLY);
return actionDCN;
}
else
{
uWhichDCS = 1;
// goto NodeD; // send new DCS??
goto NodeDprime; // resend *same* DCS, same baudrate
}
case ifrDIS:
case ifrDTC:
if(pTG->T30.uTrainCount >= 3)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got DIS/DTC after 3 TCFs\r\n"));
ICommFailureCode(pTG, T30FAILS_3TCFS_DISDTC);
return actionDCN;
}
else
{
pTG->T30.ifrCommand = pTG->T30.ifrResp;
return actionGONODE_A;
}
case ifrFTT:
// reset training count on FTT since we drop speed. Want to try
// 3 times _without_ a response (DIS DTC doesn't count) before
// giving up
pTG->T30.uTrainCount = 0;
uWhichDCS = 2;
goto NodeD;
case ifrCFR:
pTG->T30.uTrainCount = 0;
switch(action = pTG->Params.lpfnWhatNext(pTG, eventGOTCFR))
{
case actionGONODE_I: // Non-ECM PhaseC
case actionGONODE_IV: // ECM PhaseC, new page
return action;
case actionGONODE_D: goto NodeD; // Ricoh hook
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
default:
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Unknown Reply to TCF\r\n"));
ICommFailureCode(pTG, T30FAILS_TCF_UNKNOWN);
return actionDCN;
}
}
BG_CHK(FALSE);
}
ET30ACTION NonECMPhaseC(PThrdGlbl pTG)
{
/******** Transmitter Phase C. Fig A-7/T.30 (sheet 1) ********/
LPBUFFER lpbf=0;
ULONG lTotalLen=0;
SWORD swRet;
USHORT uMod, uEnc;
(MyDebugPrint(pTG, LOG_ALL, "T30: NonECMPhaseC Starting Page SEND.......\r\n"));
/***
switch(action = pTG->Params.lpfnWhatNext(eventSTARTSEND))
{
case actionCONTINUE: break;
case actionDCN:
case actionHANGUP: return action;
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(action);
}
***/
// already done in WhatNext
// ICommSetSendMode(FALSE, MY_BIGBUF_SIZE, MY_BIGBUF_ACTUALSIZE-4, FALSE);
// Callback to open file to send. Returns no data
(MyDebugPrint(pTG, LOG_ALL, "Waiting for Startpage in T30 at 0x%08lx\r\n", GetTickCount()));
DEBUGSTMT(IFProcProfile((HTASK)(-1), TRUE));
if((swRet=GetSendBuf(pTG, 0, SEND_STARTPAGE)) != SEND_OK)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Nonzero return %d from SendProc at Start Page\r\n", swRet));
// return actionDCN;
return actionERROR;
}
DEBUGSTMT(IFProcProfile((HTASK)(-1), FALSE));
(MyDebugPrint(pTG, LOG_ALL, "Got Startpage in T30 at 0x%08lx\r\n", GetTickCount()));
uEnc = ProtGetSendEncoding(pTG);
BG_CHK(uEnc==MR_DATA || uEnc==MH_DATA);
uMod = ProtGetSendMod(pTG);
// in non-ECM mode, PhaseC is ALWAYS with short-train.
// Only TCF uses long-train
if(uMod >= V17_START) uMod |= ST_FLAG;
//(MyDebugPrint(pTG, "Calling SendMode in T30 at 0x%08lx\r\n", GetTickCount()));
// **MUST** call RecvSilence here since it is recv-followed-by-send case
// here we should use a small timeout (100ms?) and if it fails,
// should go back to sending the previous V21 frame (which could be DCS
// or MPS or whatever, which is why it gets complicated & we havn't
// done it!). Meanwhile use a long timeout, ignore return value
// and send anyway.
if(!ModemRecvSilence(pTG, pTG->Params.hModem, RECV_PHASEC_PAUSE, LONG_RECVSILENCE_TIMEOUT))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Non-ECM PIX RecvSilence(%d, %d) FAILED!!!\r\n", RECV_PHASEC_PAUSE, LONG_RECVSILENCE_TIMEOUT));
}
if(!ModemSendMode(pTG, pTG->Params.hModem, uMod, FALSE, (IFR)((uEnc==MR_DATA) ? ifrPIX_MR : ifrPIX_MH)))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> ModemSendMode failed in Tx PhaseC\r\n"));
ICommFailureCode(pTG, T30FAILS_SENDMODE_PHASEC);
BG_CHK(FALSE);
return actionERROR;
}
//(MyDebugPrint(pTG, "Done SendMode in T30 at 0x%08lx\r\n", GetTickCount()));
// need to send these quickly to avoid underrun (see bug842).
// Also move the preamble/postamble into the ModemSendMode
// and ModemSendMem(FINAL)
// Already sent (in ModemSendMode)
// SendZeros(PAGE_PREAMBLE, FALSE); // Send some zeros to warm up....
// (MyDebugPrint(pTG, "SENT Preamble Training.....\r\n"));
// need to set line min zeros here. get from prot and call Modem
SetStuffZERO(pTG, (ProtGetMinBytesPerLine(pTG)) ); // Enable ZERO stuffing
// Start yielding *after* entering PhaseC and getting some stuff into the buffer
FComCriticalNeg(pTG, FALSE);
// DONT SEND an EOL here. See BUG#6441. We now make sure the EOL is
// added by FAXCODEC. At this level we only append the RTC
#ifdef IFAX
BroadcastMessage(pTG, IF_PSIFAX_DATAMODE, PSIFAX_SEND, (uMod & (~ST_FLAG)));
#endif
(MyDebugPrint(pTG, LOG_ALL, "SENDING Page Data.....at %ld \n", GetTickCount()));
lTotalLen = 0;
BG_CHK(lpbf == 0);
while((swRet=GetSendBuf(pTG, &lpbf, SEND_SEQ)) == SEND_OK)
{
BG_CHK(lpbf);
if(!lpbf->wLengthData)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got 0 bytes from GetSendBuf--freeing buf\r\n"));
MyFreeBuf(pTG, lpbf);
continue;
}
lTotalLen += lpbf->wLengthData;
if(!FilterSendMem(pTG, pTG->Params.hModem, lpbf->lpbBegData, lpbf->wLengthData))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> DataWrite Timeout in NON-ECM Phase C\r\n"));
ICommFailureCode(pTG, T30FAILS_MODEMSEND_PHASEC);
BG_CHK(FALSE);
return actionERROR; // goto error;
}
// (MyDebugPrint(pTG, "Freeing 0x%08lx in NON-ECM\r\n", lpbf));
if(!MyFreeBuf(pTG, lpbf))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> FReeBuf failed in NON-ECM Phase C\r\n"));
ICommFailureCode(pTG, T30FAILS_FREEBUF_PHASEC);
BG_CHK(FALSE);
return actionERROR; // goto error;
}
lpbf = 0;
}
BG_CHK(!lpbf);
if(swRet == SEND_ERROR)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Error in SendBuf\r\n"));
// return actionDCN;
return actionERROR;
}
BG_CHK(swRet == SEND_EOF);
(MyDebugPrint(pTG, LOG_ALL, "SENDING RTC.....at %ld \n", GetTickCount()));
SetStuffZERO(pTG, 0); // Disable ZERO stuffing BEFORE sending RTC!
if(!SendRTC(pTG, FALSE)) // RTC and final flag NOT set
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Error in SendMem at end-PhaseC\r\n"));
BG_CHK(FALSE);
ICommFailureCode(pTG, T30FAILS_MODEMSEND_ENDPHASEC);
return actionERROR; // error return from ModemSendMem
}
// Turn off yielding *before* the buffer completely drains
FComCriticalNeg(pTG, TRUE);
#if (PAGE_POSTAMBLE_DIV != 0)
BG_CHK(PAGE_POSTAMBLE_DIV);
// Send zeros to cool off FINAL flag SET
SendZeros(pTG, (USHORT)(TCFLen[uMod & 0x0F] / (PAGE_POSTAMBLE_DIV)), TRUE);
#else
// Need this!! Need to send the SEND_EOF flag to modem driver
SendZeros(pTG, 1, TRUE); // sending 0 bytes is a bit unsafe....
#endif // (PAGE_POSTAMBLE_DIV != 0)
// I think this can precede the SEND_FINAL code,
// to be _very_ safe put it after the whole buffer has drained
// need to set line min back to 0 here.
//+++SetStuffZERO(0); // Disable ZERO stuffing
(MyDebugPrint(pTG, LOG_ALL, "Page Send Done.....len=(%ld, 0x%08x)\r\n", lTotalLen, lTotalLen));
pTG->T30.fSendAfterSend = TRUE; // PhaseC/PIX--MPS/EOM/EOP
return actionGONODE_II;
}
ET30ACTION NonECMPhaseD(PThrdGlbl pTG)
{
USHORT uTryCount;
ET30ACTION action;
/******** Transmitter Phase D. Fig A-7/T.30 (sheet 2) ********/
// NodeII:
MyDebugPrint (pTG, LOG_ALL, "T30: NonECMPhaseD \n" );
switch(action = pTG->Params.lpfnWhatNext(pTG, eventPOSTPAGE))
{
case actionSENDMPS: pTG->T30.ifrSend = ifrMPS; break;
case actionSENDEOM: pTG->T30.ifrSend = ifrEOM; break;
case actionSENDEOP: pTG->T30.ifrSend = ifrEOP; break;
#ifdef PRI
case actionSENDPRIMPS: pTG->T30.ifrSend = ifrPRIMPS; break;
case actionSENDPRIEOM: pTG->T30.ifrSend = ifrPRIEOM; break;
case actionSENDPRIEOP: pTG->T30.ifrSend = ifrPRIEOP; break;
#endif
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
for(uTryCount=0;;)
{
(MyDebugPrint(pTG, LOG_ALL, "Sending postpage in T30 at %ld\r\n", GetTickCount()));
// RSL dont sleep here
SendSingleFrame(pTG, pTG->T30.ifrSend, 0, 0, 0);
echoretry:
pTG->T30.ifrResp = GetResponse(pTG, ifrPOSTPAGEresponse);
// if we hear our own frame, try to recv again. DONT retransmit!
if(pTG->T30.ifrResp==pTG->T30.ifrSend) { ECHOMSG(pTG->T30.ifrResp); goto echoretry; }
(MyDebugPrint(pTG, LOG_ALL, "Got postpage resp in T30 at %08ld\r\n", GetTickCount()));
if(pTG->T30.ifrResp != ifrNULL && pTG->T30.ifrResp != ifrBAD)
break;
if(++uTryCount >= 3)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got no reply after 3 post-page commands\r\n"));
ICommFailureCode(pTG, T30FAILS_3POSTPAGE_NOREPLY);
return actionDCN;
}
}
switch(pTG->T30.ifrResp)
{
case ifrBAD: // ifrBAD means *only* bad frames recvd. Treat like NULL
case ifrNULL: BG_CHK(FALSE); // these should never get here
ICommFailureCode(pTG, T30FAILS_BUG1);
return actionERROR; // in case they do
case ifrDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got ifrDCN from GetResponse after sending post-page command\r\n"));
ICommFailureCode(pTG, T30FAILS_POSTPAGE_DCN);
return actionHANGUP;
case ifrPIN:
case ifrPIP:
# ifdef PRI
goto NodeE;
# else
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Procedure interrupts not supported\r\n"));
pTG->T30.ifrResp = pTG->T30.ifrResp - ifrPIP + ifrRTP;
// ICommFailureCode(T30FAILS_POSTPAGE_PIPPIN);
// return actionERROR;
// return actionDCN;
# endif
// default: // fallthrough --- MCF, RTN, RTP
}
action = pTG->Params.lpfnWhatNext(pTG, eventGOTPOSTPAGERESP,
(WORD)pTG->T30.ifrResp, (DWORD)pTG->T30.ifrSend);
if(pTG->T30.ifrSend==ifrEOP &&
(pTG->T30.ifrResp==ifrMCF || pTG->T30.ifrResp==ifrRTP)
&& action==actionDCN)
{
ICommFailureCode(pTG, T30FAILS_SUCCESS);
return actionDCN_SUCCESS;
}
else
return action;
}
ET30ACTION RecvPhaseB(PThrdGlbl pTG, ET30ACTION action)
{
LPLPFR lplpfr;
USHORT N, i;
/******** Receiver Phase B. Fig A-7/T.30 (sheet 1) ********/
MyDebugPrint (pTG, LOG_ALL, "T30: RecvPhaseB \n" );
if(action == actionGONODE_R1)
{
// NodeR1:
// Have to redo this "DIS bit" everytime through PhaseB
pTG->T30.fReceivedDIS = FALSE; // set to FALSE when sending DIS
// also the Received EOM stuff
pTG->T30.fReceivedEOM = FALSE;
// and teh received-DTC stuff
pTG->T30.fReceivedDTC = FALSE;
N = 0;
lplpfr = 0;
action = pTG->Params.lpfnWhatNext(pTG, eventSENDDIS,(WORD)0,
(ULONG_PTR)((LPUWORD)&N),(ULONG_PTR)((LPLPLPFR)&lplpfr));
}
else
{
// NodeR2:
BG_CHK(action == actionGONODE_R2);
// fix the Received EOM stuff
pTG->T30.fReceivedEOM = FALSE;
N = 0;
lplpfr = 0;
action = pTG->Params.lpfnWhatNext(pTG, eventSENDDTC,(WORD)0,
(ULONG_PTR)((LPUWORD)&N),(ULONG_PTR)((LPLPLPFR)&lplpfr));
}
switch(action)
{
case actionDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionDCN from eventSENDDIS or SENDDTC\r\n"));
return actionDCN;
case actionSEND_DIS:
case actionSEND_DTC: break;
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
// INI file settings related stuff
if(pTG->ProtParams.RecvT1Timer)
TstartTimeOut(pTG, &pTG->T30.toT1, pTG->ProtParams.RecvT1Timer);
else
TstartTimeOut(pTG, &pTG->T30.toT1, T1_TIMEOUT);
do
{
ICommStatus(pTG, T30STATR_TRAIN, 0, 0, 0);
if(pTG->Params.lpfnWhatNext(pTG, eventNODE_R) == actionERROR)
break;
TIMESTAMP("Sending NSF-DIS")
SendManyFrames(pTG, lplpfr, N);
/**
for(i=0; i<N; i++)
SendFrame( lplpfr[i]->ifr, lplpfr[i]->fif, lplpfr[i]->cb, i==(N-1));
**/
pTG->T30.uRecvTCFMod = 0xFFFF;
echoretry:
TIMESTAMP("Getting Response")
pTG->T30.ifrCommand=GetResponse(pTG, ifrPHASEBresponse);
// if we hear our own frame, try to recv again. DONT retransmit!
for(i=0; i<N; i++)
{
if(pTG->T30.ifrCommand == lplpfr[i]->ifr)
{
ECHOMSG(pTG->T30.ifrCommand);
goto echoretry;
}
}
switch(pTG->T30.ifrCommand)
{
case ifrNSS: // do same as for DCS
case ifrDCS: return PhaseGetTCF(pTG, pTG->T30.ifrCommand, FALSE);
case ifrBAD: // ifrBAD means *only* bad frames recvd. Treat like NULL
case ifrNULL: break; // out of the switch() and continue with loop
case ifrDCN:
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got DCN after DIS or DTC\r\n"));
ICommFailureCode(pTG, T30FAILR_PHASEB_DCN);
return actionHANGUP; //bugfix #478
default: return actionGONODE_RECVCMD;
}
}
while(TcheckTimeOut(pTG, &(pTG->T30.toT1)));
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> T1 timeout on Receiver\r\n"));
ICommFailureCode(pTG, T30FAILR_T1);
return actionDCN;
}
ET30ACTION PhaseNodeF(PThrdGlbl pTG, BOOL fEopMcf, BOOL fEnteredHere)
{
USHORT uFLoopCount;
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseNodeF \n" );
// NodeF:
uFLoopCount = 0;
NodeFprime:
for(;;)
{
pTG->T30.uRecvTCFMod = 0xFFFF;
echoretry:
pTG->T30.ifrCommand = GetCommand(pTG, (USHORT)(pTG->EchoProtect.fGotWrongMode ? ifrNODEFafterWRONGMODE : ifrNODEFcommand));
// reset the fGotWrongMode flag
pTG->EchoProtect.fGotWrongMode = 0;
// if we hear the last frame we sent, try to recv again. DONT retx!
// bug--might have matched ifrNULL...
// added: if ModemRecvMode() returns EOF then also retry. RecvMode
// returns RECV_EOF only if we pass it the ifrNODEFafterWRONGMODE hint
// and it senses silence (i.e. we saw a V21 echo but missed it). In
// this case we want to retry the high speed PIX recv
if(pTG->EchoProtect.ifrLastSent && (pTG->T30.ifrCommand==pTG->EchoProtect.ifrLastSent || pTG->T30.ifrCommand==ifrEOFfromRECVMODE))
{
ECHOMSG(pTG->T30.ifrCommand);
switch(pTG->EchoProtect.modePrevRecv)
{
default:
case modeNONE: goto echoretry;
case modeNONECM: return actionGONODE_RECVPHASEC;
case modeECM: return actionGONODE_RECVPHASEC;
case modeECMRETX:return actionGONODE_RECVECMRETRANSMIT;
}
}
// as soon as we get anything else ZERO the pTG->EchoProtect state
_fmemset(&pTG->EchoProtect, 0, sizeof(pTG->EchoProtect));
// reset this flag if we fail completely to train after a CTC. We may
// get an echo of our last command, go to NodeF, reject the echo and
// loop back to ECMRecvPhaseC. AT that point we need this flag STILL
// set so we try to recv LONG train. HOWEVER should the ECMRecvPhaseC
// fail for some OTHER reason than echo, this flag need to be reset
// here or it may stay alive confusing something much later.
pTG->ECM.fRecvdCTC = FALSE;
#ifdef RICOHAI
if(fEnteredHere)
{
if(pTG->T30.ifrCommand!=ifrNSS)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Recv-side AI protocol failed\r\n"));
fUsingOEMProt = 0;
return actionGONODE_R1;
}
}
#else
BG_CHK(!fEnteredHere);
#endif
switch(pTG->T30.ifrCommand)
{
// ifrNULL means T2 timeout, or loss of carrier, or no flags
// or no frame. ifrBAD means *only* bad frame(s) recvd.
case ifrNSS: // do same as for DCS
case ifrDCS: return PhaseGetTCF(pTG, pTG->T30.ifrCommand, fEnteredHere);
// ifrDCS is highly time-critical!!
case ifrBAD: SendCRP(pTG); // and fall thru to ifrNULL
case ifrNULL: break; // Loop again, until timeout
case ifrTIMEOUT: goto T2Timeout;
// ifrTIMEOUT means T2 timed out without flags...
case ifrDCN: if(fEopMcf)
{
ICommFailureCode(pTG, T30FAILR_SUCCESS);
return actionHANGUP_SUCCESS;
}
else
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> DCN following UNKNOWN on Receiver\r\n"));
ICommFailureCode(pTG, T30FAILR_UNKNOWN_DCN1);
return actionHANGUP;
}
default: return actionGONODE_RECVCMD;
}
}
T2Timeout:
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> T2 timed out\r\n"));
// restart PhaseB after T2 timeout IFF (a) EOM or PPS-EOM was recvd
// AND (b) If we are in ECM mode, the last response sent was n MCF
// This avoids us doing this after sending a CTR, RNR or PPR
// Ricoh's protocol conformance tester doesnt like this. This is
// Ricoh's bug numbbers B3-0142, 0143, 0144
if(pTG->T30.fReceivedEOM && (ProtReceivingECM(pTG) ? (pTG->ECM.ifrPrevResponse==ifrMCF) : TRUE) )
return actionGONODE_R1;
else if(ProtReceivingECM(pTG) && ++uFLoopCount<3)
goto NodeFprime;
#ifdef PRI
else // no EOM && (no ECM || 3 times)
{
if((pTG->Params.lpfnWhatNext(pTG, eventQUERYLOCALINT))==actionTRUE)
goto NodeF;
}
#endif
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got T2 timeout in GetCommand\r\n"));
ICommFailureCode(pTG, T30FAILR_T2);
return actionHANGUP;
}
ET30ACTION PhaseRecvCmd(PThrdGlbl pTG)
{
ET30ACTION action;
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseRecvCmd \n" );
if(pTG->T30.ifrCommand == ifrDCN)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got DCN in GetCommand\r\n"));
ICommFailureCode(pTG, T30FAILR_UNKNOWN_DCN2);
return actionHANGUP;
}
if( pTG->T30.ifrCommand==ifrDTC || pTG->T30.ifrCommand==ifrDCS ||
pTG->T30.ifrCommand==ifrDIS || pTG->T30.ifrCommand==ifrNSS)
{
switch(action = pTG->Params.lpfnWhatNext(pTG, eventRECVCMD, (WORD)pTG->T30.ifrCommand))
{
case actionGETTCF: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> MainBody: Wrong Way to GETTCF\r\n"));
ICommFailureCode(pTG, T30FAILR_BUG2);
BG_CHK(FALSE);
return actionERROR;
case actionGONODE_A: return actionGONODE_A;
case actionGONODE_D: return action;
case actionHANGUP: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionHANGUP from eventRECVCMD\r\n"));
return action;
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
}
if( pTG->T30.ifrCommand == ifrEOM || pTG->T30.ifrCommand == ifrPRI_EOM ||
pTG->T30.ifrCommand == ifrPPS_EOM || pTG->T30.ifrCommand == ifrPPS_PRI_EOM ||
pTG->T30.ifrCommand == ifrEOR_EOM || pTG->T30.ifrCommand == ifrEOR_PRI_EOM)
pTG->T30.fReceivedEOM = TRUE;
if(ProtReceivingECM(pTG))
return actionGONODE_VII;
else if(pTG->T30.ifrCommand >= ifrPRI_FIRST && pTG->T30.ifrCommand <= ifrPRI_LAST)
{
#ifdef PRI
return actionGONODE_RECVPRIQ;
#else
pTG->T30.ifrCommand = pTG->T30.ifrCommand-ifrPRI_MPS+ifrMPS;
// fall thru to GONODEIII
#endif
}
if(pTG->T30.ifrCommand >= ifrMPS && pTG->T30.ifrCommand <= ifrEOP)
return actionGONODE_III;
else
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got UNKNOWN in GetCommand\r\n"));
ICommFailureCode(pTG, T30FAILR_UNKNOWN_UNKNOWN2);
return actionHANGUP;
}
/***
else if(pTG->T30.ifrCommand >= ifrMPS && pTG->T30.ifrCommand <= ifrEOP)
return actionGONODE_III;
else if(pTG->T30.ifrCommand >= ifrECM_FIRST && ProtReceivingECM())
return actionGONODE_VII;
#ifdef PRI
else if(pTG->T30.ifrCommand >= ifrPRI_FIRST && pTG->T30.ifrCommand <= ifrPRI_LAST)
return actionGONODE_RECVPRIQ;
#endif
***/
}
ET30ACTION PhaseGetTCF(PThrdGlbl pTG, IFR ifr, BOOL fEnteredHere)
{
SWORD swRet;
IFR ifrDummy;
ET30ACTION action;
MyDebugPrint (pTG, LOG_ALL, "T30: PhaseGetTCF \n" );
if(pTG->T30.uRecvTCFMod == 0xFFFF) // uninitialised
{
ECHOPROTECT(0, 0);
CLEAR_MISSED_TCFS();
action = actionGONODE_F;
goto error;
}
#ifdef OEMNSF
else if(pTG->T30.uRecvTCFMod == 0xFF)
{
swRet = 0;
goto sendcfr;
}
#endif
swRet = GetTCF(pTG); // swRet = errs per 1000, +ve or 0 if we think its good
// -ve if we think its bad. -1111 if other error
// -1000 if too short
if(swRet < -1000)
{
BG_CHK(swRet == -1112 || swRet == -1113);
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Timed out/missed getting TCF\r\n"));
ECHOPROTECT(0, 0);
#ifndef MDDI
pTG->T30.uMissedTCFs++;
if (pTG->T30.uMissedTCFs >= MAX_MISSED_TCFS_BEFORE_FTT)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Sending FTT after %u missed TCFs\r\n",
(unsigned) pTG->T30.uMissedTCFs));
CLEAR_MISSED_TCFS();
swRet = -1000; // We pretend we got a too-short TCF.
}
else
#endif // !MDDI
{
action = actionGONODE_F;
goto error;
}
}
#ifdef OEMNSF
sendcfr:
#endif
// Here we can also signal the frames received before DCS!
// Were no longer in time-critical mode, so call all the
// callbacks we skipped. One for recording the received frames
// and one for handling teh received command, i.e. DCS.
// (the only options the protocol has is actionGETTCF or HANGUP)
ifrDummy = ifr;
action = pTG->Params.lpfnWhatNext(pTG, eventGOTFRAMES, pTG->T30.Nframes,
(ULONG_PTR)((LPLPFR)(pTG->T30.lpfs->rglpfr)), (ULONG_PTR)((LPIFR)(&ifrDummy)));
if(action == actionERROR)
goto error;
if(ifr != ifrDummy)
{
switch(ifrDummy)
{
case ifrNULL:
case ifrBAD:
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got ifrBAD from whatnext after recv TCF\r\n"));
ECHOPROTECT(0, 0);
CLEAR_MISSED_TCFS();
action = actionGONODE_F;
goto error;
default:
BG_CHK(FALSE);
break;
}
}
// BG_CHK(action == actionNULL);
// BG_CHK(ifrDummy == ifr);
// Now call teh callback to check the received TCF and
// return either FTT or CFR
switch(action = pTG->Params.lpfnWhatNext(pTG, eventGOTTCF,(WORD)swRet))
{
case actionSENDCFR:
(MyDebugPrint(pTG, LOG_ALL, "Good TCF\r\n"));
EnterPageCrit(); // start the CFR--PAGE critsection
SendCFR(pTG); // after sending CFR we are again in a race
ECHOPROTECT(ifrCFR, 0); // dunno recv mode yet
return actionGONODE_RECVPHASEC;
case actionSENDFTT:
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Bad TCF.....!!!\r\n"));
SendFTT(pTG);
ECHOPROTECT(ifrFTT, 0);
CLEAR_MISSED_TCFS();
return actionGONODE_F;
case actionERROR:
goto error;
default:
action = BadAction(pTG, action);
goto error;
}
error:
// missed TCF or no NSS. Did _not_ reply to TCF
// if we sent a reply, must _not_ come here
#ifdef RICOHAI
if(fEnteredHere)
return actionGONODE_R1;
#else
BG_CHK(!fEnteredHere);
#endif
return action;
}
ET30ACTION NonECMRecvPhaseC(PThrdGlbl pTG)
{
/******** Receiver Phase C. Fig A-7/T.30 (sheet 1) ********/
LPBUFFER lpbf;
ULONG lTotalLen=0;
USHORT uRet, uMod, uEnc;
DWORD tiffCompression;
LPSTR lpsTemp;
DWORD HiRes;
// There is a race between sending the CFR and sending out an
// +FRM=xx command, so we want to do it ASAP.
MyDebugPrint (pTG, LOG_ALL, "T30: NonECMRecvPhaseC \n" );
uEnc = ProtGetRecvEncoding(pTG);
BG_CHK(uEnc==MR_DATA || uEnc==MH_DATA);
if (uEnc == MR_DATA) {
tiffCompression = TIFF_COMPRESSION_MR;
}
else {
tiffCompression = TIFF_COMPRESSION_MH;
}
if (pTG->ProtInst.RecvParams.Fax.AwRes & (AWRES_mm080_077 | AWRES_200_200) ) {
HiRes = 1;
}
else {
HiRes = 0;
}
//
// do it once per RX
//
if ( !pTG->fTiffOpenOrCreated) {
//
// top 32bits of 64bit handle are guaranteed to be zero
//
pTG->Inst.hfile = PtrToUlong(TiffCreateW ( pTG->lpwFileName,
tiffCompression,
1728,
FILLORDER_LSB2MSB,
HiRes
) );
if (! (pTG->Inst.hfile)) {
lpsTemp = UnicodeStringToAnsiString(pTG->lpwFileName);
MyDebugPrint(pTG, LOG_ERR, "ERROR:Can't create tiff file %s compr=%d \n",
lpsTemp,
tiffCompression);
MemFree(lpsTemp);
return actionERROR;
}
pTG->fTiffOpenOrCreated = 1;
lpsTemp = UnicodeStringToAnsiString(pTG->lpwFileName);
MyDebugPrint(pTG, LOG_ALL, "Created tiff file %s compr=%d HiRes=%d \n",
lpsTemp, tiffCompression, HiRes);
MemFree(lpsTemp);
}
uMod = ProtGetRecvMod(pTG);
// in non-ECM mode, PhaseC is ALWAYS with short-train.
// Only TCF uses long-train
if(uMod >= V17_START) uMod |= ST_FLAG;
pTG->T30.sRecvBufSize = MY_BIGBUF_SIZE;
if((uRet = ModemRecvMode(pTG, pTG->Params.hModem, uMod, FALSE, PHASEC_TIMEOUT,
(IFR)((uEnc==MR_DATA) ? ifrPIX_MR : ifrPIX_MH))) != RECV_OK)
{
ExitPageCrit();
// reset Page ack. In case we miss page completely
ProtResetRecvPageAck(pTG);
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> NonECMPhC: RecvMode returned %d\r\n", uRet));
pTG->EchoProtect.modePrevRecv = modeNONECM;
// set global flag if we got WRONGMODE
pTG->EchoProtect.fGotWrongMode = (uRet==RECV_WRONGMODE);
// elim flush--does no good & wastes 10ms
// ModemFlush(pTG->Params.hModem);
CLEAR_MISSED_TCFS();
return actionGONODE_F;
// try to get 300bps command
// return actionHANGUP;
// goto error instead. No way to recover from here!
}
ExitPageCrit();
// as soon as we get good carrier ZERO the EchoProtect state
_fmemset(&pTG->EchoProtect, 0, sizeof(pTG->EchoProtect));
#ifdef IFAX
BroadcastMessage(pTG, IF_PSIFAX_DATAMODE, PSIFAX_RECV, (uMod & (~ST_FLAG)));
#endif
(MyDebugPrint(pTG, LOG_ALL, "RECEIVING Page.......\r\n"));
// reset Page ack
ProtResetRecvPageAck(pTG);
// Turn yielding on *after* entering receive mode safely!
FComCriticalNeg(pTG, FALSE);
/***
switch(action = pTG->Params.lpfnWhatNext(eventSTARTRECV))
{
case actionCONTINUE: break;
case actionDCN: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionDCN from eventSTARTRECV\r\n"));
return actionDCN; // goto NodeC;
case actionHANGUP: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionHANGUP from eventSTARTRECV\r\n"));
return actionHANGUP; // goto NodeB;
case actionERROR: return action; // goto PhaseLoop & exit
default: return BadAction(action);
}
***/
/***
BG_CHK((BOOL)Comm.dcb.fOutX == FALSE);
// Paranoia! Dont want to take ^Q/^S from modem to
// be XON/XOFF in the receive data phase!!
FComInFilterInit();
***/
// to mark start of Page
if(!PutRecvBuf(pTG, NULL, RECV_STARTPAGE))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Zero return from PutRecvBuf(start page)\r\n"));
return actionERROR;
}
// make it large, in case of large buffers & slow modems
#define READ_TIMEOUT 25000
lTotalLen = 0;
do
{
uRet=ModemRecvBuf(pTG, pTG->Params.hModem, FALSE, &lpbf, READ_TIMEOUT);
// lpbf==0 && uRet==RECV_OK just does nothing & loops back
if (uRet == RECV_EOF) {
// indicate that this is actually last recv_seq (we've got dle/etx already).
(MyDebugPrint(pTG, LOG_ALL, "RECV: NonECMRecvPhaseC fLastReadBlock = 1 \n"));
pTG->fLastReadBlock = 1;
}
if(lpbf)
{
lTotalLen += lpbf->wLengthData;
if(!PutRecvBuf(pTG, lpbf, RECV_SEQ))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Zero return from PutRecvBuf in page\r\n"));
return actionERROR;
}
lpbf = 0;
}
else {
if ( pTG->fLastReadBlock == 1) {
PutRecvBuf(pTG, lpbf, RECV_FLUSH);
}
}
}
while(uRet == RECV_OK);
if(uRet == RECV_EOF)
{
FComCriticalNeg(pTG, TRUE);
pTG->T30.fAtEndOfRecvPage = TRUE;
// call this *after* getting MPS/EOM/EOP
// PutRecvBuf(NULL, RECV_ENDPAGE); // to mark end of Page
}
else
{
// Timeout from ModemRecvBuf
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> DataRead Timeout or Error=%d\r\n", uRet));
// BG_CHK(FALSE);
ICommFailureCode(pTG, T30FAILR_MODEMRECV_PHASEC);
return actionERROR; // goto error;
}
(MyDebugPrint(pTG, LOG_ALL, "Page Recv Done.....len=(%ld, 0x%08x)\r\n", lTotalLen, lTotalLen));
ECHOPROTECT(0, 0);
CLEAR_MISSED_TCFS();
return actionGONODE_F; // goto NodeF; // get post-message command
}
ET30ACTION NonECMRecvPhaseD(PThrdGlbl pTG)
{
ET30ACTION action;
ET30ACTION ret;
MyDebugPrint (pTG, LOG_ALL, "T30: NonECMRecvPhaseD \n" );
/******** Receiver Phase D. Fig A-7/T.30 (sheet 2) ********/
// NodeIII:
/** Here the T30 flowchart is all BS. Fundamentally relying on
a +FCERROR response is not possible, so what we do here really
depends on what we've got. (According to the T30 flowchart we
return to NodeF after sending MCF/RTP/RTN, in all cases. What we
now know is that,
after MPS/MCF goto RecvPhaseC to get the next page
after EOM/MCF goto NodeR1 and send NSF etc all over again
***changed*** go back to NodeF, wait for T2 timeout
before sending DIS all over again.
after EOP/MCF goto NodeF, where GetResponse() will get
a DCN and we end up in NodeB (disconnect)
after xxx/RTN or xxx/RTP, I don't know what to do, but my guess
(looking at the Sender side of T30 flowchart) is:-
after MPS/RTx goto NodeF (sender goes to NodeD)
after EOP/RTx goto NodeF (sender goes to D or C)
after EOM/RTx goto NodeR1 (sender goes to NodeT)
***changed*** go back to NodeF, wait for T2 timeout
before sending DIS all over again.
****/
// only MPS/EOM/EOP commands come here
if(pTG->T30.fAtEndOfRecvPage) // so we won't call this twice
{
// This calls ET30ProtRecvPageAck so that WhatNext can choose
// MCF or RTN respectively. So it *must* be done before the
// call to WhatNext below
// RECV_ENDDOC if EOP or EOM
PutRecvBuf(pTG, NULL, ((pTG->T30.ifrCommand==ifrMPS) ? RECV_ENDPAGE : RECV_ENDDOC));
// ignore error/abort. We'll catch it soon enough
pTG->T30.fAtEndOfRecvPage = FALSE;
}
// returns MCF if page was OK, or RTN if it was bad
ret=actionGONODE_F;
ECHOPROTECT(0, 0);
CLEAR_MISSED_TCFS();
switch(action = pTG->Params.lpfnWhatNext(pTG, eventRECVPOSTPAGECMD,(WORD)pTG->T30.ifrCommand))
{
case actionSENDMCF: if(pTG->T30.ifrCommand==ifrMPS)
{
EnterPageCrit(); //start MPS--PAGE critsection
ECHOPROTECT(ifrMCF, modeNONECM);
ret=actionGONODE_RECVPHASEC;
}
else if(pTG->T30.ifrCommand==ifrEOP)
{
ECHOPROTECT(ifrMCF, 0);
CLEAR_MISSED_TCFS();
ret=actionNODEF_SUCCESS;
}
SendMCF(pTG);
break;
case actionSENDRTN: ECHOPROTECT(ifrRTN, 0);
SendRTN(pTG);
break;
case actionHANGUP: (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got actionHANGUP from eventRECVPOSTPAGECMD\r\n"));
ret=actionHANGUP;
break;
// case actionSENDRTP: SendRTP(pTG); break; // never send RTP
case actionERROR: ret=action; break; // goto PhaseLoop & exit
default: return BadAction(pTG, action);
}
return ret;
}
/***************************************************************************
Name : IFR GetCmdResp(BOOL fCommand)
Purpose : Implement the "Command Received" and "Response Received"
subroutines in the T.30 flowchart. The following macros are
defined:-
#define GetResponse() GetCmdResp(FALSE)
#define GetCommand() GetCmdResp(TRUE)
where the first form results in a faithful implem.
of the "Response Received" subroutine, the 2nd one
changes two things (i) the timeout for getting falgs goes from
T4 to T2, and (ii) if the routine times out on the very
first frame without getting flags (i.e. T2 times out)
it returns ifrTIMEOUT. This results in the "Command Recvd"
and the enclosing T2 loop being implemented in this
routine.
Upon receiving a set of frames, this routine assembles them
into the ET30FR structs pointed to by rglpfr, and if any
of them have a non-null FIF, or if >1 frame are received, it
calls (*Callbacks.Callbacks.lpWhatNext)().
Finally it returns the ifr of the last frame received, if
all frames were good, or ifrBAD if *all* frames were bad.
The algorithm it implements is very close to the "Response
Received" flowchart, minus the "Transmit DCN" box & below.
It returns ifrNULL or ifrBAD corresponding to return via
Node 2, (i.e. ifrNULL for timeout and other errors), ifrBAD
for bad frames, *iff* it can resync and get 200ms of silence
following these. ifrERROR for Node 1 (i.e. any error or timeout
after which we cannot resync or get silence for 200ms),
and ifrDCN for Node 3 (i.e. DCN was received).
<<<Node1 (ifrERROR is bad, so we dont use it. On error
we immediately return BAD or NULL or TIMEOUT and allow retry>>>
The "Optional Response" box is replaced by a "Not final frame".
A "start T4" box is assumed at "Enter" and a "Reset T4" box
is assumed after "process optional response"
It also modifies the flowchart in that *all* the frames are got
until a final frame, even if an FCS error frame is recvd. This
is partly because after an erroneous frame we have absolutely
no idea how long to wait until we either get silence or timeout
and hangup. ALso it may be more robust. The main routine
throws away the entire set of frames if one is bad.
The callback function is called before any return and it gets
a pointer to the desired return vakue, so it can modify this.
Arguments: whether it is called as "Command Received" or as
"Response Received"
Returns : ifrNULL -- timeout
ifrTIMEOUT -- T2 timed out before first flag
(returns this if and only if fCommand==TRUE)
ifrBAD -- all frames received were bad
ifrDCN -- DCN was received. The only valid action is "goto NodeB"
ifrXXXX -- last received frame
Calls :
Called By: GetResponse() GetCommand1() and GetCommand2() macros (ONLY!)
These are called only from ET30MainBody.
Comment : Watch out for timeouts > 65535 (max UWORD)
Revision Log
Date Name Description
-------- ----- ---------------------------------------------------------
06/15/92 arulm Created for first time (completely from scratch)
***************************************************************************/
IFR GetCmdResp(PThrdGlbl pTG, BOOL fCommand, USHORT ifrHint)
{
/*** we need to try for T4 to get a frame here. The way Class1 modems
works, as long as no flags are received, the AT+FRH=3 command will
time out and return ERROR. CONNECT is returned as soon as flags
are received. Some modems return <CR><LF>OK<CR><LF><CR><LF>CONNECT
which is still taken care of by out new ModemDialog which throws
away blank lines and looks for the expected answer i.e. "CONNECT"
in this case, in multi-line responses.
***/
BYTE bRecv[MAXFRAMESIZE];
BOOL fFinal, fNoFlags, fGotFIF, fGotBad;
IFR ifr, ifrLastGood;
USHORT uRet, uRecv, j;
ET30ACTION action;
LPLPFR lplpfr;
LPFR lpfrNext;
BOOL fResync=0;
ULONG ulTimeout;
BOOL fGotEofFromRecvMode=0;
// need to init these first
pTG->T30.Nframes=0, fFinal=FALSE, fNoFlags=TRUE, fGotFIF=FALSE, fGotBad=FALSE;
ifrLastGood=ifrNULL;
// figure out the timeout
if(fCommand)
ulTimeout = T2_TIMEOUT;
else
ulTimeout = T4_TIMEOUT;
// if we're sending DCS-TCF and waiting for CFR, we increase the timeout each time
// we fail, to avoid getting an infinite collision. This fixes bug#6847
if(ifrHint==ifrTCFresponse && pTG->T30.uTrainCount>1)
{
ulTimeout += TCFRESPONSE_TIMEOUT_SLACK;
(MyDebugPrint(pTG, LOG_ALL, "Get TCF response: traincount=%d timeout=%ld\r\n", pTG->T30.uTrainCount, ulTimeout));
}
lplpfr = pTG->T30.lpfs->rglpfr;
lpfrNext = (LPFR)(pTG->T30.lpfs->b);
pTG->T30.sRecvBufSize = 0;
uRet = ModemRecvMode(pTG, pTG->Params.hModem, V21_300, TRUE, ulTimeout, ifrHint);
if(uRet == RECV_TIMEOUT || uRet == RECV_ERROR)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> RecvMode failed=%d\r\n", uRet));
fResync = TRUE;
goto error;
}
else if(uRet == RECV_WRONGMODE)
{
// BG_CHK(FALSE); // just so we know who gives it!!
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got FCERROR from FRH=3\r\n"));
// treat like timeout
// fReSync = FALSE; // no need to resync
goto error;
}
else if(uRet == RECV_EOF)
{
// ModemRecvMode() returns EOF then return ifrEOF immediately. RecvMode
// returns RECV_EOF only if we pass it the ifrNODEFafterWRONGMODE hint
// and it senses silence (i.e. we saw a V21 echo but missed it). In
// this case we want to retry the high speed PIX recv again immediately
(MyDebugPrint(pTG, LOG_ERR, "WARNING: ECHO--Got EOF from V21 RecvMode\r\n"));
fGotEofFromRecvMode=TRUE;
goto error;
}
BG_CHK(uRet == RECV_OK);
for( ;!fFinal; )
{
if((((LPB)lpfrNext+sizeof(FR)+30-(LPB)pTG->T30.lpfs) > TOTALRECVDFRAMESPACE)
|| (pTG->T30.Nframes >= MAXRECVFRAMES))
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Out of space for received frames. Haaalp!!\r\n"));
pTG->T30.Nframes = 0;
// fGotBad = TRUE; // fGotBad = FALSE;
lplpfr = pTG->T30.lpfs->rglpfr;
lpfrNext = (LPFR)(pTG->T30.lpfs->b);
// BG_CHK(FALSE);
// fall through
}
uRet = ModemRecvMem(pTG, pTG->Params.hModem, bRecv, MAXFRAMESIZE, ulTimeout, &uRecv);
if(uRet == RECV_TIMEOUT)
{
fResync = TRUE;
goto error; // timeout in FRH=3
}
fNoFlags = FALSE;
if(uRet == RECV_EOF)
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Got NO CARRIER, but no final bit\r\n"));
BG_CHK(uRecv == 0);
goto error; // ignore good frames if we got any, because we
// must've missed the last (most important) one
// fFinal = TRUE; // pretend we got fFinal
// continue;
}
else if(uRet == RECV_ERROR)
{
fResync = TRUE;
goto error; // error in FRH=3
}
/** Sometimes modems give use ERROR even when teh frame is good.
Happens a lot from Though to PP144MT on CFR. So we ignore the
BADFRAME response, and check the recvd data for ourselves.
First stage--check the FF-03/13-FCF. If this is OK (i.e. got
a known FCF) and no FIF reqd and data length is correct
then accept it. But if the frame requires an FIF and we
got a BADFRAME response then don't accept it even if it
looks good--too dangerous.
2nd-stage(NYI)--keep a table of CRCs for each fixed frame & check
them ourselves.
**/
#ifdef MDDI // badframe handling
else if(uRet == RECV_BADFRAME)
goto badframe;
BG_CHK(uRet==RECV_OK);
#else
BG_CHK(uRet == RECV_OK || uRet==RECV_BADFRAME);
#endif
// faxT2log(("FRAME>>> \r\n"));
// ST_FRAMES(D_HexPrint(bRecv, uRecv));
/*** Got Valid OK Frame. Now analyse it. In general here we
want to be pretty lax in recognizing frames. Oftentimes
we have mucho garbage thrown in.
<CHANGE> Actually we're checking strictly now....
If we get a bad frame, we mark this in the lpfrfr[] array
and go on to try and get another frame.
<CHANGE> We've stopped saving bad frames
***/
// AT&T modem gives us frames w/o CRC, so we get just 1 byte here!!
// <CHNAGE> fixed that at driver level, so we always get 3 bytes here
// IFF it is a valid frame
if(uRecv < 3)
goto badframe;
if(bRecv[0] != 0xFF)
goto badframe;
if(bRecv[1] == 0x03)
fFinal = FALSE;
else if(bRecv[1] == 0x13)
{
fFinal = TRUE;
#ifdef CL0
// tell modem that recv is done
// but keep calling RecvMem until we get RECV_EOF
ModemEndRecv(pTG, pTG->Params.hModem);
#endif //CL0
}
else
goto badframe;
for(ifr=1; ifr<ifrMAX; ifr++)
{
if(rgFrameInfo[ifr].fInsertDISBit)
{
// Mask off the DIS bit
if(rgFrameInfo[ifr].bFCF1 != (BYTE)(bRecv[2] & 0xFE))
continue;
}
else
{
if(rgFrameInfo[ifr].bFCF1 != bRecv[2])
continue;
}
j=3;
if(rgFrameInfo[ifr].bFCF2)
{
// AT&T modem gives us frames w/o CRC, so we get just 1 byte frames!!
if(uRecv < 4)
goto badframe;
if((BYTE)(rgFrameInfo[ifr].bFCF2-1) != bRecv[3])
continue;
j++;
}
BG_CHK(j <= uRecv);
if(rgFrameInfo[ifr].wFIFLength == 0xFF) // var length FIF
{
// Var length frames
// Cant accept it if the modem thought they it was bad
// accept it IFF RECV_OK & FIFlen>0
if(uRet==RECV_OK && (j < uRecv))
fGotFIF = TRUE;
else
{
(MyDebugPrint(pTG, LOG_ERR, "Discarding Bad Frame: uRet=%d FIFlen=%d Reqd=Var\r\n", uRet, (uRecv-j)));
goto badframe;
}
}
else if(rgFrameInfo[ifr].wFIFLength) // fixed length FIF
{
// if frame length is exactly right then accept it
// else if modem said frame was ok and length is
// longer than expected then accept it too
if((j+rgFrameInfo[ifr].wFIFLength) == uRecv)
fGotFIF = TRUE;
else if(uRet==RECV_OK && (j+rgFrameInfo[ifr].wFIFLength < uRecv))
fGotFIF = TRUE;
else
{
(MyDebugPrint(pTG, LOG_ERR, "Discarding Bad Frame: uRet=%d FIFlen=%d Reqd=%d\r\n", uRet, (uRecv-j), rgFrameInfo[ifr].wFIFLength));
goto badframe;
}
}
else // no FIF reqd
{
if(j != uRecv)
{
(MyDebugPrint(pTG, LOG_ALL, "Weird frame(2)\r\n"));
BG_CHK(j < uRecv);
// see the BADFRAME comment block above on why we do this
if(uRet != RECV_OK)
goto badframe;
// accept it even if wrong length *iff* uRet==RECV_OK
// goto badframe;
// **additional** reason to do this that 2 bytes of extra
// CRC may be present. This happens because of the AT&T and
// NEC modems that doen't give CRC, so we do no CRC lopping-off
// in the modem driver. So accept it anyway here.
}
}
goto goodframe;
}
// fall through here when ifr == ifrMAX
BG_CHK(ifr==ifrMAX);
badframe:
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> IGNORING Bad Frame (Size=%d) -->\r\n", uRecv));
// D_HexPrint(bRecv, uRecv);
//Protocol Dump
DumpFrame(pTG, FALSE, 0, uRecv, bRecv);
fGotBad = TRUE; // remember we got a bad frame
// Don't want to save bad frames
/***
// mark frame as bad
ifr = ifrBAD;
lpfrNext->ifr = ifrBAD;
lpfrNext->cb = (BYTE)uRecv;
_fmemcpy(lpfrNext->fif, bRecv, uRecv);
lplpfr[pTG->T30.Nframes] = lpfrNext;
pTG->T30.Nframes++;
lpfrNext++; // skips over only the base part of frame
lpfrNext = ((LPBYTE)lpfrNext + uRecv); // skips over FIF too
***/
continue;
// loop back here for the next frame
goodframe:
// (MyDebugPrint(pTG, "FRAME RECEIVED: (%s) 0x%02x -->\r\n", (LPSTR)(rgFrameInfo[ifr].szName), bRecv[2]));
// ST_T30_2(D_HexPrint(bRecv, uRecv));
ifrLastGood = ifr; // save last good ifr
lpfrNext->ifr = ifr;
if(uRecv > j)
{
lpfrNext->cb = uRecv-j;
_fmemcpy(lpfrNext->fif, bRecv+j, uRecv-j);
//Protocol Dump
DumpFrame(pTG, FALSE, ifr, (USHORT)(uRecv-j), bRecv+j);
}
else
{
lpfrNext->cb = 0;
BG_CHK(uRecv == j);
//Protocol Dump
DumpFrame(pTG, FALSE, ifr, 0, 0);
}
lplpfr[pTG->T30.Nframes] = lpfrNext;
pTG->T30.Nframes++;
///////////// back to NSS-DCS form ////////////
if(ifr==ifrDCS && fFinal)
{
// Need to set receive speed, since bypass all callbacks
// now and go straight to AT+FRM=xx
pTG->T30.uRecvTCFMod = (((lpfrNext->fif[1])>>2) & 0x0F);
(MyDebugPrint(pTG, LOG_ALL, "cmdresp-DCS fx sp=%d\r\n", pTG->T30.uRecvTCFMod));
// fastexit:
BG_CHK(ifr==ifrDCS);
return ifr; // ifr of final frame
}
#ifdef OEMNSF
else if(ifr==ifrNSS && lpfnOEMGetBaudRate && fUsingOEMProt)
{
WORD wRet = lpfnOEMGetBaudRate(pTG, lpfrNext->fif, lpfrNext->cb, fFinal);
if(wRet)
{
BG_CHK(fFinal);
if(LOBYTE(wRet) == OEMNSF_GET_TCF)
{
(MyDebugPrint(pTG, LOG_ALL, "cmdresp-NSS fx sp=%d\r\n", pTG->T30.uRecvTCFMod));
pTG->T30.uRecvTCFMod = HIBYTE(wRet);
BG_CHK(ifr==ifrNSS);
return ifr;
}
else if(LOBYTE(wRet) == OEMNSF_NO_TCF)
{
(MyDebugPrint(pTG, LOG_ALL, "cmdresp-NSS fx sp=%d\r\n", pTG->T30.uRecvTCFMod));
pTG->T30.uRecvTCFMod = 0xFF;
BG_CHK(ifr==ifrNSS);
return ifr;
}
}
else if(fFinal)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> OEMGetBaudRate returned %d for FINAL frame. Using standard T30\r\n"));
fUsingOEMProt = 0;
return ifrNULL;
}
}
#endif
lpfrNext++; // skips over only the base part of frame
lpfrNext = (LPFR)((LPBYTE)lpfrNext + (uRecv-j)); // skips over FIF too
if(ifr == ifrCRP)
{
fGotBad = TRUE; // pretend we got a bad response (so the T30 chart says!)
// fResync = FALSE;
goto error; // exit, but get 200ms of silence
// caller will resend command
}
if(ifr == ifrDCN)
goto exit2; // exit. Don't call callback function!
// loop back here for the next frame
}
BG_CHK(fFinal); // only reason we ever drop through here
// let ifrDCN also come through here. No hurry, so we can report
// it to the protocol module. If we got *only* bad frames, here we'll
// return ifrBAD
//exit:
if(!pTG->T30.Nframes || // we got no good frames
(ifr != ifrLastGood)) // final frame was bad
{
(MyDebugPrint(pTG, LOG_ERR, "<<WARNING>> Got %d good frames. LastGood=%d Last=%d. Ignoring ALL\r\n",
pTG->T30.Nframes, ifrLastGood, ifr ));
// fResync = FALSE;
goto error;
}
// ifr of final frame
BG_CHK(ifr == ifrLastGood);
if(pTG->T30.Nframes>1 || fGotFIF)
{
action=pTG->Params.lpfnWhatNext(pTG, eventGOTFRAMES, pTG->T30.Nframes,
(ULONG_PTR)lplpfr, (ULONG_PTR)((LPIFR)&ifr));
BG_CHK(action==actionNULL || action==actionERROR);
}
exit2:
// ifr can be changed by callback function (e.g. if bad FIFs)
if(ifr==ifrDTC)
pTG->T30.fReceivedDTC = TRUE;
return ifr; // ifr of final frame
error:
// comes here on getting RECV_TIMEOUT, RECV_WRONGMODE, RECV_ERROR
// and ifrCRP, or on getting no Frames
// fReSync is set on RECV_TIMEOUT or RECV_ERROR
#ifdef DEBUG
if(pTG->T30.Nframes>0 && ifr!=ifrCRP && ifrLastGood!=ifrCRP)
{
(MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Got some good frames--Throwing away!! IFR=%d Last=%d\r\n", ifr, ifrLastGood));
// BG_CHK(FALSE);
}
#endif
if(fGotBad)
ifr = ifrBAD; // caller can send CRP if desired
else
ifr = ifrNULL; // caller can repeat command & try again
if(fCommand && fNoFlags) ifr = ifrTIMEOUT; // hook for "Command Recvd?"
// may not need this on TIMEOUT since AT-OK has already been
// called in ModemDialog. Probably need it on ERROR
if(uRet==RECV_ERROR)
{
BG_CHK(fResync);
ModemSync(pTG, pTG->Params.hModem, RESYNC_TIMEOUT2);
}
// eliminate Flush--serves no purpose
// else if(uRet==RECV_WRONGMODE)
// {
// ModemFlush(pTG->Params.hModem);
// }
// ModemRecvMode() returns EOF then return ifrEOF immediately. RecvMode
// returns RECV_EOF only if we pass it the ifrNODEFafterWRONGMODE hint
// and it senses silence (i.e. we saw a V21 echo but missed it). In
// this case we want to retry the high speed PIX recv again immediately
if(fGotEofFromRecvMode) ifr=ifrEOFfromRECVMODE;
BG_CHK(ifr==ifrBAD || ifr==ifrNULL || ifr==ifrTIMEOUT || ifr==ifrEOFfromRECVMODE);
return ifr;
/*-----------------------------------------------------------------
* Don't want to return ifrERROR ever. Nobody wants to deal with it.
* Don't try to get the 200ms of silence, because it causes us to
* get badly out of sync, i.e. after a few bad frames we don't
* resync and we lose the call. Just do a local-resync (AT-OK)
* and try to recv again.
* -----------------------------------------------------------------
*
* if( !ModemSync(pTG->Params.hModem, RESYNC_TIMEOUT) ||
* !ModemRecvSilence(pTG->Params.hModem, 20, RESYNC_TIMEOUT))
* {
* (MyDebugPrint(pTG, LOG_ERR, "<<ERROR>> Can't get silence after Response timeout--hanging up\r\n"));
* // ifr = ifrERROR;
* // caller must send DCN and hangup
* }
* action=pTG->Params.lpfnWhatNext(eventGOTFRAMES, pTG->T30.Nframes,
* (DWORD)lplpfr, (DWORD)((LPIFR)&ifr));
* BG_CHK(action == actionNULL);
*
* -----------------------------------------------------------------
* and don't want to regsiter garbage frame sets. Nobody does anything
* but retry on BAD/NULL/TIMEOUT returns anyway
*-----------------------------------------------------------------*/
}