~ruther/stm32h747i-disco-usb-image-viewer

241e20059313285b3c69d72d5360f27dbcee7573 — Rutherther 3 months ago d73097e
feat: allow reading larger packets without data loss
M firmware/include/usb_device.h => firmware/include/usb_device.h +1 -1
@@ 58,7 58,7 @@ typedef enum {
typedef struct {
  packet_status_t packet_status;
  packet_dpid_t dpid;
  uint8_t byte_count;
  uint16_t byte_count;
  uint8_t endpoint_num;
  uint8_t status_phase_start;
} packet_info_t;

M firmware/include/usb_device_cdc.h => firmware/include/usb_device_cdc.h +4 -0
@@ 124,6 124,10 @@ typedef struct {
  cdc_tx_state_t tx_state;

  uint8_t got_setup;

  bool endpoint_nak;
  uint32_t lost_data;
  bool reenabled_endpoint;
} usb_device_cdc_t;

/**

M firmware/src/main.c => firmware/src/main.c +17 -8
@@ 239,7 239,7 @@ usb_device_t* init_usb()
  void *usb_dev = usb_device_init(USB_OTG_HS1, &USB_CLASS_CDC_ACM,
                                  0x1234, 0x1111, u"Frantisek Bohacek",
                                  u"Display", 1, NULL);
  cdc_acm_configure(usb_dev, 512 * 4); // 4 full packets maximum
  cdc_acm_configure(usb_dev, 512 * 8); // 8 full packets maximum
  usb_device_setup(usb_dev);

  return usb_dev;


@@ 337,8 337,8 @@ void init_clock(void) {
  /* clocks_system_clock_source(CLOCK_SOURCE_PLL_1_P_CK, 1, 1, 1, 300); */
}

#define DISPLAY_WIDTH 800
#define DISPLAY_HEIGHT 800
#define DISPLAY_WIDTH 800UL
#define DISPLAY_HEIGHT 480UL

typedef struct {
  uint8_t b;


@@ 492,16 492,19 @@ void main()

void app_loop(usb_device_t* usb_dev, display_t* display, pixel_rgb888_t* framebuffer)
{
  uint8_t data[64];
  uint8_t data[1024];

  bool handling_cmd = false;
  uint8_t cmd;
  uint32_t pos = 0;
  uint32_t max_size = 1;

  display_refresh(display);

  while (1) {
    uint16_t received = cdc_data_receive(usb_dev, data, 1);
    uint16_t received = cdc_data_receive(usb_dev, data, max_size);

    max_size = 1;

    if (received == 0) {
      continue;


@@ 562,13 565,19 @@ void app_loop(usb_device_t* usb_dev, display_t* display, pixel_rgb888_t* framebu
      break;
    case 'i':
      handling_cmd = true;
      ((uint8_t *)framebuffer)[pos] = data[0];
      max_size = 1024;

      pos++;
      for (uint16_t i = 0; i < received; i++) {
        if (pos > 0) {
          ((uint8_t *)framebuffer)[pos - 1] = data[i];
        }
        pos++;
      }

      if (pos == DISPLAY_WIDTH * DISPLAY_HEIGHT * 3) {
      if (pos >= DISPLAY_WIDTH * DISPLAY_HEIGHT * 3) {
        handling_cmd = false;
        display_refresh(display);
        max_size = 1;
      }
      break;
    }

M firmware/src/usb_device_cdc.c => firmware/src/usb_device_cdc.c +27 -16
@@ 118,6 118,9 @@ void cdc_acm_configure(usb_device_t* device, uint16_t rx_queue_size) {
  usb_device_cdc_t* cdc = (usb_device_cdc_t*)device->class;
  usb_class_header_t* header = &cdc->header;

  cdc->endpoint_nak = false;
  cdc->lost_data = 0;

  queue_t* rx_queue = queue_malloc(1, rx_queue_size);
  cdc->rx_buffer = rx_queue;
  queue_init(rx_queue, 1, rx_queue_size);


@@ 571,6 574,15 @@ void usb_device_cdc_tx_done_callback(usb_device_t *device,
  cdc_process_tx(device, CDC_TX_DONE);
}

void usb_device_cdc_enable_endpoint_maybe(usb_device_t* device, usb_device_cdc_t *cdc) {
  if (queue_space(cdc->rx_buffer) >= MAX_PACKET_SIZE) {
    reg_write_bits_pos(&device->out[DATA_ENDPOINT].DOEPTSIZ, MAX_PACKET_SIZE, USB_OTG_DOEPTSIZ_XFRSIZ_Pos, 0xFFFF);
    reg_write_bits_pos(&device->out[DATA_ENDPOINT].DOEPTSIZ, 1, USB_OTG_DOEPTSIZ_PKTCNT_Pos, 0x3FF);
    device->out[DATA_ENDPOINT].DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
    cdc->endpoint_nak = false;
  }
}

void usb_device_cdc_received_data_callback(usb_device_t *device, packet_info_t *packet) {
  usb_device_cdc_t* cdc = (usb_device_cdc_t*)device->class;
  uint8_t data[4*MAX_PACKET_SIZE];


@@ 581,22 593,17 @@ void usb_device_cdc_received_data_callback(usb_device_t *device, packet_info_t *
  if (packet->packet_status == PACKET_OUT_DATA_PACKET_RCVD) {
    // TODO: could be optimized until pointer has to wrap
    usb_generic_read(data, len, fifo);

    for (uint16_t i = 0; i < len; i++) {
      queue_enqueue(cdc->rx_buffer, (void*)&data[i]);
    if (queue_space(cdc->rx_buffer) < len) {
      cdc->lost_data += len - queue_space(cdc->rx_buffer);
    }

    if (queue_space(cdc->rx_buffer) >= MAX_PACKET_SIZE) {
      reg_write_bits_pos(&device->out[2].DOEPTSIZ, MAX_PACKET_SIZE, USB_OTG_DOEPTSIZ_XFRSIZ_Pos, 0xFFFF);
      reg_write_bits_pos(&device->out[2].DOEPTSIZ, 1, USB_OTG_DOEPTSIZ_PKTCNT_Pos, 0x3FF);
      device->out[DATA_ENDPOINT].DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
    for (uint16_t i = 0; i < len; i++) {
       queue_enqueue(cdc->rx_buffer, (void*)&data[i]);
    }
  } else if(packet->packet_status == PACKET_OUT_TRANSFER_COMPLETED) {
      if (queue_space(cdc->rx_buffer) >= MAX_PACKET_SIZE) {
        reg_write_bits_pos(&device->out[2].DOEPTSIZ, MAX_PACKET_SIZE, USB_OTG_DOEPTSIZ_XFRSIZ_Pos, 0xFFFF);
        reg_write_bits_pos(&device->out[2].DOEPTSIZ, 1, USB_OTG_DOEPTSIZ_PKTCNT_Pos, 0x3FF);
        device->out[DATA_ENDPOINT].DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
      }
    if (!(device->out[DATA_ENDPOINT].DOEPINT & USB_OTG_DOEPINT_NAK)) {
      usb_device_cdc_enable_endpoint_maybe(device, cdc);
    }
  } else {
    usb_generic_read(data, packet->byte_count, fifo);
  }


@@ 604,10 611,10 @@ void usb_device_cdc_received_data_callback(usb_device_t *device, packet_info_t *

void usb_device_cdc_nak_callback(usb_device_t *device, uint8_t endpoint) {
  usb_device_cdc_t* cdc = (usb_device_cdc_t*)device->class;
  if (queue_space(cdc->rx_buffer) >= MAX_PACKET_SIZE) {
    reg_write_bits_pos(&device->out[2].DOEPTSIZ, MAX_PACKET_SIZE, USB_OTG_DOEPTSIZ_XFRSIZ_Pos, 0xFFFF);
    reg_write_bits_pos(&device->out[2].DOEPTSIZ, 1, USB_OTG_DOEPTSIZ_PKTCNT_Pos, 0x3FF);
    device->out[DATA_ENDPOINT].DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_CNAK;
  if (queue_space(cdc->rx_buffer) > MAX_PACKET_SIZE) {
    usb_device_cdc_enable_endpoint_maybe(device, cdc);
  } else {
    cdc->endpoint_nak = true;
  }
}
void usb_device_cdc_nyet_callback(usb_device_t *device, uint8_t endpoint) {


@@ 671,6 678,10 @@ uint16_t cdc_data_receive(usb_device_t *device, uint8_t* buffer,
    }
  }

  if (cdc->endpoint_nak) {
    usb_device_cdc_enable_endpoint_maybe(device, cdc);
  }

  return size;
}


Do not follow this link