usb.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. #include "tusb.h"
  2. #include "ide_async.h"
  3. // Invoked to determine max LUN
  4. uint8_t tud_msc_get_maxlun_cb(void) {
  5. return 1;
  6. }
  7. // Invoked when received SCSI_CMD_INQUIRY
  8. // Application fill vendor id, product id and revision with string up to 8, 16, 4 characters respectively
  9. void tud_msc_inquiry_cb(uint8_t lun, uint8_t vendor_id[8], uint8_t product_id[16], uint8_t product_rev[4]) {
  10. const char vid[] = "Sweproj";
  11. const char pid[] = "IDEUSB adapter";
  12. const char rev[] = "1.0";
  13. memcpy(vendor_id , vid, strlen(vid));
  14. memcpy(product_id , pid, strlen(pid));
  15. memcpy(product_rev, rev, strlen(rev));
  16. }
  17. // Invoked when received Test Unit Ready command.
  18. // return true allowing host to read/write this LUN e.g SD card inserted
  19. bool tud_msc_test_unit_ready_cb(uint8_t lun) {
  20. return true;
  21. }
  22. // Invoked when received SCSI_CMD_READ_CAPACITY_10 and SCSI_CMD_READ_FORMAT_CAPACITY to determine the disk size
  23. // Application update block count and block size
  24. void tud_msc_capacity_cb(uint8_t lun, uint32_t* block_count, uint16_t* block_size) {
  25. *block_count = ide_async_get_num_sectors();
  26. *block_size = 512;
  27. }
  28. // Invoked when received Start Stop Unit command
  29. // - Start = 0 : stopped power mode, if load_eject = 1 : unload disk storage
  30. // - Start = 1 : active mode, if load_eject = 1 : load disk storage
  31. bool tud_msc_start_stop_cb(uint8_t lun, uint8_t power_condition, bool start, bool load_eject) {
  32. return true;
  33. }
  34. // Callback invoked when received READ10 command.
  35. // Copy disk's data to buffer (up to bufsize) and return number of copied bytes.
  36. int32_t tud_msc_read10_cb(uint8_t lun, uint32_t lba, uint32_t offset, void* buffer, uint32_t bufsize) {
  37. return ide_async_read(lba, offset, buffer, bufsize);
  38. }
  39. // Callback invoked when received WRITE10 command.
  40. // Process data in buffer to disk's storage and return number of written bytes
  41. int32_t tud_msc_write10_cb(uint8_t lun, uint32_t lba, uint32_t offset, uint8_t* buffer, uint32_t bufsize) {
  42. return ide_async_write(lba, offset, buffer, bufsize);
  43. }
  44. // Callback invoked when received an SCSI command not in built-in list below
  45. // - READ_CAPACITY10, READ_FORMAT_CAPACITY, INQUIRY, MODE_SENSE6, REQUEST_SENSE
  46. // - READ10 and WRITE10 has their own callbacks
  47. int32_t tud_msc_scsi_cb (uint8_t lun, uint8_t const scsi_cmd[16], void* buffer, uint16_t bufsize) {
  48. // read10 & write10 has their own callback and MUST not be handled here
  49. void const* response = NULL;
  50. uint16_t resplen = 0;
  51. // most scsi handled is input
  52. bool in_xfer = true;
  53. switch (scsi_cmd[0]) {
  54. case SCSI_CMD_PREVENT_ALLOW_MEDIUM_REMOVAL:
  55. // Host is about to read/write etc ... better not to disconnect disk
  56. resplen = 0;
  57. break;
  58. case SCSI_CMD_START_STOP_UNIT:
  59. // Host try to eject/safe remove/poweroff us. We could safely disconnect with disk storage, or go into lower power
  60. /* scsi_start_stop_unit_t const * start_stop = (scsi_start_stop_unit_t const *) scsi_cmd;
  61. // Start bit = 0 : low power mode, if load_eject = 1 : unmount disk storage as well
  62. // Start bit = 1 : Ready mode, if load_eject = 1 : mount disk storage
  63. start_stop->start;
  64. start_stop->load_eject;
  65. */
  66. resplen = 0;
  67. break;
  68. default:
  69. // Set Sense = Invalid Command Operation
  70. tud_msc_set_sense(lun, SCSI_SENSE_ILLEGAL_REQUEST, 0x20, 0x00);
  71. // negative means error -> tinyusb could stall and/or response with failed status
  72. resplen = -1;
  73. break;
  74. }
  75. // return resplen must not larger than bufsize
  76. if (resplen > bufsize) resplen = bufsize;
  77. if (response && (resplen > 0)) {
  78. if(in_xfer) {
  79. memcpy(buffer, response, resplen);
  80. } else {
  81. // SCSI output
  82. }
  83. }
  84. return resplen;
  85. }