windows-nt/Source/XPSP1/NT/drivers/net/irda/sigmatel/sys/receive.c
2020-09-26 16:20:57 +08:00

2739 lines
76 KiB
C

/**************************************************************************************************************************
* RECEIVE.C SigmaTel STIR4200 packet reception and decoding module
**************************************************************************************************************************
* (C) Unpublished Copyright of Sigmatel, Inc. All Rights Reserved.
*
*
* Created: 04/06/2000
* Version 0.9
* Edited: 04/24/2000
* Version 0.91
* Edited: 04/27/2000
* Version 0.92
* Edited: 05/03/2000
* Version 0.93
* Edited: 05/12/2000
* Version 0.94
* Edited: 05/19/2000
* Version 0.95
* Edited: 07/13/2000
* Version 1.00
* Edited: 08/22/2000
* Version 1.02
* Edited: 09/25/2000
* Version 1.10
* Edited: 10/13/2000
* Version 1.11
* Edited: 11/09/2000
* Version 1.12
* Edited: 12/29/2000
* Version 1.13
* Edited: 01/16/2001
* Version 1.14
* Edited: 02/20/2001
* Version 1.15
*
**************************************************************************************************************************/
#define DOBREAKS // enable debug breaks
#include <ndis.h>
#include <ntdef.h>
#include <windef.h>
#include "stdarg.h"
#include "stdio.h"
#include "debug.h"
#include "usbdi.h"
#include "usbdlib.h"
#include "ircommon.h"
#include "irusb.h"
/*****************************************************************************
*
* Function: ReceiveProcessFifoData
*
* Synopsis: Processes the received data and indicates packets to the protocol
*
* Arguments: pThisDev - pointer to current ir device object
*
* Returns: None
*
*
*****************************************************************************/
VOID
ReceiveProcessFifoData(
IN OUT PIR_DEVICE pThisDev
)
{
ULONG BytesProcessed;
BOOLEAN ReturnValue = TRUE;
while( ReturnValue )
{
if( pThisDev->currentSpeed<=MAX_SIR_SPEED )
{
ReturnValue = ReceiveSirStepFSM( pThisDev, &BytesProcessed );
}
else if( pThisDev->currentSpeed<=MAX_MIR_SPEED )
{
ReturnValue = ReceiveMirStepFSM( pThisDev, &BytesProcessed );
}
else
{
ReturnValue = ReceiveFirStepFSM( pThisDev, &BytesProcessed );
}
}
//
// Indicate that we are no more receiving
//
InterlockedExchange( (PLONG)&pThisDev->fCurrentlyReceiving, FALSE );
}
/*****************************************************************************
*
* Function: ReceiveResetPointers
*
* Synopsis: Reset the receive pointers as the data is gone when we are sending
*
* Arguments: pThisDev - pointer to current ir device object
*
* Returns: None
*
*
*****************************************************************************/
VOID
ReceiveResetPointers(
IN OUT PIR_DEVICE pThisDev
)
{
pThisDev->rcvState = STATE_INIT;
pThisDev->readBufPos = 0;
}
/*****************************************************************************
*
* Function: ReceivePreprocessFifo
*
* Synopsis: Verifies if there is data to be received
*
* Arguments: MiniportAdapterContext - pointer to current ir device object
* pFifoCount - pinter to count to return
*
* Returns: NT status code
*
*
*****************************************************************************/
NTSTATUS
ReceivePreprocessFifo(
IN OUT PIR_DEVICE pThisDev,
OUT PULONG pFifoCount
)
{
NTSTATUS Status;
#ifdef WORKAROUND_POLLING_FIFO_COUNT
LARGE_INTEGER CurrentTime;
BOOLEAN SlowReceive;
ULONG OldFifoCount = 0;
LONG Delay;
//
// Set the receive algorithm
//
#if defined(SUPPORT_LA8)
if( pThisDev->ChipRevision >= CHIP_REVISION_8 )
SlowReceive = FALSE;
else
#endif
SlowReceive = TRUE;
if( SlowReceive )
{
Status = St4200GetFifoCount( pThisDev, pFifoCount );
if( Status != STATUS_SUCCESS )
{
DEBUGMSG(DBG_ERR, (" ReceivePreprocessFifo(): USB failure\n"));
return Status;
}
}
else
{
*pFifoCount = 1;
}
//
// Receive the data
//
if( *pFifoCount || pThisDev->fReadHoldingReg )
{
//
// See if we need to take care of the fake empty FIFO
//
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
if( *pFifoCount )
{
#endif
//
// If we are in SIR read again until we see a stable value
//
if( (pThisDev->currentSpeed <= MAX_SIR_SPEED) && (pThisDev->currentSpeed != SPEED_9600) && SlowReceive )
{
//
// Make also sure we don't ever wrap
//
while( (OldFifoCount != *pFifoCount) && (*pFifoCount < 9*STIR4200_FIFO_SIZE/10) )
{
OldFifoCount = *pFifoCount;
St4200GetFifoCount( pThisDev, pFifoCount );
}
}
//
// If we are in FIR we need to delay
//
if( (pThisDev->currentSpeed > MAX_MIR_SPEED) && SlowReceive )
{
if( pThisDev->ChipRevision < CHIP_REVISION_7 )
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_ERR, (" ReceivePreprocessFifo(): Delaying\n"));
#endif
Delay = STIR4200_READ_DELAY - (STIR4200_READ_DELAY*(*pFifoCount))/STIR4200_ESC_PACKET_SIZE;
if( Delay > 0 )
{
NdisStallExecution( (ULONG)Delay );
}
}
else //if( pThisDev->dongleCaps.windowSize == 2 )
{
/*if( !(*pFifoCount%10) )
{
DEBUGMSG(DBG_ERR, (" ReceivePreprocessFifo(): Forcing wrap\n"));
NdisMSleep( 1000 );
}*/
Delay = pThisDev->ReceiveAdaptiveDelay -
(pThisDev->ReceiveAdaptiveDelay*(*pFifoCount))/STIR4200_MULTIPLE_READ_THREHOLD;
if( Delay > 0 )
{
NdisStallExecution( (ULONG)Delay );
}
}
}
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
}
//
// Read after a successful bulk-in with count of zero
//
else
{
pThisDev->fReadHoldingReg = FALSE;
}
#endif
//
// Perform the read
//
pThisDev->PreReadBuffer.DataLen = 0;
Status = ReceivePacketRead(
pThisDev,
&pThisDev->PreReadBuffer
);
if( Status == STATUS_SUCCESS )
{
*pFifoCount = pThisDev->PreReadBuffer.DataLen;
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
//
// If we got data restore the flag
//
if( *pFifoCount )
{
pThisDev->fReadHoldingReg = TRUE;
}
#endif
#if !defined(ONLY_ERROR_MESSAGES) && defined( WORKAROUND_FAKE_EMPTY_FIFO )
if( *pFifoCount && !pThisDev->fReadHoldingReg )
DEBUGMSG(DBG_ERR, (" ReceivePreprocessFifo(): Final byte(s) workaround\n"));
#endif
#if defined(RECEIVE_LOGGING)
if( pThisDev->ReceiveFileHandle && *pFifoCount )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pThisDev->PreReadBuffer.pDataBuf,
pThisDev->PreReadBuffer.DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveFilePosition,
NULL
);
pThisDev->ReceiveFilePosition += pThisDev->PreReadBuffer.DataLen;
}
#endif
}
else
{
DEBUGMSG(DBG_ERR, (" ReceivePreprocessFifo(): USB failure\n"));
pThisDev->PreReadBuffer.DataLen = 0;
*pFifoCount = 0;
}
}
#else
Status = ReceivePacketRead(
pThisDev,
&pThisDev->PreReadBuffer
);
if( Status == STATUS_SUCCESS )
*pFifoCount = pThisDev->PreReadBuffer.DataLen;
#endif
return Status;
}
/*****************************************************************************
*
* Function: ReceiveGetFifoData
*
* Synopsis: Load the preprocessed data if any is vailable, otherwise tries to read and load new data
*
* Arguments: pThisDev - pointer to current ir device object
* pData - buffer to copy to
* pBytesRead - pointer to return bytes read
* BytesToRead - requested number of bytes
*
* Returns: Number of bytes in the FIFO
*
*
*****************************************************************************/
NTSTATUS
ReceiveGetFifoData(
IN OUT PIR_DEVICE pThisDev,
OUT PUCHAR pData,
OUT PULONG pBytesRead,
ULONG BytesToRead
)
{
NTSTATUS Status;
#ifdef WORKAROUND_POLLING_FIFO_COUNT
LARGE_INTEGER CurrentTime;
BOOLEAN SlowReceive;
ULONG FifoCount = 0, OldFifoCount = 0;
LONG Delay;
//
// Make sure if there is data in the preread buffer
//
if( pThisDev->PreReadBuffer.DataLen )
{
ULONG OutputBufferSize;
IRUSB_ASSERT( pThisDev->PreReadBuffer.DataLen <= BytesToRead );
//
// Copy the data
//
RtlCopyMemory( pData, pThisDev->PreReadBuffer.pDataBuf, pThisDev->PreReadBuffer.DataLen );
#if !defined(WORKAROUND_BROKEN_MIR)
//
// Consider MIR
//
if( pThisDev->currentSpeed == SPEED_1152000 )
ReceiveMirUnstuff(
pThisDev,
pData,
pThisDev->PreReadBuffer.DataLen,
pThisDev->pRawUnstuffedBuf,
&OutputBufferSize
);
#endif
*pBytesRead = pThisDev->PreReadBuffer.DataLen;
pThisDev->PreReadBuffer.DataLen = 0;
return STATUS_SUCCESS;
}
//
// Try to read if no data is already available
//
else
{
//
// Set the receive algorithm
//
#if defined(SUPPORT_LA8)
if( pThisDev->ChipRevision >= CHIP_REVISION_8 )
SlowReceive = FALSE;
else
#endif
SlowReceive = TRUE;
if( SlowReceive )
{
Status = St4200GetFifoCount( pThisDev, &FifoCount );
if( Status != STATUS_SUCCESS )
{
DEBUGMSG(DBG_ERR, (" ReceiveGetFifoData(): USB failure\n"));
return Status;
}
}
else
{
FifoCount = 1;
}
//
// Receive the data
//
if( FifoCount || pThisDev->fReadHoldingReg )
{
//
// See if we need to take care of the fake empty FIFO
//
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
if( FifoCount )
{
#endif
//
// If we are in SIR read again until we see a stable value
//
#if defined( WORKAROUND_9600_ANTIBOUNCING )
if( (pThisDev->currentSpeed <= MAX_SIR_SPEED) && SlowReceive )
{
if( pThisDev->currentSpeed != SPEED_9600 )
{
//
// Make also sure we don't ever wrap
//
while( (OldFifoCount != FifoCount) && (FifoCount < 9*STIR4200_FIFO_SIZE/10) )
{
OldFifoCount = FifoCount;
St4200GetFifoCount( pThisDev, &FifoCount );
}
}
else
{
if( pThisDev->rcvState != STATE_INIT )
{
while( OldFifoCount != FifoCount )
{
OldFifoCount = FifoCount;
St4200GetFifoCount( pThisDev, &FifoCount );
}
}
}
}
#else
if( (pThisDev->currentSpeed <= MAX_SIR_SPEED) && ( pThisDev->currentSpeed != SPEED_9600) && SlowReceive )
{
while( OldFifoCount != FifoCount )
{
OldFifoCount = FifoCount;
St4200GetFifoCount( pThisDev, &FifoCount );
}
}
#endif
//
// If we are in FIR we need to delay
//
if( (pThisDev->currentSpeed > MAX_MIR_SPEED) && SlowReceive )
{
if( pThisDev->ChipRevision <= CHIP_REVISION_6 )
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_ERR, (" ReceiveGetFifoData(): Delaying\n"));
#endif
Delay = STIR4200_READ_DELAY - (STIR4200_READ_DELAY*FifoCount)/STIR4200_ESC_PACKET_SIZE;
if( Delay > 0 )
{
NdisStallExecution( (ULONG)Delay );
}
}
else //if( pThisDev->dongleCaps.windowSize == 2 )
{
/*if( !(FifoCount%10) )
{
DEBUGMSG(DBG_ERR, (" ReceiveGetFifoData(): Forcing wrap\n"));
NdisMSleep( 1000 );
}*/
Delay = pThisDev->ReceiveAdaptiveDelay -
(pThisDev->ReceiveAdaptiveDelay*FifoCount)/STIR4200_MULTIPLE_READ_THREHOLD;
if( Delay > 0 )
{
NdisStallExecution( (ULONG)Delay );
}
}
}
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
}
else
{
// Force antibouncing to take care of OHCI
if( pThisDev->currentSpeed <= MAX_SIR_SPEED )
{
if( pThisDev->rcvState != STATE_INIT )
{
OldFifoCount = 1;
while( OldFifoCount != FifoCount )
{
OldFifoCount = FifoCount;
St4200GetFifoCount( pThisDev, &FifoCount );
}
}
}
pThisDev->fReadHoldingReg = FALSE;
}
#endif
//
// Perform the read
//
pThisDev->PreReadBuffer.DataLen = 0;
Status = ReceivePacketRead(
pThisDev,
&pThisDev->PreReadBuffer
);
if( Status == STATUS_SUCCESS )
{
IRUSB_ASSERT( pThisDev->PreReadBuffer.DataLen <= BytesToRead );
//
// Copy the data
//
RtlCopyMemory( pData, pThisDev->PreReadBuffer.pDataBuf, pThisDev->PreReadBuffer.DataLen );
FifoCount = pThisDev->PreReadBuffer.DataLen;
#if defined( WORKAROUND_FAKE_EMPTY_FIFO )
//
// If we got data restore the flag
//
if( FifoCount )
{
pThisDev->fReadHoldingReg = TRUE;
}
#endif
#if !defined(ONLY_ERROR_MESSAGES) && defined( WORKAROUND_FAKE_EMPTY_FIFO )
if( FifoCount && !pThisDev->fReadHoldingReg )
DEBUGMSG(DBG_ERR, (" ReceiveGetFifoData(): Final byte(s) workaround\n"));
#endif
#if defined(RECEIVE_LOGGING)
if( pThisDev->ReceiveFileHandle && FifoCount )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pThisDev->PreReadBuffer.pDataBuf,
pThisDev->PreReadBuffer.DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveFilePosition,
NULL
);
pThisDev->ReceiveFilePosition += pThisDev->PreReadBuffer.DataLen;
}
#endif
pThisDev->PreReadBuffer.DataLen = 0;
}
else
{
DEBUGMSG(DBG_ERR, (" ReceiveGetFifoData(): USB failure\n"));
pThisDev->PreReadBuffer.DataLen = 0;
FifoCount = 0;
}
}
}
*pBytesRead = FifoCount;
return Status;
#else
if( pThisDev->PreReadBuffer.DataLen )
{
IRUSB_ASSERT( pThisDev->PreReadBuffer.DataLen <= BytesToRead );
//
// Copy the data
//
RtlCopyMemory( pData, pThisDev->PreReadBuffer.pDataBuf, pThisDev->PreReadBuffer.DataLen );
*pBytesRead = pThisDev->PreReadBuffer.DataLen;
pThisDev->PreReadBuffer.DataLen = 0;
return STATUS_SUCCESS;
}
else
{
Status = ReceivePacketRead(
pThisDev,
&pThisDev->PreReadBuffer
);
if( Status == STATUS_SUCCESS )
{
RtlCopyMemory( pData, pThisDev->PreReadBuffer.pDataBuf, pThisDev->PreReadBuffer.DataLen );
*pBytesRead = pThisDev->PreReadBuffer.DataLen;
pThisDev->PreReadBuffer.DataLen = 0;
}
return Status;
}
#endif
}
/*****************************************************************************
*
* Function: ReceiveFirStepFSM
*
* Synopsis: Step the receive FSM to read in a piece of an IrDA frame.
* Strip the BOFs and EOF, and eliminate escape sequences.
*
* Arguments: pIrDev - pointer to the current IR device object
* pBytesProcessed - pointer to bytes processed
*
* Returns: TRUE after an entire frame has been read in
* FALSE otherwise
*
*****************************************************************************/
BOOLEAN
ReceiveFirStepFSM(
IN OUT PIR_DEVICE pIrDev,
OUT PULONG pBytesProcessed
)
{
ULONG rawBufPos, rawBytesRead;
BOOLEAN FrameProcessed = FALSE, ForceExit = FALSE;
UCHAR ThisChar;
PUCHAR pRawBuf, pReadBuf;
PRCV_BUFFER pRecBuf;
*pBytesProcessed = 0;
if( !pIrDev->pCurrentRecBuf )
{
UINT Index;
pRecBuf = ReceiveGetBuf( pIrDev, &Index, RCV_STATE_FULL );
if( !pRecBuf )
{
//
// no buffers available; stop
//
DEBUGMSG(DBG_ERR, (" ReceiveSirStepFSM out of buffers\n"));
pIrDev->packetsReceivedNoBuffer ++;
return FALSE;
}
pIrDev->pCurrentRecBuf = pRecBuf;
}
else
pRecBuf = pIrDev->pCurrentRecBuf;
pReadBuf = pRecBuf->pDataBuf;
pRawBuf = pIrDev->pRawBuf;
/***********************************************/
/* Read in and process groups of incoming */
/* bytes from the FIFO. */
/***********************************************/
while( (pIrDev->rcvState != STATE_SAW_EOF) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + FAST_IR_FCS_SIZE)) &&
!ForceExit )
{
if( pIrDev->rcvState == STATE_CLEANUP )
{
/***********************************************/
/* We returned a complete packet last time, */
/* but we had read some extra bytes, which */
/* we stored into the rawBuf after */
/* returning the previous complete buffer */
/* to the user. So instead of calling */
/* DoRcvDirect() in this first execution of */
/* this loop, we just use these previously */
/* read bytes. (This is typically only 1 or */
/* 2 bytes). */
/***********************************************/
rawBytesRead = pIrDev->rawCleanupBytesRead;
pIrDev->rcvState = STATE_INIT;
}
else
{
if( ReceiveGetFifoData( pIrDev, pRawBuf, &rawBytesRead, STIR4200_FIFO_SIZE ) == STATUS_SUCCESS )
{
if( rawBytesRead == (ULONG)-1 )
{
/***********************************************/
/* Receive error...back to INIT state... */
/***********************************************/
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
continue;
}
else if( rawBytesRead == 0 )
{
/***********************************************/
/* No more receive bytes...break out... */
/***********************************************/
break;
}
}
else
break;
}
/***********************************************/
/* Let the receive state machine process */
/* this group of bytes. */
/* */
/* NOTE: We have to loop once more after */
/* getting MAX_RCV_DATA_SIZE bytes so that */
/* we can see the 'EOF'; hence <= and not */
/* <. */
/***********************************************/
for( rawBufPos = 0;
((pIrDev->rcvState != STATE_SAW_EOF) && (rawBufPos < rawBytesRead) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + FAST_IR_FCS_SIZE)));
rawBufPos++ )
{
*pBytesProcessed += 1;
ThisChar = pRawBuf[rawBufPos];
switch( pIrDev->rcvState )
{
case STATE_INIT:
switch( ThisChar )
{
case STIR4200_FIR_BOF:
pIrDev->rcvState = STATE_GOT_FIR_BOF;
break;
#if defined(WORKAROUND_XX_HANG)
case 0x3F:
if( (rawBufPos+1) < rawBytesRead )
{
if( pRawBuf[rawBufPos+1] == 0x3F )
{
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): hang sequence in INIT state\n"));
St4200ResetFifo( pIrDev );
}
}
break;
#endif
#if defined(WORKAROUND_FF_HANG)
case 0xFF:
if( (rawBufPos+2) < rawBytesRead )
{
if( (pRawBuf[rawBufPos+2] == 0xFF) && (pRawBuf[rawBufPos+1] == 0xFF) &&
(rawBytesRead>STIR4200_FIFO_OVERRUN_THRESHOLD) )
{
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): overflow sequence in INIT state\n"));
St4200ResetFifo( pIrDev );
rawBufPos = rawBytesRead;
ForceExit = TRUE;
}
}
break;
#endif
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid char in INIT state\n"));
break;
}
break;
case STATE_GOT_FIR_BOF:
switch( ThisChar )
{
case STIR4200_FIR_BOF:
pIrDev->rcvState = STATE_GOT_BOF;
break;
#if defined(WORKAROUND_BAD_ESC)
case STIR4200_FIR_ESC_CHAR:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid char in BOF state, bufpos=%d, char=%X\n", pIrDev->readBufPos, (ULONG)ThisChar));
if( rawBufPos < (rawBytesRead-1) )
{
pIrDev->rcvState = STATE_GOT_BOF;
rawBufPos ++;
}
else
{
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
}
break;
#endif
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid char in BOF state, bufpos=%d, char=%X\n", pIrDev->readBufPos, (ULONG)ThisChar));
#if defined(WORKAROUND_BAD_SOF)
pIrDev->rcvState = STATE_GOT_BOF;
rawBufPos --;
#else
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
#endif
break;
}
break;
case STATE_GOT_BOF:
switch( ThisChar )
{
case STIR4200_FIR_BOF:
/***********************************************/
/* It's a mistake, but could still be valid data
/***********************************************/
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): More than legal BOFs, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->rcvState = STATE_GOT_BOF;
pIrDev->readBufPos = 0;
break;
case STIR4200_FIR_PREAMBLE:
/***********************************************/
/* Garbage */
/***********************************************/
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid char in BOF state, bufpos=%d, char=%X\n", pIrDev->readBufPos, (ULONG)ThisChar));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
case STIR4200_FIR_ESC_CHAR:
/***********************************************/
/* Start of data. Our first data byte */
/* happens to be an ESC sequence. */
/***********************************************/
pIrDev->rcvState = STATE_ESC_SEQUENCE;
pIrDev->readBufPos = 0;
break;
default:
pReadBuf[0] = ThisChar;
pIrDev->rcvState = STATE_ACCEPTING;
pIrDev->readBufPos = 1;
break;
}
break;
case STATE_ACCEPTING:
switch( ThisChar )
{
case STIR4200_FIR_EOF:
#if defined( WORKAROUND_33_HANG )
if( pIrDev->readBufPos < (IRDA_A_C_TOTAL_SIZE + FAST_IR_FCS_SIZE - 1) )
#else
if( pIrDev->readBufPos < (IRDA_A_C_TOTAL_SIZE + FAST_IR_FCS_SIZE) )
#endif
{
DEBUGMSG(DBG_INT_ERR,
("ReceiveFirStepFSM(): WARNING: EOF encountered in short packet, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->packetsReceivedRunt ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
}
else
{
#if defined( WORKAROUND_MISSING_7E )
// Force to get out if there is one EOF and we have no more data
if( rawBufPos == (rawBytesRead-1) )
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR, ("ReceiveFirStepFSM(): Using a single 7E EOF\n"));
#endif
pIrDev->rcvState = STATE_SAW_EOF;
}
else
pIrDev->rcvState = STATE_SAW_FIR_BOF;
#else
pIrDev->rcvState = STATE_SAW_FIR_BOF;
#endif
}
break;
case STIR4200_FIR_ESC_CHAR:
pIrDev->rcvState = STATE_ESC_SEQUENCE;
break;
case STIR4200_FIR_PREAMBLE:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid preamble char in ACCEPTING state, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
default:
pReadBuf[pIrDev->readBufPos++] = ThisChar;
break;
}
break;
case STATE_ESC_SEQUENCE:
switch( ThisChar )
{
case STIR4200_FIR_ESC_DATA_7D:
pReadBuf[pIrDev->readBufPos++] = 0x7d;
pIrDev->rcvState = STATE_ACCEPTING;
break;
case STIR4200_FIR_ESC_DATA_7E:
pReadBuf[pIrDev->readBufPos++] = 0x7e;
pIrDev->rcvState = STATE_ACCEPTING;
break;
case STIR4200_FIR_ESC_DATA_7F:
pReadBuf[pIrDev->readBufPos++] = 0x7f;
pIrDev->rcvState = STATE_ACCEPTING;
break;
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid escaped char=%X\n", (ULONG)ThisChar));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
}
break;
case STATE_SAW_FIR_BOF:
switch( ThisChar )
{
case STIR4200_FIR_EOF:
pIrDev->rcvState = STATE_SAW_EOF;
break;
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid char=%X, expected EOF\n", (ULONG)ThisChar));
pIrDev->rcvState = STATE_SAW_EOF;
#if !defined(WORKAROUND_MISSING_7E)
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
#endif
break;
}
break;
case STATE_SAW_EOF:
default:
DEBUGMSG(DBG_ERR, (" ReceiveFirStepFSM(): Illegal state, bufpos=%d\n", pIrDev->readBufPos));
IRUSB_ASSERT( 0 );
pIrDev->readBufPos = 0;
pIrDev->rcvState = STATE_INIT;
return FALSE;
}
}
}
// * Set result and do any post-cleanup.
switch( pIrDev->rcvState )
{
case STATE_SAW_EOF:
/***********************************************/
/* We've read in the entire packet. Queue */
/* it and return TRUE. */
/***********************************************/
pRecBuf->DataLen = pIrDev->readBufPos;
pIrDev->pCurrentRecBuf = NULL;
ReceiveDeliverBuffer(
pIrDev,
pRecBuf
);
FrameProcessed = TRUE;
if( rawBufPos < rawBytesRead )
{
/***********************************************/
/* This is ugly. We have some more */
/* unprocessed bytes in the raw buffer. */
/* Move these to the beginning of the raw */
/* buffer go to the CLEANUP state, which */
/* indicates that these bytes be used up */
/* during the next call. (This is typically */
/* only 1 or 2 bytes). */
/* */
/* Note: We can't just leave these in the */
/* raw buffer because we might be */
/* supporting connections to multiple COM */
/* ports. */
/* */
/***********************************************/
RtlMoveMemory( pRawBuf, &pRawBuf[rawBufPos], rawBytesRead - rawBufPos );
pIrDev->rawCleanupBytesRead = rawBytesRead - rawBufPos;
pIrDev->rcvState = STATE_CLEANUP;
}
else
{
pIrDev->rcvState = STATE_INIT;
}
pIrDev->readBufPos = 0;
break;
default:
if( pIrDev->readBufPos > (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + FAST_IR_FCS_SIZE) )
{
DEBUGMSG( DBG_INT_ERR,(" ReceiveFirStepFSM() Overflow\n"));
St4200ResetFifo( pIrDev );
pIrDev->packetsReceivedOverflow ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
pIrDev->pCurrentRecBuf = NULL;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
}
else
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): returning with partial packet, read %d bytes\n", pIrDev->readBufPos));
#endif
}
FrameProcessed = FALSE;
break;
}
return FrameProcessed;
}
#if !defined(WORKAROUND_BROKEN_MIR)
/*****************************************************************************
*
* Function: ReceiveMirUnstuff
*
* Synopsis: Software unstuffing for a MIR frmae
*
* Arguments: pIrDev - pointer to the current IR device object
* pBytesProcessed - pointer to bytes processed
*
* Returns: TRUE after an entire frame has been read in
*
*****************************************************************************/
BOOLEAN
ReceiveMirUnstuff(
IN OUT PIR_DEVICE pIrDev,
IN PUCHAR pInputBuffer,
ULONG InputBufferSize,
OUT PUCHAR pOutputBuffer,
OUT PULONG pOutputBufferSize
)
{
ULONG MirIncompleteBitCount = pIrDev->MirIncompleteBitCount;
ULONG MirOneBitCount = pIrDev->MirOneBitCount;
UCHAR MirIncompleteByte = pIrDev->MirIncompleteByte;
ULONG ByteCounter, BitCounter;
BOOL MirUnstuffNext = FALSE;
*pOutputBufferSize = 0;
if( MirOneBitCount == 5 )
{
MirUnstuffNext = TRUE;
}
//
// Loop on the input buffer
//
for( ByteCounter=0; ByteCounter<InputBufferSize; ByteCounter++ )
{
//
// Loop on the byte
//
for( BitCounter=0; BitCounter<8; BitCounter++ )
{
//
// test for one
//
if( pInputBuffer[ByteCounter] & (0x01<<BitCounter) )
{
//
// Sixth one, reset
//
if( MirUnstuffNext )
{
MirOneBitCount = 0;
MirUnstuffNext = FALSE;
pIrDev->MirFlagCount ++;
}
//
// Increase the one count
//
else
{
MirOneBitCount ++;
if( MirOneBitCount == 5 )
{
MirUnstuffNext = TRUE;
}
}
//
// Copy to the temp byte
//
MirIncompleteByte += 0x01<<MirIncompleteBitCount;
//
// Increase the output bit count
//
MirIncompleteBitCount ++;
}
else
{
//
// Increase the output bit count if we are not stuffing
//
if( !MirUnstuffNext )
{
MirIncompleteBitCount ++;
}
//
// Reset
//
MirOneBitCount = 0;
MirUnstuffNext = FALSE;
//
// No copy needs to be done
//
}
//
// Flush to output buffer
//
if( MirIncompleteBitCount == 8 )
{
pOutputBuffer[*pOutputBufferSize] = MirIncompleteByte;
(*pOutputBufferSize) ++;
MirIncompleteBitCount = 0;
MirIncompleteByte = 0;
}
//
// Check for complete packet
//
if( pIrDev->MirFlagCount == 2 )
{
pIrDev->MirFlagCount = 0;
pIrDev->MirIncompleteBitCount = 0;
pIrDev->MirOneBitCount = 0;
pIrDev->MirIncompleteByte = 0;
return TRUE;
}
}
}
//
// roll over
//
pIrDev->MirIncompleteBitCount = MirIncompleteBitCount;
pIrDev->MirOneBitCount = MirOneBitCount;
pIrDev->MirIncompleteByte = MirIncompleteByte;
return FALSE;
}
#endif
/*****************************************************************************
*
* Function: ReceiveMirStepFSM
*
* Synopsis: Step the receive FSM to read in a piece of an IrDA frame.
* Strip the BOFs and EOF, and eliminate escape sequences.
*
* Arguments: pIrDev - pointer to the current IR device object
* pBytesProcessed - pointer to bytes processed
*
* Returns: TRUE after an entire frame has been read in
* FALSE otherwise
*
*****************************************************************************/
BOOLEAN
ReceiveMirStepFSM(
IN OUT PIR_DEVICE pIrDev,
OUT PULONG pBytesProcessed
)
{
ULONG rawBufPos, rawBytesRead;
BOOLEAN FrameProcessed = FALSE, ForceExit = FALSE;
UCHAR ThisChar;
PUCHAR pRawBuf, pReadBuf;
PRCV_BUFFER pRecBuf;
*pBytesProcessed = 0;
if( !pIrDev->pCurrentRecBuf )
{
UINT Index;
pRecBuf = ReceiveGetBuf( pIrDev, &Index, RCV_STATE_FULL );
if ( !pRecBuf)
{
//
// no buffers available; stop
//
DEBUGMSG(DBG_ERR, (" ReceiveMirStepFSM out of buffers\n"));
pIrDev->packetsReceivedNoBuffer ++;
return FALSE;
}
pIrDev->pCurrentRecBuf = pRecBuf;
}
else
pRecBuf = pIrDev->pCurrentRecBuf;
pReadBuf = pRecBuf->pDataBuf;
pRawBuf = pIrDev->pRawBuf;
/***********************************************/
/* Read in and process groups of incoming */
/* bytes from the FIFO. */
/***********************************************/
while( (pIrDev->rcvState != STATE_SAW_EOF) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + MEDIUM_IR_FCS_SIZE)) &&
!ForceExit )
{
if( pIrDev->rcvState == STATE_CLEANUP )
{
/***********************************************/
/* We returned a complete packet last time, */
/* but we had read some extra bytes, which */
/* we stored into the rawBuf after */
/* returning the previous complete buffer */
/* to the user. So instead of calling */
/* DoRcvDirect() in this first execution of */
/* this loop, we just use these previously */
/* read bytes. (This is typically only 1 or */
/* 2 bytes). */
/***********************************************/
rawBytesRead = pIrDev->rawCleanupBytesRead;
pIrDev->rcvState = STATE_INIT;
}
else
{
if( ReceiveGetFifoData( pIrDev, pRawBuf, &rawBytesRead, STIR4200_FIFO_SIZE ) == STATUS_SUCCESS )
{
if( rawBytesRead == (ULONG)-1 )
{
/***********************************************/
/* Receive error...back to INIT state... */
/***********************************************/
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
continue;
}
else if( rawBytesRead == 0 )
{
/***********************************************/
/* No more receive bytes...break out... */
/***********************************************/
break;
}
}
else
break;
}
/***********************************************/
/* Let the receive state machine process */
/* this group of bytes. */
/* */
/* NOTE: We have to loop once more after */
/* getting MAX_RCV_DATA_SIZE bytes so that */
/* we can see the 'EOF'; hence <= and not */
/* <. */
/***********************************************/
for( rawBufPos = 0;
((pIrDev->rcvState != STATE_SAW_EOF) && (rawBufPos < rawBytesRead) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + MEDIUM_IR_FCS_SIZE)));
rawBufPos++ )
{
*pBytesProcessed += 1;
ThisChar = pRawBuf[rawBufPos];
switch( pIrDev->rcvState )
{
case STATE_INIT:
switch( ThisChar )
{
case STIR4200_MIR_BOF:
pIrDev->rcvState = STATE_GOT_MIR_BOF;
break;
case 0xFF:
if( ((rawBufPos+2) < rawBytesRead) && (rawBufPos==0) )
{
if( (pRawBuf[rawBufPos+2] == 0xFF) && (pRawBuf[rawBufPos+1] == 0xFF) )
{
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): overflow sequence in INIT state\n"));
St4200ResetFifo( pIrDev );
St4200SoftReset( pIrDev );
//rawBufPos = rawBytesRead;
//ForceExit = TRUE;
}
}
break;
default:
break;
}
break;
case STATE_GOT_MIR_BOF:
switch( ThisChar )
{
case STIR4200_MIR_BOF:
pIrDev->rcvState = STATE_GOT_BOF;
break;
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): invalid char in BOF state, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
}
break;
case STATE_GOT_BOF:
switch( ThisChar )
{
case STIR4200_MIR_BOF:
/***********************************************/
/* It's a mistake, but could still be valid data
/***********************************************/
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): More than legal BOFs, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->readBufPos = 0;
pIrDev->rcvState = STATE_GOT_BOF;
break;
case STIR4200_MIR_ESC_CHAR:
/***********************************************/
/* Start of data. Our first data byte */
/* happens to be an ESC sequence. */
/***********************************************/
pIrDev->readBufPos = 0;
pIrDev->rcvState = STATE_ESC_SEQUENCE;
break;
default:
pReadBuf[0] = ThisChar;
pIrDev->readBufPos = 1;
pIrDev->rcvState = STATE_ACCEPTING;
break;
}
break;
case STATE_ACCEPTING:
switch( ThisChar )
{
case STIR4200_MIR_EOF:
if( pIrDev->readBufPos < (IRDA_A_C_TOTAL_SIZE + MEDIUM_IR_FCS_SIZE) )
{
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): WARNING: EOF encountered in short packet, bufpos=%d\n", pIrDev->readBufPos));
pIrDev->packetsReceivedRunt ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
}
else
{
pIrDev->rcvState = STATE_SAW_FIR_BOF;
}
break;
case STIR4200_MIR_ESC_CHAR:
pIrDev->rcvState = STATE_ESC_SEQUENCE;
break;
default:
pReadBuf[pIrDev->readBufPos++] = ThisChar;
break;
}
break;
case STATE_ESC_SEQUENCE:
switch( ThisChar )
{
case STIR4200_MIR_ESC_DATA_7D:
pReadBuf[pIrDev->readBufPos++] = 0x7d;
pIrDev->rcvState = STATE_ACCEPTING;
break;
case STIR4200_MIR_ESC_DATA_7E:
pReadBuf[pIrDev->readBufPos++] = 0x7e;
pIrDev->rcvState = STATE_ACCEPTING;
break;
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): invalid escaped char=%X\n", (ULONG)ThisChar));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
}
break;
case STATE_SAW_MIR_BOF:
switch( ThisChar )
{
case STIR4200_MIR_EOF:
pIrDev->rcvState = STATE_SAW_EOF;
break;
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): invalid char=%X, expected EOF\n", (ULONG)ThisChar));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
}
break;
case STATE_SAW_EOF:
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): Illegal state, bufpos=%d\n", pIrDev->readBufPos));
IRUSB_ASSERT( 0 );
pIrDev->readBufPos = 0;
pIrDev->rcvState = STATE_INIT;
return FALSE;
}
}
}
// * Set result and do any post-cleanup.
switch( pIrDev->rcvState )
{
case STATE_SAW_EOF:
/***********************************************/
/* We've read in the entire packet. Queue */
/* it and return TRUE. */
/***********************************************/
pRecBuf->DataLen = pIrDev->readBufPos;
pIrDev->pCurrentRecBuf = NULL;
ReceiveDeliverBuffer(
pIrDev,
pRecBuf
);
FrameProcessed = TRUE;
if( rawBufPos < rawBytesRead )
{
/***********************************************/
/* This is ugly. We have some more */
/* unprocessed bytes in the raw buffer. */
/* Move these to the beginning of the raw */
/* buffer go to the CLEANUP state, which */
/* indicates that these bytes be used up */
/* during the next call. (This is typically */
/* only 1 or 2 bytes). */
/* */
/* Note: We can't just leave these in the */
/* raw buffer because we might be */
/* supporting connections to multiple COM */
/* ports. */
/* */
/***********************************************/
RtlMoveMemory( pRawBuf, &pRawBuf[rawBufPos], rawBytesRead - rawBufPos );
pIrDev->rawCleanupBytesRead = rawBytesRead - rawBufPos;
pIrDev->rcvState = STATE_CLEANUP;
}
else
{
pIrDev->rcvState = STATE_INIT;
}
pIrDev->readBufPos = 0;
break;
default:
if( pIrDev->readBufPos > (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + MEDIUM_IR_FCS_SIZE) )
{
DEBUGMSG( DBG_INT_ERR,(" ReceiveMirStepFSM() Overflow\n"));
pIrDev->packetsReceivedOverflow ++;
pIrDev->readBufPos = 0;
pIrDev->rcvState = STATE_INIT;
pIrDev->pCurrentRecBuf = NULL;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
}
else
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR,
(" ReceiveMirStepFSM(): returning with partial packet, read %d bytes\n", pIrDev->readBufPos));
#endif
}
FrameProcessed = FALSE;
break;
}
return FrameProcessed;
}
/*****************************************************************************
*
* Function: ReceiveSirStepFSM
*
* Synopsis: Step the receive FSM to read in a piece of an IrDA frame.
* Strip the BOFs and EOF, and eliminate escape sequences.
*
* Arguments: pIrDev - pointer to the current IR device object
* pBytesProcessed - pointer to bytes processed
*
* Returns: TRUE after an entire frame has been read in
* FALSE otherwise
*
*****************************************************************************/
BOOLEAN
ReceiveSirStepFSM(
IN OUT PIR_DEVICE pIrDev,
OUT PULONG pBytesProcessed
)
{
ULONG rawBufPos, rawBytesRead;
BOOLEAN FrameProcessed = FALSE, ForceExit = FALSE;
UCHAR ThisChar;
PUCHAR pRawBuf, pReadBuf;
PRCV_BUFFER pRecBuf;
*pBytesProcessed = 0;
if( !pIrDev->pCurrentRecBuf )
{
UINT Index;
pRecBuf = ReceiveGetBuf( pIrDev, &Index, RCV_STATE_FULL );
if ( !pRecBuf)
{
//
// no buffers available; stop
//
DEBUGMSG(DBG_ERR, (" ReceiveSirStepFSM out of buffers\n"));
pIrDev->packetsReceivedNoBuffer ++;
return FALSE;
}
pIrDev->pCurrentRecBuf = pRecBuf;
}
else
pRecBuf = pIrDev->pCurrentRecBuf;
pReadBuf = pRecBuf->pDataBuf;
pRawBuf = pIrDev->pRawBuf;
// Read in and process groups of incoming bytes from the FIFO.
// NOTE: We have to loop once more after getting MAX_RCV_DATA_SIZE
// bytes so that we can see the 'EOF'; hence <= and not <.
while( (pIrDev->rcvState != STATE_SAW_EOF) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + SLOW_IR_FCS_SIZE)) &&
!ForceExit )
{
if( pIrDev->rcvState == STATE_CLEANUP )
{
/***********************************************/
/* We returned a complete packet last time, */
/* but we had read some extra bytes, which */
/* we stored into the rawBuf after */
/* returning the previous complete buffer */
/* to the user. So instead of calling */
/* DoRcvDirect() in this first execution of */
/* this loop, we just use these previously */
/* read bytes. (This is typically only 1 or */
/* 2 bytes). */
/***********************************************/
rawBytesRead = pIrDev->rawCleanupBytesRead;
pIrDev->rcvState = STATE_INIT;
}
else
{
if( ReceiveGetFifoData( pIrDev, pRawBuf, &rawBytesRead, STIR4200_FIFO_SIZE ) == STATUS_SUCCESS )
{
if( rawBytesRead == (ULONG)-1 )
{
/***********************************************/
/* Receive error...back to INIT state... */
/***********************************************/
DEBUGMSG( DBG_ERR,(" ReceiveSirStepFSM() Error in receiving packet\n"));
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
continue;
}
else if( rawBytesRead == 0 )
{
/***********************************************/
/* No more receive bytes...break out... */
/***********************************************/
#if defined(WORKAROUND_MISSING_C1)
if( (pIrDev->rcvState == STATE_ACCEPTING) && (pIrDev->ChipRevision <= CHIP_REVISION_7) )
{
pIrDev->rcvState = STATE_SAW_EOF;
DEBUGMSG(DBG_INT_ERR, (" ReceiveSirStepFSM(): Missing C1 workaround\n"));
pRecBuf->MissingC1Detected = TRUE;
}
#endif
break;
}
}
else
break;
}
/***********************************************/
/* Let the receive state machine process */
/* this group of bytes. */
/* */
/* NOTE: We have to loop once more after */
/* getting MAX_RCV_DATA_SIZE bytes so that */
/* we can see the 'EOF'; hence <= and not */
/* <. */
/***********************************************/
for( rawBufPos = 0;
((pIrDev->rcvState != STATE_SAW_EOF) && (rawBufPos < rawBytesRead) &&
(pIrDev->readBufPos <= (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + SLOW_IR_FCS_SIZE)));
rawBufPos ++ )
{
*pBytesProcessed += 1;
ThisChar = pRawBuf[rawBufPos];
switch( pIrDev->rcvState )
{
case STATE_INIT:
switch( ThisChar )
{
#if defined(WORKAROUND_FF_HANG)
case 0xFF:
if( (rawBufPos+2) < rawBytesRead )
{
if( (pRawBuf[rawBufPos+2] == 0xFF) && (pRawBuf[rawBufPos+1] == 0xFF) &&
(rawBytesRead>STIR4200_FIFO_OVERRUN_THRESHOLD) )
{
DEBUGMSG(DBG_INT_ERR,
(" ReceiveFirStepFSM(): overflow sequence in INIT state\n"));
St4200DoubleResetFifo( pIrDev );
rawBufPos = rawBytesRead;
ForceExit = TRUE;
}
}
break;
#endif
#if defined( WORKAROUND_E0_81_FLAG )
// This will take care of wrong start flags at low rates
case 0x81:
case 0xe0:
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR,
(" ReceiveSirStepFSM(): WORKAROUND_E0_81_FLAG\n"));
#endif
#endif
case SLOW_IR_BOF:
pIrDev->rcvState = STATE_GOT_BOF;
break;
case SLOW_IR_EOF:
case SLOW_IR_ESC:
default:
/***********************************************/
/* Byte is garbage...scan past it.... */
/***********************************************/
DEBUGMSG(DBG_INT_ERR,
(" ReceiveSirStepFSM(): invalid char in INIT state\n"));
break;
}
break;
case STATE_GOT_BOF:
switch( ThisChar )
{
case SLOW_IR_BOF:
break;
case SLOW_IR_EOF:
/***********************************************/
/* Garbage */
/***********************************************/
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() Invalid char in BOF state\n"));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
case SLOW_IR_ESC:
/***********************************************/
/* Start of data. Our first data byte */
/* happens to be an ESC sequence. */
/***********************************************/
pIrDev->rcvState = STATE_ESC_SEQUENCE;
pIrDev->readBufPos = 0;
break;
default:
pReadBuf[0] = ThisChar;
pIrDev->rcvState = STATE_ACCEPTING;
pIrDev->readBufPos = 1;
break;
}
break;
case STATE_ACCEPTING:
switch( ThisChar )
{
case SLOW_IR_BOF:
//
// Either a new packet is starting here and we're missing parts of the old one
// or it's garbage
//
#if !defined(WORKAROUND_MISSING_C1)
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() Invalid char in ACCEPTING state\n"));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
#else
//
// Take the packet and decrement the pointer in the FIFO decoding so that
// the new packet can be processed
//
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() C0 in ACCEPTING state, trying workaround\n"));
rawBufPos --;
pRecBuf->MissingC1Detected = TRUE;
#endif
case SLOW_IR_EOF:
if( pIrDev->readBufPos < (IRDA_A_C_TOTAL_SIZE + SLOW_IR_FCS_SIZE) )
{
pIrDev->packetsReceivedRunt ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
#if defined(WORKAROUND_MISSING_C1)
if( pRecBuf->MissingC1Detected )
pRecBuf->MissingC1Detected = FALSE;
else
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() Error packet too small\n"));
#else
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() Error packet too small\n"));
#endif
}
else
{
pIrDev->rcvState = STATE_SAW_EOF;
}
break;
case SLOW_IR_ESC:
pIrDev->rcvState = STATE_ESC_SEQUENCE;
break;
default:
pReadBuf[pIrDev->readBufPos++] = ThisChar;
break;
}
break;
case STATE_ESC_SEQUENCE:
switch( ThisChar )
{
case SLOW_IR_EOF:
case SLOW_IR_BOF:
case SLOW_IR_ESC:
/***********************************************/
/* ESC + {EOF|BOF|ESC} is an abort sequence */
/***********************************************/
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
case SLOW_IR_EOF ^ SLOW_IR_ESC_COMP:
case SLOW_IR_BOF ^ SLOW_IR_ESC_COMP:
case SLOW_IR_ESC ^ SLOW_IR_ESC_COMP:
pReadBuf[pIrDev->readBufPos++] = ThisChar ^ SLOW_IR_ESC_COMP;
pIrDev->rcvState = STATE_ACCEPTING;
break;
default:
// junk
DEBUGMSG(DBG_INT_ERR,
(" ReceiveSirStepFSM(): invalid escaped char=%X\n", (ULONG)ThisChar));
pIrDev->packetsReceivedDropped ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
break;
}
break;
case STATE_SAW_EOF:
default:
DEBUGMSG(DBG_INT_ERR,
(" ReceiveSirStepFSM(): Illegal state, bufpos=%d\n", pIrDev->readBufPos));
IRUSB_ASSERT( 0 );
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
return FALSE;
}
}
}
// * Set result and do any post-cleanup.
switch( pIrDev->rcvState )
{
case STATE_SAW_EOF:
// We've read in the entire packet.
// Queue it and return TRUE.
pRecBuf->DataLen = pIrDev->readBufPos;
pIrDev->pCurrentRecBuf = NULL;
ReceiveDeliverBuffer(
pIrDev,
pRecBuf
);
FrameProcessed = TRUE;
if( rawBufPos < rawBytesRead )
{
/***********************************************/
/* This is ugly. We have some more */
/* unprocessed bytes in the raw buffer. */
/* Move these to the beginning of the raw */
/* buffer go to the CLEANUP state, which */
/* indicates that these bytes be used up */
/* during the next call. (This is typically */
/* only 1 or 2 bytes). */
/* */
/* Note: We can't just leave these in the */
/* raw buffer because we might be */
/* supporting connections to multiple COM */
/* ports. */
/* */
/***********************************************/
RtlMoveMemory( pRawBuf, &pRawBuf[rawBufPos], rawBytesRead - rawBufPos );
pIrDev->rawCleanupBytesRead = rawBytesRead - rawBufPos;
pIrDev->rcvState = STATE_CLEANUP;
#if defined( WORKAROUND_9600_ANTIBOUNCING )
if( (pIrDev->currentSpeed == SPEED_9600) && (pIrDev->ChipRevision <= CHIP_REVISION_7) )
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR, (" ReceiveSirStepFSM(): Delaying\n"));
#endif
NdisMSleep( 10*1000 );
}
#endif
}
else
{
pIrDev->rcvState = STATE_INIT;
}
pIrDev->readBufPos = 0;
break;
default:
if( pIrDev->readBufPos > (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + SLOW_IR_FCS_SIZE) )
{
DEBUGMSG( DBG_INT_ERR,(" ReceiveSirStepFSM() Overflow\n"));
pIrDev->packetsReceivedOverflow ++;
pIrDev->rcvState = STATE_INIT;
pIrDev->readBufPos = 0;
pIrDev->pCurrentRecBuf = NULL;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
}
else
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG(DBG_INT_ERR, (" ReceiveSirStepFSM(): returning with partial packet, read %d bytes\n", pIrDev->readBufPos));
#endif
}
FrameProcessed = FALSE;
break;
}
return FrameProcessed;
}
/*****************************************************************************
*
* Function: ReceiveProcessReturnPacket
*
* Synopsis: Returns the packet to the free pool after preparing for reuse
*
* Arguments: pThisDev - pointer to the current ir device object
* pReceiveBuffer - pointer to a RCV_BUFFER struct
*
* Returns: None
*
*
*****************************************************************************/
VOID
ReceiveProcessReturnPacket(
OUT PIR_DEVICE pThisDev,
OUT PRCV_BUFFER pReceiveBuffer
)
{
PNDIS_BUFFER pBuffer;
DEBUGONCE(DBG_FUNC, ("+ReceiveProcessReturnPacket\n"));
//
// Deallocate the buffer
//
NdisUnchainBufferAtFront( (PNDIS_PACKET)pReceiveBuffer->pPacket, &pBuffer );
IRUSB_ASSERT( pBuffer );
if( pBuffer )
{
NdisFreeBuffer( pBuffer );
}
//
// Get ready to reuse
//
InterlockedExchange( &pReceiveBuffer->DataLen, 0 );
InterlockedExchange( &pReceiveBuffer->fInRcvDpc, FALSE );
InterlockedExchange( (PULONG)&pReceiveBuffer->BufferState, RCV_STATE_FREE );
#if DBG
if( InterlockedDecrement(&pThisDev->packetsHeldByProtocol)<0 )
{
IRUSB_ASSERT(0);
}
#endif
DEBUGMSG(DBG_FUNC, ("-ReceiveProcessReturnPacket\n"));
}
/*****************************************************************************
*
* Function: ReceiveDeliverBuffer
*
* Synopsis: Delivers the buffer to the protocol via
* NdisMIndicateReceivePacket.
*
* Arguments: pThisDev - pointer to the current ir device object
* pRecBuf - poiter to descriptor to deliver
*
* Returns: None
*
*
*****************************************************************************/
VOID
ReceiveDeliverBuffer(
IN OUT PIR_DEVICE pThisDev,
IN PRCV_BUFFER pRecBuf
)
{
PNDIS_BUFFER pBuffer;
NDIS_STATUS Status;
DEBUGMSG(DBG_FUNC, ("+ReceiveDeliverBuffer\n"));
if( pThisDev->currentSpeed <= MAX_MIR_SPEED )
{
USHORT sirfcs;
/***********************************************/
/* The packet we have already has had BOFs, */
/* EOF, and * escape-sequences removed. It */
/* contains an FCS code at the end, which */
/* we need to verify and then remove before */
/* delivering the frame. We compute the FCS */
/* on the packet with the packet FCS */
/* attached; this should produce the */
/* constant value GOOD_FCS. */
/***********************************************/
if( (sirfcs = ComputeFCS16(pRecBuf->pDataBuf, pRecBuf->DataLen)) != GOOD_FCS )
{
#if !defined(WORKAROUND_EXTRA_BYTE) && !defined(WORKAROUND_MISSING_C1)
//
// FCS error...drop frame...
//
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->DataLen));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
goto done;
#else
//
// Calculate again stripping off the last byte
//
if( pRecBuf->MissingC1Detected )
{
if( (sirfcs = ComputeFCS16(pRecBuf->pDataBuf, pRecBuf->DataLen-1)) != GOOD_FCS )
{
#if defined(RECEIVE_ERROR_LOGGING)
if( pThisDev->ReceiveErrorFileHandle )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveErrorFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pRecBuf->pDataBuf,
pRecBuf->DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveErrorFilePosition,
NULL
);
pThisDev->ReceiveErrorFilePosition += pRecBuf->DataLen;
}
#endif
//
// It is really junk
//
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->DataLen));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
pRecBuf->MissingC1Detected = FALSE;
goto done;
}
else
{
//
// Readjust to get rid of the extra byte
//
pRecBuf->DataLen --;
pRecBuf->MissingC1Detected = FALSE;
}
}
else
{
//
// Or maybe the first one
//
if( (sirfcs = ComputeFCS16(pRecBuf->pDataBuf+1, pRecBuf->DataLen-1)) != GOOD_FCS )
{
#if defined(RECEIVE_ERROR_LOGGING)
if( pThisDev->ReceiveErrorFileHandle )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveErrorFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pRecBuf->pDataBuf,
pRecBuf->DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveErrorFilePosition,
NULL
);
pThisDev->ReceiveErrorFilePosition += pRecBuf->DataLen;
}
#endif
//
// It is really junk
//
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->DataLen));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
goto done;
}
//
else
{
//
// Readjust to get rid of the extra byte
//
pRecBuf->DataLen --;
RtlMoveMemory( pRecBuf->pDataBuf, &pRecBuf->pDataBuf[1], pRecBuf->DataLen );
}
}
#endif
}
/***********************************************/
/* Remove FCS from end of packet... */
/***********************************************/
pRecBuf->DataLen -= SLOW_IR_FCS_SIZE;
}
else
{
LONG firfcs;
#if !defined(WORKAROUND_33_HANG)
if( (firfcs = ComputeFCS32(pRecBuf->dataBuf, pRecBuf->dataLen)) != FIR_GOOD_FCS )
{
/***********************************************/
/* FCS error...drop frame... */
/***********************************************/
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->dataLen));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->dataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->state, RCV_STATE_FREE );
goto done;
}
#else
if( (firfcs = ComputeFCS32(pRecBuf->pDataBuf, pRecBuf->DataLen)) != FIR_GOOD_FCS )
{
NTSTATUS rc;
//
// Try again with the data stuffed with 0x33
//
if( pRecBuf->DataLen < (MAX_TOTAL_SIZE_WITH_ALL_HEADERS + FAST_IR_FCS_SIZE) )
{
pRecBuf->pDataBuf[pRecBuf->DataLen] = 0x33;
pRecBuf->DataLen ++;
if( (firfcs = ComputeFCS32(pRecBuf->pDataBuf, pRecBuf->DataLen)) != FIR_GOOD_FCS )
{
#if defined(RECEIVE_ERROR_LOGGING)
if( pThisDev->ReceiveErrorFileHandle )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveErrorFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pRecBuf->pDataBuf,
pRecBuf->DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveErrorFilePosition,
NULL
);
pThisDev->ReceiveErrorFilePosition += pRecBuf->DataLen;
}
#endif
/***********************************************/
/* FCS error...drop frame... */
/***********************************************/
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->DataLen));
pThisDev->ReceiveAdaptiveDelayBoost += STIR4200_DELTA_DELAY;
if( pThisDev->ReceiveAdaptiveDelayBoost <= STIR4200_MAX_BOOST_DELAY )
pThisDev->ReceiveAdaptiveDelay += STIR4200_DELTA_DELAY;
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Delay: %d\n",pThisDev->ReceiveAdaptiveDelay));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
goto done;
}
}
else
{
#if defined(RECEIVE_ERROR_LOGGING)
if( pThisDev->ReceiveErrorFileHandle )
{
IO_STATUS_BLOCK IoStatusBlock;
ZwWriteFile(
pThisDev->ReceiveErrorFileHandle,
NULL,
NULL,
NULL,
&IoStatusBlock,
pRecBuf->pDataBuf,
pRecBuf->DataLen,
(PLARGE_INTEGER)&pThisDev->ReceiveErrorFilePosition,
NULL
);
pThisDev->ReceiveErrorFilePosition += pRecBuf->DataLen;
}
#endif
/***********************************************/
/* FCS error...drop frame... */
/***********************************************/
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Bad FCS, size: %d\n",pRecBuf->DataLen));
pThisDev->ReceiveAdaptiveDelayBoost += STIR4200_DELTA_DELAY;
if( pThisDev->ReceiveAdaptiveDelayBoost <= STIR4200_MAX_BOOST_DELAY )
pThisDev->ReceiveAdaptiveDelay += STIR4200_DELTA_DELAY;
DEBUGMSG( DBG_INT_ERR,(" ReceiveDeliverBuffer(): Delay: %d\n",pThisDev->ReceiveAdaptiveDelay));
pThisDev->packetsReceivedChecksum ++;
InterlockedExchange( &pRecBuf->DataLen, 0 );
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
goto done;
}
//
// Reset the USB of the part
//
if( pThisDev->ChipRevision <= CHIP_REVISION_7 )
{
St4200ResetFifo( pThisDev );
}
}
#endif
/***********************************************/
/* Remove FCS from end of packet... */
/***********************************************/
pRecBuf->DataLen -= FAST_IR_FCS_SIZE;
}
//
// If in normal mode, give the packet to the protocol
//
#if defined(DIAGS)
if( !pThisDev->DiagsActive )
{
#endif
NdisAllocateBuffer(
&Status,
&pBuffer,
pThisDev->hBufferPool,
(PVOID)pRecBuf->pDataBuf,
pRecBuf->DataLen
);
if( Status != NDIS_STATUS_SUCCESS )
{
DEBUGMSG( DBG_ERR,(" ReceiveDeliverBuffer(): No packets available...\n"));
InterlockedExchange( &pRecBuf->DataLen, 0);
InterlockedExchange( (PULONG)&pRecBuf->BufferState, RCV_STATE_FREE );
goto done;
}
NdisChainBufferAtFront( (PNDIS_PACKET)pRecBuf->pPacket, pBuffer );
//
// Fix up some other packet fields.
// Remember, we only account for A and C fields
//
NDIS_SET_PACKET_HEADER_SIZE(
(PNDIS_PACKET)pRecBuf->pPacket,
IRDA_CONTROL_FIELD_SIZE + IRDA_ADDRESS_FIELD_SIZE
);
#if DBG
InterlockedIncrement( &pThisDev->packetsHeldByProtocol );
if( pThisDev->packetsHeldByProtocol > pThisDev->MaxPacketsHeldByProtocol )
{
pThisDev->MaxPacketsHeldByProtocol = pThisDev->packetsHeldByProtocol; //keep record of our longest attained len
}
#endif
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG( DBG_INT_ERR,
(" ReceiveDeliverBuffer() Handed packet to protocol, size: %d\n", pRecBuf->DataLen ));
#endif
//
// Indicate the packet to NDIS
//
NDIS_SET_PACKET_STATUS( (PNDIS_PACKET)pRecBuf->pPacket, NDIS_STATUS_PENDING );
InterlockedExchange( &pRecBuf->fInRcvDpc, TRUE );
NdisMIndicateReceivePacket(
pThisDev->hNdisAdapter,
&((PNDIS_PACKET)pRecBuf->pPacket),
1
);
//
// Check to see if the packet is not pending (patch for 98)
//
#if defined(LEGACY_NDIS5)
Status = NDIS_GET_PACKET_STATUS( (PNDIS_PACKET)pRecBuf->pPacket );
if( (Status == NDIS_STATUS_SUCCESS) || (Status == NDIS_STATUS_RESOURCES) )
{
ReceiveProcessReturnPacket( pThisDev, pRecBuf ) ;
}
#endif
#if defined(DIAGS)
}
//
// Do a diagnostic receive
//
else
{
#if !defined(ONLY_ERROR_MESSAGES)
DEBUGMSG( DBG_INT_ERR,
(" ReceiveDeliverBuffer() Queued packet, size: %d\n", pRecBuf->DataLen ));
#endif
//
// Put the buffer in the diagnostic queue
//
ExInterlockedInsertTailList(
&pThisDev->DiagsReceiveQueue,
&pRecBuf->ListEntry,
&pThisDev->DiagsReceiveLock
);
}
#endif
done:
DEBUGMSG(DBG_FUNC, ("-ReceiveDeliverBuffer\n"));
}
/*****************************************************************************
*
* Function: StIrUsbReturnPacket
*
* Synopsis: The protocol returns ownership of a receive packet to
* the ir device object.
*
* Arguments: Context - a pointer to the current ir device obect.
* pReturnedPacket - a pointer the packet which the protocol
* is returning ownership.
*
* Returns: None.
*
*
*
*****************************************************************************/
VOID
StIrUsbReturnPacket(
IN OUT NDIS_HANDLE Context,
IN OUT PNDIS_PACKET pReturnedPacket
)
{
PIR_DEVICE pThisDev;
PNDIS_BUFFER pBuffer;
PRCV_BUFFER pRecBuffer;
UINT Index;
BOOLEAN found = FALSE;
DEBUGONCE(DBG_FUNC, ("+StIrUsbReturnPacket\n"));
//
// The context is just the pointer to the current ir device object.
//
pThisDev = CONTEXT_TO_DEV( Context );
NdisInterlockedIncrement( (PLONG)&pThisDev->packetsReceived );
//
// Search the queue to find the right packet.
//
for( Index=0; Index < NUM_RCV_BUFS; Index ++ )
{
pRecBuffer = &(pThisDev->rcvBufs[Index]);
if( ((PNDIS_PACKET) pRecBuffer->pPacket) == pReturnedPacket )
{
if( pRecBuffer->fInRcvDpc )
{
ReceiveProcessReturnPacket( pThisDev, pRecBuffer );
found = TRUE;
}
else
{
DEBUGMSG(DBG_ERR, (" StIrUsbReturnPacket, queues are corrupt\n"));
IRUSB_ASSERT( 0 );
}
break;
}
}
//
// Ensure that the packet was found.
//
IRUSB_ASSERT( found );
DEBUGMSG(DBG_FUNC, ("-StIrUsbReturnPacket\n"));
}
/*****************************************************************************
*
* Function: ReceiveGetBuf
*
* Synopsis: Gets a receive buffer
*
* Arguments: pThisDev - a pointer to the current ir device obect
* pIndex - pointer to return the buffer index
* state - state to set the buffer to
*
* Returns: buffer
*
*
*****************************************************************************/
PRCV_BUFFER
ReceiveGetBuf(
IN PIR_DEVICE pThisDev,
OUT PUINT pIndex,
IN RCV_BUFFER_STATE BufferState
)
{
UINT Index;
PRCV_BUFFER pBuf = NULL;
DEBUGMSG(DBG_FUNC, ("+ReceiveGetBuf()\n"));
//
// Look for a free buffer to return
//
for( Index=0; Index<NUM_RCV_BUFS; Index++ )
{
if( pThisDev->rcvBufs[Index].BufferState == RCV_STATE_FREE )
{
//
// set to input state
//
InterlockedExchange( (PULONG)&pThisDev->rcvBufs[Index].BufferState, (ULONG)BufferState );
*pIndex = Index;
pBuf = &(pThisDev->rcvBufs[*pIndex]);
break;
}
}
DEBUGMSG(DBG_FUNC, ("-ReceiveGetBuf()\n"));
return pBuf;
}
/*****************************************************************************
*
* Function: ReceivePacketRead
*
* Synopsis: Reads a packet from the US device
* the inbound USB header, check for overrun,
* deliver to the protocol
*
* Arguments: pThisDev - pointer to the current ir device object
* pRecBuf - pointer to a RCV_BUFFER struct
*
* Returns: NT status code
*
*
*****************************************************************************/
NTSTATUS
ReceivePacketRead(
IN PIR_DEVICE pThisDev,
OUT PFIFO_BUFFER pRecBuf
)
{
ULONG UrbSize;
ULONG TransferLength;
PURB pUrb = NULL;
PDEVICE_OBJECT pUrbTargetDev;
PIO_STACK_LOCATION pNextStack;
NTSTATUS Status = STATUS_UNSUCCESSFUL;
DEBUGMSG(DBG_FUNC, ("+ReceivePacketRead()\n"));
IRUSB_ASSERT( KeGetCurrentIrql() == PASSIVE_LEVEL );
UrbSize = sizeof(struct _URB_BULK_OR_INTERRUPT_TRANSFER);
TransferLength = STIR4200_FIFO_SIZE;
//
// Stop if a halt/reset/suspend is going on
//
if( pThisDev->fPendingReadClearStall || pThisDev->fPendingHalt ||
pThisDev->fPendingReset || pThisDev->fPendingClearTotalStall || !pThisDev->fProcessing )
{
//
// USB reset going on?
//
DEBUGMSG( DBG_ERR,(" ReceivePacketRead() rejecting a packet due to pendig halt/reset\n"));
Status = STATUS_UNSUCCESSFUL;
goto done;
}
pUrb = pThisDev->pUrb;
//
// Build our URB for USBD
//
NdisZeroMemory( pUrb, UrbSize );
IRUSB_ASSERT( pThisDev->BulkInPipeHandle );
//
// Now that we have created the urb, we will send a
// request to the USB device object.
//
KeClearEvent( &pThisDev->EventSyncUrb );
pUrbTargetDev = pThisDev->pUsbDevObj;
IRUSB_ASSERT( pUrbTargetDev );
//
// make an irp sending to usbhub
//
pRecBuf->pIrp = IoAllocateIrp( (CCHAR)(pThisDev->pUsbDevObj->StackSize + 1), FALSE );
if( NULL == pRecBuf->pIrp )
{
DEBUGMSG(DBG_ERR, (" read failed to alloc IRP\n"));
Status = STATUS_INSUFFICIENT_RESOURCES;
goto done;
}
((PIRP)pRecBuf->pIrp)->IoStatus.Status = STATUS_PENDING;
((PIRP)pRecBuf->pIrp)->IoStatus.Information = 0;
//
// Build our URB for USBD
//
pUrb->UrbBulkOrInterruptTransfer.Hdr.Length = (USHORT)UrbSize;
pUrb->UrbBulkOrInterruptTransfer.Hdr.Function = URB_FUNCTION_BULK_OR_INTERRUPT_TRANSFER;
pUrb->UrbBulkOrInterruptTransfer.PipeHandle = pThisDev->BulkInPipeHandle;
pUrb->UrbBulkOrInterruptTransfer.TransferFlags = USBD_TRANSFER_DIRECTION_IN ;
//
// short packet is not treated as an error.
//
pUrb->UrbBulkOrInterruptTransfer.TransferFlags |= USBD_SHORT_TRANSFER_OK;
//
// not using linked urb's
//
pUrb->UrbBulkOrInterruptTransfer.UrbLink = NULL;
pUrb->UrbBulkOrInterruptTransfer.TransferBufferMDL = NULL;
pUrb->UrbBulkOrInterruptTransfer.TransferBuffer = pRecBuf->pDataBuf;
pUrb->UrbBulkOrInterruptTransfer.TransferBufferLength = TransferLength;
//
// Call the class driver to perform the operation.
//
pNextStack = IoGetNextIrpStackLocation( (PIRP)pRecBuf->pIrp );
IRUSB_ASSERT( pNextStack != NULL );
//
// pass the URB to the USB driver stack
//
pNextStack->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
pNextStack->Parameters.Others.Argument1 = pUrb;
pNextStack->Parameters.DeviceIoControl.IoControlCode = IOCTL_INTERNAL_USB_SUBMIT_URB;
IoSetCompletionRoutine(
((PIRP)pRecBuf->pIrp), // irp to use
ReceiveCompletePacketRead, // routine to call when irp is done
pRecBuf, // context to pass routine is the RCV_BUFFER
TRUE, // call on success
TRUE, // call on error
TRUE // call on cancel
);
//
// Call IoCallDriver to send the irp to the usb port.
//
InterlockedExchange( (PLONG)&pRecBuf->BufferState, RCV_STATE_PENDING );
Status = MyIoCallDriver( pThisDev, pUrbTargetDev, (PIRP)pRecBuf->pIrp ); // Start UsbRead()
DEBUGMSG(DBG_FUNC, (" ReceivePacketRead() after IoCallDriver () status = 0x%x\n", Status));
IRUSB_ASSERT( STATUS_SUCCESS != Status );
//
// Wait for completion
//
Status = MyKeWaitForSingleObject(
pThisDev,
&pThisDev->EventSyncUrb, // event to wait on
NULL, // irp to cancel on halt/reset or timeout
0
);
if( Status == STATUS_TIMEOUT )
{
IrUsb_CancelIo( pThisDev, pRecBuf->pIrp, &pThisDev->EventSyncUrb );
}
else
{
//
// Update the status to reflect the real return code
//
Status = pThisDev->StatusSendReceive;
}
IRUSB_ASSERT( NULL == pRecBuf->pIrp ); // Will be nulled by completion routine
DEBUGMSG(DBG_FUNC, (" ReceivePacketRead() after KeWaitForSingleObject() Status = 0x%x\n", Status));
done:
DEBUGMSG(DBG_FUNC, ("-ReceivePacketRead()\n"));
return Status;
}
/*****************************************************************************
*
* Function: ReceiveCompletePacketRead
*
* Synopsis: Completes USB read operation
*
* Arguments: pUsbDevObj - pointer to the USB device object which
* completed the irp
* pIrp - the irp which was completed by the
* device object
* Context - the context given to IoSetCompletionRoutine
* before calling IoCallDriver on the irp
* The Context is a pointer to the ir device object.
*
* Returns: STATUS_MORE_PROCESSING_REQUIRED - allows the completion routine
* (IofCompleteRequest) to stop working on the irp.
*
*
*****************************************************************************/
NTSTATUS
ReceiveCompletePacketRead(
IN PDEVICE_OBJECT pUsbDevObj,
IN PIRP pIrp,
IN PVOID Context
)
{
PIR_DEVICE pThisDev;
NTSTATUS status;
ULONG_PTR BytesRead;
PFIFO_BUFFER pFifoBuf;
DEBUGMSG(DBG_FUNC, ("+ReceiveCompletePacketRead\n"));
//
// The context given to ReceiveCompletePacketRead is the receive buffer object
//
pFifoBuf = (PFIFO_BUFFER)Context;
pThisDev = (PIR_DEVICE)pFifoBuf->pThisDev;
IRUSB_ASSERT( pFifoBuf->pIrp == pIrp );
IRUSB_ASSERT( NULL != pThisDev );
//
// We have a number of cases:
// 1) The USB read timed out and we received no data.
// 2) The USB read timed out and we received some data.
// 3) The USB read was successful and fully filled our irp buffer.
// 4) The irp was cancelled.
// 5) Some other failure from the USB device object.
//
status = pIrp->IoStatus.Status;
//
// IoCallDriver has been called on this Irp;
// Set the length based on the TransferBufferLength
// value in the URB
//
pIrp->IoStatus.Information = pThisDev->pUrb->UrbBulkOrInterruptTransfer.TransferBufferLength;
BytesRead = pIrp->IoStatus.Information;
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead Bytes Read = 0x%x, dec %d\n", BytesRead,BytesRead ));
switch( status )
{
case STATUS_SUCCESS:
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead STATUS_SUCCESS\n"));
if( BytesRead > 0 )
{
pFifoBuf->DataLen = (UINT)pIrp->IoStatus.Information;
}
break; // STATUS_SUCCESS
case STATUS_TIMEOUT:
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead STATUS_TIMEOUT\n"));
break;
case STATUS_PENDING:
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead STATUS_PENDING\n"));
break;
case STATUS_DEVICE_DATA_ERROR:
// can get during shutdown
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead STATUS_DEVICE_DATA_ERROR\n"));
break;
case STATUS_UNSUCCESSFUL:
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead STATUS_UNSUCCESSFUL\n"));
break;
case STATUS_INSUFFICIENT_RESOURCES:
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead STATUS_INSUFFICIENT_RESOURCES\n"));
break;
case STATUS_INVALID_PARAMETER:
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead STATUS_INVALID_PARAMETER\n"));
break;
case STATUS_CANCELLED:
DEBUGMSG(DBG_FUNC, (" ReceiveCompletePacketRead STATUS_CANCELLED\n"));
break;
case STATUS_DEVICE_NOT_CONNECTED:
// can get during shutdown
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead STATUS_DEVICE_NOT_CONNECTED\n"));
break;
case STATUS_DEVICE_POWER_FAILURE:
// can get during shutdown
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead STATUS_DEVICE_POWER_FAILURE\n"));
break;
default:
InterlockedIncrement( (PLONG)&pThisDev->NumDataErrors );
DEBUGMSG(DBG_ERR, (" ReceiveCompletePacketRead UNKNOWN WEIRD STATUS = 0x%x, dec %d\n",status,status ));
break;
}
//
// change the status
//
if( STATUS_SUCCESS != status )
{
InterlockedExchange( (PLONG)&pFifoBuf->BufferState, RCV_STATE_FREE );
}
else
{
InterlockedExchange( (PLONG)&pFifoBuf->BufferState, RCV_STATE_FULL );
}
//
// Free the IRP and its mdl because they were allocated by us
//
IoFreeIrp( pIrp );
pFifoBuf->pIrp = NULL;
InterlockedIncrement( (PLONG)&pThisDev->NumReads );
//
// we will track count of pending irps
//
IrUsb_DecIoCount( pThisDev );
if( ( STATUS_SUCCESS != status ) && ( STATUS_CANCELLED != status ) &&
( STATUS_DEVICE_NOT_CONNECTED != status ) )
{
PURB urb = pThisDev->pUrb;
DEBUGMSG(DBG_ERR, (" USBD status = 0x%x\n", urb->UrbHeader.Status));
DEBUGMSG(DBG_ERR, (" NT status = 0x%x\n", status));
if( !pThisDev->fPendingReadClearStall && !pThisDev->fPendingClearTotalStall &&
!pThisDev->fPendingHalt && !pThisDev->fPendingReset && pThisDev->fProcessing )
{
DEBUGMSG(DBG_ERR,
(" ReceiveCompletePacketRead error, will schedule a clear stall via URB_FUNCTION_RESET_PIPE (IN)\n"));
InterlockedExchange( &pThisDev->fPendingReadClearStall, TRUE );
ScheduleWorkItem( pThisDev, ResetPipeCallback, pThisDev->BulkInPipeHandle, 0 );
}
}
//
// This will only work as long as we serialize the access to the hardware
//
pThisDev->StatusSendReceive = status;
//
// Signal completion
//
KeSetEvent( &pThisDev->EventSyncUrb, 0, FALSE );
//
// We return STATUS_MORE_PROCESSING_REQUIRED so that the completion
// routine (IoCompleteRequest) will stop working on the irp.
//
DEBUGMSG(DBG_FUNC, ("-ReceiveCompletePacketRead\n"));
return STATUS_MORE_PROCESSING_REQUIRED;
}