scsi.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669
  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 "rtdm.scsi"
  14. #define DBG_LVL DBG_INFO
  15. #include <rtdbg.h>
  16. /*
  17. * Since SCSI is used for storage (almost),
  18. * we do not want to implement SCSI as a system DM bus.
  19. */
  20. struct scsi_driver
  21. {
  22. rt_err_t (*probe)(struct rt_scsi_device *sdev);
  23. rt_err_t (*remove)(struct rt_scsi_device *sdev);
  24. };
  25. extern rt_err_t scsi_sd_probe(struct rt_scsi_device *sdev);
  26. extern rt_err_t scsi_sd_remove(struct rt_scsi_device *sdev);
  27. extern rt_err_t scsi_cdrom_probe(struct rt_scsi_device *sdev);
  28. extern rt_err_t scsi_cdrom_remove(struct rt_scsi_device *sdev);
  29. static struct scsi_driver driver_table[SCSI_DEVICE_TYPE_MAX] =
  30. {
  31. #ifdef RT_SCSI_SD
  32. [SCSI_DEVICE_TYPE_DIRECT] =
  33. {
  34. .probe = scsi_sd_probe,
  35. .remove = scsi_sd_remove,
  36. },
  37. #endif /* RT_SCSI_SD */
  38. #ifdef RT_SCSI_CDROM
  39. [SCSI_DEVICE_TYPE_CDROM] =
  40. {
  41. .probe = scsi_cdrom_probe,
  42. .remove = scsi_cdrom_remove,
  43. },
  44. #endif /* RT_SCSI_CDROM */
  45. };
  46. static rt_err_t scsi_device_setup(struct rt_scsi_device *sdev)
  47. {
  48. rt_err_t err;
  49. rt_tick_t timeout;
  50. if (sdev->host->ops->reset)
  51. {
  52. if ((err = sdev->host->ops->reset(sdev)))
  53. {
  54. return err;
  55. }
  56. }
  57. if (!driver_table[sdev->devtype].probe)
  58. {
  59. LOG_E("Device type %x is not supported", sdev->devtype);
  60. return -RT_ENOSYS;
  61. }
  62. timeout = rt_tick_from_millisecond(5000);
  63. timeout += rt_tick_get();
  64. while ((err = rt_scsi_test_unit_ready(sdev)))
  65. {
  66. if (rt_tick_get() >= timeout)
  67. {
  68. return -RT_ETIMEOUT;
  69. }
  70. }
  71. return driver_table[sdev->devtype].probe(sdev);
  72. }
  73. rt_err_t rt_scsi_host_register(struct rt_scsi_host *scsi)
  74. {
  75. struct rt_scsi_device tmp_sdev, *sdev;
  76. if (!scsi || !scsi->dev || !scsi->ops)
  77. {
  78. return -RT_EINVAL;
  79. }
  80. if (!scsi->max_id || !scsi->max_lun)
  81. {
  82. return -RT_EINVAL;
  83. }
  84. rt_list_init(&scsi->lun_nodes);
  85. rt_memset(&tmp_sdev, 0, sizeof(tmp_sdev));
  86. tmp_sdev.host = scsi;
  87. for (rt_size_t id = 0; id < scsi->max_id; ++id)
  88. {
  89. for (rt_size_t lun = 0; lun < scsi->max_lun; ++lun)
  90. {
  91. tmp_sdev.id = id;
  92. tmp_sdev.lun = lun;
  93. if (rt_scsi_inquiry(&tmp_sdev, RT_NULL))
  94. {
  95. continue;
  96. }
  97. if (tmp_sdev.devtype >= SCSI_DEVICE_TYPE_MAX)
  98. {
  99. /*
  100. * This might seem odd, but we're only aiming to
  101. * support simple SCSI.
  102. * If devices appear on the bus out of order,
  103. * we won't perform additional scans.
  104. */
  105. scsi->max_id = id;
  106. scsi->max_lun = lun;
  107. LOG_D("Scan is end of ID: %u LUN: %u", id, lun);
  108. break;
  109. }
  110. if (!(sdev = rt_malloc(sizeof(*sdev))))
  111. {
  112. if (!rt_list_isempty(&scsi->lun_nodes))
  113. {
  114. LOG_E("No memory to create device ID: %u, LUN: %u", id, lun);
  115. return RT_EOK;
  116. }
  117. return -RT_ENOMEM;
  118. }
  119. rt_memcpy(sdev, &tmp_sdev, sizeof(*sdev));
  120. rt_list_init(&sdev->list);
  121. if (scsi_device_setup(sdev))
  122. {
  123. rt_free(sdev);
  124. continue;
  125. }
  126. rt_list_insert_before(&scsi->lun_nodes, &sdev->list);
  127. }
  128. }
  129. return rt_list_isempty(&scsi->lun_nodes) ? -RT_EEMPTY : RT_EOK;
  130. }
  131. rt_err_t rt_scsi_host_unregister(struct rt_scsi_host *scsi)
  132. {
  133. struct rt_scsi_device *sdev, *next_sdev;
  134. if (!scsi)
  135. {
  136. return -RT_EINVAL;
  137. }
  138. rt_list_for_each_entry_safe(sdev, next_sdev, &scsi->lun_nodes, list)
  139. {
  140. rt_list_remove(&sdev->list);
  141. if (sdev->host->ops->reset)
  142. {
  143. sdev->host->ops->reset(sdev);
  144. }
  145. if (!driver_table[sdev->devtype].remove)
  146. {
  147. driver_table[sdev->devtype].remove(sdev);
  148. }
  149. rt_free(sdev);
  150. }
  151. return RT_EOK;
  152. }
  153. rt_inline rt_err_t scsi_transfer(struct rt_scsi_device *sdev, struct rt_scsi_cmd *cmd)
  154. {
  155. return sdev->host->ops->transfer(sdev, cmd);
  156. }
  157. rt_err_t rt_scsi_request_sense(struct rt_scsi_device *sdev,
  158. struct rt_scsi_request_sense_data *out_data)
  159. {
  160. rt_err_t err;
  161. struct rt_scsi_cmd cmd;
  162. rt_memset(&cmd, 0, sizeof(cmd));
  163. cmd.op.request_sense.opcode = RT_SCSI_CMD_REQUEST_SENSE;
  164. cmd.op.request_sense.config = 0;
  165. cmd.op.request_sense.alloc_length = 0x12;
  166. cmd.op.request_sense.control = 0;
  167. cmd.op_size = sizeof(cmd.op.request_sense);
  168. cmd.data.ptr = &cmd.data.request_sense;
  169. cmd.data.size = sizeof(cmd.data.request_sense);
  170. err = scsi_transfer(sdev, &cmd);
  171. if (!err && out_data)
  172. {
  173. rt_memcpy(out_data, &cmd.data.request_sense, sizeof(*out_data));
  174. }
  175. return err;
  176. }
  177. rt_err_t rt_scsi_test_unit_ready(struct rt_scsi_device *sdev)
  178. {
  179. struct rt_scsi_cmd cmd;
  180. rt_memset(&cmd, 0, sizeof(cmd));
  181. cmd.op.test_unit_ready.opcode = RT_SCSI_CMD_TEST_UNIT_READY;
  182. cmd.op.test_unit_ready.control = 0;
  183. cmd.op_size = sizeof(cmd.op.test_unit_ready);
  184. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  185. }
  186. rt_err_t rt_scsi_inquiry(struct rt_scsi_device *sdev,
  187. struct rt_scsi_inquiry_data *out_data)
  188. {
  189. rt_err_t err;
  190. struct rt_scsi_cmd cmd;
  191. rt_memset(&cmd, 0, sizeof(cmd));
  192. cmd.op.inquiry.opcode = RT_SCSI_CMD_INQUIRY;
  193. cmd.op.inquiry.config = 0;
  194. cmd.op.inquiry.page = 0;
  195. cmd.op.inquiry.alloc_length = 0x24;
  196. cmd.op.inquiry.control = 0;
  197. cmd.op_size = sizeof(cmd.op.inquiry);
  198. cmd.data.ptr = &cmd.data.inquiry;
  199. cmd.data.size = sizeof(cmd.data.inquiry);
  200. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  201. if (!err)
  202. {
  203. sdev->devtype = cmd.data.inquiry.devtype & RT_SCSI_DEVTYPE_MASK;
  204. sdev->removable = cmd.data.inquiry.rmb >> RT_SCSI_REMOVABLE_BIT;
  205. if (out_data)
  206. {
  207. rt_memcpy(out_data, &cmd.data.inquiry, sizeof(*out_data));
  208. }
  209. }
  210. return err;
  211. }
  212. rt_err_t rt_scsi_read_capacity10(struct rt_scsi_device *sdev,
  213. struct rt_scsi_read_capacity10_data *out_data)
  214. {
  215. rt_err_t err;
  216. struct rt_scsi_cmd cmd;
  217. rt_memset(&cmd, 0, sizeof(cmd));
  218. cmd.op.read_capacity10.opcode = RT_SCSI_CMD_READ_CAPACITY10;
  219. cmd.op.read_capacity10.config = 0;
  220. cmd.op.read_capacity10.logical_block_addr = 0;
  221. cmd.op.read_capacity10.pmi = 0;
  222. cmd.op.read_capacity10.control = 0;
  223. cmd.op_size = sizeof(cmd.op.read_capacity10);
  224. cmd.data.ptr = &cmd.data.read_capacity10;
  225. cmd.data.size = sizeof(cmd.data.read_capacity10);
  226. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  227. if (!err)
  228. {
  229. sdev->last_block = rt_be32_to_cpu(cmd.data.read_capacity10.last_block);
  230. sdev->block_size = rt_be32_to_cpu(cmd.data.read_capacity10.block_size);
  231. if (out_data)
  232. {
  233. rt_memcpy(out_data, &cmd.data.read_capacity10, sizeof(*out_data));
  234. }
  235. }
  236. return err;
  237. }
  238. rt_err_t rt_scsi_read_capacity16(struct rt_scsi_device *sdev,
  239. struct rt_scsi_read_capacity16_data *out_data)
  240. {
  241. rt_err_t err;
  242. struct rt_scsi_cmd cmd;
  243. rt_memset(&cmd, 0, sizeof(cmd));
  244. cmd.op.read_capacity16.opcode = RT_SCSI_CMD_READ_CAPACITY16;
  245. cmd.op.read_capacity16.config = 0x10;
  246. cmd.op.read_capacity16.logical_block_addr = 0;
  247. cmd.op.read_capacity16.alloc_len = rt_cpu_to_be32(sizeof(cmd.data.read_capacity16));
  248. cmd.op.read_capacity16.pmi = 0;
  249. cmd.op.read_capacity16.control = 0;
  250. cmd.op_size = sizeof(cmd.op.read_capacity16);
  251. cmd.data.ptr = &cmd.data.read_capacity16;
  252. cmd.data.size = sizeof(cmd.data.read_capacity16);
  253. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  254. if (!err)
  255. {
  256. sdev->last_block = rt_be64_to_cpu(cmd.data.read_capacity16.last_block);
  257. sdev->block_size = rt_be32_to_cpu(cmd.data.read_capacity16.block_size);
  258. if (out_data)
  259. {
  260. rt_memcpy(out_data, &cmd.data.read_capacity16, sizeof(*out_data));
  261. }
  262. }
  263. return err;
  264. }
  265. rt_err_t rt_scsi_read10(struct rt_scsi_device *sdev,
  266. rt_off_t lba, void *buffer, rt_size_t size)
  267. {
  268. struct rt_scsi_cmd cmd;
  269. rt_memset(&cmd, 0, sizeof(cmd));
  270. cmd.op.read10.opcode = RT_SCSI_CMD_READ10;
  271. cmd.op.read10.config = 0;
  272. cmd.op.read10.lba = rt_cpu_to_be32(lba);
  273. cmd.op.read10.size = rt_cpu_to_be16(size);
  274. cmd.op_size = sizeof(cmd.op.read10);
  275. cmd.data.ptr = buffer;
  276. cmd.data.size = size * sdev->block_size;
  277. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  278. }
  279. rt_err_t rt_scsi_read12(struct rt_scsi_device *sdev,
  280. rt_off_t lba, void *buffer, rt_size_t size)
  281. {
  282. struct rt_scsi_cmd cmd;
  283. rt_memset(&cmd, 0, sizeof(cmd));
  284. cmd.op.read12.opcode = RT_SCSI_CMD_READ12;
  285. cmd.op.read12.config = 0;
  286. cmd.op.read12.lba = rt_cpu_to_be32(lba);
  287. cmd.op.read12.size = rt_cpu_to_be32(size);
  288. cmd.op.read12.control = 0;
  289. cmd.op_size = sizeof(cmd.op.read12);
  290. cmd.data.ptr = buffer;
  291. cmd.data.size = size * sdev->block_size;
  292. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  293. }
  294. rt_err_t rt_scsi_read16(struct rt_scsi_device *sdev,
  295. rt_off_t lba, void *buffer, rt_size_t size)
  296. {
  297. struct rt_scsi_cmd cmd;
  298. rt_memset(&cmd, 0, sizeof(cmd));
  299. cmd.op.read16.opcode = RT_SCSI_CMD_READ16;
  300. cmd.op.read16.config = 0;
  301. cmd.op.read16.lba = rt_cpu_to_be64(lba);
  302. cmd.op.read16.size = rt_cpu_to_be32(size);
  303. cmd.op.read16.control = 0;
  304. cmd.op_size = sizeof(cmd.op.read16);
  305. cmd.data.ptr = buffer;
  306. cmd.data.size = size * sdev->block_size;
  307. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  308. }
  309. rt_err_t rt_scsi_write10(struct rt_scsi_device *sdev,
  310. rt_off_t lba, const void *buffer, rt_size_t size)
  311. {
  312. struct rt_scsi_cmd cmd;
  313. rt_memset(&cmd, 0, sizeof(cmd));
  314. cmd.op.write10.opcode = RT_SCSI_CMD_WRITE10;
  315. cmd.op.write10.config = 0;
  316. cmd.op.write10.lba = rt_cpu_to_be32(lba);
  317. cmd.op.write10.size = rt_cpu_to_be16(size);
  318. cmd.op_size = sizeof(cmd.op.write10);
  319. cmd.data.ptr = (void *)buffer;
  320. cmd.data.size = size * sdev->block_size;
  321. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  322. }
  323. rt_err_t rt_scsi_write12(struct rt_scsi_device *sdev,
  324. rt_off_t lba, const void *buffer, rt_size_t size)
  325. {
  326. struct rt_scsi_cmd cmd;
  327. rt_memset(&cmd, 0, sizeof(cmd));
  328. cmd.op.write12.opcode = RT_SCSI_CMD_WRITE12;
  329. cmd.op.write12.config = 0;
  330. cmd.op.write12.lba = rt_cpu_to_be32(lba);
  331. cmd.op.write12.size = rt_cpu_to_be32(size);
  332. cmd.op.write12.control = 0;
  333. cmd.op_size = sizeof(cmd.op.write12);
  334. cmd.data.ptr = (void *)buffer;
  335. cmd.data.size = size * sdev->block_size;
  336. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  337. }
  338. rt_err_t rt_scsi_write16(struct rt_scsi_device *sdev,
  339. rt_off_t lba, const void *buffer, rt_size_t size)
  340. {
  341. struct rt_scsi_cmd cmd;
  342. rt_memset(&cmd, 0, sizeof(cmd));
  343. cmd.op.write16.opcode = RT_SCSI_CMD_WRITE16;
  344. cmd.op.write16.config = 0;
  345. cmd.op.write16.lba = rt_cpu_to_be64(lba);
  346. cmd.op.write16.size = rt_cpu_to_be32(size);
  347. cmd.op.write16.control = 0;
  348. cmd.op_size = sizeof(cmd.op.write16);
  349. cmd.data.ptr = (void *)buffer;
  350. cmd.data.size = size * sdev->block_size;
  351. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  352. }
  353. rt_err_t rt_scsi_synchronize_cache10(struct rt_scsi_device *sdev,
  354. rt_off_t lba, rt_size_t size)
  355. {
  356. struct rt_scsi_cmd cmd;
  357. rt_memset(&cmd, 0, sizeof(cmd));
  358. cmd.op.synchronize_cache10.opcode = RT_SCSI_CMD_SYNCHRONIZE_CACHE10;
  359. cmd.op.synchronize_cache10.config = 0;
  360. cmd.op.synchronize_cache10.lba = rt_cpu_to_be32(lba);
  361. cmd.op.synchronize_cache10.size = rt_cpu_to_be16(size);
  362. cmd.op.synchronize_cache10.control = 0;
  363. cmd.op_size = sizeof(cmd.op.synchronize_cache10);
  364. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  365. }
  366. rt_err_t rt_scsi_synchronize_cache16(struct rt_scsi_device *sdev,
  367. rt_off_t lba, rt_size_t size)
  368. {
  369. struct rt_scsi_cmd cmd;
  370. rt_memset(&cmd, 0, sizeof(cmd));
  371. cmd.op.synchronize_cache16.opcode = RT_SCSI_CMD_SYNCHRONIZE_CACHE16;
  372. cmd.op.synchronize_cache16.config = 0;
  373. cmd.op.synchronize_cache16.lba = rt_cpu_to_be64(lba);
  374. cmd.op.synchronize_cache16.size = rt_cpu_to_be32(size);
  375. cmd.op.synchronize_cache16.control = 0;
  376. cmd.op_size = sizeof(cmd.op.synchronize_cache16);
  377. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  378. }
  379. rt_err_t rt_scsi_write_same10(struct rt_scsi_device *sdev,
  380. rt_off_t lba, rt_size_t size)
  381. {
  382. struct rt_scsi_cmd cmd;
  383. rt_memset(&cmd, 0, sizeof(cmd));
  384. cmd.op.write_same10.opcode = RT_SCSI_CMD_WRITE_SAME10;
  385. cmd.op.write_same10.config = RT_BIT(RT_SCSI_UNMAP_SHIFT);
  386. cmd.op.write_same10.lba = rt_cpu_to_be32(lba);
  387. cmd.op.write_same10.size = rt_cpu_to_be16(size);
  388. cmd.op.write_same10.control = 0;
  389. cmd.op_size = sizeof(cmd.op.write_same10);
  390. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  391. }
  392. rt_err_t rt_scsi_write_same16(struct rt_scsi_device *sdev,
  393. rt_off_t lba, rt_size_t size)
  394. {
  395. struct rt_scsi_cmd cmd;
  396. rt_memset(&cmd, 0, sizeof(cmd));
  397. cmd.op.write_same16.opcode = RT_SCSI_CMD_WRITE_SAME16;
  398. cmd.op.write_same16.config = RT_BIT(RT_SCSI_UNMAP_SHIFT);
  399. cmd.op.write_same16.lba = rt_cpu_to_be64(lba);
  400. cmd.op.write_same16.size = rt_cpu_to_be32(size);
  401. cmd.op.write_same16.control = 0;
  402. cmd.op_size = sizeof(cmd.op.write_same16);
  403. return scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  404. }
  405. rt_err_t rt_scsi_mode_select6(struct rt_scsi_device *sdev,
  406. rt_uint8_t pf, rt_uint8_t sp, void *buffer, rt_size_t size,
  407. struct rt_scsi_mode_select_data *data)
  408. {
  409. rt_err_t err;
  410. rt_uint8_t *real_buffer;
  411. struct rt_scsi_cmd cmd;
  412. real_buffer = rt_malloc(4 + size);
  413. if (!real_buffer)
  414. {
  415. return -RT_ENOMEM;
  416. }
  417. rt_memcpy(real_buffer + 4, buffer, size);
  418. size += 4;
  419. real_buffer[0] = 0;
  420. real_buffer[1] = data->medium_type;
  421. real_buffer[2] = data->device_specific;
  422. real_buffer[3] = data->block_descriptor_length;
  423. rt_memset(&cmd, 0, sizeof(cmd));
  424. cmd.op.mode_select6.opcode = RT_SCSI_CMD_MODE_SELECT;
  425. cmd.op.mode_select6.config = pf ? RT_BIT(RT_SCSI_PF_SHIFT) : 0;
  426. cmd.op.mode_select6.config |= sp ? RT_BIT(RT_SCSI_SP_SHIFT) : 0;
  427. cmd.op.mode_select6.param_list_len = (rt_uint8_t)size;
  428. cmd.op.mode_select6.control = 0;
  429. cmd.op_size = sizeof(cmd.op.mode_select6);
  430. cmd.data.ptr = real_buffer;
  431. cmd.data.size = size;
  432. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  433. rt_free(real_buffer);
  434. return err;
  435. }
  436. rt_err_t rt_scsi_mode_select10(struct rt_scsi_device *sdev,
  437. rt_uint8_t pf, rt_uint8_t sp, void *buffer, rt_size_t size,
  438. struct rt_scsi_mode_select_data *data)
  439. {
  440. rt_err_t err;
  441. rt_uint8_t *real_buffer;
  442. struct rt_scsi_cmd cmd;
  443. real_buffer = rt_malloc(8 + size);
  444. if (!real_buffer)
  445. {
  446. return -RT_ENOMEM;
  447. }
  448. rt_memcpy(real_buffer + 8, buffer, size);
  449. size += 8;
  450. real_buffer[0] = 0;
  451. real_buffer[1] = 0;
  452. real_buffer[2] = data->medium_type;
  453. real_buffer[3] = data->device_specific;
  454. real_buffer[4] = data->longlba ? 0x01 : 0;
  455. real_buffer[5] = 0;
  456. real_buffer[6] = rt_cpu_to_be16(data->block_descriptor_length);
  457. rt_memset(&cmd, 0, sizeof(cmd));
  458. cmd.op.mode_select10.opcode = RT_SCSI_CMD_MODE_SELECT10;
  459. cmd.op.mode_select10.config = pf ? RT_BIT(RT_SCSI_PF_SHIFT) : 0;
  460. cmd.op.mode_select10.config |= sp ? RT_BIT(RT_SCSI_SP_SHIFT) : 0;
  461. cmd.op.mode_select10.param_list_len = rt_cpu_to_be16(size);
  462. cmd.op.mode_select10.control = 0;
  463. cmd.op_size = sizeof(cmd.op.mode_select10);
  464. cmd.data.ptr = real_buffer;
  465. cmd.data.size = size;
  466. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  467. rt_free(real_buffer);
  468. return err;
  469. }
  470. static void scsi_mode_sense_fill(struct rt_scsi_mode_select_data *data,
  471. rt_uint8_t modepage, rt_uint8_t *buffer, rt_bool_t use10)
  472. {
  473. if (buffer[0] == 0x86 && buffer[1] == 0x0b && (modepage == 6 || modepage == 8))
  474. {
  475. data->header_length = 0;
  476. data->length = 13;
  477. data->medium_type = 0;
  478. data->device_specific = 0;
  479. data->longlba = 0;
  480. data->block_descriptor_length = 0;
  481. }
  482. else if (use10)
  483. {
  484. data->length = rt_be16_to_cpu(buffer[0]) + 2;
  485. data->medium_type = buffer[2];
  486. data->device_specific = buffer[3];
  487. data->longlba = buffer[4] & 0x01;
  488. data->block_descriptor_length = rt_be16_to_cpu(buffer[6]);
  489. }
  490. else
  491. {
  492. data->length = buffer[0] + 1;
  493. data->medium_type = buffer[1];
  494. data->device_specific = buffer[2];
  495. data->block_descriptor_length = buffer[3];
  496. }
  497. }
  498. rt_err_t rt_scsi_mode_sense6(struct rt_scsi_device *sdev,
  499. rt_uint8_t dbd, rt_uint8_t modepage, rt_uint8_t subpage, void *buffer, rt_size_t size,
  500. struct rt_scsi_mode_select_data *data)
  501. {
  502. rt_err_t err;
  503. struct rt_scsi_cmd cmd;
  504. rt_memset(buffer, 0, size);
  505. rt_memset(&cmd, 0, sizeof(cmd));
  506. cmd.op.mode_sense6.opcode = RT_SCSI_CMD_MODE_SENSE;
  507. cmd.op.mode_sense6.config = dbd & 0x18;
  508. cmd.op.mode_sense6.page_control_code = modepage;
  509. cmd.op.mode_sense6.subpage_code = subpage;
  510. cmd.op.mode_sense6.allocation_len = (rt_uint8_t)size;
  511. cmd.op.mode_sense6.control = 0;
  512. cmd.op_size = sizeof(cmd.op.mode_sense6);
  513. cmd.data.ptr = buffer;
  514. cmd.data.size = size;
  515. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  516. if (!err)
  517. {
  518. data->header_length = 4;
  519. scsi_mode_sense_fill(data, modepage, buffer, RT_FALSE);
  520. }
  521. return err;
  522. }
  523. rt_err_t rt_scsi_mode_sense10(struct rt_scsi_device *sdev,
  524. rt_uint8_t dbd, rt_uint8_t modepage, rt_uint8_t subpage, void *buffer, rt_size_t size,
  525. struct rt_scsi_mode_select_data *data)
  526. {
  527. rt_err_t err;
  528. struct rt_scsi_cmd cmd;
  529. rt_memset(buffer, 0, size);
  530. rt_memset(&cmd, 0, sizeof(cmd));
  531. cmd.op.mode_sense6.opcode = RT_SCSI_CMD_MODE_SENSE10;
  532. cmd.op.mode_sense6.config = dbd & 0x18;
  533. cmd.op.mode_sense6.page_control_code = modepage;
  534. cmd.op.mode_sense6.subpage_code = subpage;
  535. cmd.op.mode_sense6.allocation_len = rt_cpu_to_be16(size);
  536. cmd.op.mode_sense6.control = 0;
  537. cmd.op_size = sizeof(cmd.op.mode_sense6);
  538. cmd.data.ptr = buffer;
  539. cmd.data.size = size;
  540. err = scsi_transfer(sdev, &cmd) ? : rt_scsi_request_sense(sdev, RT_NULL);
  541. if (!err)
  542. {
  543. data->header_length = 8;
  544. scsi_mode_sense_fill(data, modepage, buffer, RT_FALSE);
  545. }
  546. return err;
  547. }