add TUD_EPBUF_DEF for ecm rndis, also separate notify and control buffer, use edpt_claim() to prevent race condition
This commit is contained in:
		@@ -35,13 +35,18 @@
 | 
			
		||||
#include "net_device.h"
 | 
			
		||||
#include "rndis_protocol.h"
 | 
			
		||||
 | 
			
		||||
void rndis_class_set_handler(uint8_t *data, int size); /* found in ./misc/networking/rndis_reports.c */
 | 
			
		||||
extern void rndis_class_set_handler(uint8_t *data, int size); /* found in ./misc/networking/rndis_reports.c */
 | 
			
		||||
 | 
			
		||||
#define CFG_TUD_NET_PACKET_PREFIX_LEN sizeof(rndis_data_packet_t)
 | 
			
		||||
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 0
 | 
			
		||||
 | 
			
		||||
#define NETD_PACKET_SIZE  (CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN)
 | 
			
		||||
#define NETD_CONTROL_SIZE 120
 | 
			
		||||
 | 
			
		||||
//--------------------------------------------------------------------+
 | 
			
		||||
// MACRO CONSTANT TYPEDEF
 | 
			
		||||
//--------------------------------------------------------------------+
 | 
			
		||||
typedef struct
 | 
			
		||||
{
 | 
			
		||||
typedef struct {
 | 
			
		||||
  uint8_t itf_num;      // Index number of Management Interface, +1 for Data Interface
 | 
			
		||||
  uint8_t itf_data_alt; // Alternate setting of Data Interface. 0 : inactive, 1 : active
 | 
			
		||||
 | 
			
		||||
@@ -55,78 +60,44 @@ typedef struct
 | 
			
		||||
  // TODO since configuration descriptor may not be long-lived memory, we should
 | 
			
		||||
  // keep a copy of endpoint attribute instead
 | 
			
		||||
  uint8_t const * ecm_desc_epdata;
 | 
			
		||||
 | 
			
		||||
} netd_interface_t;
 | 
			
		||||
 | 
			
		||||
#define CFG_TUD_NET_PACKET_PREFIX_LEN sizeof(rndis_data_packet_t)
 | 
			
		||||
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 0
 | 
			
		||||
 | 
			
		||||
CFG_TUD_MEM_SECTION CFG_TUSB_MEM_ALIGN tu_static
 | 
			
		||||
uint8_t received[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
 | 
			
		||||
 | 
			
		||||
CFG_TUD_MEM_SECTION CFG_TUSB_MEM_ALIGN tu_static
 | 
			
		||||
uint8_t transmitted[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
 | 
			
		||||
 | 
			
		||||
struct ecm_notify_struct
 | 
			
		||||
{
 | 
			
		||||
typedef struct ecm_notify_struct {
 | 
			
		||||
  tusb_control_request_t header;
 | 
			
		||||
  uint32_t downlink, uplink;
 | 
			
		||||
};
 | 
			
		||||
} ecm_notify_t;
 | 
			
		||||
 | 
			
		||||
tu_static const struct ecm_notify_struct ecm_notify_nc =
 | 
			
		||||
{
 | 
			
		||||
  .header = {
 | 
			
		||||
    .bmRequestType = 0xA1,
 | 
			
		||||
    .bRequest = 0 /* NETWORK_CONNECTION aka NetworkConnection */,
 | 
			
		||||
    .wValue = 1 /* Connected */,
 | 
			
		||||
    .wLength = 0,
 | 
			
		||||
  },
 | 
			
		||||
};
 | 
			
		||||
typedef struct {
 | 
			
		||||
  TUD_EPBUF_DEF(rx, NETD_PACKET_SIZE);
 | 
			
		||||
  TUD_EPBUF_DEF(tx, NETD_PACKET_SIZE);
 | 
			
		||||
 | 
			
		||||
tu_static const struct ecm_notify_struct ecm_notify_csc =
 | 
			
		||||
{
 | 
			
		||||
  .header = {
 | 
			
		||||
    .bmRequestType = 0xA1,
 | 
			
		||||
    .bRequest = 0x2A /* CONNECTION_SPEED_CHANGE aka ConnectionSpeedChange */,
 | 
			
		||||
    .wLength = 8,
 | 
			
		||||
  },
 | 
			
		||||
  .downlink = 9728000,
 | 
			
		||||
  .uplink = 9728000,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// TODO remove CFG_TUD_MEM_SECTION, control internal buffer is already in this special section
 | 
			
		||||
CFG_TUD_MEM_SECTION CFG_TUSB_MEM_ALIGN tu_static union
 | 
			
		||||
{
 | 
			
		||||
  uint8_t rndis_buf[120];
 | 
			
		||||
  struct ecm_notify_struct ecm_buf;
 | 
			
		||||
} notify;
 | 
			
		||||
  TUD_EPBUF_DEF(notify, sizeof(ecm_notify_t));
 | 
			
		||||
  TUD_EPBUF_DEF(ctrl, NETD_CONTROL_SIZE);
 | 
			
		||||
} netd_epbuf_t;
 | 
			
		||||
 | 
			
		||||
//--------------------------------------------------------------------+
 | 
			
		||||
// INTERNAL OBJECT & FUNCTION DECLARATION
 | 
			
		||||
//--------------------------------------------------------------------+
 | 
			
		||||
// TODO remove CFG_TUD_MEM_SECTION
 | 
			
		||||
CFG_TUD_MEM_SECTION tu_static netd_interface_t _netd_itf;
 | 
			
		||||
static netd_interface_t _netd_itf;
 | 
			
		||||
CFG_TUD_MEM_SECTION static netd_epbuf_t _netd_epbuf;
 | 
			
		||||
static bool can_xmit;
 | 
			
		||||
 | 
			
		||||
tu_static bool can_xmit;
 | 
			
		||||
 | 
			
		||||
void tud_network_recv_renew(void)
 | 
			
		||||
{
 | 
			
		||||
  usbd_edpt_xfer(0, _netd_itf.ep_out, received, sizeof(received));
 | 
			
		||||
void tud_network_recv_renew(void) {
 | 
			
		||||
  usbd_edpt_xfer(0, _netd_itf.ep_out, _netd_epbuf.rx, NETD_PACKET_SIZE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void do_in_xfer(uint8_t *buf, uint16_t len)
 | 
			
		||||
{
 | 
			
		||||
static void do_in_xfer(uint8_t *buf, uint16_t len) {
 | 
			
		||||
  can_xmit = false;
 | 
			
		||||
  usbd_edpt_xfer(0, _netd_itf.ep_in, buf, len);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void netd_report(uint8_t *buf, uint16_t len)
 | 
			
		||||
{
 | 
			
		||||
  uint8_t const rhport = 0;
 | 
			
		||||
void netd_report(uint8_t *buf, uint16_t len) {
 | 
			
		||||
  const uint8_t rhport = 0;
 | 
			
		||||
  len = tu_min16(len, sizeof(ecm_notify_t));
 | 
			
		||||
 | 
			
		||||
  // skip if previous report not yet acknowledged by host
 | 
			
		||||
  if ( usbd_edpt_busy(rhport, _netd_itf.ep_notif) ) return;
 | 
			
		||||
  usbd_edpt_xfer(rhport, _netd_itf.ep_notif, buf, len);
 | 
			
		||||
  TU_VERIFY(usbd_edpt_claim(rhport, _netd_itf.ep_notif), );
 | 
			
		||||
  memcpy(_netd_epbuf.notify, buf, len);
 | 
			
		||||
  usbd_edpt_xfer(rhport, _netd_itf.ep_notif, _netd_epbuf.notify, len);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//--------------------------------------------------------------------+
 | 
			
		||||
@@ -140,15 +111,12 @@ bool netd_deinit(void) {
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void netd_reset(uint8_t rhport)
 | 
			
		||||
{
 | 
			
		||||
void netd_reset(uint8_t rhport) {
 | 
			
		||||
  (void) rhport;
 | 
			
		||||
 | 
			
		||||
  netd_init();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
uint16_t netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t max_len)
 | 
			
		||||
{
 | 
			
		||||
uint16_t netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t max_len) {
 | 
			
		||||
  bool const is_rndis = (TUD_RNDIS_ITF_CLASS    == itf_desc->bInterfaceClass    &&
 | 
			
		||||
                         TUD_RNDIS_ITF_SUBCLASS == itf_desc->bInterfaceSubClass &&
 | 
			
		||||
                         TUD_RNDIS_ITF_PROTOCOL == itf_desc->bInterfaceProtocol);
 | 
			
		||||
@@ -172,21 +140,19 @@ uint16_t netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint1
 | 
			
		||||
  uint8_t const * p_desc = tu_desc_next( itf_desc );
 | 
			
		||||
 | 
			
		||||
  // Communication Functional Descriptors
 | 
			
		||||
  while ( TUSB_DESC_CS_INTERFACE == tu_desc_type(p_desc) && drv_len <= max_len )
 | 
			
		||||
  {
 | 
			
		||||
  while (TUSB_DESC_CS_INTERFACE == tu_desc_type(p_desc) && drv_len <= max_len) {
 | 
			
		||||
    drv_len += tu_desc_len(p_desc);
 | 
			
		||||
    p_desc   = tu_desc_next(p_desc);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // notification endpoint (if any)
 | 
			
		||||
  if ( TUSB_DESC_ENDPOINT == tu_desc_type(p_desc) )
 | 
			
		||||
  {
 | 
			
		||||
    TU_ASSERT( usbd_edpt_open(rhport, (tusb_desc_endpoint_t const *) p_desc), 0 );
 | 
			
		||||
  if (TUSB_DESC_ENDPOINT == tu_desc_type(p_desc)) {
 | 
			
		||||
    TU_ASSERT(usbd_edpt_open(rhport, (tusb_desc_endpoint_t const *) p_desc), 0);
 | 
			
		||||
 | 
			
		||||
    _netd_itf.ep_notif = ((tusb_desc_endpoint_t const *) p_desc)->bEndpointAddress;
 | 
			
		||||
    _netd_itf.ep_notif = ((tusb_desc_endpoint_t const*)p_desc)->bEndpointAddress;
 | 
			
		||||
 | 
			
		||||
    drv_len += tu_desc_len(p_desc);
 | 
			
		||||
    p_desc   = tu_desc_next(p_desc);
 | 
			
		||||
    p_desc = tu_desc_next(p_desc);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  //------------- Data Interface -------------//
 | 
			
		||||
@@ -196,27 +162,24 @@ uint16_t netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint1
 | 
			
		||||
  //   - 1 : IN & OUT endpoints for active networking
 | 
			
		||||
  TU_ASSERT(TUSB_DESC_INTERFACE == tu_desc_type(p_desc), 0);
 | 
			
		||||
 | 
			
		||||
  do
 | 
			
		||||
  {
 | 
			
		||||
  do {
 | 
			
		||||
    tusb_desc_interface_t const * data_itf_desc = (tusb_desc_interface_t const *) p_desc;
 | 
			
		||||
    TU_ASSERT(TUSB_CLASS_CDC_DATA == data_itf_desc->bInterfaceClass, 0);
 | 
			
		||||
 | 
			
		||||
    drv_len += tu_desc_len(p_desc);
 | 
			
		||||
    p_desc   = tu_desc_next(p_desc);
 | 
			
		||||
  }while( _netd_itf.ecm_mode && (TUSB_DESC_INTERFACE == tu_desc_type(p_desc)) && (drv_len <= max_len) );
 | 
			
		||||
  } while (_netd_itf.ecm_mode && (TUSB_DESC_INTERFACE == tu_desc_type(p_desc)) && (drv_len <= max_len));
 | 
			
		||||
 | 
			
		||||
  // Pair of endpoints
 | 
			
		||||
  TU_ASSERT(TUSB_DESC_ENDPOINT == tu_desc_type(p_desc), 0);
 | 
			
		||||
 | 
			
		||||
  if ( _netd_itf.ecm_mode )
 | 
			
		||||
  {
 | 
			
		||||
  if (_netd_itf.ecm_mode) {
 | 
			
		||||
    // ECM by default is in-active, save the endpoint attribute
 | 
			
		||||
    // to open later when received setInterface
 | 
			
		||||
    _netd_itf.ecm_desc_epdata = p_desc;
 | 
			
		||||
  }else
 | 
			
		||||
  {
 | 
			
		||||
  } else {
 | 
			
		||||
    // Open endpoint pair for RNDIS
 | 
			
		||||
    TU_ASSERT( usbd_open_edpt_pair(rhport, p_desc, 2, TUSB_XFER_BULK, &_netd_itf.ep_out, &_netd_itf.ep_in), 0 );
 | 
			
		||||
    TU_ASSERT(usbd_open_edpt_pair(rhport, p_desc, 2, TUSB_XFER_BULK, &_netd_itf.ep_out, &_netd_itf.ep_in), 0);
 | 
			
		||||
 | 
			
		||||
    tud_network_init_cb();
 | 
			
		||||
 | 
			
		||||
@@ -232,38 +195,50 @@ uint16_t netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint1
 | 
			
		||||
  return drv_len;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void ecm_report(bool nc)
 | 
			
		||||
{
 | 
			
		||||
  notify.ecm_buf = (nc) ? ecm_notify_nc : ecm_notify_csc;
 | 
			
		||||
  notify.ecm_buf.header.wIndex = _netd_itf.itf_num;
 | 
			
		||||
  netd_report((uint8_t *)¬ify.ecm_buf, (nc) ? sizeof(notify.ecm_buf.header) : sizeof(notify.ecm_buf));
 | 
			
		||||
static void ecm_report(bool nc) {
 | 
			
		||||
  const ecm_notify_t ecm_notify_nc = {
 | 
			
		||||
    .header = {
 | 
			
		||||
      .bmRequestType = 0xA1,
 | 
			
		||||
      .bRequest = 0, /* NETWORK_CONNECTION aka NetworkConnection */
 | 
			
		||||
      .wValue = 1,   /* Connected */
 | 
			
		||||
      .wLength = 0,
 | 
			
		||||
    },
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  const ecm_notify_t ecm_notify_csc = {
 | 
			
		||||
    .header = {
 | 
			
		||||
      .bmRequestType = 0xA1,
 | 
			
		||||
      .bRequest = 0x2A, /* CONNECTION_SPEED_CHANGE aka ConnectionSpeedChange */
 | 
			
		||||
      .wLength = 8,
 | 
			
		||||
    },
 | 
			
		||||
    .downlink = 9728000,
 | 
			
		||||
    .uplink = 9728000,
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  ecm_notify_t notify = (nc) ? ecm_notify_nc : ecm_notify_csc;
 | 
			
		||||
  notify.header.wIndex = _netd_itf.itf_num;
 | 
			
		||||
  netd_report((uint8_t *)¬ify, (nc) ? sizeof(notify.header) : sizeof(notify));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Invoked when a control transfer occurred on an interface of this class
 | 
			
		||||
// Driver response accordingly to the request and the transfer stage (setup/data/ack)
 | 
			
		||||
// return false to stall control endpoint (e.g unsupported request)
 | 
			
		||||
bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t const * request)
 | 
			
		||||
{
 | 
			
		||||
  if ( stage == CONTROL_STAGE_SETUP )
 | 
			
		||||
  {
 | 
			
		||||
    switch ( request->bmRequestType_bit.type )
 | 
			
		||||
    {
 | 
			
		||||
bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t const * request) {
 | 
			
		||||
  if (stage == CONTROL_STAGE_SETUP) {
 | 
			
		||||
    switch (request->bmRequestType_bit.type) {
 | 
			
		||||
      case TUSB_REQ_TYPE_STANDARD:
 | 
			
		||||
        switch ( request->bRequest )
 | 
			
		||||
        {
 | 
			
		||||
          case TUSB_REQ_GET_INTERFACE:
 | 
			
		||||
          {
 | 
			
		||||
            uint8_t const req_itfnum = (uint8_t) request->wIndex;
 | 
			
		||||
        switch (request->bRequest) {
 | 
			
		||||
          case TUSB_REQ_GET_INTERFACE: {
 | 
			
		||||
            uint8_t const req_itfnum = (uint8_t)request->wIndex;
 | 
			
		||||
            TU_VERIFY(_netd_itf.itf_num+1 == req_itfnum);
 | 
			
		||||
 | 
			
		||||
            tud_control_xfer(rhport, request, &_netd_itf.itf_data_alt, 1);
 | 
			
		||||
          }
 | 
			
		||||
          break;
 | 
			
		||||
 | 
			
		||||
          case TUSB_REQ_SET_INTERFACE:
 | 
			
		||||
          {
 | 
			
		||||
            uint8_t const req_itfnum = (uint8_t) request->wIndex;
 | 
			
		||||
            uint8_t const req_alt    = (uint8_t) request->wValue;
 | 
			
		||||
          case TUSB_REQ_SET_INTERFACE: {
 | 
			
		||||
            uint8_t const req_itfnum = (uint8_t)request->wIndex;
 | 
			
		||||
            uint8_t const req_alt = (uint8_t)request->wValue;
 | 
			
		||||
 | 
			
		||||
            // Only valid for Data Interface with Alternate is either 0 or 1
 | 
			
		||||
            TU_VERIFY(_netd_itf.itf_num+1 == req_itfnum && req_alt < 2);
 | 
			
		||||
@@ -273,14 +248,14 @@ bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t
 | 
			
		||||
 | 
			
		||||
            _netd_itf.itf_data_alt = req_alt;
 | 
			
		||||
 | 
			
		||||
            if ( _netd_itf.itf_data_alt )
 | 
			
		||||
            {
 | 
			
		||||
            if (_netd_itf.itf_data_alt) {
 | 
			
		||||
              // TODO since we don't actually close endpoint
 | 
			
		||||
              // hack here to not re-open it
 | 
			
		||||
              if ( _netd_itf.ep_in == 0 && _netd_itf.ep_out == 0 )
 | 
			
		||||
              {
 | 
			
		||||
              if (_netd_itf.ep_in == 0 && _netd_itf.ep_out == 0) {
 | 
			
		||||
                TU_ASSERT(_netd_itf.ecm_desc_epdata);
 | 
			
		||||
                TU_ASSERT( usbd_open_edpt_pair(rhport, _netd_itf.ecm_desc_epdata, 2, TUSB_XFER_BULK, &_netd_itf.ep_out, &_netd_itf.ep_in) );
 | 
			
		||||
                TU_ASSERT(
 | 
			
		||||
                  usbd_open_edpt_pair(rhport, _netd_itf.ecm_desc_epdata, 2, TUSB_XFER_BULK, &_netd_itf.ep_out, &
 | 
			
		||||
                    _netd_itf.ep_in));
 | 
			
		||||
 | 
			
		||||
                // TODO should be merge with RNDIS's after endpoint opened
 | 
			
		||||
                // Also should have opposite callback for application to disable network !!
 | 
			
		||||
@@ -288,8 +263,7 @@ bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t
 | 
			
		||||
                can_xmit = true; // we are ready to transmit a packet
 | 
			
		||||
                tud_network_recv_renew(); // prepare for incoming packets
 | 
			
		||||
              }
 | 
			
		||||
            }else
 | 
			
		||||
            {
 | 
			
		||||
            } else {
 | 
			
		||||
              // TODO close the endpoint pair
 | 
			
		||||
              // For now pretend that we did, this should have no harm since host won't try to
 | 
			
		||||
              // communicate with the endpoints again
 | 
			
		||||
@@ -303,50 +277,39 @@ bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t
 | 
			
		||||
          // unsupported request
 | 
			
		||||
          default: return false;
 | 
			
		||||
        }
 | 
			
		||||
      break;
 | 
			
		||||
        break;
 | 
			
		||||
 | 
			
		||||
      case TUSB_REQ_TYPE_CLASS:
 | 
			
		||||
        TU_VERIFY (_netd_itf.itf_num == request->wIndex);
 | 
			
		||||
        TU_VERIFY(_netd_itf.itf_num == request->wIndex);
 | 
			
		||||
 | 
			
		||||
        if (_netd_itf.ecm_mode)
 | 
			
		||||
        {
 | 
			
		||||
        if (_netd_itf.ecm_mode) {
 | 
			
		||||
          /* the only required CDC-ECM Management Element Request is SetEthernetPacketFilter */
 | 
			
		||||
          if (0x43 /* SET_ETHERNET_PACKET_FILTER */ == request->bRequest)
 | 
			
		||||
          {
 | 
			
		||||
          if (0x43 /* SET_ETHERNET_PACKET_FILTER */ == request->bRequest) {
 | 
			
		||||
            tud_control_xfer(rhport, request, NULL, 0);
 | 
			
		||||
            ecm_report(true);
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
        else
 | 
			
		||||
        {
 | 
			
		||||
          if (request->bmRequestType_bit.direction == TUSB_DIR_IN)
 | 
			
		||||
          {
 | 
			
		||||
            rndis_generic_msg_t *rndis_msg = (rndis_generic_msg_t *) ((void*) notify.rndis_buf);
 | 
			
		||||
        } else {
 | 
			
		||||
          if (request->bmRequestType_bit.direction == TUSB_DIR_IN) {
 | 
			
		||||
            rndis_generic_msg_t* rndis_msg = (rndis_generic_msg_t*)((void*)_netd_epbuf.ctrl);
 | 
			
		||||
            uint32_t msglen = tu_le32toh(rndis_msg->MessageLength);
 | 
			
		||||
            TU_ASSERT(msglen <= sizeof(notify.rndis_buf));
 | 
			
		||||
            tud_control_xfer(rhport, request, notify.rndis_buf, (uint16_t) msglen);
 | 
			
		||||
          }
 | 
			
		||||
          else
 | 
			
		||||
          {
 | 
			
		||||
            tud_control_xfer(rhport, request, notify.rndis_buf, (uint16_t) sizeof(notify.rndis_buf));
 | 
			
		||||
            TU_ASSERT(msglen <= NETD_CONTROL_SIZE);
 | 
			
		||||
            tud_control_xfer(rhport, request, _netd_epbuf.ctrl, (uint16_t)msglen);
 | 
			
		||||
          } else {
 | 
			
		||||
            tud_control_xfer(rhport, request, _netd_epbuf.ctrl, NETD_CONTROL_SIZE);
 | 
			
		||||
          }
 | 
			
		||||
        }
 | 
			
		||||
      break;
 | 
			
		||||
        break;
 | 
			
		||||
 | 
			
		||||
      // unsupported request
 | 
			
		||||
      default: return false;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  else if ( stage == CONTROL_STAGE_DATA )
 | 
			
		||||
  {
 | 
			
		||||
  } else if (stage == CONTROL_STAGE_DATA) {
 | 
			
		||||
    // Handle RNDIS class control OUT only
 | 
			
		||||
    if (request->bmRequestType_bit.type == TUSB_REQ_TYPE_CLASS &&
 | 
			
		||||
        request->bmRequestType_bit.direction == TUSB_DIR_OUT   &&
 | 
			
		||||
        _netd_itf.itf_num == request->wIndex)
 | 
			
		||||
    {
 | 
			
		||||
      if ( !_netd_itf.ecm_mode )
 | 
			
		||||
      {
 | 
			
		||||
        rndis_class_set_handler(notify.rndis_buf, request->wLength);
 | 
			
		||||
        request->bmRequestType_bit.direction == TUSB_DIR_OUT &&
 | 
			
		||||
        _netd_itf.itf_num == request->wIndex) {
 | 
			
		||||
      if (!_netd_itf.ecm_mode) {
 | 
			
		||||
        rndis_class_set_handler(_netd_epbuf.ctrl, request->wLength);
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
@@ -354,92 +317,77 @@ bool netd_control_xfer_cb (uint8_t rhport, uint8_t stage, tusb_control_request_t
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void handle_incoming_packet(uint32_t len)
 | 
			
		||||
{
 | 
			
		||||
  uint8_t *pnt = received;
 | 
			
		||||
static void handle_incoming_packet(uint32_t len) {
 | 
			
		||||
  uint8_t* pnt = _netd_epbuf.rx;
 | 
			
		||||
  uint32_t size = 0;
 | 
			
		||||
 | 
			
		||||
  if (_netd_itf.ecm_mode)
 | 
			
		||||
  {
 | 
			
		||||
  if (_netd_itf.ecm_mode) {
 | 
			
		||||
    size = len;
 | 
			
		||||
  }
 | 
			
		||||
  else
 | 
			
		||||
  {
 | 
			
		||||
    rndis_data_packet_t *r = (rndis_data_packet_t *) ((void*) pnt);
 | 
			
		||||
    if (len >= sizeof(rndis_data_packet_t))
 | 
			
		||||
      if ( (r->MessageType == REMOTE_NDIS_PACKET_MSG) && (r->MessageLength <= len))
 | 
			
		||||
        if ( (r->DataOffset + offsetof(rndis_data_packet_t, DataOffset) + r->DataLength) <= len)
 | 
			
		||||
        {
 | 
			
		||||
          pnt = &received[r->DataOffset + offsetof(rndis_data_packet_t, DataOffset)];
 | 
			
		||||
  } else {
 | 
			
		||||
    rndis_data_packet_t* r = (rndis_data_packet_t*)((void*)pnt);
 | 
			
		||||
    if (len >= sizeof(rndis_data_packet_t)) {
 | 
			
		||||
      if ((r->MessageType == REMOTE_NDIS_PACKET_MSG) && (r->MessageLength <= len)) {
 | 
			
		||||
        if ((r->DataOffset + offsetof(rndis_data_packet_t, DataOffset) + r->DataLength) <= len) {
 | 
			
		||||
          pnt = &_netd_epbuf.rx[r->DataOffset + offsetof(rndis_data_packet_t, DataOffset)];
 | 
			
		||||
          size = r->DataLength;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if (!tud_network_recv_cb(pnt, (uint16_t) size))
 | 
			
		||||
  {
 | 
			
		||||
  if (!tud_network_recv_cb(pnt, (uint16_t)size)) {
 | 
			
		||||
    /* if a buffer was never handled by user code, we must renew on the user's behalf */
 | 
			
		||||
    tud_network_recv_renew();
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool netd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
 | 
			
		||||
{
 | 
			
		||||
  (void) rhport;
 | 
			
		||||
  (void) result;
 | 
			
		||||
bool netd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes) {
 | 
			
		||||
  (void)rhport;
 | 
			
		||||
  (void)result;
 | 
			
		||||
 | 
			
		||||
  /* new packet received */
 | 
			
		||||
  if ( ep_addr == _netd_itf.ep_out )
 | 
			
		||||
  {
 | 
			
		||||
  if (ep_addr == _netd_itf.ep_out) {
 | 
			
		||||
    handle_incoming_packet(xferred_bytes);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /* data transmission finished */
 | 
			
		||||
  if ( ep_addr == _netd_itf.ep_in )
 | 
			
		||||
  {
 | 
			
		||||
  if (ep_addr == _netd_itf.ep_in) {
 | 
			
		||||
    /* TinyUSB requires the class driver to implement ZLP (since ZLP usage is class-specific) */
 | 
			
		||||
 | 
			
		||||
    if ( xferred_bytes && (0 == (xferred_bytes % CFG_TUD_NET_ENDPOINT_SIZE)) )
 | 
			
		||||
    {
 | 
			
		||||
    if (xferred_bytes && (0 == (xferred_bytes % CFG_TUD_NET_ENDPOINT_SIZE))) {
 | 
			
		||||
      do_in_xfer(NULL, 0); /* a ZLP is needed */
 | 
			
		||||
    }
 | 
			
		||||
    else
 | 
			
		||||
    {
 | 
			
		||||
    } else {
 | 
			
		||||
      /* we're finally finished */
 | 
			
		||||
      can_xmit = true;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  if ( _netd_itf.ecm_mode && (ep_addr == _netd_itf.ep_notif) )
 | 
			
		||||
  {
 | 
			
		||||
    if (sizeof(notify.ecm_buf.header) == xferred_bytes) ecm_report(false);
 | 
			
		||||
  if (_netd_itf.ecm_mode && (ep_addr == _netd_itf.ep_notif)) {
 | 
			
		||||
    if (sizeof(tusb_control_request_t) == xferred_bytes) {
 | 
			
		||||
      ecm_report(false);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool tud_network_can_xmit(uint16_t size)
 | 
			
		||||
{
 | 
			
		||||
bool tud_network_can_xmit(uint16_t size) {
 | 
			
		||||
  (void)size;
 | 
			
		||||
 | 
			
		||||
  return can_xmit;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void tud_network_xmit(void *ref, uint16_t arg)
 | 
			
		||||
{
 | 
			
		||||
  uint8_t *data;
 | 
			
		||||
  uint16_t len;
 | 
			
		||||
 | 
			
		||||
  if (!can_xmit)
 | 
			
		||||
void tud_network_xmit(void *ref, uint16_t arg) {
 | 
			
		||||
  if (!can_xmit) {
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  len = (_netd_itf.ecm_mode) ? 0 : CFG_TUD_NET_PACKET_PREFIX_LEN;
 | 
			
		||||
  data = transmitted + len;
 | 
			
		||||
  uint16_t len = (_netd_itf.ecm_mode) ? 0 : CFG_TUD_NET_PACKET_PREFIX_LEN;
 | 
			
		||||
  uint8_t* data = _netd_epbuf.tx + len;
 | 
			
		||||
 | 
			
		||||
  len += tud_network_xmit_cb(data, ref, arg);
 | 
			
		||||
 | 
			
		||||
  if (!_netd_itf.ecm_mode)
 | 
			
		||||
  {
 | 
			
		||||
    rndis_data_packet_t *hdr = (rndis_data_packet_t *) ((void*) transmitted);
 | 
			
		||||
  if (!_netd_itf.ecm_mode) {
 | 
			
		||||
    rndis_data_packet_t *hdr = (rndis_data_packet_t *) ((void*) _netd_epbuf.tx);
 | 
			
		||||
    memset(hdr, 0, sizeof(rndis_data_packet_t));
 | 
			
		||||
    hdr->MessageType = REMOTE_NDIS_PACKET_MSG;
 | 
			
		||||
    hdr->MessageLength = len;
 | 
			
		||||
@@ -447,7 +395,7 @@ void tud_network_xmit(void *ref, uint16_t arg)
 | 
			
		||||
    hdr->DataLength = len - sizeof(rndis_data_packet_t);
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  do_in_xfer(transmitted, len);
 | 
			
		||||
  do_in_xfer(_netd_epbuf.tx, len);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user