2010-05-08 05:12:14 +02:00
|
|
|
/*
|
|
|
|
LUFA Library
|
2011-01-01 14:00:56 +01:00
|
|
|
Copyright (C) Dean Camera, 2011.
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
dean [at] fourwalledcubicle [dot] com
|
2010-10-28 08:08:58 +02:00
|
|
|
www.lufa-lib.org
|
2010-05-08 05:12:14 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2011-01-01 14:00:56 +01:00
|
|
|
Copyright 2011 Dean Camera (dean [at] fourwalledcubicle [dot] com)
|
2010-05-08 05:12:14 +02:00
|
|
|
|
2010-10-13 16:05:35 +02:00
|
|
|
Permission to use, copy, modify, distribute, and sell this
|
2010-05-08 05:12:14 +02:00
|
|
|
software and its documentation for any purpose is hereby granted
|
2010-10-13 16:05:35 +02:00
|
|
|
without fee, provided that the above copyright notice appear in
|
2010-05-08 05:12:14 +02:00
|
|
|
all copies and that both that the copyright notice and this
|
2010-10-13 16:05:35 +02:00
|
|
|
permission notice and warranty disclaimer appear in supporting
|
|
|
|
documentation, and that the name of the author not be used in
|
|
|
|
advertising or publicity pertaining to distribution of the
|
2010-05-08 05:12:14 +02:00
|
|
|
software without specific, written prior permission.
|
|
|
|
|
|
|
|
The author disclaim all warranties with regard to this
|
|
|
|
software, including all implied warranties of merchantability
|
|
|
|
and fitness. In no event shall the author be liable for any
|
|
|
|
special, indirect or consequential damages or any damages
|
|
|
|
whatsoever resulting from loss of use, data or profits, whether
|
|
|
|
in an action of contract, negligence or other tortious action,
|
|
|
|
arising out of or in connection with the use or performance of
|
|
|
|
this software.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define __INCLUDE_FROM_USB_DRIVER
|
2011-02-19 23:59:27 +01:00
|
|
|
#include "../../Core/USBMode.h"
|
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
#if defined(USB_CAN_BE_DEVICE)
|
|
|
|
|
|
|
|
#define __INCLUDE_FROM_RNDIS_DRIVER
|
2010-10-25 00:53:57 +02:00
|
|
|
#define __INCLUDE_FROM_RNDIS_DEVICE_C
|
2010-05-08 05:12:14 +02:00
|
|
|
#include "RNDIS.h"
|
|
|
|
|
|
|
|
static const uint32_t PROGMEM AdapterSupportedOIDList[] =
|
|
|
|
{
|
|
|
|
OID_GEN_SUPPORTED_LIST,
|
|
|
|
OID_GEN_PHYSICAL_MEDIUM,
|
|
|
|
OID_GEN_HARDWARE_STATUS,
|
|
|
|
OID_GEN_MEDIA_SUPPORTED,
|
|
|
|
OID_GEN_MEDIA_IN_USE,
|
|
|
|
OID_GEN_MAXIMUM_FRAME_SIZE,
|
|
|
|
OID_GEN_MAXIMUM_TOTAL_SIZE,
|
|
|
|
OID_GEN_LINK_SPEED,
|
|
|
|
OID_GEN_TRANSMIT_BLOCK_SIZE,
|
|
|
|
OID_GEN_RECEIVE_BLOCK_SIZE,
|
|
|
|
OID_GEN_VENDOR_ID,
|
|
|
|
OID_GEN_VENDOR_DESCRIPTION,
|
|
|
|
OID_GEN_CURRENT_PACKET_FILTER,
|
|
|
|
OID_GEN_MAXIMUM_TOTAL_SIZE,
|
|
|
|
OID_GEN_MEDIA_CONNECT_STATUS,
|
|
|
|
OID_GEN_XMIT_OK,
|
|
|
|
OID_GEN_RCV_OK,
|
|
|
|
OID_GEN_XMIT_ERROR,
|
|
|
|
OID_GEN_RCV_ERROR,
|
|
|
|
OID_GEN_RCV_NO_BUFFER,
|
|
|
|
OID_802_3_PERMANENT_ADDRESS,
|
|
|
|
OID_802_3_CURRENT_ADDRESS,
|
|
|
|
OID_802_3_MULTICAST_LIST,
|
|
|
|
OID_802_3_MAXIMUM_LIST_SIZE,
|
|
|
|
OID_802_3_RCV_ERROR_ALIGNMENT,
|
|
|
|
OID_802_3_XMIT_ONE_COLLISION,
|
|
|
|
OID_802_3_XMIT_MORE_COLLISIONS,
|
|
|
|
};
|
|
|
|
|
|
|
|
void RNDIS_Device_ProcessControlRequest(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo)
|
|
|
|
{
|
|
|
|
if (!(Endpoint_IsSETUPReceived()))
|
|
|
|
return;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
if (USB_ControlRequest.wIndex != RNDISInterfaceInfo->Config.ControlInterfaceNumber)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (USB_ControlRequest.bRequest)
|
|
|
|
{
|
2010-09-28 14:14:06 +02:00
|
|
|
case RNDIS_REQ_SendEncapsulatedCommand:
|
2010-05-08 05:12:14 +02:00
|
|
|
if (USB_ControlRequest.bmRequestType == (REQDIR_HOSTTODEVICE | REQTYPE_CLASS | REQREC_INTERFACE))
|
|
|
|
{
|
|
|
|
Endpoint_ClearSETUP();
|
|
|
|
Endpoint_Read_Control_Stream_LE(RNDISInterfaceInfo->State.RNDISMessageBuffer, USB_ControlRequest.wLength);
|
|
|
|
Endpoint_ClearIN();
|
2010-08-22 04:45:20 +02:00
|
|
|
|
|
|
|
RNDIS_Device_ProcessRNDISControlMessage(RNDISInterfaceInfo);
|
2010-05-08 05:12:14 +02:00
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
break;
|
2010-09-28 14:14:06 +02:00
|
|
|
case RNDIS_REQ_GetEncapsulatedResponse:
|
2010-05-08 05:12:14 +02:00
|
|
|
if (USB_ControlRequest.bmRequestType == (REQDIR_DEVICETOHOST | REQTYPE_CLASS | REQREC_INTERFACE))
|
|
|
|
{
|
|
|
|
RNDIS_Message_Header_t* MessageHeader = (RNDIS_Message_Header_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
|
|
|
|
if (!(MessageHeader->MessageLength))
|
|
|
|
{
|
|
|
|
RNDISInterfaceInfo->State.RNDISMessageBuffer[0] = 0;
|
|
|
|
MessageHeader->MessageLength = 1;
|
|
|
|
}
|
|
|
|
|
2010-08-22 05:26:20 +02:00
|
|
|
Endpoint_ClearSETUP();
|
2010-10-13 16:05:35 +02:00
|
|
|
Endpoint_Write_Control_Stream_LE(RNDISInterfaceInfo->State.RNDISMessageBuffer, MessageHeader->MessageLength);
|
2010-05-08 05:12:14 +02:00
|
|
|
Endpoint_ClearOUT();
|
|
|
|
|
|
|
|
MessageHeader->MessageLength = 0;
|
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool RNDIS_Device_ConfigureEndpoints(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo)
|
|
|
|
{
|
|
|
|
memset(&RNDISInterfaceInfo->State, 0x00, sizeof(RNDISInterfaceInfo->State));
|
|
|
|
|
2010-09-30 08:23:38 +02:00
|
|
|
for (uint8_t EndpointNum = 1; EndpointNum < ENDPOINT_TOTAL_ENDPOINTS; EndpointNum++)
|
2010-05-08 05:12:14 +02:00
|
|
|
{
|
2010-09-30 08:23:38 +02:00
|
|
|
uint16_t Size;
|
|
|
|
uint8_t Type;
|
|
|
|
uint8_t Direction;
|
|
|
|
bool DoubleBanked;
|
|
|
|
|
|
|
|
if (EndpointNum == RNDISInterfaceInfo->Config.DataINEndpointNumber)
|
|
|
|
{
|
|
|
|
Size = RNDISInterfaceInfo->Config.DataINEndpointSize;
|
|
|
|
Direction = ENDPOINT_DIR_IN;
|
|
|
|
Type = EP_TYPE_BULK;
|
|
|
|
DoubleBanked = RNDISInterfaceInfo->Config.DataINEndpointDoubleBank;
|
|
|
|
}
|
|
|
|
else if (EndpointNum == RNDISInterfaceInfo->Config.DataOUTEndpointNumber)
|
|
|
|
{
|
|
|
|
Size = RNDISInterfaceInfo->Config.DataOUTEndpointSize;
|
|
|
|
Direction = ENDPOINT_DIR_OUT;
|
|
|
|
Type = EP_TYPE_BULK;
|
|
|
|
DoubleBanked = RNDISInterfaceInfo->Config.DataOUTEndpointDoubleBank;
|
|
|
|
}
|
|
|
|
else if (EndpointNum == RNDISInterfaceInfo->Config.NotificationEndpointNumber)
|
|
|
|
{
|
|
|
|
Size = RNDISInterfaceInfo->Config.NotificationEndpointSize;
|
|
|
|
Direction = ENDPOINT_DIR_IN;
|
|
|
|
Type = EP_TYPE_INTERRUPT;
|
|
|
|
DoubleBanked = RNDISInterfaceInfo->Config.NotificationEndpointDoubleBank;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-09-30 08:23:38 +02:00
|
|
|
if (!(Endpoint_ConfigureEndpoint(EndpointNum, Type, Direction, Size,
|
2010-12-02 02:56:06 +01:00
|
|
|
DoubleBanked ? ENDPOINT_BANK_DOUBLE : ENDPOINT_BANK_SINGLE)))
|
2010-09-30 08:23:38 +02:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2010-05-08 05:12:14 +02:00
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RNDIS_Device_USBTask(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo)
|
|
|
|
{
|
|
|
|
if (USB_DeviceState != DEVICE_STATE_Configured)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Endpoint_SelectEndpoint(RNDISInterfaceInfo->Config.NotificationEndpointNumber);
|
|
|
|
|
|
|
|
if (Endpoint_IsINReady() && RNDISInterfaceInfo->State.ResponseReady)
|
|
|
|
{
|
|
|
|
USB_Request_Header_t Notification = (USB_Request_Header_t)
|
|
|
|
{
|
|
|
|
.bmRequestType = (REQDIR_DEVICETOHOST | REQTYPE_CLASS | REQREC_INTERFACE),
|
2010-09-28 16:59:16 +02:00
|
|
|
.bRequest = RNDIS_NOTIF_ResponseAvailable,
|
2010-05-08 05:12:14 +02:00
|
|
|
.wValue = 0,
|
|
|
|
.wIndex = 0,
|
|
|
|
.wLength = 0,
|
|
|
|
};
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2011-01-10 19:43:34 +01:00
|
|
|
Endpoint_Write_Stream_LE(&Notification, sizeof(USB_Request_Header_t), NULL);
|
2010-05-08 05:12:14 +02:00
|
|
|
|
|
|
|
Endpoint_ClearIN();
|
|
|
|
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = false;
|
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
}
|
2010-05-08 05:12:14 +02:00
|
|
|
|
|
|
|
void RNDIS_Device_ProcessRNDISControlMessage(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo)
|
|
|
|
{
|
|
|
|
/* Note: Only a single buffer is used for both the received message and its response to save SRAM. Because of
|
|
|
|
this, response bytes should be filled in order so that they do not clobber unread data in the buffer. */
|
|
|
|
|
|
|
|
RNDIS_Message_Header_t* MessageHeader = (RNDIS_Message_Header_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
|
|
|
|
switch (MessageHeader->MessageType)
|
|
|
|
{
|
|
|
|
case REMOTE_NDIS_INITIALIZE_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = true;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDIS_Initialize_Message_t* INITIALIZE_Message =
|
|
|
|
(RNDIS_Initialize_Message_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
RNDIS_Initialize_Complete_t* INITIALIZE_Response =
|
|
|
|
(RNDIS_Initialize_Complete_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
INITIALIZE_Response->MessageType = REMOTE_NDIS_INITIALIZE_CMPLT;
|
|
|
|
INITIALIZE_Response->MessageLength = sizeof(RNDIS_Initialize_Complete_t);
|
|
|
|
INITIALIZE_Response->RequestId = INITIALIZE_Message->RequestId;
|
|
|
|
INITIALIZE_Response->Status = REMOTE_NDIS_STATUS_SUCCESS;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
INITIALIZE_Response->MajorVersion = REMOTE_NDIS_VERSION_MAJOR;
|
2010-10-13 16:05:35 +02:00
|
|
|
INITIALIZE_Response->MinorVersion = REMOTE_NDIS_VERSION_MINOR;
|
2010-05-08 05:12:14 +02:00
|
|
|
INITIALIZE_Response->DeviceFlags = REMOTE_NDIS_DF_CONNECTIONLESS;
|
|
|
|
INITIALIZE_Response->Medium = REMOTE_NDIS_MEDIUM_802_3;
|
|
|
|
INITIALIZE_Response->MaxPacketsPerTransfer = 1;
|
|
|
|
INITIALIZE_Response->MaxTransferSize = (sizeof(RNDIS_Packet_Message_t) + ETHERNET_FRAME_SIZE_MAX);
|
|
|
|
INITIALIZE_Response->PacketAlignmentFactor = 0;
|
|
|
|
INITIALIZE_Response->AFListOffset = 0;
|
|
|
|
INITIALIZE_Response->AFListSize = 0;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDISInterfaceInfo->State.CurrRNDISState = RNDIS_Initialized;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
break;
|
|
|
|
case REMOTE_NDIS_HALT_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = false;
|
|
|
|
MessageHeader->MessageLength = 0;
|
|
|
|
|
|
|
|
RNDISInterfaceInfo->State.CurrRNDISState = RNDIS_Uninitialized;
|
|
|
|
|
|
|
|
break;
|
|
|
|
case REMOTE_NDIS_QUERY_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = true;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDIS_Query_Message_t* QUERY_Message = (RNDIS_Query_Message_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
RNDIS_Query_Complete_t* QUERY_Response = (RNDIS_Query_Complete_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
uint32_t Query_Oid = QUERY_Message->Oid;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2011-04-13 08:12:52 +02:00
|
|
|
void* QueryData = &RNDISInterfaceInfo->State.RNDISMessageBuffer[sizeof(RNDIS_Message_Header_t) +
|
|
|
|
QUERY_Message->InformationBufferOffset];
|
2010-10-13 16:05:35 +02:00
|
|
|
void* ResponseData = &RNDISInterfaceInfo->State.RNDISMessageBuffer[sizeof(RNDIS_Query_Complete_t)];
|
2010-05-08 05:12:14 +02:00
|
|
|
uint16_t ResponseSize;
|
|
|
|
|
|
|
|
QUERY_Response->MessageType = REMOTE_NDIS_QUERY_CMPLT;
|
|
|
|
QUERY_Response->MessageLength = sizeof(RNDIS_Query_Complete_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
if (RNDIS_Device_ProcessNDISQuery(RNDISInterfaceInfo, Query_Oid, QueryData, QUERY_Message->InformationBufferLength,
|
|
|
|
ResponseData, &ResponseSize))
|
|
|
|
{
|
|
|
|
QUERY_Response->Status = REMOTE_NDIS_STATUS_SUCCESS;
|
|
|
|
QUERY_Response->MessageLength += ResponseSize;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
QUERY_Response->InformationBufferLength = ResponseSize;
|
|
|
|
QUERY_Response->InformationBufferOffset = (sizeof(RNDIS_Query_Complete_t) - sizeof(RNDIS_Message_Header_t));
|
|
|
|
}
|
|
|
|
else
|
2010-10-13 16:05:35 +02:00
|
|
|
{
|
2010-05-08 05:12:14 +02:00
|
|
|
QUERY_Response->Status = REMOTE_NDIS_STATUS_NOT_SUPPORTED;
|
|
|
|
|
|
|
|
QUERY_Response->InformationBufferLength = 0;
|
|
|
|
QUERY_Response->InformationBufferOffset = 0;
|
|
|
|
}
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
break;
|
|
|
|
case REMOTE_NDIS_SET_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = true;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDIS_Set_Message_t* SET_Message = (RNDIS_Set_Message_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
RNDIS_Set_Complete_t* SET_Response = (RNDIS_Set_Complete_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
uint32_t SET_Oid = SET_Message->Oid;
|
|
|
|
|
|
|
|
SET_Response->MessageType = REMOTE_NDIS_SET_CMPLT;
|
|
|
|
SET_Response->MessageLength = sizeof(RNDIS_Set_Complete_t);
|
|
|
|
SET_Response->RequestId = SET_Message->RequestId;
|
|
|
|
|
|
|
|
void* SetData = &RNDISInterfaceInfo->State.RNDISMessageBuffer[sizeof(RNDIS_Message_Header_t) +
|
|
|
|
SET_Message->InformationBufferOffset];
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
SET_Response->Status = RNDIS_Device_ProcessNDISSet(RNDISInterfaceInfo, SET_Oid, SetData,
|
|
|
|
SET_Message->InformationBufferLength) ?
|
|
|
|
REMOTE_NDIS_STATUS_SUCCESS : REMOTE_NDIS_STATUS_NOT_SUPPORTED;
|
|
|
|
break;
|
|
|
|
case REMOTE_NDIS_RESET_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = true;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDIS_Reset_Complete_t* RESET_Response = (RNDIS_Reset_Complete_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
|
|
|
|
RESET_Response->MessageType = REMOTE_NDIS_RESET_CMPLT;
|
|
|
|
RESET_Response->MessageLength = sizeof(RNDIS_Reset_Complete_t);
|
|
|
|
RESET_Response->Status = REMOTE_NDIS_STATUS_SUCCESS;
|
|
|
|
RESET_Response->AddressingReset = 0;
|
|
|
|
|
|
|
|
break;
|
|
|
|
case REMOTE_NDIS_KEEPALIVE_MSG:
|
|
|
|
RNDISInterfaceInfo->State.ResponseReady = true;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
RNDIS_KeepAlive_Message_t* KEEPALIVE_Message =
|
|
|
|
(RNDIS_KeepAlive_Message_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
RNDIS_KeepAlive_Complete_t* KEEPALIVE_Response =
|
|
|
|
(RNDIS_KeepAlive_Complete_t*)&RNDISInterfaceInfo->State.RNDISMessageBuffer;
|
|
|
|
|
|
|
|
KEEPALIVE_Response->MessageType = REMOTE_NDIS_KEEPALIVE_CMPLT;
|
|
|
|
KEEPALIVE_Response->MessageLength = sizeof(RNDIS_KeepAlive_Complete_t);
|
|
|
|
KEEPALIVE_Response->RequestId = KEEPALIVE_Message->RequestId;
|
|
|
|
KEEPALIVE_Response->Status = REMOTE_NDIS_STATUS_SUCCESS;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool RNDIS_Device_ProcessNDISQuery(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo,
|
2010-07-21 16:00:51 +02:00
|
|
|
const uint32_t OId,
|
|
|
|
void* const QueryData,
|
|
|
|
const uint16_t QuerySize,
|
|
|
|
void* ResponseData,
|
|
|
|
uint16_t* const ResponseSize)
|
2010-05-08 05:12:14 +02:00
|
|
|
{
|
|
|
|
(void)QueryData;
|
|
|
|
(void)QuerySize;
|
|
|
|
|
|
|
|
switch (OId)
|
|
|
|
{
|
|
|
|
case OID_GEN_SUPPORTED_LIST:
|
|
|
|
*ResponseSize = sizeof(AdapterSupportedOIDList);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
memcpy_P(ResponseData, AdapterSupportedOIDList, sizeof(AdapterSupportedOIDList));
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_PHYSICAL_MEDIUM:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Indicate that the device is a true ethernet link */
|
|
|
|
*((uint32_t*)ResponseData) = 0;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_HARDWARE_STATUS:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
*((uint32_t*)ResponseData) = NDIS_HardwareStatus_Ready;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_MEDIA_SUPPORTED:
|
|
|
|
case OID_GEN_MEDIA_IN_USE:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
*((uint32_t*)ResponseData) = REMOTE_NDIS_MEDIUM_802_3;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_VENDOR_ID:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Vendor ID 0x0xFFFFFF is reserved for vendors who have not purchased a NDIS VID */
|
|
|
|
*((uint32_t*)ResponseData) = 0x00FFFFFF;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_MAXIMUM_FRAME_SIZE:
|
|
|
|
case OID_GEN_TRANSMIT_BLOCK_SIZE:
|
|
|
|
case OID_GEN_RECEIVE_BLOCK_SIZE:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
*((uint32_t*)ResponseData) = ETHERNET_FRAME_SIZE_MAX;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_VENDOR_DESCRIPTION:
|
|
|
|
*ResponseSize = (strlen(RNDISInterfaceInfo->Config.AdapterVendorDescription) + 1);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
memcpy(ResponseData, RNDISInterfaceInfo->Config.AdapterVendorDescription, *ResponseSize);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_MEDIA_CONNECT_STATUS:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
*((uint32_t*)ResponseData) = REMOTE_NDIS_MEDIA_STATE_CONNECTED;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_LINK_SPEED:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Indicate 10Mb/s link speed */
|
|
|
|
*((uint32_t*)ResponseData) = 100000;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
case OID_802_3_PERMANENT_ADDRESS:
|
|
|
|
case OID_802_3_CURRENT_ADDRESS:
|
|
|
|
*ResponseSize = sizeof(MAC_Address_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
memcpy(ResponseData, &RNDISInterfaceInfo->Config.AdapterMACAddress, sizeof(MAC_Address_t));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
case OID_802_3_MAXIMUM_LIST_SIZE:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Indicate only one multicast address supported */
|
|
|
|
*((uint32_t*)ResponseData) = 1;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_CURRENT_PACKET_FILTER:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
*((uint32_t*)ResponseData) = RNDISInterfaceInfo->State.CurrPacketFilter;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
|
|
|
return true;
|
2010-05-08 05:12:14 +02:00
|
|
|
case OID_GEN_XMIT_OK:
|
|
|
|
case OID_GEN_RCV_OK:
|
|
|
|
case OID_GEN_XMIT_ERROR:
|
|
|
|
case OID_GEN_RCV_ERROR:
|
|
|
|
case OID_GEN_RCV_NO_BUFFER:
|
|
|
|
case OID_802_3_RCV_ERROR_ALIGNMENT:
|
|
|
|
case OID_802_3_XMIT_ONE_COLLISION:
|
|
|
|
case OID_802_3_XMIT_MORE_COLLISIONS:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Unused statistic OIDs - always return 0 for each */
|
|
|
|
*((uint32_t*)ResponseData) = 0;
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_GEN_MAXIMUM_TOTAL_SIZE:
|
|
|
|
*ResponseSize = sizeof(uint32_t);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
/* Indicate maximum overall buffer (Ethernet frame and RNDIS header) the adapter can handle */
|
|
|
|
*((uint32_t*)ResponseData) = (RNDIS_MESSAGE_BUFFER_SIZE + ETHERNET_FRAME_SIZE_MAX);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-21 16:00:51 +02:00
|
|
|
static bool RNDIS_Device_ProcessNDISSet(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo,
|
|
|
|
const uint32_t OId,
|
|
|
|
const void* SetData,
|
|
|
|
const uint16_t SetSize)
|
2010-05-08 05:12:14 +02:00
|
|
|
{
|
|
|
|
(void)SetSize;
|
|
|
|
|
|
|
|
switch (OId)
|
|
|
|
{
|
|
|
|
case OID_GEN_CURRENT_PACKET_FILTER:
|
|
|
|
RNDISInterfaceInfo->State.CurrPacketFilter = *((uint32_t*)SetData);
|
|
|
|
RNDISInterfaceInfo->State.CurrRNDISState = ((RNDISInterfaceInfo->State.CurrPacketFilter) ?
|
|
|
|
RNDIS_Data_Initialized : RNDIS_Data_Initialized);
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
case OID_802_3_MULTICAST_LIST:
|
|
|
|
/* Do nothing - throw away the value from the host as it is unused */
|
2010-10-13 16:05:35 +02:00
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-05-18 16:33:34 +02:00
|
|
|
bool RNDIS_Device_IsPacketReceived(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo)
|
|
|
|
{
|
|
|
|
if ((USB_DeviceState != DEVICE_STATE_Configured) ||
|
|
|
|
(RNDISInterfaceInfo->State.CurrRNDISState != RNDIS_Data_Initialized))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Endpoint_SelectEndpoint(RNDISInterfaceInfo->Config.DataOUTEndpointNumber);
|
|
|
|
return Endpoint_IsOUTReceived();
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t RNDIS_Device_ReadPacket(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo,
|
|
|
|
void* Buffer,
|
|
|
|
uint16_t* const PacketLength)
|
|
|
|
{
|
|
|
|
if ((USB_DeviceState != DEVICE_STATE_Configured) ||
|
|
|
|
(RNDISInterfaceInfo->State.CurrRNDISState != RNDIS_Data_Initialized))
|
|
|
|
{
|
|
|
|
return ENDPOINT_RWSTREAM_DeviceDisconnected;
|
|
|
|
}
|
|
|
|
|
|
|
|
Endpoint_SelectEndpoint(RNDISInterfaceInfo->Config.DataOUTEndpointNumber);
|
|
|
|
|
|
|
|
*PacketLength = 0;
|
|
|
|
|
|
|
|
if (!(Endpoint_IsOUTReceived()))
|
|
|
|
return ENDPOINT_RWSTREAM_NoError;
|
|
|
|
|
|
|
|
RNDIS_Packet_Message_t RNDISPacketHeader;
|
|
|
|
Endpoint_Read_Stream_LE(&RNDISPacketHeader, sizeof(RNDIS_Packet_Message_t), NULL);
|
|
|
|
|
|
|
|
if (RNDISPacketHeader.DataLength > ETHERNET_FRAME_SIZE_MAX)
|
|
|
|
{
|
|
|
|
Endpoint_StallTransaction();
|
|
|
|
|
|
|
|
return RNDIS_ERROR_LOGICAL_CMD_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
*PacketLength = (uint16_t)RNDISPacketHeader.DataLength;
|
|
|
|
|
|
|
|
Endpoint_Read_Stream_LE(Buffer, RNDISPacketHeader.DataLength, NULL);
|
|
|
|
Endpoint_ClearOUT();
|
|
|
|
|
|
|
|
return ENDPOINT_RWSTREAM_NoError;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t RNDIS_Device_SendPacket(USB_ClassInfo_RNDIS_Device_t* const RNDISInterfaceInfo,
|
|
|
|
void* Buffer,
|
|
|
|
const uint16_t PacketLength)
|
|
|
|
{
|
|
|
|
uint8_t ErrorCode;
|
|
|
|
|
|
|
|
if ((USB_DeviceState != DEVICE_STATE_Configured) ||
|
|
|
|
(RNDISInterfaceInfo->State.CurrRNDISState != RNDIS_Data_Initialized))
|
|
|
|
{
|
|
|
|
return ENDPOINT_RWSTREAM_DeviceDisconnected;
|
|
|
|
}
|
|
|
|
|
|
|
|
Endpoint_SelectEndpoint(RNDISInterfaceInfo->Config.DataINEndpointNumber);
|
|
|
|
|
|
|
|
if ((ErrorCode = Endpoint_WaitUntilReady()) != ENDPOINT_READYWAIT_NoError)
|
|
|
|
return ErrorCode;
|
|
|
|
|
|
|
|
RNDIS_Packet_Message_t RNDISPacketHeader;
|
|
|
|
|
|
|
|
memset(&RNDISPacketHeader, 0, sizeof(RNDIS_Packet_Message_t));
|
|
|
|
|
|
|
|
RNDISPacketHeader.MessageType = REMOTE_NDIS_PACKET_MSG;
|
|
|
|
RNDISPacketHeader.MessageLength = (sizeof(RNDIS_Packet_Message_t) + PacketLength);
|
|
|
|
RNDISPacketHeader.DataOffset = (sizeof(RNDIS_Packet_Message_t) - sizeof(RNDIS_Message_Header_t));
|
|
|
|
RNDISPacketHeader.DataLength = PacketLength;
|
|
|
|
|
|
|
|
Endpoint_Write_Stream_LE(&RNDISPacketHeader, sizeof(RNDIS_Packet_Message_t), NULL);
|
|
|
|
Endpoint_Write_Stream_LE(Buffer, PacketLength, NULL);
|
|
|
|
Endpoint_ClearIN();
|
|
|
|
|
|
|
|
return ENDPOINT_RWSTREAM_NoError;
|
|
|
|
}
|
|
|
|
|
2010-05-08 05:12:14 +02:00
|
|
|
#endif
|
2010-10-13 16:05:35 +02:00
|
|
|
|