1722 lines
64 KiB
C
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++;
|
|
}
|
|
}
|
|
|