2018-05-18 22:23:45 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Intel PRO/1000 Driver
|
|
|
|
* LICENSE: GPL-2.0+ (https://spdx.org/licenses/GPL-2.0+)
|
|
|
|
* PURPOSE: Interrupt handlers
|
|
|
|
* COPYRIGHT: Copyright 2013 Cameron Gutman (cameron.gutman@reactos.org)
|
|
|
|
* Copyright 2018 Mark Jansen (mark.jansen@reactos.org)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "nic.h"
|
|
|
|
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
MiniportISR(
|
|
|
|
OUT PBOOLEAN InterruptRecognized,
|
|
|
|
OUT PBOOLEAN QueueMiniportHandleInterrupt,
|
|
|
|
IN NDIS_HANDLE MiniportAdapterContext)
|
|
|
|
{
|
2018-05-29 20:52:43 +00:00
|
|
|
ULONG Value;
|
2018-05-18 22:23:45 +00:00
|
|
|
PE1000_ADAPTER Adapter = (PE1000_ADAPTER)MiniportAdapterContext;
|
|
|
|
|
2018-05-29 20:52:43 +00:00
|
|
|
Value = NICInterruptRecognized(Adapter, InterruptRecognized);
|
|
|
|
InterlockedOr(&Adapter->InterruptPending, Value);
|
2018-05-18 22:23:45 +00:00
|
|
|
|
|
|
|
if (!(*InterruptRecognized))
|
|
|
|
{
|
|
|
|
/* This is not ours. */
|
|
|
|
*QueueMiniportHandleInterrupt = FALSE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-29 20:52:43 +00:00
|
|
|
/* Mark the events pending service */
|
2018-05-18 22:23:45 +00:00
|
|
|
*QueueMiniportHandleInterrupt = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
MiniportHandleInterrupt(
|
|
|
|
IN NDIS_HANDLE MiniportAdapterContext)
|
|
|
|
{
|
2018-05-29 20:52:43 +00:00
|
|
|
ULONG Value;
|
|
|
|
PE1000_ADAPTER Adapter = (PE1000_ADAPTER)MiniportAdapterContext;
|
|
|
|
volatile PE1000_TRANSMIT_DESCRIPTOR TransmitDescriptor;
|
|
|
|
|
2018-05-18 22:23:45 +00:00
|
|
|
NDIS_DbgPrint(MAX_TRACE, ("Called.\n"));
|
2018-05-29 20:52:43 +00:00
|
|
|
|
|
|
|
Value = InterlockedExchange(&Adapter->InterruptPending, 0);
|
|
|
|
|
|
|
|
NdisDprAcquireSpinLock(&Adapter->Lock);
|
|
|
|
|
|
|
|
if (Value & E1000_IMS_LSC)
|
|
|
|
{
|
|
|
|
ULONG Status;
|
2018-06-03 20:43:50 +00:00
|
|
|
|
2018-05-29 20:52:43 +00:00
|
|
|
NdisDprReleaseSpinLock(&Adapter->Lock);
|
|
|
|
Value &= ~E1000_IMS_LSC;
|
|
|
|
NDIS_DbgPrint(MIN_TRACE, ("Link status changed!.\n"));
|
|
|
|
|
|
|
|
NICUpdateLinkStatus(Adapter);
|
|
|
|
|
|
|
|
Status = Adapter->MediaState == NdisMediaStateConnected ? NDIS_STATUS_MEDIA_CONNECT : NDIS_STATUS_MEDIA_DISCONNECT;
|
|
|
|
NdisMIndicateStatus(Adapter->AdapterHandle, Status, NULL, 0);
|
|
|
|
NdisMIndicateStatusComplete(Adapter->AdapterHandle);
|
|
|
|
|
|
|
|
NdisDprAcquireSpinLock(&Adapter->Lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Value & E1000_IMS_TXDW)
|
|
|
|
{
|
|
|
|
while (Adapter->TxFull || Adapter->LastTxDesc != Adapter->CurrentTxDesc)
|
|
|
|
{
|
|
|
|
TransmitDescriptor = Adapter->TransmitDescriptors + Adapter->LastTxDesc;
|
|
|
|
|
|
|
|
if (!(TransmitDescriptor->Status & E1000_TDESC_STATUS_DD))
|
|
|
|
{
|
|
|
|
/* Not processed yet */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
Adapter->LastTxDesc = (Adapter->LastTxDesc + 1) % NUM_TRANSMIT_DESCRIPTORS;
|
|
|
|
Value &= ~E1000_IMS_TXDW;
|
|
|
|
Adapter->TxFull = FALSE;
|
|
|
|
NDIS_DbgPrint(MAX_TRACE, ("CurrentTxDesc:%u, LastTxDesc:%u\n", Adapter->CurrentTxDesc, Adapter->LastTxDesc));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-03 20:43:50 +00:00
|
|
|
if (Value & (E1000_IMS_RXDMT0 | E1000_IMS_RXT0))
|
|
|
|
{
|
|
|
|
volatile PE1000_RECEIVE_DESCRIPTOR ReceiveDescriptor;
|
|
|
|
PETH_HEADER EthHeader;
|
|
|
|
ULONG BufferOffset;
|
|
|
|
|
|
|
|
/* Clear out these interrupts */
|
|
|
|
Value &= ~(E1000_IMS_RXDMT0 | E1000_IMS_RXT0);
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
|
|
|
BufferOffset = Adapter->CurrentRxDesc * Adapter->ReceiveBufferEntrySize;
|
|
|
|
ReceiveDescriptor = Adapter->ReceiveDescriptors + Adapter->CurrentRxDesc;
|
|
|
|
|
|
|
|
if (!(ReceiveDescriptor->Status & E1000_RDESC_STATUS_DD))
|
|
|
|
{
|
|
|
|
/* Not received yet */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ReceiveDescriptor->Length != 0)
|
|
|
|
{
|
|
|
|
EthHeader = Adapter->ReceiveBuffer + BufferOffset;
|
|
|
|
|
|
|
|
NdisMEthIndicateReceive(Adapter->AdapterHandle,
|
|
|
|
NULL,
|
|
|
|
EthHeader,
|
|
|
|
sizeof(ETH_HEADER),
|
|
|
|
EthHeader + 1,
|
|
|
|
ReceiveDescriptor->Length - sizeof(ETH_HEADER),
|
|
|
|
ReceiveDescriptor->Length - sizeof(ETH_HEADER));
|
|
|
|
|
|
|
|
if (ReceiveDescriptor->Status & E1000_RDESC_STATUS_EOP)
|
|
|
|
{
|
|
|
|
NdisMEthIndicateReceiveComplete(Adapter->AdapterHandle);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
__debugbreak();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore the descriptor Address, incase we received a NULL descriptor */
|
|
|
|
ReceiveDescriptor->Address = Adapter->ReceiveBufferPa.QuadPart + BufferOffset;
|
|
|
|
/* Give the descriptor back */
|
|
|
|
ReceiveDescriptor->Status = 0;
|
|
|
|
E1000WriteUlong(Adapter, E1000_REG_RDT, Adapter->CurrentRxDesc);
|
|
|
|
Adapter->CurrentRxDesc = (Adapter->CurrentRxDesc + 1) % NUM_RECEIVE_DESCRIPTORS;
|
|
|
|
}
|
|
|
|
}
|
2018-05-29 20:52:43 +00:00
|
|
|
|
|
|
|
ASSERT(Value == 0);
|
2018-05-18 22:23:45 +00:00
|
|
|
}
|