windows-nt/Source/XPSP1/NT/printscan/fax/provider/t30/cl2and20/cl2and20.c
2020-09-26 16:20:57 +08:00

1722 lines
64 KiB
C

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
class20.c
Abstract:
This is the main source for Classes 2 and 2.0 fax-modem T.30 driver
Author:
Source base was originated by Win95 At Work Fax package.
RafaelL - July 1997 - port to NT
Revision History:
--*/
#include "prep.h"
#include "oemint.h"
#include "efaxcb.h"
#include "tiff.h"
#include "glbproto.h"
#include "t30gl.h"
#include "cl2spec.h"
WORD Class2CodeToBPS[16] =
{
/* V27_2400 0 */ 2400,
/* V27_4800 1 */ 4800,
/* V29_V17_7200 2 */ 7200,
/* V29_V17_9600 3 */ 9600,
/* V33_V17_12000 4 */ 12000,
/* V33_V17_14400 5 */ 14400,
0,
0,
0,
0,
0,
0,
0,
0,
0,
0
};
// Speeds for Class2HayesSyncSpeed to try
UWORD rguwClass2Speeds[] = {19200, 9600, 2400, 1200, 0};
BOOL ParseFPTS_SendAck(PThrdGlbl pTG)
{
BOOL fPageAck = FALSE;
UWORD count;
// Acknowledge that we sent the page
// Parse the FPTS response and see if the page is good or bad.
pTG->class2_commands.parameters[0][1] = 0;
pTG->class2_commands.parameters[1][1] = 0;
pTG->class2_commands.parameters[2][1] = 0;
if (pTG->ModemClass == MODEM_CLASS2) {
Class2Parse(pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
else {
Class20Parse(pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
for(count=0; count < pTG->class2_commands.comm_count; ++count)
{
switch (pTG->class2_commands.command[count])
{
case CL2DCE_FPTS:
if( pTG->class2_commands.parameters[count][0] == 1)
{
(MyDebugPrint (pTG, LOG_ALL, "Found good FPTS\n\r"));
fPageAck = TRUE;
// Exar hack!!!
// Exar modems give FPTS:1 for good pages and
// FPTS:1,2,0,0 for bad. So, look for the 1,2
// if this is an EXAR. Otherwise, 1 means good.
if (pTG->CurrentMFRSpec.bIsExar)
{
if( pTG->class2_commands.parameters[count][1] == 2)
{
(MyDebugPrint (pTG, LOG_ALL, "Nope - really Found bad FPTS\n\r"));
fPageAck = FALSE;
}
}
}
else if (pTG->class2_commands.parameters[count][0] == 2 )
{
(MyDebugPrint (pTG, LOG_ALL, "Found bad FPTS\n\r"));
fPageAck = FALSE;
}
break;
default:
break;
}
}
ICommSendPageAck(pTG, fPageAck);
return fPageAck;
}
USHORT Class2Dial(PThrdGlbl pTG, LPSTR lpszDial)
{
UWORD uwLen, uwDialStringLen;
ULONG ulTimeout;
SWORD swLen;
USHORT uRet;
BYTE bBuf[DIALBUFSIZE];
char chMod = pTG->NCUParams2.chDialModifier;
BG_CHK(lpszDial);
//// faxTlog(("Entering Class 2 Dial\r\n"));
// Send the predial command
if(pTG->lpCmdTab->szPreDial && (swLen=(SWORD)_fstrlen(pTG->lpCmdTab->szPreDial)))
{
if(Class2iModemDialog(pTG, (LPSTR)pTG->lpCmdTab->szPreDial, swLen,
10000L, TRUE, 0,
pTG->cbszCLASS2_OK, pTG->cbszCLASS2_ERROR, (C2PSTR)NULL) != 1)
{
(MyDebugPrint (pTG, LOG_ALL, "<<WARNING>> Error on User's PreDial string: %s\r\n", (LPSTR)pTG->lpCmdTab->szPreDial));
}
}
#if 1
// If the dial string already has a T or P prefix, we use that
// instead.
{
char c=0;
while((c=*lpszDial) && c==' ') *lpszDial++;
if (c=='t'|| c=='T' || c=='p'|| c=='P')
{
chMod = c;
lpszDial++;
while((c=*lpszDial) && c==' ') *lpszDial++;
}
}
#endif //1
uwLen = (UWORD)wsprintf(bBuf, pTG->cbszCLASS2_DIAL,
chMod, (LPSTR)lpszDial);
// Need to set an approriate timeout here. A minimum of 15secs is too short
// (experiment calling machines within a PABX), plus one has to give extra
// time for machines that pick up after 2 or 4 rings and also for long distance
// calls. I take a minumum of 30secs and add 3secs for each digits over 7
// (unless it's pulse dial in which case I add 8secs/digit).
// (I'm assuming that a long-distance call will take a minimum of 8 digits
// anywhere in ths world!). Fax machines I've tested wait about 30secs
// independent of everything.
uwDialStringLen = (UWORD)_fstrlen(lpszDial);
ulTimeout = 60000L;
if(uwDialStringLen > 7)
ulTimeout += ((chMod=='p' || chMod=='P')?8000:3000) * (uwDialStringLen - 7);
if(pTG->NCUParams2.AnswerTimeout != -1 &&
(((ULONG)pTG->NCUParams2.AnswerTimeout * 1000L) > ulTimeout))
ulTimeout = 1000L * (ULONG)pTG->NCUParams2.AnswerTimeout;
(MyDebugPrint (pTG, LOG_ALL, "\n\rDial String: %s size is %d\n\r",(LPB)bBuf,uwLen));
if(pTG->fAbort)
{
(MyDebugPrint (pTG, LOG_ALL, "Class2Dial--aborting\r\n"));
pTG->fAbort = FALSE;
Class2ModemHangup(pTG );
return CONNECT_ERROR;
}
ICommStatus(pTG, T30STATS_DIALING, 0, 0, 0);
uRet = Class2iModemDialog(pTG, (LPB)bBuf, uwLen,
ulTimeout, TRUE, 0,
pTG->cbszFCON, pTG->cbszCLASS2_BUSY, pTG->cbszCLASS2_NOANSWER,
pTG->cbszCLASS2_NODIALTONE, pTG->cbszCLASS2_ERROR, (NPSTR)NULL);
//// faxTlog(pTG, ("\n\rModemDial -- got %d response from Dialog\r\n", uRet));
// If it was "ERROR", try again - maybe the predial command screwed
// up somehow and left and ERROR hanging around?
if (uRet == 5)
{
uRet = Class2iModemDialog(pTG, (LPB)bBuf, uwLen,
ulTimeout, TRUE, 0,
pTG->cbszFCON, pTG->cbszCLASS2_BUSY, pTG->cbszCLASS2_NOANSWER,
pTG->cbszCLASS2_NODIALTONE, pTG->cbszCLASS2_ERROR, (NPSTR)NULL);
//// faxTlog(pTG, ("\n\rModemDial -- 2nd %d response from Dialog\r\n", uRet));
}
#if ((CONNECT_TIMEOUT==NCUDIAL_ERROR) && (CONNECT_BUSY==NCUDIAL_BUSY) && (CONNECT_NOANSWER==NCUDIAL_NOANSWER))
# if ((CONNECT_NODIALTONE==NCUDIAL_NODIALTONE) && (CONNECT_ERROR==NCUDIAL_MODEMERROR))
# pragma message("verified CONNECT defines")
# else
# error CONNECT defines not correct
# endif
#else
# error CONNECT defines not correct
#endif
switch(uRet)
{
case CONNECT_TIMEOUT:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got timeout\r\n"));
pTG->fFatalErrorWasSignaled = 1;
SignalStatusChange(pTG, FS_NO_ANSWER);
break;
case CONNECT_OK:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got CONNECT\r\n"));
break;
case CONNECT_BUSY:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got BUSY\r\n"));
pTG->fFatalErrorWasSignaled = 1;
SignalStatusChange(pTG, FS_BUSY);
break;
case CONNECT_NOANSWER:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got NOANSWER\r\n"));
pTG->fFatalErrorWasSignaled = 1;
SignalStatusChange(pTG, FS_NO_ANSWER);
break;
case CONNECT_NODIALTONE:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got NODIALTONE\r\n"));
pTG->fFatalErrorWasSignaled = 1;
SignalStatusChange(pTG, FS_NO_DIAL_TONE);
break;
case CONNECT_ERROR:
(MyDebugPrint (pTG, LOG_ALL, "Dial: Got an ERROR\r\n"));
pTG->fFatalErrorWasSignaled = 1;
SignalStatusChange(pTG, FS_NO_ANSWER);
break;
default:
uRet = CONNECT_ERROR;
BG_CHK(FALSE);
break;
}
BG_CHK(uRet>=0 && uRet<=5);
if(uRet != CONNECT_OK)
{
if(!Class2ModemHangup(pTG ))
return CONNECT_ERROR;
}
return uRet;
}
// ACTIVESLICE defined in msched.h
#define IDLESLICE 500
#ifdef WFWBG
# ifdef NOPRE
# define MySleep(x) \
BG_CHK(x==IDLESLICE); IFSetTimeSlice(IDLESLICE); \
ExitDLLCritSection(); IFDllSleep(0); EnterDLLCritSection(); \
IFSetTimeSlice(ACTIVESLICE);
# else
# define MySleep(x) \
BG_CHK(x==IDLESLICE); IFSetTimeSlice(IDLESLICE); \
IFDllSleep(0); IFSetTimeSlice(ACTIVESLICE);
# endif
#endif //WFWBG
#if defined(WIN32) && defined(THREAD)
# define MySleep(x) BG_CHK(x); Sleep(x);
#endif //WIN32 && THREAD
#ifdef IFK
# define MySleep(x) IFProcSleep(x)
#endif
// fImmediate indicates a manual answer
USHORT Class2Answer(PThrdGlbl pTG, BOOL fImmediate)
{
USHORT uRet;
SWORD swLen;
int i, PrevRings, Rings;
NPSTR sz;
ULONG ulWaitTime;
(MyDebugPrint (pTG, LOG_ALL, "Entering Class2Answer\r\n"));
// Default time we will wait after getting right number of rings
// to allow input buffer to flush
ulWaitTime = 500L;
if(!fImmediate && pTG->NCUParams2.RingsBeforeAnswer>3)
{
startTimeOut(pTG, &pTG->toAnswer, (6000L * (pTG->NCUParams2.RingsBeforeAnswer-3)));
for(PrevRings=0;;)
{
if(pTG->fAbort)
{
(MyDebugPrint (pTG, LOG_ALL, "Class2Answer--aborting\r\n"));
pTG->fAbort = FALSE;
Class2ModemHangup(pTG);
return CONNECT_ERROR;
}
// get S1 value & check it here.
// If we get an error don't look
// at return value. Just try again
if( (uRet=Class2iModemDialog(pTG, pTG->cbszQUERY_S1,
(UWORD) (strlen(pTG->cbszQUERY_S1) ),
2000L, TRUE, 0,
pTG->cbszCLASS2_OK, pTG->cbszZERO, pTG->cbszRING, pTG->cbszCLASS2_ERROR,
(C2PSTR)NULL) ) > 2)
goto xxxx;
// If the OK was matched, the answer to the ATS1 is in pTG->bLastReply2.
// If a string like "001" was matched, the answer is in
// pTG->bFoundReply.
if (uRet == 2) sz=pTG->bFoundReply;
else sz=pTG->bLastReply2;
for(i=0, Rings=0;
i<REPLYBUFSIZE && sz[i]; i++)
{
if(sz[i] >= '0' && sz[i] <= '9')
Rings = Rings*10 + (sz[i] - '0');
// ignore all non-numeric chars
}
(MyDebugPrint (pTG, LOG_ALL, "Got %d Rings. Want %d\r\n", Rings, pTG->NCUParams2.RingsBeforeAnswer));
// See if the number of rings we have is more than the
// Autoanswer number of rings. Also, for those
// modems that always return 000 to ATS1, we will answer
// right away. 000 cannot be correct, and we don't want
// to get stuck looking at 000.
if( Rings >= (pTG->NCUParams2.RingsBeforeAnswer-3))
break;
// If we saw a 000, estimate how much time we had to wait
// to get to the total number of rings we wanted. Figure
// we were already at three rings or so.
// If we counted the rings, we will still wait 1/2 second
// to allow any "OK" that might be coming from the modem to
// appear. We then will flush the queue.
if (Rings == 0)
{
if (pTG->NCUParams2.RingsBeforeAnswer >= 4)
ulWaitTime = (pTG->NCUParams2.RingsBeforeAnswer-3)*6000L;
else ulWaitTime = 500L;
break;
}
MySleep(IDLESLICE);
xxxx:
if((Rings < PrevRings) || !checkTimeOut(pTG, &pTG->toAnswer))
break;
PrevRings = Rings;
}
}
// We may still have an "OK" coming from the modem as a result of the
// ATS1 command. We need to wait a bit and flush it. In most cases we
// just wait 500 milliseconds. But, if we saw a 000 from the ATS1, we
// broke immediately out of the loop above, setting ulWaitTime to be
// the approximate wait we need before answering the phone.
startTimeOut(pTG, &pTG->toAnswer, ulWaitTime);
while (checkTimeOut(pTG, &pTG->toAnswer)) {}
FComFlush(pTG );
// Send the preanswer command
if(pTG->lpCmdTab->szPreAnswer &&
(swLen=(SWORD)_fstrlen(pTG->lpCmdTab->szPreAnswer)))
{
if(Class2iModemDialog(pTG, (LPSTR)pTG->lpCmdTab->szPreAnswer, swLen,
10000L, TRUE, 0,
pTG->cbszCLASS2_OK, pTG->cbszCLASS2_ERROR, (C2PSTR)NULL) != 1)
{
(MyDebugPrint (pTG, LOG_ALL, "<<WARNING>> Error on User's PreAnswer str: %s\r\n", (LPSTR)pTG->lpCmdTab->szPreAnswer));
}
}
#define ANSWER_TIMEOUT 35000
// Need to wait reasonably long, so that we don't give up too easily
// 7/25/95 JosephJ This used to be 15000. Changed to 35000 because
// MWAVE devices needed that timeout. Also, T.30 says that callee
// should try to negotiate for T1 seconds. T1 = 35 +/- 5s.
/*
* Send ATA command. The result will be stored in the global
* variable pTG->lpbResponseBuf2. We will parse that in the Class2Receive
* routine.
*/
ICommStatus(pTG, T30STATR_ANSWERING, 0, 0, 0);
// Look for ERROR return first and try again if it happened
if(( uRet = Class2iModemDialog(pTG, pTG->cbszATA, (UWORD) (strlen(pTG->cbszATA) ),
ANSWER_TIMEOUT, TRUE, 0, pTG->cbszCLASS2_OK, pTG->cbszCLASS2_ERROR,
pTG->cbszCLASS2_FHNG, (C2PSTR) NULL)) == 2)
{
(MyDebugPrint (pTG, LOG_ALL, "ATA returned ERROR on first try!\n\r"));
// dunno why we try this a 2nd time. But this time if we get ERROR
// dont exit. The Racal modem (bug#1982) gives ERROR followed by a
// good response! Cant ignore ERROR the first time otherwise we'll
// change the ATA--ERROR--ATA(repeat) behaviour which seems to be
// explicitly desired for some cases. However we dont take any
// action based on the return value of the 2nd try, so it's safe to
// ignore ERROR here. Worst case we take longer to timeout.
uRet = Class2iModemDialog(pTG, pTG->cbszATA, (UWORD) (strlen(pTG->cbszATA) ),
ANSWER_TIMEOUT, TRUE, 0, pTG->cbszCLASS2_OK,
pTG->cbszCLASS2_FHNG, (C2PSTR) NULL);
}
if (uRet != 1) // a '1' return indicates OK.
{
(MyDebugPrint (pTG, LOG_ALL, "Class2Answer: Can't get OK after ATA\r\n"));
// try to hangup and sync with modem. This should work
// even if phone is not really off hook
if(!Class2ModemHangup(pTG ))
{
// In WFW this can occur if an external modem has been
// powered down. so just drop thru & return ERROR
(MyDebugPrint (pTG, LOG_ALL, "Can't Hangup after ANSWERFAIL\r\n"));
}
return CONNECT_ERROR;
}
else
{
(MyDebugPrint (pTG, LOG_ALL, "Class2Answer: SUCCESS!!!\n\r"));
(MyDebugPrint (pTG, LOG_ALL, "ATA Received %s\r", (LPSTR)(&(pTG->lpbResponseBuf2))));
return CONNECT_OK;
}
}
#ifdef OBSOLETE_CODE
BOOL Class2ModemGoClass0(PThrdGlbl pTG)
{
int i;
for(i=0; i<3; i++)
{
if(!FComDirectSyncWriteFast(pTG, (LPB)pTG->cbszCLASS2_GO_CLASS0,
strlen(pTG->cbszCLASS2_GO_CLASS0) ))
break;
Class2TwiddleThumbs(pTG, 50L);
FComSetBaudRate(pTG, 2400);
FComFlush(pTG, );
if(Class2ModemSync(pTG, ) >= 0)
return TRUE;
}
(MyDebugPrint (pTG, LOG_ALL, "Cant go to 2400 Baud/Data mode\r\n"));
return FALSE;
}
#endif // OBSOLETE_CODE
SWORD Class2ModemSync(PThrdGlbl pTG)
{
// The command used here must be guaranteed to be harmless,
// side-effect free & non-dstructive. i.e. we can issue it
// at any point in command mode without chnageing the state
// of teh modem or disrupting anything.
// ATZ does not qualify. AT does, I think.....
SWORD ret_value;
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMSYNC: CALLING CLASS2HAYESSYNC.\r\n"));
ret_value = Class2HayesSyncSpeed(pTG, TRUE, pTG->cbszCLASS2_ATTEN,
(UWORD) (strlen(pTG->cbszCLASS2_ATTEN) ) );
(MyDebugPrint (pTG, LOG_ALL, "RETURN VALUE FROM HAYESSYNCSPEED: %d\r\n",ret_value));
return ret_value;
}
BOOL Class2ModemHangup(PThrdGlbl pTG)
{
(MyDebugPrint (pTG, LOG_ALL, "IN HANGUP.\r\n"));
if(Class2HayesSyncSpeed(pTG, TRUE, pTG->cbszCLASS2_HANGUP,
(UWORD) (strlen(pTG->cbszCLASS2_HANGUP) ) ) < 0)
{
(MyDebugPrint (pTG, LOG_ALL, "CLASS2 HANGUP: FAILED ONCE\r\n"));
FComDTR(pTG, FALSE); // Lower DTR on stubborn hangups in ModemHangup
MY_TWIDDLETHUMBS(1000); // pause 1 second
FComDTR(pTG, TRUE); // raise it again. Some modems return to cmd state
// only when this is raised again
if(Class2HayesSyncSpeed(pTG, TRUE, pTG->cbszCLASS2_HANGUP,
(UWORD) (strlen(pTG->cbszCLASS2_HANGUP) ) ) < 0)
{
(MyDebugPrint (pTG, LOG_ALL, "HANGUP: FAILED AGAIN\r\n"));
return FALSE;
}
}
(MyDebugPrint (pTG, LOG_ALL, "---->HANGUP: Completed Class2HayesSyncSpeed.\r\n"));
if(!iModemGoClass(pTG, 0)) return FALSE;
// Can also ignore this return value. Just for tidier cleanup
(MyDebugPrint (pTG, LOG_ALL, "HANGUP: Completed GoClass0.\r\n"));
// Bug1982: Racal modem, doesnt accept ATA. So we send it a PreAnswer
// command of ATS0=1, i.r. turning ON autoanswer. And we ignore the
// ERROR response it gives to the subsequent ATAs. It then answers
// 'automatically' and gives us all the right responses. On hangup
// however we need to send an ATS0=0 to turn auto-answer off. The
// ExitCommand is not sent at all in Class2 and in Class1 it is only
// sent on releasing the modem, not between calls. So send an S0=0
// after ATH0. If the modem doesnt like it we ignore the resp anyway.
Class2iModemDialog(pTG, pTG->cbszCLASS2_CALLDONE, (UWORD) (strlen(pTG->cbszCLASS2_CALLDONE) ) ,
LOCALCOMMAND_TIMEOUT, TRUE, 0, pTG->cbszCLASS2_OK, pTG->cbszCLASS2_ERROR,
(C2PSTR)NULL);
return TRUE;
}
void Class2GetRecvPageAck(PThrdGlbl pTG)
{
// Called by icomfile after a page has been received
// If the page was good, the argument will be TRUE. The
// global variable pTG->fRecvPageOK is set to the passed in
// value.
if (pTG->fPageIsBad) {
pTG->fRecvPageOK = 0;
}
else {
pTG->fRecvPageOK = 1;
}
// RSL: ICommGetRecvPageAck is irrelevant- pTG->fRecvPageOK is the real status from rx_thrd.
// pTG->fRecvPageOK = ICommGetRecvPageAck(pTG, TRUE); // sleep until we get it
(MyDebugPrint (pTG, LOG_ALL, "Class2RecvPageAck: Page check =%d\n\r", pTG->fRecvPageOK));
switch(pTG->fRecvPageOK)
{
case 0:
case 1: break;
default:
pTG->fRecvPageOK = 0;
Class2Abort(pTG, TRUE);
BG_CHK( FALSE);
break;
}
}
void Class2Abort(PThrdGlbl pTG, BOOL fEnable)
{
// Called when user invokes an abort - simply set the
// abort flag to be true. Various places in the code look
// for this flag and perform an abort. Flag was originally
// set to false in LibMain (called upon startup).
(MyDebugPrint (pTG, LOG_ALL, "Class2Abort: %s...\n\r", ((LPSTR)(fEnable?"Aborting":"Abort Done")) ));
pTG->fAbort = fEnable;
}
BOOL Class2ModemAbort(PThrdGlbl pTG)
{
// Try to abort modem in reasonable fashion - send the
// abort command and then send hangup. The abort command
// should hangup, but I am not convinced it always does!!!
// It should not hurt to hang up again (I hope).
// We'll use a long timeout here to let the abort take place.
if(!Class2iModemDialog(pTG, pTG->cbszCLASS2_ABORT, (UWORD) (strlen(pTG->cbszCLASS2_ABORT) ),
STARTSENDMODE_TIMEOUT, TRUE, 1, pTG->cbszCLASS2_OK,
pTG->cbszCLASS2_ERROR, (C2PSTR) NULL))
{
//Ignore failure
(MyDebugPrint (pTG, LOG_ALL, "Class2ModemAbort: FK Failed!\n\r"));
}
return Class2ModemHangup(pTG );
}
#ifdef OBSOLETE_CODE
BOOL Class2ModemGoClass2(PThrdGlbl pTG)
{
int i, speed;
for(i=0; i<3; i++)
{
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMGOCLASS2: Iteration %d\r\n",i));
if(!FComDirectSyncWriteFast(pTG, (LPB)pTG->cbszGO_CLASS2,
strlen(pTG->cbszGO_CLASS2) ))
goto error;
Class2TwiddleThumbs(50L);
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMGOCLASS2: SETTING BAUD RATE.\r\n"));
FComSetBaudRate(pTG, CLASS2_BAUDRATE);
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMGOCLASS2: FLUSHING.\r\n"));
FComFlush(pTG, );
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMGOCLASS2: SYNCING.\r\n"));
speed=Class2ModemSync(pTG, );
if(speed==0)
{
(MyDebugPrint (pTG, LOG_ALL, "CLASS2MODEMGOCLASS2: RETURNING TRUE.\r\n"));
return TRUE;
}
else if(speed==CLASS2_BAUDRATE)
{
(MyDebugPrint (pTG, LOG_ALL, "Go Class-2. Didn't sync the first time!!??\r\n"));
return TRUE;
}
}
error:
// error is already set to ERR_NO_RESPONSE inside Class2HayesSync()
(MyDebugPrint (pTG, LOG_ALL, "Cant go to 19.2 kBaud/Fax mode\r\n"));
return FALSE;
}
#endif // OBSOLETE_CODE
SWORD Class2HayesSyncSpeed(PThrdGlbl pTG, BOOL fTryCurrent, C2PSTR cbszCommand, UWORD uwLen)
{
/* Internal routine to synchronize with the modem's speed. Tries to
get a response from the modem by trying the speeds in rglSpeeds
in order (terminated by a 0). If fTryCurrent is nonzero, checks for
a response before trying to reset the speeds.
Returns the speed it found, 0 if they're in sync upon entry (only
checked if fTryCurrent!=0), or -1 if it couldn't sync.
*/
// short i;
/* This was initially set to -1, which has to be wrong, since you
do a BG_CHK to make sure fTryCurrent is TRUE, meaning you want
to try the current speed. In this case, -1 would be an invalid
index. KDB */
short ilWhich = 0;
BG_CHK( fTryCurrent);
// has to be TRUE, or we won't work with autobauding modems
/* I don't understand how this would ever get executed. KDB */
if (!fTryCurrent)
if(!FComSetBaudRate(pTG, rguwClass2Speeds[++ilWhich]))
return -1;
for(;;)
{
//// faxTlog(("Sync Trying: ilWhich=%d speed=%d\r\n", ilWhich, rguwClass2Speeds[ilWhich]));
if(Class2SyncModemDialog(pTG, cbszCommand, uwLen, pTG->cbszCLASS2_OK))
{
//// faxTlog(("Succeeded in Syncing at Speed = %d (il=%d)\r\n",rguwClass2Speeds[ilWhich], ilWhich));
return (ilWhich>=0 ? rguwClass2Speeds[ilWhich] : 0);
}
/* failed. try next speed. */
if (rguwClass2Speeds[++ilWhich]==0)
{
// Tried all speeds. No response
(MyDebugPrint (pTG, LOG_ALL, "Cannot Sync with Modem on Command %s\r\n", (LPSTR)cbszCommand));
return -1;
}
if(!FComSetBaudRate(pTG, rguwClass2Speeds[ilWhich]))
return -1;
}
}
USHORT Class2ModemRecvBuf(PThrdGlbl pTG, LPBUFFER far* lplpbf, USHORT uTimeout)
{
USHORT uRet;
*lplpbf = MyAllocBuf(pTG, MY_BIGBUF_SIZE);
// //// faxTlog(("In ModemRecvBuf lplpbf=%08lx uTimeout=%d\r\n", lplpbf, uTimeout));
uRet = Class2ModemRecvData(pTG, (*lplpbf)->lpbBegBuf,
(*lplpbf)->wLengthBuf, uTimeout, &((*lplpbf)->wLengthData));
if(!((*lplpbf)->wLengthData))
{
MyFreeBuf(pTG, *lplpbf);
*lplpbf = NULL;
}
else
{
// If necessary, bit-reverse...
if (pTG->CurrentMFRSpec.fSWFBOR && pTG->CurrentMFRSpec.iReceiveBOR==1)
{
(MyDebugPrint (pTG, LOG_ALL, "<<WARNING>> SWFBOR Enabled. bit-reversing data\r\n"));
cl2_flip_bytes( (*lplpbf)->lpbBegBuf, ((*lplpbf)->wLengthData));
}
}
/*
if(*lplpbf)
//// faxTlog(("Ex ModemRecvBuf lpbf=%08lx uCount=%d uRet=%d\r\n",
*lplpbf, (*lplpbf)->wLengthData, uRet));
else
//// faxTlog(("Ex ModemRecvBuf lpbf=null uRet=%d\r\n", uRet));
*/
return uRet;
}
USHORT Class2ModemRecvData(PThrdGlbl pTG, LPB lpb, USHORT cbMax, USHORT uTimeout, USHORT far* lpcbRecv)
{
SWORD swEOF;
BG_CHK(lpb && cbMax && lpcbRecv);
startTimeOut(pTG, &pTG->toRecv, uTimeout);
// 4th arg must be TRUE for Class2
*lpcbRecv = FComFilterReadBuf(pTG, lpb, cbMax, &pTG->toRecv, TRUE, &swEOF);
switch(swEOF)
{
case 1: // Class1 eof
case -1:// Class2 eof
return RECV_EOF;
case 0:
return RECV_OK;
default:
BG_CHK(FALSE);
// fall through
case -2:
return RECV_ERROR;
case -3:
return RECV_TIMEOUT;
}
}
BOOL Class2ModemSendMem(PThrdGlbl pTG, LPBYTE lpb, USHORT uCount)
{
BG_CHK(lpb);
(MyDebugPrint (pTG, LOG_ALL, "SENDING data....\r\n"));
if(!FComFilterAsyncWrite(pTG, lpb, uCount, FILTER_DLEZERO))
{
goto error;
}
return TRUE;
error:
(MyDebugPrint (pTG, LOG_ALL, "Failed on AsyncWrite in Class2ModemSendMem\n\r"));
FComOutFilterClose(pTG);
FComXon(pTG, FALSE);
return FALSE;
}
BOOL Class2ModemDrain(PThrdGlbl pTG)
{
//// faxTlog((pTG, "Entering Class2ModemDrain\r\n"));
if(!FComDrain(pTG, TRUE, TRUE))
return FALSE;
// Must turn XON/XOFF off immediately *after* drain, but before we
// send the next AT command, since Received frames have 0x13 or
// even 0x11 in them!! MUST GO AFTER the getOK ---- See BELOW!!!!
if(!Class2iModemDialog(pTG, NULL, 0,
STARTSENDMODE_TIMEOUT, FALSE, 0, pTG->cbszCLASS2_OK, (C2PSTR)NULL))
{
(MyDebugPrint (pTG, LOG_ALL, "Failed to terminate page with DLE-ETX\n\r"));
return FALSE;
}
// Must change FlowControl State *after* getting OK because in Windows
// this call takes 500 ms & resets chips, blows away data etc.
// So do this *only* when you *know* both RX & TX are empty.
// Turn off flow control.
FComXon(pTG, FALSE);
return TRUE;
}
void Class2TwiddleThumbs( ULONG ulTime)
{
MY_TWIDDLETHUMBS(ulTime);
}
LPSTR Class2_fstrstr(LPSTR sz1, LPSTR sz2)
{
int i, len1, len2;
len1 = _fstrlen(sz1);
len2 = _fstrlen(sz2);
for(i=0; i<=(len1-len2); i++)
{
if(_fmemcmp(sz1+i, sz2, len2) == 0)
return sz1+i;
}
return NULL;
}
UWORD Class2iModemDialog(PThrdGlbl pTG, LPSTR szSend, UWORD uwLen, ULONG ulTimeout,
BOOL fMultiLine, UWORD uwRepeatCount, ...)
{
/** Takes a command string, and it's lengt writes it out to the modem
and tries to get one of the allowed responses. It writes the command
out, waits ulTimeOut millisecs for a response. If it gets one of the
expected responses it returns immediately.
If it gets an unexpected/illegal response it tries (without any
waiting) for subsequent lines to the same response. When all the
lines (if > 1) of the response lines are exhausted, if none is among the
expected responses, it writes the command again and tries again,
until ulTimeout has expired. Note that if no response is received,
the command will be written just once.
The whole above thing will be repeated upto uwRepeatCount times
if uwRepeatCount is non-zero
<<<<<NOTE:::uwRepeatCount != 0 should not be used except for local sync>>>>>
It returns when (a) one of the specified responses is received or
(b) uwRepeatCount tries have failed (each having returned an
illegal response or having returned no response in ulTimeout
millsecs) or (c) the command write failed, in which
case it returns immediately.
It flushes the modem inque before each Command Write.
Returns 0 on failure and the 1 based index of the successful
response on success.
This can be used in the following way:-
for Local Dialogs (AT, AT+FTH=? etc), set ulTimeout to a lowish
value, of the order of the transmission time of the longest
possible (erroneous or correct) line of response plus the size
of the command. eg. at 1200baud we have about 120cps = about
10ms/char. Therefore a timeout of about 500ms is more than
adequate, except for really long command lines.
for Local Sync dialogs, used to sync up with the modem which may
be in an unsure state, use the same timeout, but also a repeat
count of 2 or 3.
for remote-driven dialogs, eg. AT+FRH=xx which returns a CONNECT
after the flags have been received, and which may incur a delay
before a response (ATDT is teh same. CONNECT is issued after a
long delay & anything the DTE sends will abort the process).
For these cases the caller should supply a long timeout and
probably a repeatcount of 1, so that the
routine will timeout after one try but go on issuing teh command
as long as an error repsonse is received.
For +FRH etc, the long timeout should be T1 or T2 in teh case of
CommandRecv and ResponseRecv respectively.
**/
BYTE bReply[REPLYBUFSIZE];
UWORD i, j, uwRet, uwWantCount;
SWORD swNumRead;
C2PSTR rgcbszWant[10];
va_list args;
LPTO lpto, lptoRead, lpto0;
BOOL fTimedOut;
// extract the (variable length) list of acceptable responses.
// each is a C2SZ, code based 2 byte ptr
va_start(args, uwRepeatCount); // Ansi Defintion
for(j=1; j<10; j++)
{
if((rgcbszWant[j] = va_arg(args, C2PSTR)) == NULL)
break;
}
uwWantCount = j-1;
va_end(args);
pTG->lpbResponseBuf2[0] = 0;
BG_CHK(uwWantCount>0);
#if 0 //// RSL
ST_DIA(
if(szSend)
{
faxT2log(("Dialog: Send (%s\r\n) len=%d WantCount=%d time=%ld rep=%d\r\n", (LPSTR)szSend,
uwLen, uwWantCount, ulTimeout, uwRepeatCount));
}
else
{
faxT2log(("Response: WantCount=%d time=%ld rep=%d\r\n",
uwWantCount, ulTimeout, uwRepeatCount));
}
for(j=1; j<=uwWantCount; j++)
faxT2log(pTG, ("Want %s\r\n", (LPSTR)(rgcbszWant[j])));
);
#endif
lpto = &pTG->toDialog;
lpto0 = &pTG->toZero;
// Try the dialog upto uwRepeatCount times
for(uwRet=0, i=0; i<=uwRepeatCount; i++)
{
startTimeOut(pTG, lpto, ulTimeout);
fTimedOut = FALSE;
do
{
if(fTimedOut)
{
// Need to send anychar to abort the previous command.
// We could recurse on this function, but the function
// uses static (lpto vars etc)!!
fTimedOut = FALSE;
BG_CHK(swNumRead == 0);
// use random 20ms timeout
FComDirectSyncWriteFast(pTG, "\r", 1);
startTimeOut(pTG, lpto0, 500);
swNumRead = FComFilterReadLine(pTG, bReply, REPLYBUFSIZE-1, lpto0);
(MyDebugPrint (pTG, LOG_ALL, "AnykeyAbort got<<%s>>\r\n", (LPSTR)bReply));
}
if(szSend)
{
// If a command is supplied, write it out, flushing input
// first to get rid of spurious input.
// FComInputFlush();
FComFlush(pTG, ); // Need to flush output too?
// Need to check that we are sending only ASCII or pre-stuffed data here
(MyDebugPrint (pTG, LOG_ALL, "WRITING THE FOLLOWING COMMAND: %s\r\n",(LPSTR)szSend));
if(!FComDirectSyncWriteFast(pTG, szSend, uwLen))
{
(MyDebugPrint (pTG, LOG_ALL, "Modem Dialog Write timed Out\r\n"));
uwRet = 0;
goto done;
// If Write fails, fail & return immediately.
// SetMyError() will already have been called.
}
}
// Try to get a response until timeout or bad response
pTG->bLastReply2[0] = 0;
for(lptoRead=lpto;;startTimeOut(pTG, lpto0, ulTimeout), lptoRead=lpto0)
{
// First, check for abort. If we are aborting,
// return failure from here. That will cause
// many commands to stop.
if (pTG->fAbort)
{
(MyDebugPrint (pTG, LOG_ALL, "ABORTING...\n\r"));
pTG->fAbort = FALSE;
uwRet = 0;
goto end;
}
// get a CR-LF terminated line
// for the first line use macro timeout, for multi-line
// responses use 0 timeout.
swNumRead = FComFilterReadLine(pTG, bReply, REPLYBUFSIZE-1, lptoRead);
if(swNumRead < 0)
swNumRead = (-swNumRead); // error-but lets see what we got anyway
else if(swNumRead == 0)
break; // Timeout -- restart dialog or exit
if(swNumRead == 2 && bReply[0] == '\r' && bReply[1] == '\n')
continue; // blank line -- throw away & get another
// COPIED THIS FROM DUP FUNCTION IN MODEM.C!!
// Fix Bug#1226. Elsa Microlink returns this garbage line in
// response to AT+FDIS?, followed by the real reply. Since
// we only look at the first line, we see only this garbage line
// and we never see the real reply
if(swNumRead==3 && bReply[0]==0x13 && bReply[1]=='\r' && bReply[2]=='\n')
continue;
for(bReply[REPLYBUFSIZE-1]=0, j=1; j<=uwWantCount; j++)
{
if(Class2_fstrstr(bReply, rgcbszWant[j]) != NULL)
{
uwRet = j;
// It matched!!!
// Save this reply. This is used when checking
// ATS1 responses
_fmemcpy(pTG->bFoundReply, bReply, REPLYBUFSIZE);
goto end;
}
}
if(!fMultiLine)
continue;
// go to ulTimeout check. i.e. *don't* set fTimedOut
// but don't exit either. Retry command and response until
// timeout
// We reach here it IFF we got a non blank reply, but it wasn't what
// we wanted. Squirrel teh first line away somewhere so that we can
// retrieve is later. We use this hack to get multi-line informational
// responses to things like +FTH=? Very important to ensure that
// blank-line replies don't get recorded here. (They may override
// the preceding line that we need!).
// Use the far pointer version
_fmemcpy(pTG->bLastReply2, bReply, REPLYBUFSIZE);
// In pTG->lpbResponseBuf2, all received lines are
// saved, not just the last one. This is used
// for multiline responses, like the response
// to a Class 2 ATD command
// pTG->lpbResponseBuf2[0] was initialized at the
// start of this routine.
// Ignore lines past the buffer size...
// no valid response would be that long anyway
if ( (_fstrlen((LPB)pTG->lpbResponseBuf2)+
_fstrlen((LPB)bReply) ) < RESPONSE_BUF_SIZE)
_fstrcat((LPB)pTG->lpbResponseBuf2, (LPB)bReply);
else
{
(MyDebugPrint (pTG, LOG_ALL, "<<ERROR>> ModemDialog:Response too long!\n\r"));
uwRet = 0;
goto end;
}
}
// we come here only on timeout.
fTimedOut = TRUE;
}
while(checkTimeOut(pTG, lpto));
}
end:
if(uwRet == 0)
{
(MyDebugPrint (pTG, LOG_ALL, "ModemDialog: (%s\r\n) --> (%d)(%s, etc) Failed\r\n",
(LPSTR)(szSend?szSend:"null"), uwWantCount,
(LPSTR)rgcbszWant[1]));
}
else
{
(MyDebugPrint (pTG, LOG_ALL, "ModemDialog: GOT IT %d (%s)\r\n", uwRet, (LPSTR)(rgcbszWant[uwRet])));
}
done:
(MyDebugPrint (pTG, LOG_ALL, "returning from ModemDialog....\n\r"));
return uwRet;
}
/* Converts the code for a speed to the speed in BPS */
// These are in the same order as the return values
// for DIS/DCS frames defined in the Class 2 spec in
// Table 8.4
/* Converts a DCS min-scan field code into millisecs */
// One array is for normal (100 dpi) res, the other for high (200 dpi) res...
// The ordering of the arraies is based on the values that
// are defined in filet30.h - THEY ARE NOT THE SAME AS THE VALUES
// RETURNED IN THE DCS FRAME!!!! This is inconsistent with baud rate
// but it is consistent with the Class 1 code...
BYTE msPerLineNormalRes[8] = { 20, 5, 10, 20, 40, 40, 10, 0 };
BYTE msPerLineHighRes[8] = { 20, 5, 10, 10, 40, 20, 5, 0 };
USHORT Class2MinScanToBytesPerLine(PThrdGlbl pTG, BYTE Minscan, BYTE Baud, BYTE Resolution)
{
USHORT uStuff;
BYTE ms;
uStuff = Class2CodeToBPS[Baud];
BG_CHK(uStuff);
if ( Resolution & AWRES_mm080_077)
ms = msPerLineHighRes[Minscan];
else ms = msPerLineNormalRes[Minscan];
uStuff /= 100; // StuffBytes = (BPS * ms)/8000
uStuff *= ms; // take care not to use longs
uStuff /= 80; // or overflow WORD or lose precision
uStuff += 1; // Rough fix for truncation problems
(MyDebugPrint (pTG, LOG_ALL, "Stuffing %d bytes\n\r", uStuff));
return uStuff;
}
// Convert the SEND_CAPS or SEND_PARAMS BC structure into values used by
// the +FDCC, +FDIS, and +FDT commands
void Class2SetDIS_DCSParams(PThrdGlbl pTG, BCTYPE bctype, LPUWORD Encoding, LPUWORD Resolution,
LPUWORD PageWidth, LPUWORD PageLength, LPSTR szID)
{
LPBC lpbc;
(MyDebugPrint (pTG, LOG_ALL, "Class2SetDIS_DCSParamas: entering. Type = %d\n\r", (USHORT)bctype));
if (bctype == SEND_PARAMS)
lpbc = (LPBC)&pTG->bcSendParams;
else
lpbc = (LPBC)&pTG->bcSendCaps;
// Set the ID
szID[0] = '\0';
BG_CHK(lpbc->wTotalSize >= sizeof(BC));
BG_CHK(lpbc->wTotalSize < sizeof(pTG->bcSendCaps));
BG_CHK(sizeof(pTG->bcSendCaps) == sizeof(pTG->bcSendParams));
// GetNumId(lpbc, szID, 21); else
//GetTextId(lpbc, szID, 21); // 20 + 1 for 0-term
if (pTG->LocalID) {
strcpy (szID, pTG->LocalID);
}
switch(lpbc->Fax.Encoding)
{
case MH_DATA:
*Encoding = 0;
break;
case MR_DATA:
case (MR_DATA | MH_DATA):
*Encoding = 1;
break;
case MMR_DATA:
case (MMR_DATA | MH_DATA):
case (MMR_DATA | MR_DATA):
case (MMR_DATA | MR_DATA | MH_DATA):
*Encoding = 3;
break;
default:
(MyDebugPrint (pTG, LOG_ALL, "<<ERROR>> Bad Encoding type %x\n\r",lpbc->Fax.Encoding));
break;
}
if( (lpbc->Fax.AwRes) & AWRES_mm080_077) *Resolution = 1;
else if( (lpbc->Fax.AwRes) & AWRES_mm080_038) *Resolution = 0;
else
{
(MyDebugPrint (pTG, LOG_ALL, "<<ERROR>> Bad Resolution type %x\n\r", lpbc->Fax.AwRes));
}
switch(lpbc->Fax.PageWidth & 0x3)
{
case WIDTH_A4: // 1728 pixels
*PageWidth = 0;
break;
case WIDTH_B4: // 2048 pixels
*PageWidth = 1;
break;
case WIDTH_A3: // 2432 pixels
*PageWidth = 2;
break;
default:
(MyDebugPrint (pTG, LOG_ALL, "<<ERROR>> Bad PageWidth type %x\n\r", lpbc->Fax.PageWidth));
break;
}
switch(lpbc->Fax.PageLength)
{
case LENGTH_A4:
*PageLength = 0;
break;
case LENGTH_B4:
*PageLength = 1;
break;
case LENGTH_UNLIMITED:
*PageLength = 2;
break;
default:
(MyDebugPrint (pTG, LOG_ALL, "<<ERROR>> Bad PageLength type %x\n\r", lpbc->Fax.PageLength));
break;
}
}
BOOL Class2ResponseAction(PThrdGlbl pTG, LPPCB lpPcb)
{
USHORT count, i;
BOOL fFoundDIS_DCS;
fFoundDIS_DCS = FALSE;
(MyDebugPrint (pTG, LOG_ALL, "Class2ResponseAction: entering\n\r"));
_fmemset(lpPcb, 0, sizeof(PCB));
if (pTG->ModemClass == MODEM_CLASS2) {
Class2Parse( pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
else {
Class20Parse( pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
(MyDebugPrint (pTG, LOG_ALL, "Number of commands is %d\n\r",pTG->class2_commands.comm_count));
for(count=0; count < pTG->class2_commands.comm_count; ++count)
{
// (MyDebugPrint (pTG, LOG_ALL, "Loading PCB for command %d\n\r",
// pTG->class2_commands.command[count]));
switch (pTG->class2_commands.command[count])
{
case CL2DCE_FDIS:
case CL2DCE_FDCS:
(MyDebugPrint (pTG, LOG_ALL, "Found DCS or DIS \n\r"));
fFoundDIS_DCS = TRUE;
// Assign resolution.
if( pTG->class2_commands.parameters[count][0] == 0)
lpPcb->Resolution = AWRES_mm080_038;
else if (pTG->class2_commands.parameters[count][0] == 1 )
{
// Resolution when reported by a DIS frame indicates
// it accepts either fine or normal. When reported
// in a DCS, it means the negotiated value is FINE.
if (pTG->class2_commands.command[count] ==
CL2DCE_FDIS)
lpPcb->Resolution = AWRES_mm080_038 | AWRES_mm080_077;
else
lpPcb->Resolution = AWRES_mm080_077;
}
else
{
(MyDebugPrint (pTG, LOG_ALL, "FAILED TO ASSIGN RESOLUTION.\r\n"));
return FALSE;
}
// Assign encoding scheme.
if( pTG->class2_commands.parameters[count][4] == 0)
{
lpPcb->Encoding = MH_DATA;
}
else if ((pTG->class2_commands.parameters[count][4] == 1) ||
(pTG->class2_commands.parameters[count][4] == 2) ||
(pTG->class2_commands.parameters[count][4] == 3) )
{
lpPcb->Encoding = MH_DATA | MR_DATA;
}
else
{
(MyDebugPrint (pTG, LOG_ALL, "FAILED TO ASSIGN ENCODING.\r\n"));
return FALSE;
}
// Assign page width.
if( pTG->class2_commands.parameters[count][2] == 0)
lpPcb->PageWidth = WIDTH_A4;
else if (pTG->class2_commands.parameters[count][2] == 1)
lpPcb->PageWidth = WIDTH_B4;
else if (pTG->class2_commands.parameters[count][2] == 2)
lpPcb->PageWidth = WIDTH_A3;
// We don't support 3 and 4 (A5, A6)
// but we'll still allow them and map them to A4
// This is for Elliot bug #1252 - it should have
// no deleterious effect, since this width field
// is not used for anything at the point in where
// bug 1252 occurs. FrankFi
else if (pTG->class2_commands.parameters[count][2] == 3)
lpPcb->PageWidth = WIDTH_A4;
else if (pTG->class2_commands.parameters[count][2] == 4)
lpPcb->PageWidth = WIDTH_A4;
else
{
(MyDebugPrint (pTG, LOG_ALL, "FAILED TO ASSIGN WIDTH.\r\n"));
return FALSE;
}
// Assign page length.
if( pTG->class2_commands.parameters[count][3] == 0)
lpPcb->PageLength = LENGTH_A4;
else if (pTG->class2_commands.parameters[count][3] == 1)
lpPcb->PageLength = LENGTH_B4;
else if (pTG->class2_commands.parameters[count][3] == 2)
lpPcb->PageLength = LENGTH_UNLIMITED;
else
{
(MyDebugPrint (pTG, LOG_ALL, "INVALID LENGTH.\r\n"));
// assume it is unlimited! Some modems
// mess up on length.
lpPcb->PageLength = LENGTH_UNLIMITED;
}
// Assign baud rate
// For now, we will use the raw numbers returned in the
// DCS command. Dangerous - should fix later!
// These numbers will be tied to the baud rate array in
// the routine that figures out zero byte stuffing from
// the scan line and baud rate.
// Fixed the Hack--added a Baud field
lpPcb->Baud = pTG->class2_commands.parameters[count][1];
// Assign minimum scan time - the first number
// in the MINSCAN_num_num_num constant
// refers to scan time in ms for 100dpi, the
// second for 200dpi, and the last for 400dpi
// Class 2 does not use the 400dpi number,
// but these variables are shared with Class 1
if( pTG->class2_commands.parameters[count][7] == 0)
lpPcb->MinScan = MINSCAN_0_0_0;
else if (pTG->class2_commands.parameters[count][7] == 1)
lpPcb->MinScan = MINSCAN_5_5_5;
else if (pTG->class2_commands.parameters[count][7] == 2)
lpPcb->MinScan = MINSCAN_10_5_5;
else if (pTG->class2_commands.parameters[count][7] == 3)
lpPcb->MinScan = MINSCAN_10_10_10;
else if (pTG->class2_commands.parameters[count][7] == 4)
lpPcb->MinScan = MINSCAN_20_10_10;
else if (pTG->class2_commands.parameters[count][7] == 5)
lpPcb->MinScan = MINSCAN_20_20_20;
else if (pTG->class2_commands.parameters[count][7] == 6)
lpPcb->MinScan = MINSCAN_40_20_20;
else if (pTG->class2_commands.parameters[count][7] == 7)
lpPcb->MinScan = MINSCAN_40_40_40;
break;
case CL2DCE_FCSI:
case CL2DCE_FTSI:
for( i = 0; (lpPcb->szID[i] =
pTG->class2_commands.parameters[count][i]) != '\0'; ++i)
;
// prepare CSID for logging by FaxSvc
pTG->RemoteID = AnsiStringToUnicodeString(lpPcb->szID);
if (pTG->RemoteID) {
pTG->fRemoteIdAvail = 1;
}
break;
default:
// (MyDebugPrint (pTG, LOG_ALL, "Class2ResponseAction: Unknown token.\r\n"));
break;
}
}
return fFoundDIS_DCS;
}
USHORT Class2EndPageResponseAction(PThrdGlbl pTG)
{
USHORT csi_count = 0,
count;
if (pTG->ModemClass == MODEM_CLASS2) {
Class2Parse(pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
else {
Class20Parse(pTG, &pTG->class2_commands, pTG->lpbResponseBuf2 );
}
for(count=0; count < pTG->class2_commands.comm_count; ++count)
{
if (pTG->class2_commands.command[count] == CL2DCE_FET )
{
if (pTG->class2_commands.parameters[count][0] == 0)
{
(MyDebugPrint (pTG, LOG_ALL, "More pages coming\n\r"));
return MORE_PAGES;
}
else
{
(MyDebugPrint (pTG, LOG_ALL, "No more pages coming\n\r"));
return NO_MORE_PAGES;
}
}
}
return FALSE;
}
void Class2InitBC(PThrdGlbl pTG, LPBC lpbc, USHORT uSize, BCTYPE bctype)
{
_fmemset(lpbc, 0, uSize);
lpbc->bctype = bctype;
lpbc->wBCVer = VER_AWFXPROT100;
lpbc->wBCSize = sizeof(BC);
lpbc->wTotalSize = sizeof(BC);
/********* This is incorrect ****************************
lpbc->Std.GroupNum = GROUPNUM_STD;
lpbc->Std.GroupLength = sizeof(BCSTD);
// set Protocol Ver etc (everything else) to 00
// set all IDs to 0 also. Fax stuff set later
********* This is incorrect ****************************/
/**
lpbc->Fax.AwRes = (AWRES_mm080_038 | AWRES_mm080_077 | AWRES_200_200 | AWRES_300_300);
lpbc->Fax.Encoding = MH_DATA; // ENCODE_ALL eventually!
lpbc->Fax.PageWidth = WIDTH_A4;
lpbc->Fax.PageLength = LENGTH_UNLIMITED;
lpbc->Fax.MinScan = MINSCAN_0_0_0;
**/
}
void Class2PCBtoBC(PThrdGlbl pTG, LPBC lpbc, USHORT uMaxSize, LPPCB lppcb)
{
USHORT uLen;
lpbc->Fax.AwRes = lppcb->Resolution;
lpbc->Fax.Encoding = lppcb->Encoding;
lpbc->Fax.PageWidth = lppcb->PageWidth;
lpbc->Fax.PageLength = lppcb->PageLength;
// lpbc->Fax.MinScan = lppcb->MinScan;
BG_CHK(lpbc->wTotalSize >= sizeof(BC));
BG_CHK(lpbc->wTotalSize < uMaxSize);
if(uLen = (USHORT)_fstrlen(lppcb->szID))
{
PutTextId( lpbc, uMaxSize, lppcb->szID, uLen, TEXTCODE_ASCII);
// PutNumId(lpbc, szID, uLen);
}
}
BOOL Class2GetBC(PThrdGlbl pTG, BCTYPE bctype)
{
USHORT uLen;
LPBC lpbc;
if(bctype == BC_NONE)
{
(MyDebugPrint (pTG, LOG_ALL, "Class2GetBC: entering, type = BC_NONE\n\r", bctype));
Class2InitBC(pTG, (LPBC)&pTG->bcSendCaps, sizeof(pTG->bcSendCaps), SEND_CAPS);
pTG->bcSendCaps.Fax.AwRes = (AWRES_mm080_038 | AWRES_mm080_077);
pTG->bcSendCaps.Fax.Encoding = MH_DATA;
pTG->bcSendCaps.Fax.PageWidth = WIDTH_A4;
pTG->bcSendCaps.Fax.PageLength = LENGTH_UNLIMITED;
return TRUE;
}
if(!(lpbc = ICommGetBC(pTG, bctype, TRUE)))
{
BG_CHK(FALSE);
return FALSE;
}
(MyDebugPrint (pTG, LOG_ALL, "Class2GetBC: entering, type = %d\n\r", bctype));
(MyDebugPrint (pTG, LOG_ALL, "Some params: encoding = %d, res = %d\n\r", lpbc->Fax.Encoding, lpbc->Fax.AwRes));
// Depending on the type, pick the correct global BC structure
BG_CHK(lpbc->wTotalSize >= sizeof(BC));
if (bctype == SEND_CAPS)
{
BG_CHK(lpbc->wTotalSize < sizeof(pTG->bcSendCaps));
uLen = min(sizeof(pTG->bcSendCaps), lpbc->wTotalSize);
_fmemcpy(&pTG->bcSendCaps, lpbc, uLen);
return TRUE;
}
else if (bctype == SEND_PARAMS)
{
BG_CHK(lpbc->wTotalSize < sizeof(pTG->bcSendParams));
uLen = min(sizeof(pTG->bcSendParams), lpbc->wTotalSize);
_fmemcpy(&pTG->bcSendParams, lpbc, uLen);
return TRUE;
}
else
return FALSE;
}
BOOL Class2NCUSet(PThrdGlbl pTG, LPNCUPARAMS NCUParams2)
{
BG_CHK(NCUParams2);
(MyDebugPrint (pTG, LOG_ALL, "In Class2NCUSet\r\n"));
// Copy params into our local pTG->NCUParams2 struct
pTG->NCUParams2 = *NCUParams2;
return TRUE;
}
BOOL Class2SetProtParams(PThrdGlbl pTG, LPPROTPARAMS lp)
{
pTG->ProtParams2 = *lp;
(MyDebugPrint (pTG, LOG_ALL, "Set Class2ProtParams: fV17Send=%d fV17Recv=%d uMinScan=%d\r\n",
"HighestSend=%d LowestSend=%d\r\n",
pTG->ProtParams2.fEnableV17Send, pTG->ProtParams2.fEnableV17Recv,
pTG->ProtParams2.uMinScan, pTG->ProtParams2.HighestSendSpeed,
pTG->ProtParams2.LowestSendSpeed));
return TRUE;
}
void iNCUParamsReset(PThrdGlbl pTG)
{
_fmemset(&pTG->NCUParams2, 0, sizeof(pTG->NCUParams2));
pTG->lpCmdTab = 0;
pTG->NCUParams2.uSize = sizeof(pTG->NCUParams2);
// These are used to set S regs etc.
// -1 means leave modem at default
pTG->NCUParams2.DialtoneTimeout = -1;
pTG->NCUParams2.DialPauseTime = pTG->NCUParams2.FlashTime = -1;
// pTG->NCUParams2.PulseMakeBreak = pTG->NCUParams2.DialBlind = -1;
pTG->NCUParams2.DialBlind = -1;
pTG->NCUParams2.SpeakerVolume = pTG->NCUParams2.SpeakerControl = -1;
pTG->NCUParams2.SpeakerRing = -1;
// should be used in answer
pTG->NCUParams2.RingsBeforeAnswer = 0;
// should be used in Dial
pTG->NCUParams2.AnswerTimeout = 60;
// used in Dial
pTG->NCUParams2.chDialModifier = 'T';
}
BYTE rgbFlip256[256] = {
0x0, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,
0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,
0x8, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,
0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,
0x4, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,
0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,
0xc, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,
0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,
0x2, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,
0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,
0xa, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,
0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,
0x6, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,
0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,
0xe, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,
0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,
0x1, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,
0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,
0x9, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,
0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,
0x5, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,
0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,
0xd, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,
0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,
0x3, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,
0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,
0xb, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,
0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,
0x7, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,
0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,
0xf, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,
0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff
};
#define FLIP(index) (lpb[(index)] = rgbFlip256[lpb[(index)]])
void cl2_flip_bytes(LPB lpb, DWORD dw)
{
while (dw>8)
{
FLIP(0); FLIP(1); FLIP(2); FLIP(3);
FLIP(4); FLIP(5); FLIP(6); FLIP(7);
dw-=8;
lpb+=8;
}
while(dw)
{
FLIP(0);
dw--;
lpb++;
}
}