blk-uclass.c 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2016 Google, Inc
  4. * Written by Simon Glass <sjg@chromium.org>
  5. */
  6. #include <common.h>
  7. #include <blk.h>
  8. #include <dm.h>
  9. #include <dm/device-internal.h>
  10. #include <dm/lists.h>
  11. #include <dm/uclass-internal.h>
  12. static const char *if_typename_str[IF_TYPE_COUNT] = {
  13. [IF_TYPE_IDE] = "ide",
  14. [IF_TYPE_SCSI] = "scsi",
  15. [IF_TYPE_ATAPI] = "atapi",
  16. [IF_TYPE_USB] = "usb",
  17. [IF_TYPE_DOC] = "doc",
  18. [IF_TYPE_MMC] = "mmc",
  19. [IF_TYPE_SD] = "sd",
  20. [IF_TYPE_SATA] = "sata",
  21. [IF_TYPE_HOST] = "host",
  22. [IF_TYPE_NVME] = "nvme",
  23. [IF_TYPE_EFI] = "efi",
  24. [IF_TYPE_VIRTIO] = "virtio",
  25. };
  26. static enum uclass_id if_type_uclass_id[IF_TYPE_COUNT] = {
  27. [IF_TYPE_IDE] = UCLASS_IDE,
  28. [IF_TYPE_SCSI] = UCLASS_SCSI,
  29. [IF_TYPE_ATAPI] = UCLASS_INVALID,
  30. [IF_TYPE_USB] = UCLASS_MASS_STORAGE,
  31. [IF_TYPE_DOC] = UCLASS_INVALID,
  32. [IF_TYPE_MMC] = UCLASS_MMC,
  33. [IF_TYPE_SD] = UCLASS_INVALID,
  34. [IF_TYPE_SATA] = UCLASS_AHCI,
  35. [IF_TYPE_HOST] = UCLASS_ROOT,
  36. [IF_TYPE_NVME] = UCLASS_NVME,
  37. [IF_TYPE_EFI] = UCLASS_EFI,
  38. [IF_TYPE_VIRTIO] = UCLASS_VIRTIO,
  39. };
  40. static enum if_type if_typename_to_iftype(const char *if_typename)
  41. {
  42. int i;
  43. for (i = 0; i < IF_TYPE_COUNT; i++) {
  44. if (if_typename_str[i] &&
  45. !strcmp(if_typename, if_typename_str[i]))
  46. return i;
  47. }
  48. return IF_TYPE_UNKNOWN;
  49. }
  50. static enum uclass_id if_type_to_uclass_id(enum if_type if_type)
  51. {
  52. return if_type_uclass_id[if_type];
  53. }
  54. const char *blk_get_if_type_name(enum if_type if_type)
  55. {
  56. return if_typename_str[if_type];
  57. }
  58. struct blk_desc *blk_get_devnum_by_type(enum if_type if_type, int devnum)
  59. {
  60. struct blk_desc *desc;
  61. struct udevice *dev;
  62. int ret;
  63. ret = blk_get_device(if_type, devnum, &dev);
  64. if (ret)
  65. return NULL;
  66. desc = dev_get_uclass_platdata(dev);
  67. return desc;
  68. }
  69. /*
  70. * This function is complicated with driver model. We look up the interface
  71. * name in a local table. This gives us an interface type which we can match
  72. * against the uclass of the block device's parent.
  73. */
  74. struct blk_desc *blk_get_devnum_by_typename(const char *if_typename, int devnum)
  75. {
  76. enum uclass_id uclass_id;
  77. enum if_type if_type;
  78. struct udevice *dev;
  79. struct uclass *uc;
  80. int ret;
  81. if_type = if_typename_to_iftype(if_typename);
  82. if (if_type == IF_TYPE_UNKNOWN) {
  83. debug("%s: Unknown interface type '%s'\n", __func__,
  84. if_typename);
  85. return NULL;
  86. }
  87. uclass_id = if_type_to_uclass_id(if_type);
  88. if (uclass_id == UCLASS_INVALID) {
  89. debug("%s: Unknown uclass for interface type'\n",
  90. if_typename_str[if_type]);
  91. return NULL;
  92. }
  93. ret = uclass_get(UCLASS_BLK, &uc);
  94. if (ret)
  95. return NULL;
  96. uclass_foreach_dev(dev, uc) {
  97. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  98. debug("%s: if_type=%d, devnum=%d: %s, %d, %d\n", __func__,
  99. if_type, devnum, dev->name, desc->if_type, desc->devnum);
  100. if (desc->devnum != devnum)
  101. continue;
  102. /* Find out the parent device uclass */
  103. if (device_get_uclass_id(dev->parent) != uclass_id) {
  104. debug("%s: parent uclass %d, this dev %d\n", __func__,
  105. device_get_uclass_id(dev->parent), uclass_id);
  106. continue;
  107. }
  108. if (device_probe(dev))
  109. return NULL;
  110. debug("%s: Device desc %p\n", __func__, desc);
  111. return desc;
  112. }
  113. debug("%s: No device found\n", __func__);
  114. return NULL;
  115. }
  116. /**
  117. * blk_get_by_device() - Get the block device descriptor for the given device
  118. * @dev: Instance of a storage device
  119. *
  120. * Return: With block device descriptor on success , NULL if there is no such
  121. * block device.
  122. */
  123. struct blk_desc *blk_get_by_device(struct udevice *dev)
  124. {
  125. struct udevice *child_dev, *next;
  126. device_foreach_child_safe(child_dev, next, dev) {
  127. if (device_get_uclass_id(child_dev) != UCLASS_BLK)
  128. continue;
  129. return dev_get_uclass_platdata(child_dev);
  130. }
  131. debug("%s: No block device found\n", __func__);
  132. return NULL;
  133. }
  134. /**
  135. * get_desc() - Get the block device descriptor for the given device number
  136. *
  137. * @if_type: Interface type
  138. * @devnum: Device number (0 = first)
  139. * @descp: Returns block device descriptor on success
  140. * @return 0 on success, -ENODEV if there is no such device and no device
  141. * with a higher device number, -ENOENT if there is no such device but there
  142. * is one with a higher number, or other -ve on other error.
  143. */
  144. static int get_desc(enum if_type if_type, int devnum, struct blk_desc **descp)
  145. {
  146. bool found_more = false;
  147. struct udevice *dev;
  148. struct uclass *uc;
  149. int ret;
  150. *descp = NULL;
  151. ret = uclass_get(UCLASS_BLK, &uc);
  152. if (ret)
  153. return ret;
  154. uclass_foreach_dev(dev, uc) {
  155. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  156. debug("%s: if_type=%d, devnum=%d: %s, %d, %d\n", __func__,
  157. if_type, devnum, dev->name, desc->if_type, desc->devnum);
  158. if (desc->if_type == if_type) {
  159. if (desc->devnum == devnum) {
  160. ret = device_probe(dev);
  161. if (ret)
  162. return ret;
  163. *descp = desc;
  164. return 0;
  165. } else if (desc->devnum > devnum) {
  166. found_more = true;
  167. }
  168. }
  169. }
  170. return found_more ? -ENOENT : -ENODEV;
  171. }
  172. int blk_select_hwpart_devnum(enum if_type if_type, int devnum, int hwpart)
  173. {
  174. struct udevice *dev;
  175. int ret;
  176. ret = blk_get_device(if_type, devnum, &dev);
  177. if (ret)
  178. return ret;
  179. return blk_select_hwpart(dev, hwpart);
  180. }
  181. int blk_list_part(enum if_type if_type)
  182. {
  183. struct blk_desc *desc;
  184. int devnum, ok;
  185. int ret;
  186. for (ok = 0, devnum = 0;; ++devnum) {
  187. ret = get_desc(if_type, devnum, &desc);
  188. if (ret == -ENODEV)
  189. break;
  190. else if (ret)
  191. continue;
  192. if (desc->part_type != PART_TYPE_UNKNOWN) {
  193. ++ok;
  194. if (devnum)
  195. putc('\n');
  196. part_print(desc);
  197. }
  198. }
  199. if (!ok)
  200. return -ENODEV;
  201. return 0;
  202. }
  203. int blk_print_part_devnum(enum if_type if_type, int devnum)
  204. {
  205. struct blk_desc *desc;
  206. int ret;
  207. ret = get_desc(if_type, devnum, &desc);
  208. if (ret)
  209. return ret;
  210. if (desc->type == DEV_TYPE_UNKNOWN)
  211. return -ENOENT;
  212. part_print(desc);
  213. return 0;
  214. }
  215. void blk_list_devices(enum if_type if_type)
  216. {
  217. struct blk_desc *desc;
  218. int ret;
  219. int i;
  220. for (i = 0;; ++i) {
  221. ret = get_desc(if_type, i, &desc);
  222. if (ret == -ENODEV)
  223. break;
  224. else if (ret)
  225. continue;
  226. if (desc->type == DEV_TYPE_UNKNOWN)
  227. continue; /* list only known devices */
  228. printf("Device %d: ", i);
  229. dev_print(desc);
  230. }
  231. }
  232. int blk_print_device_num(enum if_type if_type, int devnum)
  233. {
  234. struct blk_desc *desc;
  235. int ret;
  236. ret = get_desc(if_type, devnum, &desc);
  237. if (ret)
  238. return ret;
  239. printf("\nIDE device %d: ", devnum);
  240. dev_print(desc);
  241. return 0;
  242. }
  243. int blk_show_device(enum if_type if_type, int devnum)
  244. {
  245. struct blk_desc *desc;
  246. int ret;
  247. printf("\nDevice %d: ", devnum);
  248. ret = get_desc(if_type, devnum, &desc);
  249. if (ret == -ENODEV || ret == -ENOENT) {
  250. printf("unknown device\n");
  251. return -ENODEV;
  252. }
  253. if (ret)
  254. return ret;
  255. dev_print(desc);
  256. if (desc->type == DEV_TYPE_UNKNOWN)
  257. return -ENOENT;
  258. return 0;
  259. }
  260. ulong blk_read_devnum(enum if_type if_type, int devnum, lbaint_t start,
  261. lbaint_t blkcnt, void *buffer)
  262. {
  263. struct blk_desc *desc;
  264. ulong n;
  265. int ret;
  266. ret = get_desc(if_type, devnum, &desc);
  267. if (ret)
  268. return ret;
  269. n = blk_dread(desc, start, blkcnt, buffer);
  270. if (IS_ERR_VALUE(n))
  271. return n;
  272. return n;
  273. }
  274. ulong blk_write_devnum(enum if_type if_type, int devnum, lbaint_t start,
  275. lbaint_t blkcnt, const void *buffer)
  276. {
  277. struct blk_desc *desc;
  278. int ret;
  279. ret = get_desc(if_type, devnum, &desc);
  280. if (ret)
  281. return ret;
  282. return blk_dwrite(desc, start, blkcnt, buffer);
  283. }
  284. int blk_select_hwpart(struct udevice *dev, int hwpart)
  285. {
  286. const struct blk_ops *ops = blk_get_ops(dev);
  287. if (!ops)
  288. return -ENOSYS;
  289. if (!ops->select_hwpart)
  290. return 0;
  291. return ops->select_hwpart(dev, hwpart);
  292. }
  293. int blk_dselect_hwpart(struct blk_desc *desc, int hwpart)
  294. {
  295. return blk_select_hwpart(desc->bdev, hwpart);
  296. }
  297. int blk_first_device(int if_type, struct udevice **devp)
  298. {
  299. struct blk_desc *desc;
  300. int ret;
  301. ret = uclass_find_first_device(UCLASS_BLK, devp);
  302. if (ret)
  303. return ret;
  304. if (!*devp)
  305. return -ENODEV;
  306. do {
  307. desc = dev_get_uclass_platdata(*devp);
  308. if (desc->if_type == if_type)
  309. return 0;
  310. ret = uclass_find_next_device(devp);
  311. if (ret)
  312. return ret;
  313. } while (*devp);
  314. return -ENODEV;
  315. }
  316. int blk_next_device(struct udevice **devp)
  317. {
  318. struct blk_desc *desc;
  319. int ret, if_type;
  320. desc = dev_get_uclass_platdata(*devp);
  321. if_type = desc->if_type;
  322. do {
  323. ret = uclass_find_next_device(devp);
  324. if (ret)
  325. return ret;
  326. if (!*devp)
  327. return -ENODEV;
  328. desc = dev_get_uclass_platdata(*devp);
  329. if (desc->if_type == if_type)
  330. return 0;
  331. } while (1);
  332. }
  333. int blk_find_device(int if_type, int devnum, struct udevice **devp)
  334. {
  335. struct uclass *uc;
  336. struct udevice *dev;
  337. int ret;
  338. ret = uclass_get(UCLASS_BLK, &uc);
  339. if (ret)
  340. return ret;
  341. uclass_foreach_dev(dev, uc) {
  342. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  343. debug("%s: if_type=%d, devnum=%d: %s, %d, %d\n", __func__,
  344. if_type, devnum, dev->name, desc->if_type, desc->devnum);
  345. if (desc->if_type == if_type && desc->devnum == devnum) {
  346. *devp = dev;
  347. return 0;
  348. }
  349. }
  350. return -ENODEV;
  351. }
  352. int blk_get_device(int if_type, int devnum, struct udevice **devp)
  353. {
  354. int ret;
  355. ret = blk_find_device(if_type, devnum, devp);
  356. if (ret)
  357. return ret;
  358. return device_probe(*devp);
  359. }
  360. unsigned long blk_dread(struct blk_desc *block_dev, lbaint_t start,
  361. lbaint_t blkcnt, void *buffer)
  362. {
  363. struct udevice *dev = block_dev->bdev;
  364. const struct blk_ops *ops = blk_get_ops(dev);
  365. ulong blks_read;
  366. if (!ops->read)
  367. return -ENOSYS;
  368. if (blkcache_read(block_dev->if_type, block_dev->devnum,
  369. start, blkcnt, block_dev->blksz, buffer))
  370. return blkcnt;
  371. blks_read = ops->read(dev, start, blkcnt, buffer);
  372. if (blks_read == blkcnt)
  373. blkcache_fill(block_dev->if_type, block_dev->devnum,
  374. start, blkcnt, block_dev->blksz, buffer);
  375. return blks_read;
  376. }
  377. unsigned long blk_dwrite(struct blk_desc *block_dev, lbaint_t start,
  378. lbaint_t blkcnt, const void *buffer)
  379. {
  380. struct udevice *dev = block_dev->bdev;
  381. const struct blk_ops *ops = blk_get_ops(dev);
  382. if (!ops->write)
  383. return -ENOSYS;
  384. blkcache_invalidate(block_dev->if_type, block_dev->devnum);
  385. return ops->write(dev, start, blkcnt, buffer);
  386. }
  387. unsigned long blk_derase(struct blk_desc *block_dev, lbaint_t start,
  388. lbaint_t blkcnt)
  389. {
  390. struct udevice *dev = block_dev->bdev;
  391. const struct blk_ops *ops = blk_get_ops(dev);
  392. if (!ops->erase)
  393. return -ENOSYS;
  394. blkcache_invalidate(block_dev->if_type, block_dev->devnum);
  395. return ops->erase(dev, start, blkcnt);
  396. }
  397. int blk_get_from_parent(struct udevice *parent, struct udevice **devp)
  398. {
  399. struct udevice *dev;
  400. enum uclass_id id;
  401. int ret;
  402. device_find_first_child(parent, &dev);
  403. if (!dev) {
  404. debug("%s: No block device found for parent '%s'\n", __func__,
  405. parent->name);
  406. return -ENODEV;
  407. }
  408. id = device_get_uclass_id(dev);
  409. if (id != UCLASS_BLK) {
  410. debug("%s: Incorrect uclass %s for block device '%s'\n",
  411. __func__, uclass_get_name(id), dev->name);
  412. return -ENOTBLK;
  413. }
  414. ret = device_probe(dev);
  415. if (ret)
  416. return ret;
  417. *devp = dev;
  418. return 0;
  419. }
  420. int blk_find_max_devnum(enum if_type if_type)
  421. {
  422. struct udevice *dev;
  423. int max_devnum = -ENODEV;
  424. struct uclass *uc;
  425. int ret;
  426. ret = uclass_get(UCLASS_BLK, &uc);
  427. if (ret)
  428. return ret;
  429. uclass_foreach_dev(dev, uc) {
  430. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  431. if (desc->if_type == if_type && desc->devnum > max_devnum)
  432. max_devnum = desc->devnum;
  433. }
  434. return max_devnum;
  435. }
  436. int blk_next_free_devnum(enum if_type if_type)
  437. {
  438. int ret;
  439. ret = blk_find_max_devnum(if_type);
  440. if (ret == -ENODEV)
  441. return 0;
  442. if (ret < 0)
  443. return ret;
  444. return ret + 1;
  445. }
  446. static int blk_claim_devnum(enum if_type if_type, int devnum)
  447. {
  448. struct udevice *dev;
  449. struct uclass *uc;
  450. int ret;
  451. ret = uclass_get(UCLASS_BLK, &uc);
  452. if (ret)
  453. return ret;
  454. uclass_foreach_dev(dev, uc) {
  455. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  456. if (desc->if_type == if_type && desc->devnum == devnum) {
  457. int next = blk_next_free_devnum(if_type);
  458. if (next < 0)
  459. return next;
  460. desc->devnum = next;
  461. return 0;
  462. }
  463. }
  464. return -ENOENT;
  465. }
  466. int blk_create_device(struct udevice *parent, const char *drv_name,
  467. const char *name, int if_type, int devnum, int blksz,
  468. lbaint_t lba, struct udevice **devp)
  469. {
  470. struct blk_desc *desc;
  471. struct udevice *dev;
  472. int ret;
  473. if (devnum == -1) {
  474. devnum = blk_next_free_devnum(if_type);
  475. } else {
  476. ret = blk_claim_devnum(if_type, devnum);
  477. if (ret < 0 && ret != -ENOENT)
  478. return ret;
  479. }
  480. if (devnum < 0)
  481. return devnum;
  482. ret = device_bind_driver(parent, drv_name, name, &dev);
  483. if (ret)
  484. return ret;
  485. desc = dev_get_uclass_platdata(dev);
  486. desc->if_type = if_type;
  487. desc->blksz = blksz;
  488. desc->lba = lba;
  489. desc->part_type = PART_TYPE_UNKNOWN;
  490. desc->bdev = dev;
  491. desc->devnum = devnum;
  492. *devp = dev;
  493. return 0;
  494. }
  495. int blk_create_devicef(struct udevice *parent, const char *drv_name,
  496. const char *name, int if_type, int devnum, int blksz,
  497. lbaint_t lba, struct udevice **devp)
  498. {
  499. char dev_name[30], *str;
  500. int ret;
  501. snprintf(dev_name, sizeof(dev_name), "%s.%s", parent->name, name);
  502. str = strdup(dev_name);
  503. if (!str)
  504. return -ENOMEM;
  505. ret = blk_create_device(parent, drv_name, str, if_type, devnum,
  506. blksz, lba, devp);
  507. if (ret) {
  508. free(str);
  509. return ret;
  510. }
  511. device_set_name_alloced(*devp);
  512. return 0;
  513. }
  514. int blk_unbind_all(int if_type)
  515. {
  516. struct uclass *uc;
  517. struct udevice *dev, *next;
  518. int ret;
  519. ret = uclass_get(UCLASS_BLK, &uc);
  520. if (ret)
  521. return ret;
  522. uclass_foreach_dev_safe(dev, next, uc) {
  523. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  524. if (desc->if_type == if_type) {
  525. ret = device_remove(dev, DM_REMOVE_NORMAL);
  526. if (ret)
  527. return ret;
  528. ret = device_unbind(dev);
  529. if (ret)
  530. return ret;
  531. }
  532. }
  533. return 0;
  534. }
  535. static int blk_post_probe(struct udevice *dev)
  536. {
  537. #if !defined(CONFIG_SPL_BUILD) || defined(CONFIG_SPL_LIBDISK_SUPPORT)
  538. struct blk_desc *desc = dev_get_uclass_platdata(dev);
  539. part_init(desc);
  540. #endif
  541. return 0;
  542. }
  543. UCLASS_DRIVER(blk) = {
  544. .id = UCLASS_BLK,
  545. .name = "blk",
  546. .post_probe = blk_post_probe,
  547. .per_device_platdata_auto_alloc_size = sizeof(struct blk_desc),
  548. };