windows-nt/Source/XPSP1/NT/base/mvdm/vdmredir/vrdlcdbg.c
2020-09-26 16:20:57 +08:00

4028 lines
110 KiB
C

/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
vrdlcdbg.c
Abstract:
Contains functions for dumping CCBs, parameter tables; diagnostic and
debugging functions for DOS DLC (CCB1)
Contents:
DbgOut
DbgOutStr
DumpCcb
DumpDosDlcBufferPool
DumpDosDlcBufferChain
MapCcbRetcode
(DefaultParameterTableDump)
(DumpParameterTableHeader)
(DumpBufferFreeParms)
(DumpBufferGetParms)
(DumpDirCloseAdapterParms)
(DumpDirDefineMifEnvironmentParms)
(DumpDirInitializeParms)
(DumpDirModifyOpenParmsParms)
(DumpDirOpenAdapterParms)
(DumpDirReadLog)
(DumpDirRestoreOpenParmsParms)
(DumpDirSetFunctionalAddressParms)
(DumpDirSetGroupAddressParms)
(DumpDirSetUserAppendageParms)
(DumpDirStatusParms)
(DumpDirTimerCancelParms)
(DumpDirTimerCancelGroupParms)
(DumpDirTimerSetParms)
(DumpDlcCloseSapParms)
(DumpDlcCloseStationParms)
(DumpDlcConnectStationParms)
(DumpDlcFlowControlParms)
(MapFlowControl)
(DumpDlcModifyParms)
(DumpDlcOpenSapParms)
(MapOptionsPriority)
(DumpDlcOpenStationParms)
(DumpDlcReallocateParms)
(DumpDlcResetParms)
(DumpDlcStatisticsParms)
(DumpPdtTraceOffParms)
(DumpPdtTraceOnParms)
(DumpReadParms)
(MapReadEvent)
(MapDlcStatus)
(DumpReadCancelParms)
(DumpReceiveParms)
(DumpReceiveCancelParms)
(DumpReceiveModifyParms)
(DumpTransmitDirFrameParms)
(DumpTransmitIFrameParms)
(DumpTransmitTestCmdParms)
(DumpTransmitUiFrameParms)
(DumpTransmitXidCmdParms)
(DumpTransmitXidRespFinalParms)
(DumpTransmitXidRespNotFinalParms)
(DumpTransmitParms)
(DumpTransmitQueue)
DumpReceiveDataBuffer
(MapMessageType)
DumpData
IsCcbErrorCodeAllowable
IsCcbErrorCodeValid
IsCcbCommandValid
MapCcbCommandToName
DumpDosAdapter
(MapAdapterType)
Author:
Richard L Firth (rfirth) 30-Apr-1992
Revision History:
--*/
#if DBG
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#include <nt.h>
#include <ntrtl.h> // ASSERT, DbgPrint
#include <nturtl.h>
#include <windows.h>
#include <softpc.h> // x86 virtual machine definitions
#include <vrdlctab.h>
#include <vdmredir.h>
#include <smbgtpt.h>
#include <dlcapi.h> // Official DLC API definition
#include <ntdddlc.h> // IOCTL commands
#include <dlcio.h> // Internal IOCTL API interface structures
#include "vrdlc.h"
#include "vrdebug.h"
#include "vrdlcdbg.h"
//
// defines
//
//
// standard parameters to each table dump routine
//
#define DUMP_TABLE_PARMS \
IN PVOID Parameters, \
IN BOOL IsDos, \
IN BOOL IsInput, \
IN WORD Segment, \
IN WORD Offset
//
// DumpData options
//
#define DD_NO_ADDRESS 0x00000001 // don't display address of data
#define DD_LINE_BEFORE 0x00000002 // linefeed before first dumped line
#define DD_LINE_AFTER 0x00000004 // linefeed after last dumped line
#define DD_INDENT_ALL 0x00000008 // indent all lines
#define DD_NO_ASCII 0x00000010 // don't dump ASCII respresentation
#define DD_UPPER_CASE 0x00000020 // upper-case hex dump (F4 instead of f4)
//
// misc.
//
#define DEFAULT_FIELD_WIDTH 13 // amount of description before a number
//
// local prototypes
//
VOID
DbgOutStr(
IN LPSTR Str
);
PRIVATE
VOID
DefaultParameterTableDump(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpParameterTableHeader(
IN LPSTR CommandName,
IN PVOID Table,
IN BOOL IsDos,
IN WORD Segment,
IN WORD Offset
);
PRIVATE
VOID
DumpBufferFreeParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpBufferGetParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirCloseAdapterParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirDefineMifEnvironmentParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirInitializeParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirModifyOpenParmsParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirOpenAdapterParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirReadLog(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirRestoreOpenParmsParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirSetFunctionalAddressParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirSetGroupAddressParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirSetUserAppendageParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirStatusParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirTimerCancelParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirTimerCancelGroupParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDirTimerSetParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcCloseSapParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcCloseStationParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcConnectStationParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcFlowControlParms(
DUMP_TABLE_PARMS
);
PRIVATE
LPSTR
MapFlowControl(
BYTE FlowControl
);
PRIVATE
VOID
DumpDlcModifyParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcOpenSapParms(
DUMP_TABLE_PARMS
);
PRIVATE
LPSTR
MapOptionsPriority(
UCHAR OptionsPriority
);
PRIVATE
VOID
DumpDlcOpenStationParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcReallocateParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcResetParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpDlcStatisticsParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpPdtTraceOffParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpPdtTraceOnParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpReadParms(
DUMP_TABLE_PARMS
);
PRIVATE
LPSTR
MapReadEvent(
UCHAR Event
);
PRIVATE
LPSTR
MapDlcStatus(
WORD Status
);
PRIVATE
VOID
DumpReadCancelParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpReceiveParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpReceiveCancelParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpReceiveModifyParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitDirFrameParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitIFrameParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitTestCmdParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitUiFrameParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitXidCmdParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitXidRespFinalParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitXidRespNotFinalParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitParms(
DUMP_TABLE_PARMS
);
PRIVATE
VOID
DumpTransmitQueue(
IN DOS_ADDRESS dpQueue
);
PRIVATE
LPSTR
MapMessageType(
UCHAR MessageType
);
VOID
DumpData(
IN LPSTR Title,
IN PBYTE Address,
IN DWORD Length,
IN DWORD Options,
IN DWORD Indent,
IN BOOL IsDos,
IN WORD Segment,
IN WORD Offset
);
PRIVATE
LPSTR
MapAdapterType(
IN ADAPTER_TYPE AdapterType
);
//
// explanations of error codes returned in CCB_RETCODE fields. Explanations
// taken more-or-less verbatim from IBM Local Area Network Technical Reference
// table B-1 ppB-2 to B-5. Includes all errors, even not relevant to CCB1
//
static LPSTR CcbRetcodeExplanations[] = {
"Success",
"Invalid command code",
"Duplicate command, one already outstanding",
"Adapter open, should be closed",
"Adapter closed, should be open",
"Required parameter missing",
"Invalid/incompatible option",
"Command cancelled - unrecoverable failure",
"Unauthorized access priority",
"Adapter not initialized, should be",
"Command cancelled by user request",
"Command cancelled, adapter closed while command in progress",
"Command completed Ok, adapter not open",
"Invalid error code 0x0D",
"Invalid error code 0x0E",
"Invalid error code 0x0F",
"Adapter open, NetBIOS not operational",
"Error in DIR.TIMER.SET or DIR.TIMER.CANCEL",
"Available work area exceeded",
"Invalid LOG.ID",
"Invalid shared RAM segment or size",
"Lost log data, buffer too small, log reset",
"Requested buffer size exceeds pool length",
"Command invalid, NetBIOS operational",
"Invalid SAP buffer length",
"Inadequate buffers available for request",
"USER_LENGTH value too large for buffer length",
"The CCB_PARM_TAB pointer is invalid",
"A pointer in the CCB parameter table is invalid",
"Invalid CCB_ADAPTER value",
"Invalid functional address",
"Invalid error code 0x1F",
"Lost data on receive, no buffers available",
"Lost data on receive, inadequate buffer space",
"Error on frame transmission, check TRANSMIT_FS data",
"Error on frame transmit or strip process",
"Unauthorized MAC frame",
"Maximum number of commands exceeded",
"Unrecognized command correlator",
"Link not transmitting I frames, state changed from link opened",
"Invalid transmit frame length",
"Invalid error code 0x29",
"Invalid error code 0x2a",
"Invalid error code 0x2b",
"Invalid error code 0x2c",
"Invalid error code 0x2d",
"Invalid error code 0x2e",
"Invalid error code 0x2f",
"Inadequate receive buffers for adapter to open",
"Invalid error code 0x31",
"Invalid NODE_ADDRESS",
"Invalid adapter receive buffer length defined",
"Invalid adapter transmit buffer length defined",
"Invalid error code 0x35",
"Invalid error code 0x36",
"Invalid error code 0x37",
"Invalid error code 0x38",
"Invalid error code 0x39",
"Invalid error code 0x3a",
"Invalid error code 0x3b",
"Invalid error code 0x3c",
"Invalid error code 0x3d",
"Invalid error code 0x3e",
"Invalid error code 0x3f",
"Invalid STATION_ID",
"Protocol error, link in invalid state for command",
"Parameter exceeded maximum allowed",
"Invalid SAP value or value already in use",
"Invalid routing information field",
"Requested group membership in non-existent group SAP",
"Resources not available",
"Sap cannot close unless all link stations are closed",
"Group SAP cannot close, individual SAPs not closed",
"Group SAP has reached maximum membership",
"Sequence error, incompatible command in progress",
"Station closed without remote acknowledgement",
"Sequence error, cannot close, DLC commands outstanding",
"Unsuccessful link station connection attempted",
"Member SAP not found in group SAP list",
"Invalid remote address, may not be a group address",
"Invalid pointer in CCB_POINTER field",
"Invalid error code 0x51",
"Invalid application program ID",
"Invalid application program key code",
"Invalid system key code",
"Buffer is smaller than buffer size given in DLC.OPEN.SAP",
"Adapter's system process is not installed",
"Inadequate stations available",
"Invalid CCB_PARAMETER_1 parameter",
"Inadequate queue elements to satisfy request",
"Initialization failure, cannot open adapter",
"Error detected in chained READ command",
"Direct stations not assigned to application program",
"Dd interface not installed",
"Requested adapter is not installed",
"Chained CCBs must all be for same adapter",
"Adapter initializing, command not accepted",
"Number of allowed application programs has been exceeded",
"Command cancelled by system action",
"Direct stations not available",
"Invalid DDNAME parameter",
"Inadequate GDT selectors to satisfy request",
"Invalid error code 0x66",
"Command cancelled, CCB resources purged",
"Application program ID not valid for interface",
"Segment associated with request cannot be locked"
};
#define NUMBER_OF_ERROR_MESSAGES ARRAY_ELEMENTS(CcbRetcodeExplanations)
#define LAST_DLC_ERROR_CODE LAST_ELEMENT(CcbRetcodeExplanations)
VOID
DbgOut(
IN LPSTR Format,
IN ...
)
/*++
Routine Description:
Sends formatted debug output to desired output device. If DEBUG_TO_FILE
was specified in VR environment flags, output goes to VRDEBUG.LOG in
current directory, else to standard debug output via DbgPrint
Arguments:
Format - printf-style format string
... - variable args
Return Value:
None.
--*/
{
va_list list;
char buffer[2048];
va_start(list, Format);
vsprintf(buffer, Format, list);
va_end(list);
if (hVrDebugLog) {
fputs(buffer, hVrDebugLog);
} else {
DbgPrint(buffer);
}
}
VOID
DbgOutStr(
IN LPSTR Str
)
/*++
Routine Description:
Sends formatted debug output to desired output device. If DEBUG_TO_FILE
was specified in VR environment flags, output goes to VRDEBUG.LOG in
current directory, else to standard debug output via DbgPrint
Arguments:
Str - string to print
Return Value:
None.
--*/
{
if (hVrDebugLog) {
fputs(Str, hVrDebugLog);
} else {
DbgPrint(Str);
}
}
VOID
DumpCcb(
IN PVOID Ccb,
IN BOOL DumpAll,
IN BOOL CcbIsInput,
IN BOOL IsDos,
IN WORD Segment OPTIONAL,
IN WORD Offset OPTIONAL
)
/*++
Routine Description:
Dumps (to debug terminal) a CCB and any associated parameter table. Also
displays the symbolic CCB command and an error code description if the
CCB is being returned to the caller. Dumps in either DOS format (segmented
16-bit pointers) or NT format (flat 32-bit pointers)
Arguments:
Ccb - flat 32-bit pointer to CCB1 or CCB2 to dump
DumpAll - if TRUE, dumps parameter tables and buffers, else just CCB
CcbIsInput - if TRUE, CCB is from user: don't display error code explanation
IsDos - if TRUE, CCB is DOS format
Segment - if IsDos is TRUE, segment of CCB in VDM
Offset - if IsDos is TRUE, offset of CCB in VDM
Return Value:
None.
--*/
{
PVOID parmtab = NULL;
LPSTR cmdname = "UNKNOWN CCB!";
PLLC_CCB NtCcb = (PLLC_CCB)Ccb;
PLLC_DOS_CCB DosCcb = (PLLC_DOS_CCB)Ccb;
BOOL haveParms = FALSE;
VOID (*DumpParms)(PVOID, BOOL, BOOL, WORD, WORD) = DefaultParameterTableDump;
PVOID parameterTable = NULL;
WORD seg;
WORD off;
BOOL parmsInCcb = FALSE;
switch (((PLLC_CCB)Ccb)->uchDlcCommand) {
case LLC_BUFFER_FREE:
cmdname = "BUFFER.FREE";
haveParms = TRUE;
DumpParms = DumpBufferFreeParms;
break;
case LLC_BUFFER_GET:
cmdname = "BUFFER.GET";
haveParms = TRUE;
DumpParms = DumpBufferGetParms;
break;
case LLC_DIR_CLOSE_ADAPTER:
cmdname = "DIR.CLOSE.ADAPTER";
haveParms = TRUE;
DumpParms = DumpDirCloseAdapterParms;
parmsInCcb = TRUE;
break;
case LLC_DIR_CLOSE_DIRECT:
cmdname = "DIR.CLOSE.DIRECT";
break;
case 0x2b:
//
// not supported ! (yet?)
//
cmdname = "DIR.DEFINE.MIF.ENVIRONMENT";
haveParms = TRUE;
break;
case LLC_DIR_INITIALIZE:
cmdname = "DIR.INITIALIZE";
haveParms = TRUE;
DumpParms = DumpDirInitializeParms;
break;
case LLC_DIR_INTERRUPT:
cmdname = "DIR.INTERRUPT";
break;
case LLC_DIR_MODIFY_OPEN_PARMS:
cmdname = "DIR.MODIFY.OPEN.PARMS";
haveParms = TRUE;
break;
case LLC_DIR_OPEN_ADAPTER:
cmdname = "DIR.OPEN.ADAPTER";
haveParms = TRUE;
DumpParms = DumpDirOpenAdapterParms;
break;
case LLC_DIR_OPEN_DIRECT:
//
// not supported from DOS!
//
cmdname = "DIR.OPEN.DIRECT";
haveParms = TRUE;
break;
case LLC_DIR_READ_LOG:
cmdname = "DIR.READ.LOG";
haveParms = TRUE;
break;
case LLC_DIR_RESTORE_OPEN_PARMS:
cmdname = "DIR.RESTORE.OPEN.PARMS";
break;
case LLC_DIR_SET_FUNCTIONAL_ADDRESS:
cmdname = "DIR.SET.FUNCTIONAL.ADDRESS";
haveParms = TRUE;
DumpParms = DumpDirSetFunctionalAddressParms;
parmsInCcb = TRUE;
break;
case LLC_DIR_SET_GROUP_ADDRESS:
cmdname = "DIR.SET.GROUP.ADDRESS";
haveParms = TRUE;
DumpParms = DumpDirSetGroupAddressParms;
parmsInCcb = TRUE;
break;
case LLC_DIR_SET_USER_APPENDAGE:
cmdname = "DIR.SET.USER.APPENDAGE";
haveParms = TRUE;
DumpParms = DumpDirSetUserAppendageParms;
break;
case LLC_DIR_STATUS:
cmdname = "DIR.STATUS";
haveParms = TRUE;
DumpParms = DumpDirStatusParms;
break;
case LLC_DIR_TIMER_CANCEL:
cmdname = "DIR.TIMER.CANCEL";
haveParms = TRUE;
DumpParms = DumpDirTimerCancelParms;
parmsInCcb = TRUE;
break;
case LLC_DIR_TIMER_CANCEL_GROUP:
cmdname = "DIR.TIMER.CANCEL.GROUP";
haveParms = TRUE;
DumpParms = DumpDirTimerCancelGroupParms;
parmsInCcb = TRUE;
break;
case LLC_DIR_TIMER_SET:
cmdname = "DIR.TIMER.SET";
haveParms = TRUE;
DumpParms = DumpDirTimerSetParms;
parmsInCcb = TRUE;
break;
case LLC_DLC_CLOSE_SAP:
cmdname = "DLC.CLOSE.SAP";
haveParms = TRUE;
DumpParms = DumpDlcCloseSapParms;
parmsInCcb = TRUE;
break;
case LLC_DLC_CLOSE_STATION:
cmdname = "DLC.CLOSE.STATION";
haveParms = TRUE;
DumpParms = DumpDlcCloseStationParms;
parmsInCcb = TRUE;
break;
case LLC_DLC_CONNECT_STATION:
cmdname = "DLC.CONNECT.STATION";
haveParms = TRUE;
DumpParms = DumpDlcConnectStationParms;
break;
case LLC_DLC_FLOW_CONTROL:
cmdname = "DLC.FLOW.CONTROL";
haveParms = TRUE;
DumpParms = DumpDlcFlowControlParms;
parmsInCcb = TRUE;
break;
case LLC_DLC_MODIFY:
cmdname = "DLC.MODIFY";
haveParms = TRUE;
break;
case LLC_DLC_OPEN_SAP:
cmdname = "DLC.OPEN.SAP";
haveParms = TRUE;
DumpParms = DumpDlcOpenSapParms;
break;
case LLC_DLC_OPEN_STATION:
cmdname = "DLC.OPEN.STATION";
haveParms = TRUE;
DumpParms = DumpDlcOpenStationParms;
break;
case LLC_DLC_REALLOCATE_STATIONS:
cmdname = "DLC.REALLOCATE";
haveParms = TRUE;
break;
case LLC_DLC_RESET:
cmdname = "DLC.RESET";
haveParms = TRUE;
DumpParms = DumpDlcResetParms;
parmsInCcb = TRUE;
break;
case LLC_DLC_SET_THRESHOLD:
cmdname = "DLC.SET.THRESHOLD";
haveParms = TRUE;
break;
case LLC_DLC_STATISTICS:
cmdname = "DLC.STATISTICS";
haveParms = TRUE;
break;
case 0x25:
//
// not supported !
//
cmdname = "PDT.TRACE.OFF";
break;
case 0x24:
//
// not supported !
//
cmdname = "PDT.TRACE.ON";
break;
case LLC_READ:
cmdname = "READ";
haveParms = TRUE;
DumpParms = DumpReadParms;
break;
case LLC_READ_CANCEL:
cmdname = "READ.CANCEL";
break;
case LLC_RECEIVE:
cmdname = "RECEIVE";
haveParms = TRUE;
DumpParms = DumpReceiveParms;
break;
case LLC_RECEIVE_CANCEL:
cmdname = "RECEIVE.CANCEL";
haveParms = TRUE;
DumpParms = DumpReceiveCancelParms;
parmsInCcb = TRUE;
break;
case LLC_RECEIVE_MODIFY:
cmdname = "RECEIVE.MODIFY";
haveParms = TRUE;
DumpParms = DumpReceiveModifyParms;
break;
case LLC_TRANSMIT_DIR_FRAME:
cmdname = "TRANSMIT.DIR.FRAME";
haveParms = TRUE;
DumpParms = DumpTransmitDirFrameParms;
break;
case LLC_TRANSMIT_I_FRAME:
cmdname = "TRANSMIT.I.FRAME";
haveParms = TRUE;
DumpParms = DumpTransmitIFrameParms;
break;
case LLC_TRANSMIT_TEST_CMD:
cmdname = "TRANSMIT.TEST.CMD";
haveParms = TRUE;
DumpParms = DumpTransmitTestCmdParms;
break;
case LLC_TRANSMIT_UI_FRAME:
cmdname = "TRANSMIT.UI.FRAME";
haveParms = TRUE;
DumpParms = DumpTransmitUiFrameParms;
break;
case LLC_TRANSMIT_XID_CMD:
cmdname = "TRANSMIT.XID.CMD";
haveParms = TRUE;
DumpParms = DumpTransmitXidCmdParms;
break;
case LLC_TRANSMIT_XID_RESP_FINAL:
cmdname = "TRANSMIT.XID.RESP.FINAL";
haveParms = TRUE;
DumpParms = DumpTransmitXidRespFinalParms;
break;
case LLC_TRANSMIT_XID_RESP_NOT_FINAL:
cmdname = "TRANSMIT.XID.RESP.NOT.FINAL";
haveParms = TRUE;
DumpParms = DumpTransmitXidRespNotFinalParms;
break;
}
if (IsDos) {
seg = GET_SELECTOR(&DosCcb->u.pParms);
off = GET_OFFSET(&DosCcb->u.pParms);
parmtab = POINTER_FROM_WORDS(seg, off);
} else {
parmtab = NtCcb->u.pParameterTable;
}
if (IsDos) {
PLLC_DOS_CCB DosCcb = (PLLC_DOS_CCB)Ccb;
DBGPRINT( "\n"
"------------------------------------------------------------------------------"
"\n"
);
IF_DEBUG(TIME) {
SYSTEMTIME timestruct;
GetLocalTime(&timestruct);
DBGPRINT(
"%02d:%02d:%02d.%03d\n",
timestruct.wHour,
timestruct.wMinute,
timestruct.wSecond,
timestruct.wMilliseconds
);
}
DBGPRINT( "%s DOS CCB @%04x:%04x:\n"
"adapter %02x\n"
"command %02x [%s]\n"
"retcode %02x [%s]\n"
"reserved %02x\n"
"next %04x:%04x\n"
"ANR %04x:%04x\n",
CcbIsInput ? "INPUT" : "OUTPUT",
Segment,
Offset,
DosCcb->uchAdapterNumber,
DosCcb->uchDlcCommand,
cmdname,
DosCcb->uchDlcStatus,
CcbIsInput ? "" : MapCcbRetcode(DosCcb->uchDlcStatus),
DosCcb->uchReserved1,
GET_SEGMENT(&DosCcb->pNext),
GET_OFFSET(&DosCcb->pNext),
GET_SEGMENT(&DosCcb->ulCompletionFlag),
GET_OFFSET(&DosCcb->ulCompletionFlag)
);
if (haveParms) {
if (!parmsInCcb) {
DBGPRINT(
"parms %04x:%04x\n",
GET_SEGMENT(&DosCcb->u.pParms),
GET_OFFSET(&DosCcb->u.pParms)
);
parameterTable = POINTER_FROM_WORDS(GET_SEGMENT(&DosCcb->u.pParms),
GET_OFFSET(&DosCcb->u.pParms)
);
} else {
parameterTable = (PVOID)READ_DWORD(&DosCcb->u.ulParameter);
}
}
} else {
PLLC_CCB NtCcb = (PLLC_CCB)Ccb;
DBGPRINT( "\n"
"------------------------------------------------------------------------------"
"\n"
);
IF_DEBUG(TIME) {
SYSTEMTIME timestruct;
GetLocalTime(&timestruct);
DBGPRINT(
"%02d:%02d:%02d.%03d\n",
timestruct.wHour,
timestruct.wMinute,
timestruct.wSecond,
timestruct.wMilliseconds
);
}
DBGPRINT( "%s NT CCB @ %#8x\n"
"adapter %02x\n"
"command %02x [%s]\n"
"retcode %02x [%s]\n"
"reserved %02x\n"
"next %08x\n"
"ANR %08x\n",
CcbIsInput ? "INPUT" : "OUTPUT",
Ccb,
NtCcb->uchAdapterNumber,
NtCcb->uchDlcCommand,
cmdname,
NtCcb->uchDlcStatus,
CcbIsInput ? "" : MapCcbRetcode(NtCcb->uchDlcStatus),
NtCcb->uchReserved1,
NtCcb->pNext,
NtCcb->ulCompletionFlag
);
if (haveParms) {
if (!parmsInCcb) {
DBGPRINT(
"parms %08x\n",
NtCcb->u.pParameterTable
);
}
parameterTable = NtCcb->u.pParameterTable;
}
DBGPRINT(
"hEvent %08x\n"
"reserved %02x\n"
"readflag %02x\n"
"reserved %04x\n",
NtCcb->hCompletionEvent,
NtCcb->uchReserved2,
NtCcb->uchReadFlag,
NtCcb->usReserved3
);
}
if ((parameterTable && DumpAll) || parmsInCcb) {
DumpParms(parameterTable, IsDos, CcbIsInput, seg, off);
}
}
VOID
DumpDosDlcBufferPool(
IN PDOS_DLC_BUFFER_POOL PoolDescriptor
)
/*++
Routine Description:
Dumps a DOS DLC buffer pool so we can see that it looks ok
Arguments:
PoolDescriptor - pointer to DOS_DLC_BUFFER_POOL structure
Return Value:
None.
--*/
{
int count = PoolDescriptor->BufferCount;
DBGPRINT( "DOS DLC Buffer Pool @%04x:%04x, BufferSize=%d, BufferCount=%d\n",
HIWORD(PoolDescriptor->dpBuffer),
LOWORD(PoolDescriptor->dpBuffer),
PoolDescriptor->BufferSize,
PoolDescriptor->BufferCount
);
DumpDosDlcBufferChain(PoolDescriptor->dpBuffer, PoolDescriptor->BufferCount);
}
VOID
DumpDosDlcBufferChain(
IN DOS_ADDRESS DosAddress,
IN DWORD BufferCount
)
/*++
Routine Description:
Dumps a chain of DOS buffers
Arguments:
DosAddress - address of buffer in VDM memory in DOS_ADDRESS format (16:16)
BufferCount - number of buffers to dump
Return Value:
None.
--*/
{
WORD seg = HIWORD(DosAddress);
WORD off = LOWORD(DosAddress);
LPVOID pointer = DOS_PTR_TO_FLAT(DosAddress);
int i;
for (i = 1; BufferCount; --BufferCount, ++i) {
DBGPRINT("Buffer % 3d: %04x:%04x, Next Buffer @%04x:%04x\n",
i,
seg, off,
(DWORD)GET_SELECTOR(&((PLLC_DOS_BUFFER)pointer)->pNext),
(DWORD)GET_OFFSET(&((PLLC_DOS_BUFFER)pointer)->pNext)
);
seg = GET_SELECTOR(&((PLLC_DOS_BUFFER)pointer)->pNext);
off = GET_OFFSET(&((PLLC_DOS_BUFFER)pointer)->pNext);
IF_DEBUG(DUMP_FREE_BUF) {
PLLC_DOS_BUFFER pBuf = (PLLC_DOS_BUFFER)pointer;
DBGPRINT(
"next buffer %04x:%04x\n"
"frame length %04x\n"
"data length %04x\n"
"user offset %04x\n"
"user length %04x\n"
"station id %04x\n"
"options %02x\n"
"message type %02x\n"
"buffers left %04x\n"
"rcv FS %02x\n"
"adapter num %02x\n"
"\n",
GET_SEGMENT(&pBuf->Contiguous.pNextBuffer),
GET_OFFSET(&pBuf->Contiguous.pNextBuffer),
READ_WORD(&pBuf->Contiguous.cbFrame),
READ_WORD(&pBuf->Contiguous.cbBuffer),
READ_WORD(&pBuf->Contiguous.offUserData),
READ_WORD(&pBuf->Contiguous.cbUserData),
READ_WORD(&pBuf->Contiguous.usStationId),
pBuf->Contiguous.uchOptions,
pBuf->Contiguous.uchMsgType,
READ_WORD(&pBuf->Contiguous.cBuffersLeft),
pBuf->Contiguous.uchRcvFS,
pBuf->Contiguous.uchAdapterNumber
);
}
pointer = READ_FAR_POINTER(&((PLLC_DOS_BUFFER)pointer)->pNext);
}
}
LPSTR
MapCcbRetcode(
IN BYTE Retcode
)
/*++
Routine Description:
Returns string describing error code
Arguments:
Retcode - CCB_RETCODE
Return Value:
LPSTR
--*/
{
static char errbuf[128];
if (Retcode == LLC_STATUS_PENDING) {
return "Command in progress";
} else if (Retcode > NUMBER_OF_ERROR_MESSAGES) {
sprintf(errbuf, "*** Invalid error code 0x%2x ***", Retcode);
return errbuf;
}
return CcbRetcodeExplanations[Retcode];
}
PRIVATE
VOID
DefaultParameterTableDump(
DUMP_TABLE_PARMS
)
/*++
Routine Description:
Displays default message for CCBs which have parameter tables that don't
have a dump routine yet
Arguments:
Parameters - pointer to parameter table
IsDos - if TRUE parameters in DOS (16:16) format
Segment - if IsDos is TRUE, segment of CCB in VDM
Offset - if IsDos is TRUE, offset of CCB in VDM
Return Value:
None.
--*/
{
DBGPRINT("Parameter table dump not implemented for this CCB\n");
}
PRIVATE
VOID
DumpParameterTableHeader(
IN LPSTR CommandName,
IN PVOID Table,
IN BOOL IsDos,
IN WORD Segment,
IN WORD Offset
)
/*++
Routine Description:
Displays header for parameter table dump. Displays address in DOS or NT
format (32-bit flat or 16:16)
Arguments:
CommandName - name of command which owns parameter table
Table - flat 32-bit address of parameter table
IsDos - if TRUE, use Segment:Offset in display
Segment - if IsDos is TRUE, segment of parameter table in VDM
Offset - if IsDos is TRUE, offset of parameter table in VDM
Return Value:
None.
--*/
{
DBGPRINT( IsDos ? "\n%s parameter table @%04x:%04x\n"
: "\n%s parameter table @%08x\n",
CommandName,
IsDos ? (DWORD)Segment : (DWORD)Table,
IsDos ? (DWORD)Offset : 0
);
}
PRIVATE
VOID
DumpBufferFreeParms(
DUMP_TABLE_PARMS
)
{
PLLC_BUFFER_FREE_PARMS parms = (PLLC_BUFFER_FREE_PARMS)Parameters;
DumpParameterTableHeader("BUFFER.FREE", Parameters, IsDos, Segment, Offset);
DBGPRINT( "station id %04x\n"
"buffers left %04x\n"
"reserved %02x %02x %02x %02x\n",
READ_WORD(&parms->usReserved1),
READ_WORD(&parms->cBuffersLeft),
((PBYTE)&(parms->ulReserved))[0],
((PBYTE)&(parms->ulReserved))[1],
((PBYTE)&(parms->ulReserved))[2],
((PBYTE)&(parms->ulReserved))[3]
);
if (IsDos) {
DBGPRINT(
"first buffer %04x:%04x\n",
GET_SELECTOR(&parms->pFirstBuffer),
GET_OFFSET(&parms->pFirstBuffer)
);
} else {
DBGPRINT(
"first buffer %08x\n", parms->pFirstBuffer);
}
}
PRIVATE
VOID
DumpBufferGetParms(
DUMP_TABLE_PARMS
)
{
//
// Antti's definition different from that in manual, so use IBM def.
//
typedef struct {
WORD StationId;
WORD BufferLeft;
BYTE BufferGet;
BYTE Reserved[3];
DWORD FirstBuffer;
} CCB1_BUFFER_GET_PARMS, *PCCB1_BUFFER_GET_PARMS;
PCCB1_BUFFER_GET_PARMS parms = (PCCB1_BUFFER_GET_PARMS)Parameters;
DumpParameterTableHeader("BUFFER.GET", Parameters, IsDos, Segment, Offset);
DBGPRINT( "station id %04x\n"
"buffers left %04x\n"
"buffers get %02x\n"
"reserved %02x %02x %02x\n",
READ_WORD(&parms->StationId),
READ_WORD(&parms->BufferLeft),
parms->BufferGet,
parms->Reserved[0],
parms->Reserved[1],
parms->Reserved[2]
);
if (IsDos) {
DBGPRINT(
"first buffer %04x:%04x\n",
GET_SELECTOR(&parms->FirstBuffer),
GET_OFFSET(&parms->FirstBuffer)
);
} else {
DBGPRINT(
"first buffer %08x\n", parms->FirstBuffer);
}
}
PRIVATE
VOID
DumpDirCloseAdapterParms(
DUMP_TABLE_PARMS
)
{
UNREFERENCED_PARAMETER(IsDos);
UNREFERENCED_PARAMETER(Segment);
UNREFERENCED_PARAMETER(Offset);
DBGPRINT( "lock code %04x\n", LOWORD(Parameters));
}
PRIVATE
VOID
DumpDirDefineMifEnvironmentParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpDirInitializeParms(
DUMP_TABLE_PARMS
)
{
//
// once again, invent a structure to reflect the DOS CCB parameter table
// as defined in the IBM LAN tech ref
//
typedef struct {
WORD BringUps;
WORD SharedRam;
BYTE Reserved[4];
DWORD AdapterCheckAppendage;
DWORD NetworkStatusChangeAppendage;
DWORD IoErrorAppendage;
} CCB1_DIR_INITIALIZE_PARMS, *PCCB1_DIR_INITIALIZE_PARMS;
PCCB1_DIR_INITIALIZE_PARMS parms = (PCCB1_DIR_INITIALIZE_PARMS)Parameters;
DumpParameterTableHeader("DIR.INITIALIZE", Parameters, IsDos, Segment, Offset);
DBGPRINT( "bring ups %04x\n"
"shared RAM %04x\n"
"reserved %02x %02x %02x %02x\n"
"adap. check %04x:%04x\n"
"n/w status %04x:%04x\n"
"pc error %04x:%04x\n",
READ_WORD(&parms->BringUps),
READ_WORD(&parms->SharedRam),
parms->Reserved[0],
parms->Reserved[1],
parms->Reserved[2],
parms->Reserved[3],
GET_SEGMENT(&parms->AdapterCheckAppendage),
GET_OFFSET(&parms->AdapterCheckAppendage),
GET_SEGMENT(&parms->NetworkStatusChangeAppendage),
GET_OFFSET(&parms->NetworkStatusChangeAppendage),
GET_SEGMENT(&parms->IoErrorAppendage),
GET_OFFSET(&parms->IoErrorAppendage)
);
}
PRIVATE
VOID
DumpDirModifyOpenParmsParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpDirOpenAdapterParms(
DUMP_TABLE_PARMS
)
{
PLLC_DOS_DIR_OPEN_ADAPTER_PARMS dosParms = (PLLC_DOS_DIR_OPEN_ADAPTER_PARMS)Parameters;
PLLC_DIR_OPEN_ADAPTER_PARMS ntParms = (PLLC_DIR_OPEN_ADAPTER_PARMS)Parameters;
DumpParameterTableHeader("DIR.OPEN.ADAPTER", Parameters, IsDos, Segment, Offset);
if (IsDos) {
PADAPTER_PARMS pAdapterParms = READ_FAR_POINTER(&dosParms->pAdapterParms);
PDIRECT_PARMS pDirectParms = READ_FAR_POINTER(&dosParms->pDirectParms);
PDLC_PARMS pDlcParms = READ_FAR_POINTER(&dosParms->pDlcParms);
PNCB_PARMS pNcbParms = READ_FAR_POINTER(&dosParms->pNcbParms);
ULPBYTE pProductId;
DWORD i;
DBGPRINT(
"adapter parms %04x:%04x\n"
"direct parms %04x:%04x\n"
"DLC parms %04x:%04x\n"
"NCB parms %04x:%04x\n",
GET_SEGMENT(&dosParms->pAdapterParms),
GET_OFFSET(&dosParms->pAdapterParms),
GET_SEGMENT(&dosParms->pDirectParms),
GET_OFFSET(&dosParms->pDirectParms),
GET_SEGMENT(&dosParms->pDlcParms),
GET_OFFSET(&dosParms->pDlcParms),
GET_SEGMENT(&dosParms->pNcbParms),
GET_OFFSET(&dosParms->pNcbParms)
);
if (pAdapterParms) {
DBGPRINT(
"\n"
"ADAPTER_PARMS @%04x:%04x\n"
"open error %04x\n"
"open options %04x\n"
"node address %02x-%02x-%02x-%02x-%02x-%02x\n"
"group address %08x\n"
"func. address %08x\n"
"# rcv buffers %04x\n"
"rcv buf len %04x\n"
"DHB len %04x\n"
"# DHBs %02x\n"
"Reserved %02x\n"
"Open Lock %04x\n"
"Product ID %04x:%04x\n",
GET_SEGMENT(&dosParms->pAdapterParms),
GET_OFFSET(&dosParms->pAdapterParms),
READ_WORD(&pAdapterParms->OpenErrorCode),
READ_WORD(&pAdapterParms->OpenOptions),
READ_BYTE(&pAdapterParms->NodeAddress[0]),
READ_BYTE(&pAdapterParms->NodeAddress[1]),
READ_BYTE(&pAdapterParms->NodeAddress[2]),
READ_BYTE(&pAdapterParms->NodeAddress[3]),
READ_BYTE(&pAdapterParms->NodeAddress[4]),
READ_BYTE(&pAdapterParms->NodeAddress[5]),
READ_DWORD(&pAdapterParms->GroupAddress),
READ_DWORD(&pAdapterParms->FunctionalAddress),
READ_WORD(&pAdapterParms->NumberReceiveBuffers),
READ_WORD(&pAdapterParms->ReceiveBufferLength),
READ_WORD(&pAdapterParms->DataHoldBufferLength),
READ_BYTE(&pAdapterParms->NumberDataHoldBuffers),
READ_BYTE(&pAdapterParms->Reserved),
READ_WORD(&pAdapterParms->OpenLock),
GET_SEGMENT(&pAdapterParms->ProductId),
GET_OFFSET(&pAdapterParms->ProductId)
);
pProductId = READ_FAR_POINTER(&pAdapterParms->ProductId);
if (pProductId) {
DBGPRINT("\nPRODUCT ID:\n");
for (i=0; i<18; ++i) {
DBGPRINT("%02x ", *pProductId++);
}
DBGPRINT("\n");
}
}
if (pDirectParms) {
DBGPRINT(
"\n"
"DIRECT_PARMS @%04x:%04x\n"
"dir buf size %04x\n"
"dir pool blx %04x\n"
"dir buf pool %04x:%04x\n"
"adap chk exit %04x:%04x\n"
"nw stat exit %04x:%04x\n"
"pc error exit %04x:%04x\n"
"adap wrk area %04x:%04x\n"
"adap wrk req. %04x\n"
"adap wrk act %04x\n",
GET_SEGMENT(&dosParms->pDirectParms),
GET_OFFSET(&dosParms->pDirectParms),
READ_WORD(&pDirectParms->DirectBufferSize),
READ_WORD(&pDirectParms->DirectPoolBlocks),
GET_SEGMENT(&pDirectParms->DirectBufferPool),
GET_OFFSET(&pDirectParms->DirectBufferPool),
GET_SEGMENT(&pDirectParms->AdapterCheckExit),
GET_OFFSET(&pDirectParms->AdapterCheckExit),
GET_SEGMENT(&pDirectParms->NetworkStatusExit),
GET_OFFSET(&pDirectParms->NetworkStatusExit),
GET_SEGMENT(&pDirectParms->PcErrorExit),
GET_OFFSET(&pDirectParms->PcErrorExit),
GET_SEGMENT(&pDirectParms->AdapterWorkArea),
GET_OFFSET(&pDirectParms->AdapterWorkArea),
READ_WORD(&pDirectParms->AdapterWorkAreaRequested),
READ_WORD(&pDirectParms->AdapterWorkAreaActual)
);
}
if (pDlcParms) {
DBGPRINT(
"\n"
"DLC_PARMS @%04x:%04x\n"
"max SAPs %02x\n"
"max links %02x\n"
"max grp SAPs %02x\n"
"max grp memb %02x\n"
"T1 tick 1 %02x\n"
"T2 tick 1 %02x\n"
"Ti tick 1 %02x\n"
"T1 tick 2 %02x\n"
"T2 tick 2 %02x\n"
"Ti tick 2 %02x\n",
GET_SEGMENT(&dosParms->pDlcParms),
GET_OFFSET(&dosParms->pDlcParms),
READ_BYTE(&pDlcParms->MaxSaps),
READ_BYTE(&pDlcParms->MaxStations),
READ_BYTE(&pDlcParms->MaxGroupSaps),
READ_BYTE(&pDlcParms->MaxGroupMembers),
READ_BYTE(&pDlcParms->T1Tick1),
READ_BYTE(&pDlcParms->T2Tick1),
READ_BYTE(&pDlcParms->TiTick1),
READ_BYTE(&pDlcParms->T1Tick2),
READ_BYTE(&pDlcParms->T2Tick2),
READ_BYTE(&pDlcParms->TiTick2)
);
}
if (pNcbParms) {
DBGPRINT(
"\n"
"NCB_PARMS @%04x:%04x???\n",
GET_SEGMENT(&dosParms->pNcbParms),
GET_OFFSET(&dosParms->pNcbParms)
);
}
} else {
PLLC_ADAPTER_OPEN_PARMS pAdapterParms = ntParms->pAdapterParms;
PLLC_EXTENDED_ADAPTER_PARMS pExtendedParms = ntParms->pExtendedParms;
PLLC_DLC_PARMS pDlcParms = ntParms->pDlcParms;
PVOID pNcbParms = ntParms->pReserved1;
DBGPRINT(
"adapter parms %08x\n"
"direct parms %08x\n"
"DLC parms %08x\n"
"NCB parms %08x\n",
pAdapterParms,
pExtendedParms,
pDlcParms,
pNcbParms
);
if (pAdapterParms) {
DBGPRINT(
"\n"
"ADAPTER_PARMS @%08x\n"
"open error %04x\n"
"open options %04x\n"
"node address %02x-%02x-%02x-%02x-%02x-%02x\n"
"group address %08x\n"
"func. address %08x\n"
"reserved 1 %04x\n"
"reserved 2 %04x\n"
"max frame len %04x\n"
"reserved 3[0] %04x\n"
"reserved 3[1] %04x\n"
"reserved 3[2] %04x\n"
"reserved 3[3] %04x\n"
"bring ups %04x\n"
"init warnings %04x\n"
"reserved 4[0] %04x\n"
"reserved 4[1] %04x\n"
"reserved 4[2] %04x\n",
pAdapterParms,
pAdapterParms->usOpenErrorCode,
pAdapterParms->usOpenOptions,
pAdapterParms->auchNodeAddress[0],
pAdapterParms->auchNodeAddress[1],
pAdapterParms->auchNodeAddress[2],
pAdapterParms->auchNodeAddress[3],
pAdapterParms->auchNodeAddress[4],
pAdapterParms->auchNodeAddress[5],
*(UNALIGNED DWORD *)&pAdapterParms->auchGroupAddress,
*(UNALIGNED DWORD *)&pAdapterParms->auchFunctionalAddress,
pAdapterParms->usReserved1,
pAdapterParms->usReserved2,
pAdapterParms->usMaxFrameSize,
pAdapterParms->usReserved3[0],
pAdapterParms->usReserved3[1],
pAdapterParms->usReserved3[2],
pAdapterParms->usReserved3[3],
pAdapterParms->usBringUps,
pAdapterParms->InitWarnings,
pAdapterParms->usReserved4[0],
pAdapterParms->usReserved4[1],
pAdapterParms->usReserved4[2]
);
}
if (pExtendedParms) {
DBGPRINT(
"\n"
"EXTENDED PARMS @%08x\n"
"hBufferPool %08x\n"
"pSecurityDesc %08x\n"
"Ethernet Type %08x\n",
pExtendedParms,
pExtendedParms->hBufferPool,
pExtendedParms->pSecurityDescriptor,
pExtendedParms->LlcEthernetType
);
}
if (pDlcParms) {
DBGPRINT(
"\n"
"DLC_PARMS @%04x:%04x\n"
"max SAPs %02x\n"
"max links %02x\n"
"max grp SAPs %02x\n"
"max grp memb %02x\n"
"T1 tick 1 %02x\n"
"T2 tick 1 %02x\n"
"Ti tick 1 %02x\n"
"T1 tick 2 %02x\n"
"T2 tick 2 %02x\n"
"Ti tick 2 %02x\n",
pDlcParms,
pDlcParms->uchDlcMaxSaps,
pDlcParms->uchDlcMaxStations,
pDlcParms->uchDlcMaxGroupSaps,
pDlcParms->uchDlcMaxGroupMembers,
pDlcParms->uchT1_TickOne,
pDlcParms->uchT2_TickOne,
pDlcParms->uchTi_TickOne,
pDlcParms->uchT1_TickTwo,
pDlcParms->uchT2_TickTwo,
pDlcParms->uchTi_TickTwo
);
}
if (pNcbParms) {
DBGPRINT(
"\n"
"NCB_PARMS @%08x???\n",
pNcbParms
);
}
}
}
PRIVATE
VOID
DumpDirReadLog(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("DIR.READ.LOG", Parameters, IsDos, Segment, Offset);
}
PRIVATE
VOID
DumpDirRestoreOpenParmsParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpDirSetFunctionalAddressParms(
DUMP_TABLE_PARMS
)
{
DBGPRINT( "funct addr %08lx\n", Parameters);
}
PRIVATE
VOID
DumpDirSetGroupAddressParms(
DUMP_TABLE_PARMS
)
{
DBGPRINT( "group addr %08lx\n", Parameters);
}
PRIVATE
VOID
DumpDirSetUserAppendageParms(
DUMP_TABLE_PARMS
)
{
PLLC_DIR_SET_USER_APPENDAGE_PARMS parms = (PLLC_DIR_SET_USER_APPENDAGE_PARMS)Parameters;
DumpParameterTableHeader("DIR.SET.USER.APPENDAGE", Parameters, IsDos, Segment, Offset);
if (IsDos) {
DBGPRINT( "adapt check %04x:%04x\n"
"n/w status %04x:%04x\n"
"w/s error %04x:%04x\n",
GET_SEGMENT(&parms->dpAdapterCheckExit),
GET_OFFSET(&parms->dpAdapterCheckExit),
GET_SEGMENT(&parms->dpNetworkStatusExit),
GET_OFFSET(&parms->dpNetworkStatusExit),
GET_SEGMENT(&parms->dpPcErrorExit),
GET_OFFSET(&parms->dpPcErrorExit)
);
}
}
PRIVATE
VOID
DumpDirStatusParms(
DUMP_TABLE_PARMS
)
{
PDOS_DIR_STATUS_PARMS dosParms = (PDOS_DIR_STATUS_PARMS)Parameters;
DumpParameterTableHeader("DIR.STATUS", Parameters, IsDos, Segment, Offset);
if (IsDos) {
DBGPRINT( "perm addr %02x-%02x-%02x-%02x-%02x-%02x\n"
"local addr %02x-%02x-%02x-%02x-%02x-%02x\n"
"group addr %08lx\n"
"func addr %08lx\n"
"max SAPs %02x\n"
"open SAPs %02x\n"
"max links %02x\n"
"open links %02x\n"
"avail links %02x\n"
"adapt config %02x\n"
"ucode level %02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x\n"
"adap parms %04x:%04x\n"
"adap MAC %04x:%04x\n"
"timer tick %04x:%04x\n"
"last NW stat %04x\n"
"ext. status %04x:%04x\n",
dosParms->auchPermanentAddress[0],
dosParms->auchPermanentAddress[1],
dosParms->auchPermanentAddress[2],
dosParms->auchPermanentAddress[3],
dosParms->auchPermanentAddress[4],
dosParms->auchPermanentAddress[5],
dosParms->auchNodeAddress[0],
dosParms->auchNodeAddress[1],
dosParms->auchNodeAddress[2],
dosParms->auchNodeAddress[3],
dosParms->auchNodeAddress[4],
dosParms->auchNodeAddress[5],
READ_DWORD(&dosParms->auchGroupAddress),
READ_DWORD(&dosParms->auchFunctAddr),
dosParms->uchMaxSap,
dosParms->uchOpenSaps,
dosParms->uchMaxStations,
dosParms->uchOpenStation,
dosParms->uchAvailStations,
dosParms->uchAdapterConfig,
dosParms->auchMicroCodeLevel[0],
dosParms->auchMicroCodeLevel[1],
dosParms->auchMicroCodeLevel[2],
dosParms->auchMicroCodeLevel[3],
dosParms->auchMicroCodeLevel[4],
dosParms->auchMicroCodeLevel[5],
dosParms->auchMicroCodeLevel[6],
dosParms->auchMicroCodeLevel[7],
dosParms->auchMicroCodeLevel[8],
dosParms->auchMicroCodeLevel[9],
GET_SEGMENT(&dosParms->dpAdapterParmsAddr),
GET_OFFSET(&dosParms->dpAdapterParmsAddr),
GET_SEGMENT(&dosParms->dpAdapterMacAddr),
GET_OFFSET(&dosParms->dpAdapterMacAddr),
GET_SEGMENT(&dosParms->dpTimerTick),
GET_OFFSET(&dosParms->dpTimerTick),
READ_WORD(&dosParms->usLastNetworkStatus),
GET_SEGMENT(&dosParms->dpExtendedParms),
GET_OFFSET(&dosParms->dpExtendedParms)
);
} else {
DBGPRINT("no dump for this table yet\n");
}
}
PRIVATE
VOID
DumpDirTimerCancelParms(
DUMP_TABLE_PARMS
)
{
if (IsDos) {
DBGPRINT( "cancel timer %04x:%04x\n",
HIWORD(Parameters),
LOWORD(Parameters)
);
}
}
PRIVATE
VOID
DumpDirTimerCancelGroupParms(
DUMP_TABLE_PARMS
)
{
if (IsDos) {
DBGPRINT( "cancel timer %04x:%04x\n",
HIWORD(Parameters),
LOWORD(Parameters)
);
}
}
PRIVATE
VOID
DumpDirTimerSetParms(
DUMP_TABLE_PARMS
)
{
if (IsDos) {
DBGPRINT( "timer value %04x\n",
LOWORD(Parameters)
);
}
}
PRIVATE
VOID
DumpDlcCloseSapParms(
DUMP_TABLE_PARMS
)
{
if (IsDos) {
DBGPRINT( "STATION_ID %04x\n"
"reserved %04x\n",
LOWORD(Parameters),
HIWORD(Parameters)
);
}
}
PRIVATE
VOID
DumpDlcCloseStationParms(
DUMP_TABLE_PARMS
)
{
if (IsDos) {
DBGPRINT( "STATION_ID %04x\n"
"reserved %02x %02x\n",
LOWORD(Parameters),
LOBYTE(HIWORD(Parameters)),
HIBYTE(HIWORD(Parameters))
);
}
}
PRIVATE
VOID
DumpDlcConnectStationParms(
DUMP_TABLE_PARMS
)
{
LLC_DLC_CONNECT_PARMS UNALIGNED * parms = (PLLC_DLC_CONNECT_PARMS)Parameters;
ULPBYTE routing = NULL;
int i;
DumpParameterTableHeader("DLC.CONNECT.STATION", Parameters, IsDos, Segment, Offset);
DBGPRINT( "station id %04x\n"
"reserved %02x %02x\n",
READ_WORD(&parms->usStationId),
((ULPBYTE)(&parms->usReserved))[0],
((ULPBYTE)(&parms->usReserved))[1]
);
if (IsDos) {
DBGPRINT(
"routing addr %04x:%04x\n",
GET_SEGMENT(&parms->pRoutingInfo),
GET_OFFSET(&parms->pRoutingInfo)
);
routing = READ_FAR_POINTER(&parms->pRoutingInfo);
} else {
DBGPRINT(
"routing addr %08x\n",
parms->pRoutingInfo
);
routing = parms->pRoutingInfo;
}
if (routing) {
DBGPRINT("ROUTING INFO: ");
for (i=0; i<18; ++i) {
DBGPRINT("%02x ", routing[i]);
}
DBGPRINT("\n");
}
}
PRIVATE
VOID
DumpDlcFlowControlParms(
DUMP_TABLE_PARMS
)
{
DBGPRINT(
"STATION_ID %04x\n"
"flow control %02x [%s]\n"
"reserved %02x\n",
LOWORD(Parameters),
LOBYTE(HIWORD(Parameters)),
MapFlowControl(LOBYTE(HIWORD(Parameters))),
HIBYTE(HIWORD(Parameters))
);
}
PRIVATE LPSTR MapFlowControl(BYTE FlowControl) {
if (FlowControl & 0x80) {
if (FlowControl & 0x40) {
return "reset local_busy(buffer)";
} else {
return "reset local_busy(user)";
}
} else {
return "set local_busy(user)";
}
}
PRIVATE
VOID
DumpDlcModifyParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpDlcOpenSapParms(
DUMP_TABLE_PARMS
)
{
PLLC_DLC_OPEN_SAP_PARMS parms = (PLLC_DLC_OPEN_SAP_PARMS)Parameters;
DumpParameterTableHeader("DLC.OPEN.SAP", Parameters, IsDos, Segment, Offset);
DBGPRINT( "station id %04x\n"
"user stat %04x\n"
"T1 %02x\n"
"T2 %02x\n"
"Ti %02x\n"
"max out %02x\n"
"max in %02x\n"
"max out inc %02x\n"
"max retry %02x\n"
"max members %02x\n"
"max I field %04x\n"
"SAP value %02x\n"
"options %02x [%s]\n"
"link count %02x\n"
"reserved %02x %02x\n"
"group count %02x\n",
READ_WORD(&parms->usStationId),
READ_WORD(&parms->usUserStatValue),
parms->uchT1,
parms->uchT2,
parms->uchTi,
parms->uchMaxOut,
parms->uchMaxIn,
parms->uchMaxOutIncr,
parms->uchMaxRetryCnt,
parms->uchMaxMembers,
READ_WORD(&parms->usMaxI_Field),
parms->uchSapValue,
parms->uchOptionsPriority,
MapOptionsPriority(parms->uchOptionsPriority),
parms->uchcStationCount,
parms->uchReserved2[0],
parms->uchReserved2[1],
parms->cGroupCount
);
if (IsDos) {
DBGPRINT(
"group list %04x:%04x\n"
"dlc stat app %04x:%04x\n",
GET_SEGMENT(&parms->pGroupList),
GET_OFFSET(&parms->pGroupList),
GET_SEGMENT(&parms->DlcStatusFlags),
GET_OFFSET(&parms->DlcStatusFlags)
);
//
// some code here to dump group list
//
} else {
DBGPRINT(
"group list %08x\n"
"dlc status %08x\n",
parms->pGroupList,
parms->DlcStatusFlags
);
}
DBGPRINT( "buffer size %04x\n"
"pool length %04x\n",
READ_WORD(&parms->uchReserved3[0]),
READ_WORD(&parms->uchReserved3[2])
);
if (IsDos) {
DBGPRINT(
"buffer pool %04x:%04x\n",
READ_WORD(&parms->uchReserved3[6]),
READ_WORD(&parms->uchReserved3[4])
);
} else {
DBGPRINT(
"buffer pool %08x\n",
*(LPDWORD)&parms->uchReserved3[4]
);
}
}
PRIVATE LPSTR MapOptionsPriority(UCHAR OptionsPriority) {
static char buf[80];
char* bufptr = buf;
bufptr += sprintf(buf, "Access Priority=%d", (OptionsPriority & 0xe0) >> 5);
if (OptionsPriority & 8) {
bufptr += sprintf(bufptr, " XID handled by APP");
} else {
bufptr += sprintf(bufptr, " XID handled by DLC");
}
if (OptionsPriority & 4) {
bufptr += sprintf(bufptr, " Individual SAP");
}
if (OptionsPriority & 2) {
bufptr += sprintf(bufptr, " Group SAP");
}
if (OptionsPriority & 1) {
bufptr += sprintf(bufptr, " Group Member SAP");
}
return buf;
}
PRIVATE
VOID
DumpDlcOpenStationParms(
DUMP_TABLE_PARMS
)
{
PLLC_DLC_OPEN_STATION_PARMS parms = (PLLC_DLC_OPEN_STATION_PARMS)Parameters;
ULPBYTE dest = NULL;
int i;
DumpParameterTableHeader("DLC.OPEN.STATION", Parameters, IsDos, Segment, Offset);
DBGPRINT( "sap station %04x\n"
"link station %04x\n"
"T1 %02x\n"
"T2 %02x\n"
"Ti %02x\n"
"max out %02x\n"
"max in %02x\n"
"max out inc %02x\n"
"max retry %02x\n"
"remote SAP %02x\n"
"max I field %04x\n"
"access pri %02x\n",
READ_WORD(&parms->usSapStationId),
READ_WORD(&parms->usLinkStationId),
parms->uchT1,
parms->uchT2,
parms->uchTi,
parms->uchMaxOut,
parms->uchMaxIn,
parms->uchMaxOutIncr,
parms->uchMaxRetryCnt,
parms->uchRemoteSap,
READ_WORD(&parms->usMaxI_Field),
parms->uchAccessPriority
);
if (IsDos) {
DBGPRINT(
"destination %04x:%04x\n",
GET_SEGMENT(&parms->pRemoteNodeAddress),
GET_OFFSET(&parms->pRemoteNodeAddress)
);
dest = READ_FAR_POINTER(&parms->pRemoteNodeAddress);
} else {
DBGPRINT(
"destination %08x\n",
parms->pRemoteNodeAddress
);
dest = parms->pRemoteNodeAddress;
}
if (dest) {
DBGPRINT("DESTINATION ADDRESS: ");
for (i=0; i<6; ++i) {
DBGPRINT("%02x ", dest[i]);
}
DBGPRINT("\n");
}
}
PRIVATE
VOID
DumpDlcReallocateParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpDlcResetParms(
DUMP_TABLE_PARMS
)
{
DBGPRINT( "STATION_ID %04x\n"
"reserved %02x %02x\n",
LOWORD(Parameters),
LOBYTE(HIWORD(Parameters)),
HIBYTE(HIWORD(Parameters))
);
}
PRIVATE
VOID
DumpDlcStatisticsParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpPdtTraceOffParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpPdtTraceOnParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpReadParms(
DUMP_TABLE_PARMS
)
{
PLLC_READ_PARMS parms = (PLLC_READ_PARMS)Parameters;
DumpParameterTableHeader("READ", Parameters, IsDos, Segment, Offset);
//
// this parameter table not for DOS
//
DBGPRINT( "station id %04x\n"
"option ind. %02x\n"
"event set %02x\n"
"event %02x [%s]\n"
"crit. subset %02x\n"
"notify flag %08x\n",
parms->usStationId,
parms->uchOptionIndicator,
parms->uchEventSet,
parms->uchEvent,
MapReadEvent(parms->uchEvent),
parms->uchCriticalSubset,
parms->ulNotificationFlag
);
//
// rest of table interpreted differently depending on whether status change
//
if (parms->uchEvent & 0x38) {
DBGPRINT(
"station id %04x\n"
"status code %04x [%s]\n"
"FRMR data %02x %02x %02x %02x %02x\n"
"access pri. %02x\n"
"remote addr %02x-%02x-%02x-%02x-%02x-%02x\n"
"remote SAP %02x\n"
"reserved %02x\n"
"user stat %04x\n",
parms->Type.Status.usStationId,
parms->Type.Status.usDlcStatusCode,
MapDlcStatus(parms->Type.Status.usDlcStatusCode),
parms->Type.Status.uchFrmrData[0],
parms->Type.Status.uchFrmrData[1],
parms->Type.Status.uchFrmrData[2],
parms->Type.Status.uchFrmrData[3],
parms->Type.Status.uchFrmrData[4],
parms->Type.Status.uchAccessPritority,
parms->Type.Status.uchRemoteNodeAddress[0],
parms->Type.Status.uchRemoteNodeAddress[1],
parms->Type.Status.uchRemoteNodeAddress[2],
parms->Type.Status.uchRemoteNodeAddress[3],
parms->Type.Status.uchRemoteNodeAddress[4],
parms->Type.Status.uchRemoteNodeAddress[5],
parms->Type.Status.uchRemoteSap,
parms->Type.Status.uchReserved,
parms->Type.Status.usUserStatusValue
);
} else {
DBGPRINT(
"CCB count %04x\n"
"CCB list %08x\n"
"buffer count %04x\n"
"buffer list %08x\n"
"frame count %04x\n"
"frame list %08x\n"
"error code %04x\n"
"error data %04x %04x %04x\n",
parms->Type.Event.usCcbCount,
parms->Type.Event.pCcbCompletionList,
parms->Type.Event.usBufferCount,
parms->Type.Event.pFirstBuffer,
parms->Type.Event.usReceivedFrameCount,
parms->Type.Event.pReceivedFrame,
parms->Type.Event.usEventErrorCode,
parms->Type.Event.usEventErrorData[0],
parms->Type.Event.usEventErrorData[1],
parms->Type.Event.usEventErrorData[2]
);
//
// address of CCB is in DOS memory
//
if (parms->Type.Event.usCcbCount) {
DumpCcb(DOS_PTR_TO_FLAT(parms->Type.Event.pCcbCompletionList),
TRUE, // DumpAll
FALSE, // CcbIsInput
TRUE, // IsDos
HIWORD(parms->Type.Event.pCcbCompletionList),
LOWORD(parms->Type.Event.pCcbCompletionList)
);
}
if (parms->Type.Event.usReceivedFrameCount) {
DumpReceiveDataBuffer(parms->Type.Event.pReceivedFrame, FALSE, 0, 0);
}
}
}
PRIVATE LPSTR MapReadEvent(UCHAR Event) {
switch (Event) {
case 0x80:
return "Reserved Event!";
case 0x40:
return "System Action (non-critical)";
case 0x20:
return "Network Status (non-critical)";
case 0x10:
return "Critical Exception";
case 0x8:
return "DLC Status Change";
case 0x4:
return "Receive Data";
case 0x2:
return "Transmit Completion";
case 0x1:
return "Command Completion";
}
return "Unknown Read Event";
}
PRIVATE LPSTR MapDlcStatus(WORD Status) {
if (Status & 0x8000) {
return "Link lost";
} else if (Status & 0x4000) {
return "DM/DISC Received -or- DISC ack'd";
} else if (Status & 0x2000) {
return "FRMR Received";
} else if (Status & 0x1000) {
return "FRMR Sent";
} else if (Status & 0x0800) {
return "SABME Received for open link station";
} else if (Status & 0x0400) {
return "SABME Received - link station opened";
} else if (Status & 0x0200) {
return "REMOTE Busy Entered";
} else if (Status & 0x0100) {
return "REMOTE Busy Left";
} else if (Status & 0x0080) {
return "Ti EXPIRED";
} else if (Status & 0x0040) {
return "DLC counter overflow - issue DLC.STATISTICS";
} else if (Status & 0x0020) {
return "Access Priority lowered";
} else if (Status & 0x001e) {
return "*** ERROR - INVALID STATUS ***";
} else if (Status & 0x0001) {
return "Entered LOCAL Busy";
} else {
return "Unknown DLC Status";
}
}
PRIVATE
VOID
DumpReadCancelParms(
DUMP_TABLE_PARMS
)
{
}
PRIVATE
VOID
DumpReceiveParms(
DUMP_TABLE_PARMS
)
{
//
// the format of the recieve parameter table is different depending on
// whether this is a DOS command (CCB1) or NT (CCB2)
//
PLLC_RECEIVE_PARMS ntParms = (PLLC_RECEIVE_PARMS)Parameters;
PLLC_DOS_RECEIVE_PARMS dosParms = (PLLC_DOS_RECEIVE_PARMS)Parameters;
PLLC_DOS_RECEIVE_PARMS_EX dosExParms = (PLLC_DOS_RECEIVE_PARMS_EX)Parameters;
PVOID Buffer;
DumpParameterTableHeader("RECEIVE", Parameters, IsDos, Segment, Offset);
//
// some common bits: use any structure pointer
//
DBGPRINT( "station id %04x\n"
"user length %04x\n",
READ_WORD(&ntParms->usStationId),
READ_WORD(&ntParms->usUserLength)
);
//
// dump segmented pointers for DOS, flat for NT
//
if (IsDos) {
DBGPRINT(
"receive exit %04x:%04x\n"
"first buffer %04x:%04x\n",
GET_SEGMENT(&dosParms->ulReceiveExit),
GET_OFFSET(&dosParms->ulReceiveExit),
GET_SEGMENT(&dosParms->pFirstBuffer),
GET_OFFSET(&dosParms->pFirstBuffer)
);
Buffer = READ_FAR_POINTER(&dosParms->pFirstBuffer);
//
// use Segment & Offset to address received data buffer
//
Segment = GET_SEGMENT(&dosParms->pFirstBuffer);
Offset = GET_OFFSET(&dosParms->pFirstBuffer);
} else {
DBGPRINT(
"receive flag %08x\n"
"first buffer %08x\n",
ntParms->ulReceiveFlag,
ntParms->pFirstBuffer
);
Buffer = ntParms->pFirstBuffer;
}
//
// more common bits
//
DBGPRINT( "options %02x\n",
ntParms->uchOptions
);
if (!IsDos) {
DBGPRINT(
"reserved1 %02x %02x %02x\n"
"read options %02x\n"
"reserved2 %02x %02x %02x\n"
"original CCB %08x\n"
"orig. exit %08x\n",
ntParms->auchReserved1[0],
ntParms->auchReserved1[1],
ntParms->auchReserved1[2],
ntParms->uchRcvReadOption,
((PLLC_DOS_RECEIVE_PARMS_EX)ntParms)->auchReserved2[0],
((PLLC_DOS_RECEIVE_PARMS_EX)ntParms)->auchReserved2[1],
((PLLC_DOS_RECEIVE_PARMS_EX)ntParms)->auchReserved2[2],
((PLLC_DOS_RECEIVE_PARMS_EX)ntParms)->dpOriginalCcbAddress,
((PLLC_DOS_RECEIVE_PARMS_EX)ntParms)->dpCompletionFlag
);
/* } else {
//
// we have no way of knowing from the general purpose parameters if this
// is the original DOS CCB1 RECEIVE parameter table, or the extended
// RECEIVE parameter table that we create. Dump the extended bits for
// DOS anyhow
//
DBGPRINT(
"\nExtended RECEIVE parameters for table @%08x\n"
"reserved1 %02x %02x %02x\n"
"read options %02x\n"
"reserved2 %02x %02x %02x\n"
"original CCB %04x:%04x\n"
"orig. exit %04x:%04x\n",
Parameters,
dosExParms->auchReserved1[0],
dosExParms->auchReserved1[1],
dosExParms->auchReserved1[2],
dosExParms->uchRcvReadOption,
dosExParms->auchReserved2[0],
dosExParms->auchReserved2[1],
dosExParms->auchReserved2[2],
GET_SEGMENT(&dosExParms->dpOriginalCcbAddress),
GET_OFFSET(&dosExParms->dpOriginalCcbAddress),
GET_SEGMENT(&dosExParms->dpCompletionFlag),
GET_OFFSET(&dosExParms->dpCompletionFlag)
);
*/
}
//
// only dump the buffer(s) if this is an output CCB dump
//
if (Buffer && !IsInput) {
DumpReceiveDataBuffer(Buffer, IsDos, Segment, Offset);
}
}
PRIVATE
VOID
DumpReceiveCancelParms(
DUMP_TABLE_PARMS
)
{
DBGPRINT("STATION_ID %04x\n", LOWORD(Parameters));
}
PRIVATE
VOID
DumpReceiveModifyParms(
DUMP_TABLE_PARMS
)
{
PLLC_DOS_RECEIVE_MODIFY_PARMS parms = (PLLC_DOS_RECEIVE_MODIFY_PARMS)Parameters;
PVOID Buffer;
DumpParameterTableHeader("RECEIVE.MODIFY", Parameters, IsDos, Segment, Offset);
DBGPRINT( "station id %04x\n"
"user length %04x\n"
"receive exit %04x:%04x\n"
"first buffer %04x:%04x\n"
"subroutine %04x:%04x\n",
READ_WORD(&parms->StationId),
READ_WORD(&parms->UserLength),
GET_SEGMENT(&parms->ReceivedDataExit),
GET_OFFSET(&parms->ReceivedDataExit),
GET_SEGMENT(&parms->FirstBuffer),
GET_OFFSET(&parms->FirstBuffer),
GET_SEGMENT(&parms->Subroutine),
GET_OFFSET(&parms->Subroutine)
);
Buffer = READ_FAR_POINTER(&parms->FirstBuffer);
if (Buffer) {
DumpReceiveDataBuffer(Buffer, IsDos, Segment, Offset);
}
}
PRIVATE
VOID
DumpTransmitDirFrameParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.DIR.FRAME", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitIFrameParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.I.FRAME", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitTestCmdParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.TEST.CMD", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitUiFrameParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.UI.FRAME", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitXidCmdParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.XID.CMD", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitXidRespFinalParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.XID.RESP.FINAL", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitXidRespNotFinalParms(
DUMP_TABLE_PARMS
)
{
DumpParameterTableHeader("TRANSMIT.XID.RESP.NOT.FINAL", Parameters, IsDos, Segment, Offset);
DumpTransmitParms(Parameters, IsDos, IsInput, Segment, Offset);
}
PRIVATE
VOID
DumpTransmitParms(
DUMP_TABLE_PARMS
)
{
PLLC_TRANSMIT_PARMS ntParms = (PLLC_TRANSMIT_PARMS)Parameters;
PLLC_DOS_TRANSMIT_PARMS dosParms = (PLLC_DOS_TRANSMIT_PARMS)Parameters;
DBGPRINT( "station id %04x\n"
"frame status %02x\n"
"remote SAP %02x\n",
READ_WORD(&dosParms->usStationId),
dosParms->uchTransmitFs,
dosParms->uchRemoteSap
);
if (IsDos) {
DBGPRINT(
"xmit q1 %04x:%04x\n"
"xmit q2 %04x:%04x\n"
"buf. len. 1 %04x\n"
"buf. len. 2 %04x\n"
"buffer 1 %04x:%04x\n"
"buffer 2 %04x:%04x\n",
GET_SEGMENT(&dosParms->pXmitQueue1),
GET_OFFSET(&dosParms->pXmitQueue1),
GET_SEGMENT(&dosParms->pXmitQueue2),
GET_OFFSET(&dosParms->pXmitQueue2),
READ_WORD(&dosParms->cbBuffer1),
READ_WORD(&dosParms->cbBuffer2),
GET_SEGMENT(&dosParms->pBuffer1),
GET_OFFSET(&dosParms->pBuffer1),
GET_SEGMENT(&dosParms->pBuffer2),
GET_OFFSET(&dosParms->pBuffer2)
);
IF_DEBUG(DLC_TX_DATA) {
if (READ_DWORD(&dosParms->pXmitQueue1)) {
DBGPRINT("\nXMIT_QUEUE_ONE:\n");
DumpTransmitQueue(READ_DWORD(&dosParms->pXmitQueue1));
}
if (READ_DWORD(&dosParms->pXmitQueue2)) {
DBGPRINT("\nXMIT_QUEUE_TWO:\n");
DumpTransmitQueue(READ_DWORD(&dosParms->pXmitQueue2));
}
if (dosParms->cbBuffer1) {
DBGPRINT("\nBUFFER1:\n");
DumpData(NULL,
NULL,
dosParms->cbBuffer1,
DD_UPPER_CASE,
0,
TRUE,
GET_SEGMENT(&dosParms->pBuffer1),
GET_OFFSET(&dosParms->pBuffer1)
);
}
if (dosParms->cbBuffer2) {
DBGPRINT("\nBUFFER2:\n");
DumpData(NULL,
NULL,
dosParms->cbBuffer2,
DD_UPPER_CASE,
0,
TRUE,
GET_SEGMENT(&dosParms->pBuffer2),
GET_OFFSET(&dosParms->pBuffer2)
);
}
}
} else {
DBGPRINT(
"xmit q1 %08x\n"
"xmit q2 %08x\n"
"buf. len. 1 %02x\n"
"buf. len. 2 %02x\n"
"buffer 1 %08x\n"
"buffer 2 %08x\n"
"xmt read opt %02x\n",
ntParms->pXmitQueue1,
ntParms->pXmitQueue2,
ntParms->cbBuffer1,
ntParms->cbBuffer2,
ntParms->pBuffer1,
ntParms->pBuffer2,
ntParms->uchXmitReadOption
);
}
}
PRIVATE
VOID
DumpTransmitQueue(
IN DOS_ADDRESS dpQueue
)
{
PLLC_XMIT_BUFFER pTxBuffer;
WORD userLength;
WORD dataLength;
while (dpQueue) {
pTxBuffer = (PLLC_XMIT_BUFFER)DOS_PTR_TO_FLAT(dpQueue);
dataLength = READ_WORD(&pTxBuffer->cbBuffer);
userLength = READ_WORD(&pTxBuffer->cbUserData);
DBGPRINT(
"\n"
"Transmit Buffer @%04x:%04x\n"
"next buffer %04x:%04x\n"
"reserved %02x %02x\n"
"data length %04x\n"
"user data %04x\n"
"user length %04x\n",
HIWORD(dpQueue),
LOWORD(dpQueue),
GET_SEGMENT(&pTxBuffer->pNext),
GET_OFFSET(&pTxBuffer->pNext),
((LPBYTE)(&pTxBuffer->usReserved1))[0],
((LPBYTE)(&pTxBuffer->usReserved1))[1],
dataLength,
READ_WORD(&pTxBuffer->usReserved2),
userLength
);
DumpData("user space",
(PBYTE)(&pTxBuffer->auchData),
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE, // not displaying seg:off, so no need for these 3
0,
0
);
DumpData("xmit data",
(PBYTE)(&pTxBuffer->auchData) + userLength,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE, // not displaying seg:off, so no need for these 3
0,
0
);
dpQueue = READ_DWORD(&pTxBuffer->pNext);
}
}
VOID
DumpReceiveDataBuffer(
IN PVOID Buffer,
IN BOOL IsDos,
IN WORD Segment,
IN WORD Offset
)
{
if (IsDos) {
PLLC_DOS_BUFFER pBuf = (PLLC_DOS_BUFFER)Buffer;
BOOL contiguous = pBuf->Contiguous.uchOptions & 0xc0;
WORD userLength = READ_WORD(&pBuf->Next.cbUserData);
WORD dataLength = READ_WORD(&pBuf->Next.cbBuffer);
WORD userOffset = READ_WORD(&pBuf->Next.offUserData);
//
// Buffer 1: [not] contiguous MAC/DATA
//
DBGPRINT(
"\n"
"%sContiguous MAC/DATA frame @%04x:%04x\n"
"next buffer %04x:%04x\n"
"frame length %04x\n"
"data length %04x\n"
"user offset %04x\n"
"user length %04x\n"
"station id %04x\n"
"options %02x\n"
"message type %02x [%s]\n"
"buffers left %04x\n"
"rcv FS %02x\n"
"adapter num %02x\n",
contiguous ? "" : "Not",
Segment,
Offset,
GET_SEGMENT(&pBuf->Contiguous.pNextBuffer),
GET_OFFSET(&pBuf->Contiguous.pNextBuffer),
READ_WORD(&pBuf->Contiguous.cbFrame),
READ_WORD(&pBuf->Contiguous.cbBuffer),
READ_WORD(&pBuf->Contiguous.offUserData),
READ_WORD(&pBuf->Contiguous.cbUserData),
READ_WORD(&pBuf->Contiguous.usStationId),
pBuf->Contiguous.uchOptions,
pBuf->Contiguous.uchMsgType,
MapMessageType(pBuf->Contiguous.uchMsgType),
READ_WORD(&pBuf->Contiguous.cBuffersLeft),
pBuf->Contiguous.uchRcvFS,
pBuf->Contiguous.uchAdapterNumber
);
if (!contiguous) {
DWORD cbLanHeader = (DWORD)pBuf->NotContiguous.cbLanHeader;
DWORD cbDlcHeader = (DWORD)pBuf->NotContiguous.cbDlcHeader;
DBGPRINT(
"LAN hdr len %02x\n"
"DLC hdr len %02x\n",
cbLanHeader,
cbDlcHeader
);
DumpData("LAN header",
NULL,
(DWORD)cbLanHeader,
DD_NO_ADDRESS | DD_NO_ASCII | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
(WORD)(Offset + (WORD)&((PLLC_DOS_BUFFER)0)->NotContiguous.auchLanHeader)
);
DumpData("DLC header",
NULL,
cbDlcHeader,
DD_NO_ADDRESS | DD_NO_ASCII | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
(WORD)(Offset + (WORD)&((PLLC_DOS_BUFFER)0)->NotContiguous.auchDlcHeader)
);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space",
NULL,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + userOffset
userOffset
);
} else {
DBGPRINT(
"user space\n"
);
}
if (dataLength) {
DumpData("rcvd data",
NULL,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + userOffset + userLength
(WORD)(userOffset + userLength)
);
} else {
DBGPRINT(
"rcvd data\n"
);
}
}
} else {
//
// data length is size of frame in contiguous buffer?
//
dataLength = READ_WORD(&pBuf->Contiguous.cbBuffer);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space",
NULL,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + userOffset
userOffset
);
} else {
DBGPRINT(
"user space\n"
);
}
if (dataLength) {
DumpData("rcvd data",
NULL,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + userOffset + userLength
(WORD)(userOffset + userLength)
);
} else {
DBGPRINT(
"rcvd data\n"
);
}
}
}
//
// dump second & subsequent buffers
//
Segment = GET_SEGMENT(&pBuf->pNext);
Offset = GET_OFFSET(&pBuf->pNext);
for (
pBuf = (PLLC_DOS_BUFFER)READ_FAR_POINTER(&pBuf->pNext);
pBuf;
pBuf = (PLLC_DOS_BUFFER)READ_FAR_POINTER(&pBuf->pNext)
) {
userLength = READ_WORD(&pBuf->Next.cbUserData);
dataLength = READ_WORD(&pBuf->Next.cbBuffer);
DBGPRINT(
"\n"
"Buffer 2/Subsequent @%04x:%04x\n"
"next buffer %04x:%04x\n"
"frame length %04x\n"
"data length %04x\n"
"user offset %04x\n"
"user length %04x\n",
Segment,
Offset,
GET_SEGMENT(&pBuf->pNext),
GET_OFFSET(&pBuf->pNext),
READ_WORD(&pBuf->Next.cbFrame),
dataLength,
READ_WORD(&pBuf->Next.offUserData),
userLength
);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space",
NULL,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + READ_WORD(&pBuf->Next.offUserData)
READ_WORD(&pBuf->Next.offUserData)
);
} else {
DBGPRINT(
"user space\n"
);
}
//
// there must be received data
//
DumpData("rcvd data",
NULL,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
TRUE,
Segment,
//Offset + READ_WORD(&pBuf->Next.offUserData) + userLength
(WORD)(READ_WORD(&pBuf->Next.offUserData) + userLength)
);
}
Segment = GET_SEGMENT(&pBuf->pNext);
Offset = GET_OFFSET(&pBuf->pNext);
}
} else {
PLLC_BUFFER pBuf = (PLLC_BUFFER)Buffer;
BOOL contiguous = pBuf->Contiguous.uchOptions & 0xc0;
WORD userLength = pBuf->Next.cbUserData;
WORD dataLength = pBuf->Next.cbBuffer;
WORD userOffset = pBuf->Next.offUserData;
//
// Buffer 1: [not] contiguous MAC/DATA
//
DBGPRINT(
"\n"
"%sContiguous MAC/DATA frame @%08x\n"
"next buffer %08x\n"
"frame length %04x\n"
"data length %04x\n"
"user offset %04x\n"
"user length %04x\n"
"station id %04x\n"
"options %02x\n"
"message type %02x [%s]\n"
"buffers left %04x\n"
"rcv FS %02x\n"
"adapter num %02x\n",
contiguous ? "" : "Not",
pBuf,
pBuf->Contiguous.pNextBuffer,
pBuf->Contiguous.cbFrame,
pBuf->Contiguous.cbBuffer,
pBuf->Contiguous.offUserData,
pBuf->Contiguous.cbUserData,
pBuf->Contiguous.usStationId,
pBuf->Contiguous.uchOptions,
pBuf->Contiguous.uchMsgType,
MapMessageType(pBuf->Contiguous.uchMsgType),
pBuf->Contiguous.cBuffersLeft,
pBuf->Contiguous.uchRcvFS,
pBuf->Contiguous.uchAdapterNumber
);
if (!contiguous) {
DWORD cbLanHeader = (DWORD)pBuf->NotContiguous.cbLanHeader;
DWORD cbDlcHeader = (DWORD)pBuf->NotContiguous.cbDlcHeader;
DBGPRINT(
"next frame %08x\n"
"LAN hdr len %02x\n"
"DLC hdr len %02x\n",
pBuf->NotContiguous.pNextFrame,
cbLanHeader,
cbDlcHeader
);
DumpData("LAN header",
pBuf->NotContiguous.auchLanHeader,
cbLanHeader,
DD_NO_ADDRESS | DD_NO_ASCII | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
DumpData("DLC header",
pBuf->NotContiguous.auchDlcHeader,
cbDlcHeader,
DD_NO_ADDRESS | DD_NO_ASCII | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space ",
(PBYTE)pBuf + userOffset,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
} else {
DBGPRINT(
"user space\n"
);
}
if (dataLength) {
DumpData("rcvd data",
(PBYTE)pBuf + userOffset + userLength,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
} else {
DBGPRINT(
"rcvd data\n"
);
}
}
} else {
//
// data length is size of frame in contiguous buffer?
//
dataLength = pBuf->Contiguous.cbFrame;
DBGPRINT(
"next frame %08x\n",
pBuf->NotContiguous.pNextFrame
);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space",
(PBYTE)pBuf + userOffset,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
} else {
DBGPRINT(
"user space\n"
);
}
if (dataLength) {
DumpData("rcvd data",
(PBYTE)pBuf + userOffset + userLength,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
} else {
DBGPRINT(
"rcvd data\n"
);
}
}
}
//
// dump second & subsequent buffers
//
for (pBuf = pBuf->pNext; pBuf; pBuf = pBuf->pNext) {
userLength = pBuf->Next.cbUserData;
dataLength = pBuf->Next.cbBuffer;
DBGPRINT(
"\n"
"Buffer 2/Subsequent @%08x\n"
"next buffer %08x\n"
"frame length %04x\n"
"data length %04x\n"
"user offset %04x\n"
"user length %04x\n",
pBuf,
pBuf->pNext,
pBuf->Next.cbFrame,
dataLength,
pBuf->Next.offUserData,
userLength
);
IF_DEBUG(DLC_RX_DATA) {
if (userLength) {
DumpData("user space",
(PBYTE)&pBuf + pBuf->Next.offUserData,
userLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
} else {
DBGPRINT(
"user space\n"
);
}
//
// there must be received data
//
DumpData("rcvd data",
(PBYTE)pBuf + pBuf->Next.offUserData + userLength,
dataLength,
DD_NO_ADDRESS | DD_UPPER_CASE | DD_INDENT_ALL,
DEFAULT_FIELD_WIDTH,
FALSE,
0,
0
);
}
}
}
}
PRIVATE LPSTR MapMessageType(UCHAR MessageType) {
switch (MessageType) {
case 0x02:
return "MAC Frame (Direct Station on Token Ring only)";
case 0x04:
return "I-Frame";
case 0x06:
return "UI-Frame";
case 0x08:
return "XID Command (POLL)";
case 0x0a:
return "XID Command (not POLL)";
case 0x0c:
return "XID Response (FINAL)";
case 0x0e:
return "XID Response (not FINAL)";
case 0x10:
return "TEST Response (FINAL)";
case 0x12:
return "TEST Response (not FINAL)";
case 0x14:
return "OTHER - non-MAC frame (Direct Station only)";
default:
return "*** BAD FRAME TYPE ***";
}
}
VOID
DumpData(
IN LPSTR Title,
IN PBYTE Address,
IN DWORD Length,
IN DWORD Options,
IN DWORD Indent,
IN BOOL IsDos,
IN WORD Segment,
IN WORD Offset
)
{
char dumpBuf[128];
char* bufptr;
int i, n, iterations;
char* hexptr;
if (IsDos) {
Address = LPBYTE_FROM_WORDS(Segment, Offset);
}
//
// the usual dump style: 16 columns of hex bytes, followed by 16 columns
// of corresponding ASCII characters, or '.' where the character is < 0x20
// (space) or > 0x7f (del?)
//
if (Options & DD_LINE_BEFORE) {
DbgOutStr("\n");
}
iterations = 0;
while (Length) {
bufptr = dumpBuf;
if (Title && !iterations) {
strcpy(bufptr, Title);
bufptr = strchr(bufptr, 0);
}
if (Indent && ((Options & DD_INDENT_ALL) || iterations)) {
int indentLen = (!iterations && Title)
? ((int)Indent - (int)strlen(Title) < 0)
? 1
: Indent - strlen(Title)
: Indent;
RtlFillMemory(bufptr, indentLen, ' ');
bufptr += indentLen;
}
if (!(Options & DD_NO_ADDRESS)) {
if (IsDos) {
bufptr += sprintf(bufptr, "%04x:%04x ", Segment, Offset);
} else {
bufptr += sprintf(bufptr, "%08x: ", Address);
}
}
n = (Length < 16) ? Length : 16;
hexptr = bufptr;
for (i = 0; i < n; ++i) {
bufptr += sprintf(bufptr, "%02x", Address[i]);
*bufptr++ = (i == 7) ? '-' : ' ';
}
if (Options & DD_UPPER_CASE) {
_strupr(hexptr);
}
if (!(Options & DD_NO_ASCII)) {
if (n < 16) {
for (i = 0; i < 16-n; ++i) {
bufptr += sprintf(bufptr, " ");
}
}
bufptr += sprintf(bufptr, " ");
for (i = 0; i < n; ++i) {
*bufptr++ = (Address[i] < 0x20 || Address[i] > 0x7f) ? '.' : Address[i];
}
}
*bufptr++ = '\n';
*bufptr = 0;
DbgOutStr(dumpBuf);
Length -= n;
Address += n;
++iterations;
//
// take care of segment wrap for DOS addresses
//
if (IsDos) {
DWORD x = (DWORD)Offset + n;
Offset = (WORD)x;
if (HIWORD(x)) {
Segment += 0x1000;
}
}
}
if (Options & DD_LINE_AFTER) {
DbgOutStr("\n");
}
}
//
// CCB1 error checking
//
#define BITS_PER_BYTE 8
#define CCB1_ERROR_SPREAD ((MAX_CCB1_ERROR + BITS_PER_BYTE) & ~(BITS_PER_BYTE-1))
//
// Ccb1ErrorTable - for each command described in IBM Lan Tech. Ref. (including
// those not applicable to CCB1), we keep a list of the permissable error codes
// which are taken from the "Return Codes for CCB1 Commands" table on pp B-5
// and B-6
// The error list is an 80-bit bitmap in which an ON bit indicates that the
// error number corresponding to the bit's position is allowable for the CCB1
// command corresponding to the list's index in the table
//
typedef struct {
BOOL ValidForCcb1;
BYTE ErrorList[CCB1_ERROR_SPREAD/BITS_PER_BYTE];
char* CommandName;
} CCB1_ERROR_TABLE;
#define MAX_INCLUSIVE_CCB1_COMMAND LLC_MAX_DLC_COMMAND
CCB1_ERROR_TABLE Ccb1ErrorTable[MAX_INCLUSIVE_CCB1_COMMAND + 1] = {
// DIR.INTERRUPT (0x00)
{
TRUE,
{0x83, 0x02, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.INTERRUPT"
},
// DIR.MODIFY.OPEN.PARMS (0x01)
{
TRUE,
{0x97, 0x02, 0x40, 0x39, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.MODIFY.OPEN.PARMS"
},
// DIR.RESTORE.OPEN.PARMS (0x02)
{
TRUE,
{0xd3, 0x02, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.RESTORE.OPEN.PARMS"
},
// DIR.OPEN.ADAPTER (0x03)
{
TRUE,
{0xaf, 0x02, 0x45, 0x79, 0x00, 0x00, 0x1d, 0x00, 0x00, 0x00},
"DIR.OPEN.ADAPTER"
},
// DIR.CLOSE.ADAPTER (0x04)
{
TRUE,
{0xb3, 0x02, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.CLOSE.ADAPTER"
},
// non-existent command (0x05)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x05)"
},
// DIR.SET.GROUP.ADDRESS (0x06)
{
TRUE,
{0x93, 0x0a, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.SET.GROUP.ADDRESS"
},
// DIR.SET.FUNCTIONAL.ADDRESS (0x07)
{
TRUE,
{0x93, 0x0a, 0x00, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.SET.FUNCTIONAL.ADDRESS"
},
// DIR.READ.LOG (0x08)
{
TRUE,
{0x93, 0x0a, 0x28, 0x28, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.READ.LOG"
},
// non-existent command (0x09)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x09)"
},
// TRANSMIT.DIR.FRAME (0x0a)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.DIR.FRAME"
},
// TRANSMIT.I.FRAME (0x0b)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.I.FRAME"
},
// non-existent command (0x0c)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x0c)"
},
// TRANSMIT.UI.FRAME (0x0d)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.UI.FRAME"
},
// TRANSMIT.XID.CMD (0x0e)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.XID.CMD"
},
// TRANSMIT.XID.RESP.FINAL (0x0f)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.XID.RESP.FINAL"
},
// TRANSMIT.XID.RESP.NOT.FINAL (0x10)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.XID.RESP.NOT.FINAL"
},
// TRANSMIT.TEST.CMD (0x11)
{
TRUE,
{0x93, 0x0f, 0x00, 0x28, 0xbc, 0x01, 0x00, 0x00, 0x13, 0x04},
"TRANSMIT.TEST.CMD"
},
// non-existent command (0x12)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x12)"
},
// non-existent command (0x13)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x13)"
},
// DLC.RESET (0x14)
{
TRUE,
{0x93, 0x0a, 0x00, 0x28, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"DLC.RESET"
},
// DLC.OPEN.SAP (0x15)
{
TRUE,
{0xd3, 0x0b, 0x40, 0x39, 0x00, 0x00, 0x00, 0x00, 0x6c, 0x02},
"DLC.OPEN.SAP"
},
// DLC.CLOSE.SAP (0x16)
{
TRUE,
{0x93, 0x0a, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x81, 0x11},
"DLC.CLOSE.SAP"
},
// DLC.REALLOCATE (0x17)
{
TRUE,
{0x93, 0x01, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"DLC.REALLOCATE"
},
// non-existent command (0x18)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x18)"
},
// DLC.OPEN.STATION (0x19)
{
TRUE,
{0xb3, 0x0b, 0x00, 0x38, 0x00, 0x00, 0x00, 0x00, 0x4d, 0x80},
"DLC.OPEN.STATION"
},
// DLC.CLOSE.STATION (0x1a)
{
TRUE,
{0x93, 0x0a, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x81, 0x18},
"DLC.CLOSE.STATION"
},
// DLC.CONNECT.STATION (0x1b)
{
TRUE,
{0x97, 0x0a, 0x00, 0x38, 0x00, 0x00, 0x00, 0x00, 0x13, 0x24},
"DLC.CONNECT.STATION"
},
// DLC.MODIFY (0x1c)
{
TRUE,
{0x93, 0x0b, 0x00, 0x28, 0x00, 0x00, 0x00, 0x00, 0x25, 0x42},
"DLC.MODIFY"
},
// DLC.FLOW.CONTROL (0x1d)
{
TRUE,
{0x93, 0x0a, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"DLC.FLOW.CONTROL"
},
// DLC.STATISTICS (0x1e)
{
TRUE,
{0x93, 0x0a, 0x20, 0x28, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"DLC.STATISTICS"
},
// non-existent command (0x1f)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x1f)"
},
// DIR.INITIALIZE (0x20)
{
TRUE,
{0x87, 0x00, 0x10, 0x38, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.INITIALIZE"
},
// DIR.STATUS (0x21)
{
TRUE,
{0x03, 0x12, 0x00, 0x28, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.STATUS"
},
// DIR.TIMER.SET (0x22)
{
TRUE,
{0x83, 0x0e, 0x02, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.TIMER.SET"
},
// DIR.TIMER.CANCEL (0x23)
{
TRUE,
{0x03, 0x02, 0x02, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.TIMER.CANCEL"
},
// PDT.TRACE.ON (0x24)
{
TRUE,
{0x45, 0x04, 0x00, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"PDT.TRACE.ON"
},
// PDT.TRACE.OFF (0x25)
{
TRUE,
{0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"PDT.TRACE.OFF"
},
// BUFFER.GET (0x26)
{
TRUE,
{0x13, 0x02, 0x00, 0x2a, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"BUFFER.GET"
},
// BUFFER.FREE (0x27)
{
TRUE,
{0x13, 0x02, 0x00, 0x38, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"BUFFER.FREE"
},
// RECEIVE (0x28)
{
TRUE,
{0x97, 0x0e, 0x00, 0x3c, 0x03, 0x00, 0x00, 0x00, 0x01, 0x00},
"RECEIVE"
},
// RECEIVE.CANCEL (0x29)
{
TRUE,
{0x13, 0x02, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00},
"RECEIVE.CANCEL"
},
// RECEIVE.MODIFY (0x2a)
{
TRUE,
{0x97, 0x0e, 0x00, 0x3c, 0x03, 0x00, 0x00, 0x00, 0x01, 0x00},
"RECEIVE.MODIFY"
},
// DIR.DEFINE.MIF.ENVIRONMENT (0x2b)
{
TRUE,
{0x03, 0x00, 0x00, 0x18, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.DEFINE.MIF.ENVIRONMENT"
},
// DIR.TIMER.CANCEL.GROUP (0x2c)
{
TRUE,
{0x03, 0x02, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.TIMER.CANCEL.GROUP"
},
// DIR.SET.USER.APPENDAGE (0x2d)
{
TRUE,
{0x93, 0x02, 0x00, 0x28, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.SET.USER.APPENDAGE"
},
// non-existent command (0x2e)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x2e)"
},
// non-existent command (0x2f)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x2f)"
},
// non-existent command (0x30)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT COMMAND (0x30)"
},
// READ (0x31)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"READ"
},
// READ.CANCEL (0x32)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"READ.CANCEL"
},
// DLC.SET.THRESHOLD (0x33)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DLC.SET.THRESHOLD"
},
// DIR.CLOSE.DIRECT (0x34)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.CLOSE.DIRECT"
},
// DIR.OPEN.DIRECT (0x35)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"DIR.OPEN.DIRECT"
},
// PURGE.RESOURCES (0x36)
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"PURGE.RESOURCES"
},
// LLC_MAX_DLC_COMMAND (0x37) ?
{
FALSE,
{0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
"NON-EXISTENT-COMMAND (0x37)"
}
};
BOOL
IsCcbErrorCodeAllowable(
IN BYTE CcbCommand,
IN BYTE CcbErrorCode
)
/*++
Routine Description:
Check whether an error code is allowable for a particular CCB(1) command
code. Perform range check on the error code before using as index into
allowable error table
Arguments:
CcbCommand - Command code
CcbErrorCode - Return code
Return Value:
BOOL
TRUE - CcbErrorCode is valid for CcbCommand
FALSE - CcbErrorCode should not be returned for CcbCommand
OR CcbErrorCode is invalid (out of range)
--*/
{
if (CcbErrorCode == CCB_COMMAND_IN_PROGRESS) {
return TRUE;
}
if (CcbErrorCode > MAX_CCB1_ERROR)
return FALSE;
return Ccb1ErrorTable[CcbCommand].ErrorList[CcbErrorCode/8] & (1 << (CcbErrorCode % 8));
}
BOOL
IsCcbErrorCodeValid(
IN BYTE CcbErrorCode
)
/*++
Routine Description:
Check if a return code from a CCB(1) is an allowable return code,
irrespective of command type
Arguments:
CcbErrorCode - return code to check
Return Value:
BOOL
TRUE - CcbErrorCode is in range
FALSE - CcbErrorCode is not in range
--*/
{
return (CcbErrorCode == CCB_COMMAND_IN_PROGRESS) // 0xff
// 0x00 - 0x0c
|| ((CcbErrorCode >= CCB_SUCCESS) && (CcbErrorCode <= CCB_SUCCESS_ADAPTER_NOT_OPEN))
// 0x10 - 0x1e
|| ((CcbErrorCode >= CCB_NETBIOS_FAILURE) && (CcbErrorCode <= CCB_INVALID_FUNCTION_ADDRESS))
// 0x20 - 0x28
|| ((CcbErrorCode >= CCB_DATA_LOST_NO_BUFFERS) && (CcbErrorCode <= CCB_INVALID_FRAME_LENGTH))
// 0x30
|| (CcbErrorCode == CCB_NOT_ENOUGH_BUFFERS_OPEN)
// 0x32 - 0x34
|| ((CcbErrorCode >= CCB_INVALID_NODE_ADDRESS) && (CcbErrorCode <= CCB_INVALID_TRANSMIT_LENGTH))
// 0x40 - 0x4f
|| ((CcbErrorCode >= CCB_INVALID_STATION_ID) && (CcbErrorCode <= CCB_INVALID_REMOTE_ADDRESS))
;
}
BOOL
IsCcbCommandValid(
IN BYTE CcbCommand
)
/*++
Routine Description:
Check if CCB command code is one of the allowable codes for a DOS CCB
(CCB1)
Arguments:
CcbCommand - command code to check
Return Value:
BOOL
TRUE - CcbCommand is recognized
FALSE - CcbCommand is not recognized
--*/
{
return ((CcbCommand >= LLC_DIR_INTERRUPT) && (CcbCommand <= LLC_DIR_CLOSE_ADAPTER))
|| ((CcbCommand >= LLC_DIR_SET_GROUP_ADDRESS) && (CcbCommand <= LLC_DIR_SET_FUNCTIONAL_ADDRESS))
|| ((CcbCommand >= LLC_TRANSMIT_DIR_FRAME) && (CcbCommand <= LLC_TRANSMIT_I_FRAME))
|| ((CcbCommand >= LLC_TRANSMIT_UI_FRAME) && (CcbCommand <= LLC_TRANSMIT_TEST_CMD))
|| ((CcbCommand >= LLC_DLC_RESET) && (CcbCommand <= LLC_DLC_REALLOCATE_STATIONS))
|| ((CcbCommand >= LLC_DLC_OPEN_STATION) && (CcbCommand <= LLC_DLC_STATISTICS))
|| ((CcbCommand >= LLC_DIR_INITIALIZE) && (CcbCommand <= LLC_DIR_SET_USER_APPENDAGE))
;
}
LPSTR
MapCcbCommandToName(
IN BYTE CcbCommand
)
/*++
Routine Description:
Return the name of a CCB command, given its value
Arguments:
CcbCommand - command code to map
Return Value:
char* pointer to ASCIZ name of command (in IBM format X.Y.Z)
--*/
{
return Ccb1ErrorTable[CcbCommand].CommandName;
}
VOID
DumpDosAdapter(
IN DOS_ADAPTER* pDosAdapter
)
{
DBGPRINT( "DOS_ADAPTER @ %08x\n"
"AdapterType. . . . . . . . . %s\n"
"IsOpen . . . . . . . . . . . %d\n"
"DirectStationOpen. . . . . . %d\n"
"DirectReceive. . . . . . . . %d\n"
"WaitingRestore . . . . . . . %d\n"
"BufferFree . . . . . . . . . %d\n"
"BufferPool . . . . . . . . . %08x\n"
"CurrentExceptionHandlers . . %08x %08x %08x\n"
"PreviousExceptionHandlers. . %08x %08x %08x\n"
"DlcStatusChangeAppendage . . \n"
"LastNetworkStatusChange. . . %04x\n"
"UserStatusValue. . . . . . . \n"
"AdapterParms:\n"
" OpenErrorCode . . . . . . %04x\n"
" OpenOptions . . . . . . . %04x\n"
" NodeAddress . . . . . . . %02x-%02x-%02x-%02x-%02x-%02x\n"
" GroupAddress. . . . . . . %08x\n"
" FunctionalAddress . . . . %08x\n"
" NumberReceiveBuffers. . . %04x\n"
" ReceiveBufferLength . . . %04x\n"
" DataHoldBufferLength. . . %04x\n"
" NumberDataHoldBuffers . . %02x\n"
" Reserved. . . . . . . . . %02x\n"
" OpenLock. . . . . . . . . %04x\n"
" ProductId . . . . . . . . %08x\n"
"DlcSpecified . . . . . . . . %d\n"
"DlcParms:\n"
" MaxSaps . . . . . . . . . %02x\n"
" MaxStations . . . . . . . %02x\n"
" MaxGroupSaps. . . . . . . %02x\n"
" MaxGroupMembers . . . . . %02x\n"
" T1Tick1 . . . . . . . . . %02x\n"
" T2Tick1 . . . . . . . . . %02x\n"
" TiTick1 . . . . . . . . . %02x\n"
" T1Tick2 . . . . . . . . . %02x\n"
" T2Tick2 . . . . . . . . . %02x\n"
" TiTick2 . . . . . . . . . %02x\n"
"AdapterCloseCcb. . . . . . . \n"
"DirectCloseCcb . . . . . . . \n"
"ReadCcb. . . . . . . . . . . \n"
"EventQueueCritSec. . . . . . \n"
"EventQueueHead . . . . . . . \n"
"EventQueueTail . . . . . . . \n"
"QueueElements. . . . . . . . \n"
"LocalBusyCritSec . . . . . . \n"
"DeferredReceives . . . . . . \n"
"FirstIndex . . . . . . . . . \n"
"LastIndex. . . . . . . . . . \n"
"LocalBusyInfo. . . . . . . . \n",
MapAdapterType(pDosAdapter->AdapterType),
pDosAdapter->IsOpen,
pDosAdapter->DirectStationOpen,
pDosAdapter->DirectReceive,
pDosAdapter->WaitingRestore,
pDosAdapter->BufferFree,
pDosAdapter->BufferPool,
pDosAdapter->CurrentExceptionHandlers[0],
pDosAdapter->CurrentExceptionHandlers[1],
pDosAdapter->CurrentExceptionHandlers[2],
pDosAdapter->PreviousExceptionHandlers[0],
pDosAdapter->PreviousExceptionHandlers[1],
pDosAdapter->PreviousExceptionHandlers[2],
pDosAdapter->LastNetworkStatusChange,
pDosAdapter->AdapterParms.OpenErrorCode,
pDosAdapter->AdapterParms.OpenOptions,
pDosAdapter->AdapterParms.NodeAddress[0],
pDosAdapter->AdapterParms.NodeAddress[1],
pDosAdapter->AdapterParms.NodeAddress[2],
pDosAdapter->AdapterParms.NodeAddress[3],
pDosAdapter->AdapterParms.NodeAddress[4],
pDosAdapter->AdapterParms.NodeAddress[5],
pDosAdapter->AdapterParms.GroupAddress,
pDosAdapter->AdapterParms.FunctionalAddress,
pDosAdapter->AdapterParms.NumberReceiveBuffers,
pDosAdapter->AdapterParms.ReceiveBufferLength,
pDosAdapter->AdapterParms.DataHoldBufferLength,
pDosAdapter->AdapterParms.NumberDataHoldBuffers,
pDosAdapter->AdapterParms.Reserved,
pDosAdapter->AdapterParms.OpenLock,
pDosAdapter->AdapterParms.ProductId,
pDosAdapter->DlcSpecified,
pDosAdapter->DlcParms.MaxSaps,
pDosAdapter->DlcParms.MaxStations,
pDosAdapter->DlcParms.MaxGroupSaps,
pDosAdapter->DlcParms.MaxGroupMembers,
pDosAdapter->DlcParms.T1Tick1,
pDosAdapter->DlcParms.T2Tick1,
pDosAdapter->DlcParms.TiTick1,
pDosAdapter->DlcParms.T1Tick2,
pDosAdapter->DlcParms.T2Tick2,
pDosAdapter->DlcParms.TiTick2
);
}
PRIVATE
LPSTR
MapAdapterType(
IN ADAPTER_TYPE AdapterType
)
{
switch (AdapterType) {
case TokenRing:
return "Token Ring";
case Ethernet:
return "Ethernet";
case PcNetwork:
return "PC Network";
case UnknownAdapter:
return "Unknown Adapter";
}
return "*** REALLY UNKNOWN ADAPTER! ***";
}
#endif