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

499 lines
16 KiB
C

/***************************************************************************
Name : NEGOT.C
Comment : Capability handling and negotiation
Revision Log
Date Name Description
-------- ----- ---------------------------------------------------------
***************************************************************************/
#include "prep.h"
#include "protocol.h"
#include "glbproto.h"
BOOL glLoRes=0; // Simulate LoRes from Remote when TX
#define faxTlog(m) DEBUGMSG(ZONE_NEGOT, m);
# define INST_ENCODING pTG->Inst.awfi.Encoding
# define INST_VSECURITY pTG->Inst.awfi.vSecurity
# define INST_RESOLUTION pTG->Inst.awfi.AwRes
# define INST_PAGEWIDTH pTG->Inst.awfi.PageWidth
# define INST_PAGELENGTH pTG->Inst.awfi.PageLength
//////// Move these hardcoded values to an INI file ////////
#define CAPS_WIDTH WIDTH_A4
#define ENCODE_CAPS (MH_DATA | MR_DATA ) // RSL | MMR_DATA)
// Current suppored linearized verson +++ (change to vMSG_IFAX100 when we
// have enabled Linearized published images).
#define vMSG_WIN95 vMSG_IFAX100 // vMSG_SNOW
//# define CAPS_RES 0
//#if 0
#ifdef DPI_400
# define CAPS_RES (AWRES_mm080_038 | AWRES_mm080_077 | AWRES_200_200 | AWRES_300_300 | AWRES_mm080_154 | AWRES_160_154 | AWRES_400_400)
#else
# define CAPS_RES (AWRES_mm080_038 | AWRES_mm080_077 | AWRES_200_200 | AWRES_300_300)
#endif
//#endif
/********* These are the Ricoh thresholds--they're too simplistic *****
USHORT MaxBadLines[4][4] =
{
{110, 220, 330, 440 }, // CheckLevel=1 10%
{ 82, 165, 248, 330 }, // CheckLevel=2 7.5%
{ 55, 110, 165, 220 }, // CheckLevel=3 5%
{ 27, 55, 83, 110 } // CheckLevel=4 2.5%
};
USHORT MaxConsecBad[4][4] =
{
{ 6, 12, 18, 24 }, // CheckLevel=1
{ 5, 10, 15, 20 }, // CheckLevel=2
{ 4, 8, 12, 16 }, // CheckLevel=3
{ 3, 6, 9, 12 } // CheckLevel=4
};
*****************/
////////////////////////////////////////////////////////////////
// Don't delete this -- these were my thresholds _before_ I talked to Ricoh
////////
// higher thresholds, tapering off, because
// at higher resolutions we want cleaner copy.
// USHORT MaxBadLines[4] = { 33, 66, 84, 99 }; // 3.5, 3, 2.5, 2.25 %bad
// USHORT MaxConsecBad[4] = { 5, 9, 12, 15 }; // 2/3rd of a 10pt char is 9,18,27,36 lines
////////////////////////////////////////////////////////////////
USHORT MaxBadLines[4][4] =
{
{ 77, 132, 165, 198 }, // CheckLevel=1 7, 6, 5, 4.5% bad
{ 58, 99, 124, 149 }, // CheckLevel=2 5.25, 4.5, 3.75, 3.375% bad
{ 39, 66, 83, 99 }, // CheckLevel=3 3.5, 3, 2.5, 2.25% bad
{ 19, 33, 41, 50 } // CheckLevel=4 1.75, 1.5, 1.25, 1.125% bad
};
USHORT MaxConsecBad[4][4] =
{
{ 7, 13, 18, 23 }, // CheckLevel=1
{ 6, 11, 15, 19 }, // CheckLevel=2
{ 5, 9, 12, 15 }, // CheckLevel=3
{ 4, 7, 9, 11 } // CheckLevel=4
};
// lBad = DWORD with max. consecutive badlines in low word
// and total number of bad lines in high word.
// res==resolution (using ENCODE_ values)
// i = vertical res index into table above (0=100, 1=200, 2=300 3=400)
/** Widths in pixels must be exactly correct for MH decoding to work.
the width above are for NORMAL, FINE, 200dpi and SUPER resolutions.
At 400dpi or SUPER_SUPER exactly twice as amny pixels must be supplied
and at 300dpi exactly 1.5 times.
A4 B4 A3
200 1728/216 2048/256 2432/304
300 2592/324 3072/384 3648/456
400 3456/432 4096/512 4864/608
**/
// first index is 200/300/400dpi horiz res (inch or mm based)
// second index is width A4/B4/A3
USHORT ResWidthToBytes[3][3] =
{
{ 216, 256, 304 },
{ 324, 384, 456 },
{ 432, 512, 608 }
};
BYTE BestEncoding[8] =
{
0, // none (error)
1, // MH only
2, // MR only
2, // MR & MH
4, // MMR only
4, // MMR & MH
4, // MMR & MR
4 // MMR & MR & MH
};
BOOL NegotiateCaps(PThrdGlbl pTG)
{
USHORT Res;
if (glLoRes) {
pTG->Inst.RemoteRecvCaps.Fax.AwRes = 0;
}
memset(&pTG->Inst.SendParams, 0, sizeof(pTG->Inst.SendParams));
pTG->Inst.SendParams.bctype = SEND_PARAMS;
// They should be set. This code here is correct--arulm
// +++ Following three are not set in pcfax11
pTG->Inst.SendParams.wBCSize = sizeof(BC);
pTG->Inst.SendParams.wBCVer = VER_AWFXPROT100;
pTG->Inst.SendParams.wTotalSize = sizeof(BC);
// +++ Initialize ID from our own SendCaps...
{
char rgchID[MAXFHBIDLEN+2];
GetTextId(&pTG->Inst.SendCaps, rgchID, MAXFHBIDLEN+1);
PutTextId((LPBC)&pTG->Inst.SendParams, sizeof(pTG->Inst.SendParams),
rgchID, _fstrlen(rgchID), TEXTCODE_ASCII);
}
// RSL BUGBUG this should be set from fax UI
////////////////////////////////////////////
pTG->Inst.awfi.Encoding = ENCODE_CAPS; // MR_DATA | MH_DATA;
if (! pTG->SrcHiRes) {
pTG->Inst.awfi.AwRes = 0;
}
else {
pTG->Inst.awfi.AwRes = CAPS_RES; // AWRES_200_200;
}
////////////// Width, Length, Res & Enc /////////////////////////////
/////// Encoding ///////
BG_CHK(pTG->Inst.RemoteRecvCaps.Fax.Encoding && pTG->Inst.RemoteRecvCaps.Fax.Encoding < 8);
// this next BG_CHK seems bogus...?
// BG_CHK(pTG->Inst.ProtParams.fDisableECM ? (!(pTG->Inst.RemoteRecvCaps.Fax.Encoding & MMR_DATA)) : 1);
BG_CHK(INST_ENCODING && INST_ENCODING < 8);
BG_CHK(BestEncoding[INST_ENCODING] == INST_ENCODING); // check just 1 bit set
#define SEND_RECODE_TO INST_ENCODING
// If pTG->Inst.fDisableG3ECM, we will refuse to send MMR
if (pTG->Inst.fDisableG3ECM && (pTG->Inst.RemoteRecvCaps.Fax.Encoding & MMR_DATA))
{
(MyDebugPrint(pTG, LOG_ERR, "<WARNING> - fDisableG3ECM => NOT using MMR\r\n"));
pTG->Inst.RemoteRecvCaps.Fax.Encoding &= ~MMR_DATA;
}
if(!(pTG->Inst.SendParams.Fax.Encoding =
BestEncoding[(INST_ENCODING | SEND_RECODE_TO) &
pTG->Inst.RemoteRecvCaps.Fax.Encoding]))
{
// No matching Encoding not supported
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: SendEnc %d CanRecodeTo %d RecvCapsEnc %d. No match\r\n",
INST_ENCODING, SEND_RECODE_TO, pTG->Inst.RemoteRecvCaps.Fax.Encoding));
SetFailureCode(pTG, T30FAILS_NEGOT_ENCODING);
goto error;
}
// check just 1 bit set
BG_CHK(BestEncoding[pTG->Inst.SendParams.Fax.Encoding] ==
pTG->Inst.SendParams.Fax.Encoding);
BG_CHK(pTG->Inst.SendParams.Fax.Encoding == INST_ENCODING);
/////// Width ///////
pTG->Inst.RemoteRecvCaps.Fax.PageWidth &= 0x0F; // castrate all A5/A6 widths
if(INST_PAGEWIDTH > 0x0F)
{
// A5 or A6. Can quit or send as A4
// INST_PAGEWIDTH = WIDTH_A4;
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: A5/A6 images not supported\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_A5A6);
goto error;
}
if(pTG->Inst.RemoteRecvCaps.Fax.PageWidth < INST_PAGEWIDTH)
{
// or do some scaling
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: Image too wide\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_WIDTH);
goto error;
}
else
pTG->Inst.SendParams.Fax.PageWidth = INST_PAGEWIDTH;
/////// Length ///////
if(pTG->Inst.RemoteRecvCaps.Fax.PageLength < INST_PAGELENGTH)
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: Image too long\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_LENGTH);
goto error;
}
else
pTG->Inst.SendParams.Fax.PageLength = INST_PAGELENGTH;
/////// Res ///////
// pick best resolution
pTG->Inst.HorizScaling = 0;
pTG->Inst.VertScaling = 0;
// test scaling to NORMAL
// pTG->Inst.RemoteRecvCaps.Fax.AwRes = AWRES_mm080_038;
Res = (USHORT) (INST_RESOLUTION & pTG->Inst.RemoteRecvCaps.Fax.AwRes);
if(Res) // send native
pTG->Inst.SendParams.Fax.AwRes = Res;
else
{
BG_CHK(INST_RESOLUTION != AWRES_mm080_038);
BG_CHK(pTG->Inst.RemoteRecvCaps.Fax.AwRes & AWRES_mm080_038);
switch(INST_RESOLUTION)
{
case AWRES_mm160_154:
if(AWRES_400_400 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_400_400;
}
else
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 16x15.4 image and no horiz scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
break;
case AWRES_mm080_154:
#ifdef VS
if(pTG->Inst.SendParams.Fax.Encoding == MMR_DATA)
#endif //VS
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 8x15.4 image and no vert scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
#ifdef VS
if(AWRES_mm080_077 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm080_077;
pTG->Inst.VertScaling = 2;
}
else if(AWRES_200_200 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_200_200;
pTG->Inst.VertScaling = 2;
}
else
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm080_038;
pTG->Inst.VertScaling = 4;
}
#endif //VS
break;
case AWRES_mm080_077:
if(AWRES_200_200 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_200_200;
}
#ifdef VS
else if(pTG->Inst.SendParams.Fax.Encoding == MMR_DATA)
#else //VS
else
#endif //VS
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 8x7.7 image and no vert scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
#ifdef VS
else
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm080_038;
pTG->Inst.VertScaling = 2;
}
#endif //VS
break;
case AWRES_400_400:
if(AWRES_mm160_154 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm160_154;
}
else
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 400dpi image and no horiz scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
break;
case AWRES_300_300:
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 300dpi image and no non-integer scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
break;
case AWRES_200_200:
if(AWRES_mm080_077 & pTG->Inst.RemoteRecvCaps.Fax.AwRes)
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm080_077;
}
#ifdef VS
else if(pTG->Inst.SendParams.Fax.Encoding == MMR_DATA)
#else //VS
else
#endif //VS
{
(MyDebugPrint(pTG, LOG_ERR, "Negotiation failed: 200dpi image and no vert scaling\r\n"));
SetFailureCode(pTG, T30FAILS_NEGOT_RES);
goto error;
}
#ifdef VS
else
{
pTG->Inst.SendParams.Fax.AwRes = AWRES_mm080_038;
pTG->Inst.VertScaling = 2;
}
#endif //VS
break;
default:
BG_CHK(FALSE);
}
}
#ifdef VS
if(pTG->Inst.VertScaling)
{
BG_CHK(pTG->Inst.SendParams.Fax.Encoding != MMR_DATA);
//RSL InitVertScale(pTG->Inst.VertScaling);
}
#endif //VS
(MyDebugPrint(pTG, LOG_ALL, "Negotiation Succeeded: Res=%d PageWidth=%d Len=%d Enc=%d HSc=%d VSc=%d HSc3=%d VSc3=%d\r\n",
pTG->Inst.SendParams.Fax.AwRes, pTG->Inst.SendParams.Fax.PageWidth, pTG->Inst.SendParams.Fax.PageLength,
pTG->Inst.SendParams.Fax.Encoding, pTG->Inst.HorizScaling, pTG->Inst.VertScaling,
pTG->Inst.HorizScaling300dpi, pTG->Inst.VertScaling300dpi));
return TRUE;
error:
return FALSE;
}
void InitCapsBC(PThrdGlbl pTG, LPBC lpbc, USHORT uSize, BCTYPE bctype)
{
memset(lpbc, 0, uSize);
lpbc->bctype = bctype;
// They should be set. This code here is correct--arulm
// +++ Following three lines are not in pcfax11
lpbc->wBCSize = sizeof(BC);
lpbc->wBCVer = VER_AWFXPROT100;
lpbc->wTotalSize = sizeof(BC);
lpbc->Std.GroupNum = GROUPNUM_STD;
lpbc->Std.GroupLength = sizeof(BCSTD);
lpbc->Std.vMsgProtocol = vMSG_WIN95;
lpbc->Std.fBinaryData = TRUE;
// lpbc->Std.fInwardRouting = FALSE;
// NOSECURITY is defined for France build etc.
lpbc->Std.vSecurity = 0;
lpbc->Std.OperatingSys = OS_WIN16;
// lpbc->Std.vShortFlags = 0;
// lpbc->Std.vInteractive = 0;
// lpbc->Std.DataSpeed = 0;
// lpbc->Std.DataLink = 0;
// lpbc->Fax.fPublicPoll = 0;
if (! pTG->SrcHiRes) {
lpbc->Fax.AwRes = 0;
}
else {
lpbc->Fax.AwRes = CAPS_RES;
}
lpbc->Fax.Encoding = ENCODE_CAPS;
if (0) // RSL (pTG->Inst.fDisableMRRecv)
{
ERRMSG(( SZMOD "<<WARNING>> Disabling MR Receive capability\r\n"));
lpbc->Fax.Encoding &= ~MR_DATA;
}
lpbc->Fax.PageWidth = CAPS_WIDTH; // can be upto A3
lpbc->Fax.PageLength = LENGTH_UNLIMITED;
lpbc->Image.GroupNum = GROUPNUM_IMAGE;
lpbc->Image.GroupLength = sizeof(BCIMAGE);
lpbc->Image.vRamboVer = vRAMBO_VER1;
}