uclass.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2013 Google, Inc
  4. *
  5. * (C) Copyright 2012
  6. * Pavel Herrmann <morpheus.ibis@gmail.com>
  7. */
  8. #define LOG_CATEGORY LOGC_DM
  9. #include <common.h>
  10. #include <dm.h>
  11. #include <errno.h>
  12. #include <log.h>
  13. #include <malloc.h>
  14. #include <asm/global_data.h>
  15. #include <dm/device.h>
  16. #include <dm/device-internal.h>
  17. #include <dm/lists.h>
  18. #include <dm/uclass.h>
  19. #include <dm/uclass-internal.h>
  20. #include <dm/util.h>
  21. DECLARE_GLOBAL_DATA_PTR;
  22. struct uclass *uclass_find(enum uclass_id key)
  23. {
  24. struct uclass *uc;
  25. if (!gd->dm_root)
  26. return NULL;
  27. /*
  28. * TODO(sjg@chromium.org): Optimise this, perhaps moving the found
  29. * node to the start of the list, or creating a linear array mapping
  30. * id to node.
  31. */
  32. list_for_each_entry(uc, gd->uclass_root, sibling_node) {
  33. if (uc->uc_drv->id == key)
  34. return uc;
  35. }
  36. return NULL;
  37. }
  38. /**
  39. * uclass_add() - Create new uclass in list
  40. * @id: Id number to create
  41. * @ucp: Returns pointer to uclass, or NULL on error
  42. * Return: 0 on success, -ve on error
  43. *
  44. * The new uclass is added to the list. There must be only one uclass for
  45. * each id.
  46. */
  47. static int uclass_add(enum uclass_id id, struct uclass **ucp)
  48. {
  49. struct uclass_driver *uc_drv;
  50. struct uclass *uc;
  51. int ret;
  52. *ucp = NULL;
  53. uc_drv = lists_uclass_lookup(id);
  54. if (!uc_drv) {
  55. debug("Cannot find uclass for id %d: please add the UCLASS_DRIVER() declaration for this UCLASS_... id\n",
  56. id);
  57. /*
  58. * Use a strange error to make this case easier to find. When
  59. * a uclass is not available it can prevent driver model from
  60. * starting up and this failure is otherwise hard to debug.
  61. */
  62. return -EPFNOSUPPORT;
  63. }
  64. uc = calloc(1, sizeof(*uc));
  65. if (!uc)
  66. return -ENOMEM;
  67. if (uc_drv->priv_auto) {
  68. void *ptr;
  69. ptr = calloc(1, uc_drv->priv_auto);
  70. if (!ptr) {
  71. ret = -ENOMEM;
  72. goto fail_mem;
  73. }
  74. uclass_set_priv(uc, ptr);
  75. }
  76. uc->uc_drv = uc_drv;
  77. INIT_LIST_HEAD(&uc->sibling_node);
  78. INIT_LIST_HEAD(&uc->dev_head);
  79. list_add(&uc->sibling_node, DM_UCLASS_ROOT_NON_CONST);
  80. if (uc_drv->init) {
  81. ret = uc_drv->init(uc);
  82. if (ret)
  83. goto fail;
  84. }
  85. *ucp = uc;
  86. return 0;
  87. fail:
  88. if (uc_drv->priv_auto) {
  89. free(uclass_get_priv(uc));
  90. uclass_set_priv(uc, NULL);
  91. }
  92. list_del(&uc->sibling_node);
  93. fail_mem:
  94. free(uc);
  95. return ret;
  96. }
  97. int uclass_destroy(struct uclass *uc)
  98. {
  99. struct uclass_driver *uc_drv;
  100. struct udevice *dev;
  101. int ret;
  102. /*
  103. * We cannot use list_for_each_entry_safe() here. If a device in this
  104. * uclass has a child device also in this uclass, it will be also be
  105. * unbound (by the recursion in the call to device_unbind() below).
  106. * We can loop until the list is empty.
  107. */
  108. while (!list_empty(&uc->dev_head)) {
  109. dev = list_first_entry(&uc->dev_head, struct udevice,
  110. uclass_node);
  111. ret = device_remove(dev, DM_REMOVE_NORMAL | DM_REMOVE_NO_PD);
  112. if (ret)
  113. return log_msg_ret("remove", ret);
  114. ret = device_unbind(dev);
  115. if (ret)
  116. return log_msg_ret("unbind", ret);
  117. }
  118. uc_drv = uc->uc_drv;
  119. if (uc_drv->destroy)
  120. uc_drv->destroy(uc);
  121. list_del(&uc->sibling_node);
  122. if (uc_drv->priv_auto)
  123. free(uclass_get_priv(uc));
  124. free(uc);
  125. return 0;
  126. }
  127. int uclass_get(enum uclass_id id, struct uclass **ucp)
  128. {
  129. struct uclass *uc;
  130. /* Immediately fail if driver model is not set up */
  131. if (!gd->uclass_root)
  132. return -EDEADLK;
  133. *ucp = NULL;
  134. uc = uclass_find(id);
  135. if (!uc) {
  136. if (CONFIG_IS_ENABLED(OF_PLATDATA_INST))
  137. return -ENOENT;
  138. return uclass_add(id, ucp);
  139. }
  140. *ucp = uc;
  141. return 0;
  142. }
  143. const char *uclass_get_name(enum uclass_id id)
  144. {
  145. struct uclass *uc;
  146. if (uclass_get(id, &uc))
  147. return NULL;
  148. return uc->uc_drv->name;
  149. }
  150. void *uclass_get_priv(const struct uclass *uc)
  151. {
  152. return uc->priv_;
  153. }
  154. void uclass_set_priv(struct uclass *uc, void *priv)
  155. {
  156. uc->priv_ = priv;
  157. }
  158. enum uclass_id uclass_get_by_namelen(const char *name, int len)
  159. {
  160. int i;
  161. for (i = 0; i < UCLASS_COUNT; i++) {
  162. struct uclass_driver *uc_drv = lists_uclass_lookup(i);
  163. if (uc_drv && !strncmp(uc_drv->name, name, len) &&
  164. strlen(uc_drv->name) == len)
  165. return i;
  166. }
  167. return UCLASS_INVALID;
  168. }
  169. enum uclass_id uclass_get_by_name(const char *name)
  170. {
  171. return uclass_get_by_namelen(name, strlen(name));
  172. }
  173. int dev_get_uclass_index(struct udevice *dev, struct uclass **ucp)
  174. {
  175. struct udevice *iter;
  176. struct uclass *uc = dev->uclass;
  177. int i = 0;
  178. if (list_empty(&uc->dev_head))
  179. return -ENODEV;
  180. uclass_foreach_dev(iter, uc) {
  181. if (iter == dev) {
  182. if (ucp)
  183. *ucp = uc;
  184. return i;
  185. }
  186. i++;
  187. }
  188. return -ENODEV;
  189. }
  190. int uclass_find_device(enum uclass_id id, int index, struct udevice **devp)
  191. {
  192. struct uclass *uc;
  193. struct udevice *dev;
  194. int ret;
  195. *devp = NULL;
  196. ret = uclass_get(id, &uc);
  197. if (ret)
  198. return ret;
  199. if (list_empty(&uc->dev_head))
  200. return -ENODEV;
  201. uclass_foreach_dev(dev, uc) {
  202. if (!index--) {
  203. *devp = dev;
  204. return 0;
  205. }
  206. }
  207. return -ENODEV;
  208. }
  209. int uclass_find_first_device(enum uclass_id id, struct udevice **devp)
  210. {
  211. struct uclass *uc;
  212. int ret;
  213. *devp = NULL;
  214. ret = uclass_get(id, &uc);
  215. if (ret)
  216. return ret;
  217. if (list_empty(&uc->dev_head))
  218. return 0;
  219. *devp = list_first_entry(&uc->dev_head, struct udevice, uclass_node);
  220. return 0;
  221. }
  222. int uclass_find_next_device(struct udevice **devp)
  223. {
  224. struct udevice *dev = *devp;
  225. *devp = NULL;
  226. if (list_is_last(&dev->uclass_node, &dev->uclass->dev_head))
  227. return 0;
  228. *devp = list_entry(dev->uclass_node.next, struct udevice, uclass_node);
  229. return 0;
  230. }
  231. int uclass_find_device_by_namelen(enum uclass_id id, const char *name, int len,
  232. struct udevice **devp)
  233. {
  234. struct uclass *uc;
  235. struct udevice *dev;
  236. int ret;
  237. *devp = NULL;
  238. if (!name)
  239. return -EINVAL;
  240. ret = uclass_get(id, &uc);
  241. if (ret)
  242. return ret;
  243. uclass_foreach_dev(dev, uc) {
  244. if (!strncmp(dev->name, name, len) &&
  245. strlen(dev->name) == len) {
  246. *devp = dev;
  247. return 0;
  248. }
  249. }
  250. return -ENODEV;
  251. }
  252. int uclass_find_device_by_name(enum uclass_id id, const char *name,
  253. struct udevice **devp)
  254. {
  255. return uclass_find_device_by_namelen(id, name, strlen(name), devp);
  256. }
  257. int uclass_find_next_free_seq(struct uclass *uc)
  258. {
  259. struct udevice *dev;
  260. int max = -1;
  261. /* If using aliases, start with the highest alias value */
  262. if (CONFIG_IS_ENABLED(DM_SEQ_ALIAS) &&
  263. (uc->uc_drv->flags & DM_UC_FLAG_SEQ_ALIAS))
  264. max = dev_read_alias_highest_id(uc->uc_drv->name);
  265. /* Avoid conflict with existing devices */
  266. list_for_each_entry(dev, &uc->dev_head, uclass_node) {
  267. if (dev->seq_ > max)
  268. max = dev->seq_;
  269. }
  270. /*
  271. * At this point, max will be -1 if there are no existing aliases or
  272. * devices
  273. */
  274. return max + 1;
  275. }
  276. int uclass_find_device_by_seq(enum uclass_id id, int seq, struct udevice **devp)
  277. {
  278. struct uclass *uc;
  279. struct udevice *dev;
  280. int ret;
  281. *devp = NULL;
  282. log_debug("%d\n", seq);
  283. if (seq == -1)
  284. return -ENODEV;
  285. ret = uclass_get(id, &uc);
  286. if (ret)
  287. return ret;
  288. uclass_foreach_dev(dev, uc) {
  289. log_debug(" - %d '%s'\n", dev->seq_, dev->name);
  290. if (dev->seq_ == seq) {
  291. *devp = dev;
  292. log_debug(" - found\n");
  293. return 0;
  294. }
  295. }
  296. log_debug(" - not found\n");
  297. return -ENODEV;
  298. }
  299. int uclass_find_device_by_of_offset(enum uclass_id id, int node,
  300. struct udevice **devp)
  301. {
  302. struct uclass *uc;
  303. struct udevice *dev;
  304. int ret;
  305. *devp = NULL;
  306. if (node < 0)
  307. return -ENODEV;
  308. ret = uclass_get(id, &uc);
  309. if (ret)
  310. return ret;
  311. uclass_foreach_dev(dev, uc) {
  312. if (dev_of_offset(dev) == node) {
  313. *devp = dev;
  314. return 0;
  315. }
  316. }
  317. return -ENODEV;
  318. }
  319. int uclass_find_device_by_ofnode(enum uclass_id id, ofnode node,
  320. struct udevice **devp)
  321. {
  322. struct uclass *uc;
  323. struct udevice *dev;
  324. int ret;
  325. log(LOGC_DM, LOGL_DEBUG, "Looking for %s\n", ofnode_get_name(node));
  326. *devp = NULL;
  327. if (!ofnode_valid(node))
  328. return -ENODEV;
  329. ret = uclass_get(id, &uc);
  330. if (ret)
  331. return ret;
  332. uclass_foreach_dev(dev, uc) {
  333. log(LOGC_DM, LOGL_DEBUG_CONTENT, " - checking %s\n",
  334. dev->name);
  335. if (ofnode_equal(dev_ofnode(dev), node)) {
  336. *devp = dev;
  337. goto done;
  338. }
  339. }
  340. ret = -ENODEV;
  341. done:
  342. log(LOGC_DM, LOGL_DEBUG, " - result for %s: %s (ret=%d)\n",
  343. ofnode_get_name(node), *devp ? (*devp)->name : "(none)", ret);
  344. return ret;
  345. }
  346. #if CONFIG_IS_ENABLED(OF_REAL)
  347. static int uclass_find_device_by_phandle_id(enum uclass_id id,
  348. uint find_phandle,
  349. struct udevice **devp)
  350. {
  351. struct udevice *dev;
  352. struct uclass *uc;
  353. int ret;
  354. ret = uclass_get(id, &uc);
  355. if (ret)
  356. return ret;
  357. uclass_foreach_dev(dev, uc) {
  358. uint phandle;
  359. phandle = dev_read_phandle(dev);
  360. if (phandle == find_phandle) {
  361. *devp = dev;
  362. return 0;
  363. }
  364. }
  365. return -ENODEV;
  366. }
  367. int uclass_find_device_by_phandle(enum uclass_id id, struct udevice *parent,
  368. const char *name, struct udevice **devp)
  369. {
  370. int find_phandle;
  371. *devp = NULL;
  372. find_phandle = dev_read_u32_default(parent, name, -1);
  373. if (find_phandle <= 0)
  374. return -ENOENT;
  375. return uclass_find_device_by_phandle_id(id, find_phandle, devp);
  376. }
  377. #endif
  378. int uclass_get_device_by_driver(enum uclass_id id,
  379. const struct driver *find_drv,
  380. struct udevice **devp)
  381. {
  382. struct udevice *dev;
  383. struct uclass *uc;
  384. int ret;
  385. ret = uclass_get(id, &uc);
  386. if (ret)
  387. return ret;
  388. uclass_foreach_dev(dev, uc) {
  389. if (dev->driver == find_drv)
  390. return uclass_get_device_tail(dev, 0, devp);
  391. }
  392. return -ENODEV;
  393. }
  394. int uclass_get_device_tail(struct udevice *dev, int ret, struct udevice **devp)
  395. {
  396. if (ret)
  397. return ret;
  398. assert(dev);
  399. ret = device_probe(dev);
  400. if (ret)
  401. return ret;
  402. *devp = dev;
  403. return 0;
  404. }
  405. int uclass_get_device(enum uclass_id id, int index, struct udevice **devp)
  406. {
  407. struct udevice *dev;
  408. int ret;
  409. *devp = NULL;
  410. ret = uclass_find_device(id, index, &dev);
  411. return uclass_get_device_tail(dev, ret, devp);
  412. }
  413. int uclass_get_device_by_name(enum uclass_id id, const char *name,
  414. struct udevice **devp)
  415. {
  416. struct udevice *dev;
  417. int ret;
  418. *devp = NULL;
  419. ret = uclass_find_device_by_name(id, name, &dev);
  420. return uclass_get_device_tail(dev, ret, devp);
  421. }
  422. int uclass_get_device_by_seq(enum uclass_id id, int seq, struct udevice **devp)
  423. {
  424. struct udevice *dev;
  425. int ret;
  426. *devp = NULL;
  427. ret = uclass_find_device_by_seq(id, seq, &dev);
  428. return uclass_get_device_tail(dev, ret, devp);
  429. }
  430. int uclass_get_device_by_of_offset(enum uclass_id id, int node,
  431. struct udevice **devp)
  432. {
  433. struct udevice *dev;
  434. int ret;
  435. *devp = NULL;
  436. ret = uclass_find_device_by_of_offset(id, node, &dev);
  437. return uclass_get_device_tail(dev, ret, devp);
  438. }
  439. int uclass_get_device_by_ofnode(enum uclass_id id, ofnode node,
  440. struct udevice **devp)
  441. {
  442. struct udevice *dev;
  443. int ret;
  444. log(LOGC_DM, LOGL_DEBUG, "Looking for %s\n", ofnode_get_name(node));
  445. *devp = NULL;
  446. ret = uclass_find_device_by_ofnode(id, node, &dev);
  447. log(LOGC_DM, LOGL_DEBUG, " - result for %s: %s (ret=%d)\n",
  448. ofnode_get_name(node), dev ? dev->name : "(none)", ret);
  449. return uclass_get_device_tail(dev, ret, devp);
  450. }
  451. #if CONFIG_IS_ENABLED(OF_REAL)
  452. int uclass_get_device_by_of_path(enum uclass_id id, const char *path,
  453. struct udevice **devp)
  454. {
  455. return uclass_get_device_by_ofnode(id, ofnode_path(path), devp);
  456. }
  457. int uclass_get_device_by_phandle_id(enum uclass_id id, uint phandle_id,
  458. struct udevice **devp)
  459. {
  460. struct udevice *dev;
  461. int ret;
  462. *devp = NULL;
  463. ret = uclass_find_device_by_phandle_id(id, phandle_id, &dev);
  464. return uclass_get_device_tail(dev, ret, devp);
  465. }
  466. int uclass_get_device_by_phandle(enum uclass_id id, struct udevice *parent,
  467. const char *name, struct udevice **devp)
  468. {
  469. struct udevice *dev;
  470. int ret;
  471. *devp = NULL;
  472. ret = uclass_find_device_by_phandle(id, parent, name, &dev);
  473. return uclass_get_device_tail(dev, ret, devp);
  474. }
  475. #endif
  476. /*
  477. * Starting from the given device @dev, return pointer to the first device in
  478. * the uclass that probes successfully in @devp.
  479. */
  480. static void _uclass_next_device(struct udevice *dev, struct udevice **devp)
  481. {
  482. for (; dev; uclass_find_next_device(&dev)) {
  483. if (!device_probe(dev))
  484. break;
  485. }
  486. *devp = dev;
  487. }
  488. void uclass_first_device(enum uclass_id id, struct udevice **devp)
  489. {
  490. struct udevice *dev;
  491. int ret;
  492. ret = uclass_find_first_device(id, &dev);
  493. _uclass_next_device(dev, devp);
  494. }
  495. void uclass_next_device(struct udevice **devp)
  496. {
  497. struct udevice *dev = *devp;
  498. uclass_find_next_device(&dev);
  499. _uclass_next_device(dev, devp);
  500. }
  501. int uclass_first_device_err(enum uclass_id id, struct udevice **devp)
  502. {
  503. int ret;
  504. ret = uclass_first_device_check(id, devp);
  505. if (ret)
  506. return ret;
  507. else if (!*devp)
  508. return -ENODEV;
  509. return 0;
  510. }
  511. int uclass_next_device_err(struct udevice **devp)
  512. {
  513. int ret;
  514. ret = uclass_next_device_check(devp);
  515. if (ret)
  516. return ret;
  517. else if (!*devp)
  518. return -ENODEV;
  519. return 0;
  520. }
  521. int uclass_first_device_check(enum uclass_id id, struct udevice **devp)
  522. {
  523. int ret;
  524. *devp = NULL;
  525. ret = uclass_find_first_device(id, devp);
  526. if (ret)
  527. return ret;
  528. if (!*devp)
  529. return 0;
  530. return device_probe(*devp);
  531. }
  532. int uclass_next_device_check(struct udevice **devp)
  533. {
  534. int ret;
  535. ret = uclass_find_next_device(devp);
  536. if (ret)
  537. return ret;
  538. if (!*devp)
  539. return 0;
  540. return device_probe(*devp);
  541. }
  542. int uclass_get_count(void)
  543. {
  544. const struct uclass *uc;
  545. int count = 0;
  546. if (gd->dm_root) {
  547. list_for_each_entry(uc, gd->uclass_root, sibling_node)
  548. count++;
  549. }
  550. return count;
  551. }
  552. int uclass_first_device_drvdata(enum uclass_id id, ulong driver_data,
  553. struct udevice **devp)
  554. {
  555. struct udevice *dev;
  556. struct uclass *uc;
  557. uclass_id_foreach_dev(id, dev, uc) {
  558. if (dev_get_driver_data(dev) == driver_data) {
  559. *devp = dev;
  560. return device_probe(dev);
  561. }
  562. }
  563. return -ENODEV;
  564. }
  565. int uclass_bind_device(struct udevice *dev)
  566. {
  567. struct uclass *uc;
  568. int ret;
  569. uc = dev->uclass;
  570. list_add_tail(&dev->uclass_node, &uc->dev_head);
  571. if (dev->parent) {
  572. struct uclass_driver *uc_drv = dev->parent->uclass->uc_drv;
  573. if (uc_drv->child_post_bind) {
  574. ret = uc_drv->child_post_bind(dev);
  575. if (ret)
  576. goto err;
  577. }
  578. }
  579. return 0;
  580. err:
  581. /* There is no need to undo the parent's post_bind call */
  582. list_del(&dev->uclass_node);
  583. return ret;
  584. }
  585. #if CONFIG_IS_ENABLED(DM_DEVICE_REMOVE)
  586. int uclass_pre_unbind_device(struct udevice *dev)
  587. {
  588. struct uclass *uc;
  589. int ret;
  590. uc = dev->uclass;
  591. if (uc->uc_drv->pre_unbind) {
  592. ret = uc->uc_drv->pre_unbind(dev);
  593. if (ret)
  594. return ret;
  595. }
  596. return 0;
  597. }
  598. int uclass_unbind_device(struct udevice *dev)
  599. {
  600. list_del(&dev->uclass_node);
  601. return 0;
  602. }
  603. #endif
  604. int uclass_pre_probe_device(struct udevice *dev)
  605. {
  606. struct uclass_driver *uc_drv;
  607. int ret;
  608. uc_drv = dev->uclass->uc_drv;
  609. if (uc_drv->pre_probe) {
  610. ret = uc_drv->pre_probe(dev);
  611. if (ret)
  612. return ret;
  613. }
  614. if (!dev->parent)
  615. return 0;
  616. uc_drv = dev->parent->uclass->uc_drv;
  617. if (uc_drv->child_pre_probe) {
  618. ret = uc_drv->child_pre_probe(dev);
  619. if (ret)
  620. return ret;
  621. }
  622. return 0;
  623. }
  624. int uclass_post_probe_device(struct udevice *dev)
  625. {
  626. struct uclass_driver *uc_drv;
  627. int ret;
  628. if (dev->parent) {
  629. uc_drv = dev->parent->uclass->uc_drv;
  630. if (uc_drv->child_post_probe) {
  631. ret = uc_drv->child_post_probe(dev);
  632. if (ret)
  633. return ret;
  634. }
  635. }
  636. uc_drv = dev->uclass->uc_drv;
  637. if (uc_drv->post_probe) {
  638. ret = uc_drv->post_probe(dev);
  639. if (ret)
  640. return ret;
  641. }
  642. return 0;
  643. }
  644. #if CONFIG_IS_ENABLED(DM_DEVICE_REMOVE)
  645. int uclass_pre_remove_device(struct udevice *dev)
  646. {
  647. struct uclass *uc;
  648. int ret;
  649. uc = dev->uclass;
  650. if (uc->uc_drv->pre_remove) {
  651. ret = uc->uc_drv->pre_remove(dev);
  652. if (ret)
  653. return ret;
  654. }
  655. return 0;
  656. }
  657. #endif
  658. int uclass_probe_all(enum uclass_id id)
  659. {
  660. struct udevice *dev;
  661. int ret, err;
  662. err = uclass_first_device_check(id, &dev);
  663. /* Scanning uclass to probe all devices */
  664. while (dev) {
  665. ret = uclass_next_device_check(&dev);
  666. if (ret)
  667. err = ret;
  668. }
  669. return err;
  670. }
  671. int uclass_id_count(enum uclass_id id)
  672. {
  673. struct udevice *dev;
  674. struct uclass *uc;
  675. int count = 0;
  676. uclass_id_foreach_dev(id, dev, uc)
  677. count++;
  678. return count;
  679. }
  680. UCLASS_DRIVER(nop) = {
  681. .id = UCLASS_NOP,
  682. .name = "nop",
  683. };