scsi_cdrom.c 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. /*
  2. * Copyright (c) 2006-2023, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2023-02-25 GuEe-GUI the first version
  9. */
  10. #include <rthw.h>
  11. #include <rtthread.h>
  12. #include <rtdevice.h>
  13. #define DBG_TAG "scsi.blk"
  14. #define DBG_LVL DBG_INFO
  15. #include <rtdbg.h>
  16. struct scsi_cdrom
  17. {
  18. struct rt_blk_disk parent;
  19. struct rt_scsi_device *sdev;
  20. int cdrom_id;
  21. struct rt_device_blk_geometry geometry;
  22. };
  23. #define raw_to_scsi_cdrom(raw) rt_container_of(raw, struct scsi_cdrom, parent)
  24. static struct rt_dm_ida cdrom_ida = RT_DM_IDA_INIT(CUSTOM);
  25. static struct rt_dm_ida scsi_cdrom_ida = RT_DM_IDA_INIT(SCSI_CDROM);
  26. static rt_ssize_t scsi_cdrom_read(struct rt_blk_disk *disk, rt_off_t sector,
  27. void *buffer, rt_size_t sector_count)
  28. {
  29. rt_err_t err;
  30. struct scsi_cdrom *scdrom = raw_to_scsi_cdrom(disk);
  31. struct rt_scsi_device *sdev = scdrom->sdev;
  32. sector_count &= RT_UINT32_MAX;
  33. if (sector >> 32)
  34. {
  35. err = rt_scsi_read16(sdev, sector, buffer, sector_count);
  36. }
  37. else
  38. {
  39. err = rt_scsi_read12(sdev, sector, buffer, sector_count);
  40. }
  41. return !err ? sector_count : (rt_ssize_t)err;
  42. }
  43. static rt_err_t scsi_cdrom_getgeome(struct rt_blk_disk *disk,
  44. struct rt_device_blk_geometry *geometry)
  45. {
  46. struct scsi_cdrom *scdrom = raw_to_scsi_cdrom(disk);
  47. rt_memcpy(geometry, &scdrom->geometry, sizeof(scdrom->geometry));
  48. return RT_EOK;
  49. }
  50. static const struct rt_blk_disk_ops scsi_cdrom_ops =
  51. {
  52. .read = scsi_cdrom_read,
  53. .getgeome = scsi_cdrom_getgeome,
  54. };
  55. rt_err_t scsi_cdrom_probe(struct rt_scsi_device *sdev)
  56. {
  57. rt_err_t err;
  58. union
  59. {
  60. struct rt_scsi_read_capacity10_data capacity10;
  61. struct rt_scsi_read_capacity16_data capacity16;
  62. } data;
  63. struct scsi_cdrom *scdrom = rt_calloc(1, sizeof(*scdrom));
  64. if (!scdrom)
  65. {
  66. return -RT_ENOMEM;
  67. }
  68. if ((scdrom->cdrom_id = rt_dm_ida_alloc(&cdrom_ida)) < 0)
  69. {
  70. return -RT_EFULL;
  71. }
  72. sdev->priv = scdrom;
  73. scdrom->sdev = sdev;
  74. scdrom->parent.ida = &scsi_cdrom_ida;
  75. scdrom->parent.read_only = RT_TRUE;
  76. scdrom->parent.parallel_io = RT_FALSE;
  77. scdrom->parent.ops = &scsi_cdrom_ops;
  78. scdrom->parent.max_partitions = RT_BLK_PARTITION_NONE;
  79. if ((err = rt_scsi_read_capacity10(sdev, &data.capacity10)))
  80. {
  81. goto _fail;
  82. }
  83. if (data.capacity10.last_block == 0xffffffff)
  84. {
  85. if ((err = rt_scsi_read_capacity16(sdev, &data.capacity16)))
  86. {
  87. goto _fail;
  88. }
  89. }
  90. scdrom->geometry.bytes_per_sector = sdev->block_size;
  91. scdrom->geometry.block_size = sdev->block_size;
  92. scdrom->geometry.sector_count = sdev->last_block + 1;
  93. rt_dm_dev_set_name(&scdrom->parent.parent, "cdrom%u", scdrom->cdrom_id);
  94. if ((err = rt_hw_blk_disk_register(&scdrom->parent)))
  95. {
  96. goto _fail;
  97. }
  98. return RT_EOK;
  99. _fail:
  100. rt_dm_ida_free(&cdrom_ida, scdrom->cdrom_id);
  101. rt_free(scdrom);
  102. return err;
  103. }
  104. rt_err_t scsi_cdrom_remove(struct rt_scsi_device *sdev)
  105. {
  106. struct scsi_cdrom *scdrom = sdev->priv;
  107. rt_dm_ida_free(&cdrom_ida, scdrom->cdrom_id);
  108. return rt_hw_blk_disk_unregister(&scdrom->parent);
  109. }