of_platdata.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. // SPDX-License-Identifier: GPL-2.0+
  2. #include <common.h>
  3. #include <clk.h>
  4. #include <dm.h>
  5. #include <dt-structs.h>
  6. #include <irq.h>
  7. #include <dm/test.h>
  8. #include <test/test.h>
  9. #include <test/ut.h>
  10. #include <asm-generic/gpio.h>
  11. #include <asm/global_data.h>
  12. /* Test that we can find a device using of-platdata */
  13. static int dm_test_of_plat_base(struct unit_test_state *uts)
  14. {
  15. struct udevice *dev;
  16. ut_assertok(uclass_first_device_err(UCLASS_SERIAL, &dev));
  17. ut_asserteq_str("sandbox_serial", dev->name);
  18. return 0;
  19. }
  20. DM_TEST(dm_test_of_plat_base, UT_TESTF_SCAN_PDATA);
  21. /* Test that we can read properties from a device */
  22. static int dm_test_of_plat_props(struct unit_test_state *uts)
  23. {
  24. struct dtd_sandbox_spl_test *plat;
  25. struct udevice *dev;
  26. int i;
  27. ut_assertok(uclass_get_device_by_name(UCLASS_MISC, "sandbox_spl_test",
  28. &dev));
  29. plat = dev_get_plat(dev);
  30. ut_assert(plat->boolval);
  31. ut_asserteq(1, plat->intval);
  32. ut_asserteq(3, ARRAY_SIZE(plat->intarray));
  33. ut_asserteq(2, plat->intarray[0]);
  34. ut_asserteq(3, plat->intarray[1]);
  35. ut_asserteq(4, plat->intarray[2]);
  36. ut_asserteq(5, plat->byteval);
  37. ut_asserteq(1, ARRAY_SIZE(plat->maybe_empty_int));
  38. ut_asserteq(0, plat->maybe_empty_int[0]);
  39. ut_asserteq(3, ARRAY_SIZE(plat->bytearray));
  40. ut_asserteq(6, plat->bytearray[0]);
  41. ut_asserteq(0, plat->bytearray[1]);
  42. ut_asserteq(0, plat->bytearray[2]);
  43. ut_asserteq(9, ARRAY_SIZE(plat->longbytearray));
  44. for (i = 0; i < ARRAY_SIZE(plat->longbytearray); i++)
  45. ut_asserteq(9 + i, plat->longbytearray[i]);
  46. ut_asserteq_str("message", plat->stringval);
  47. ut_asserteq(3, ARRAY_SIZE(plat->stringarray));
  48. ut_asserteq_str("multi-word", plat->stringarray[0]);
  49. ut_asserteq_str("message", plat->stringarray[1]);
  50. ut_asserteq_str("", plat->stringarray[2]);
  51. ut_assertok(uclass_next_device_err(&dev));
  52. plat = dev_get_plat(dev);
  53. ut_assert(!plat->boolval);
  54. ut_asserteq(3, plat->intval);
  55. ut_asserteq(5, plat->intarray[0]);
  56. ut_asserteq(0, plat->intarray[1]);
  57. ut_asserteq(0, plat->intarray[2]);
  58. ut_asserteq(8, plat->byteval);
  59. ut_asserteq(3, ARRAY_SIZE(plat->bytearray));
  60. ut_asserteq(1, plat->bytearray[0]);
  61. ut_asserteq(0x23, plat->bytearray[1]);
  62. ut_asserteq(0x34, plat->bytearray[2]);
  63. for (i = 0; i < ARRAY_SIZE(plat->longbytearray); i++)
  64. ut_asserteq(i < 4 ? 9 + i : 0, plat->longbytearray[i]);
  65. ut_asserteq_str("message2", plat->stringval);
  66. ut_asserteq_str("another", plat->stringarray[0]);
  67. ut_asserteq_str("multi-word", plat->stringarray[1]);
  68. ut_asserteq_str("message", plat->stringarray[2]);
  69. ut_assertok(uclass_next_device_err(&dev));
  70. plat = dev_get_plat(dev);
  71. ut_assert(!plat->boolval);
  72. ut_asserteq_str("one", plat->stringarray[0]);
  73. ut_asserteq_str("", plat->stringarray[1]);
  74. ut_asserteq_str("", plat->stringarray[2]);
  75. ut_asserteq(1, plat->maybe_empty_int[0]);
  76. ut_assertok(uclass_next_device_err(&dev));
  77. plat = dev_get_plat(dev);
  78. ut_assert(!plat->boolval);
  79. ut_asserteq_str("spl", plat->stringarray[0]);
  80. ut_asserteq(-ENODEV, uclass_next_device_err(&dev));
  81. return 0;
  82. }
  83. DM_TEST(dm_test_of_plat_props, UT_TESTF_SCAN_PDATA);
  84. /*
  85. * find_driver_info - recursively find the driver_info for a device
  86. *
  87. * This sets found[idx] to true when it finds the driver_info record for a
  88. * device, where idx is the index in the driver_info linker list.
  89. *
  90. * @uts: Test state
  91. * @parent: Parent to search
  92. * @found: bool array to update
  93. * Return: 0 if OK, non-zero on error
  94. */
  95. static int find_driver_info(struct unit_test_state *uts, struct udevice *parent,
  96. bool found[])
  97. {
  98. struct udevice *dev;
  99. /* If not the root device, find the entry that caused it to be bound */
  100. if (parent->parent) {
  101. const int n_ents =
  102. ll_entry_count(struct driver_info, driver_info);
  103. int idx = -1;
  104. int i;
  105. for (i = 0; i < n_ents; i++) {
  106. const struct driver_rt *drt = gd_dm_driver_rt() + i;
  107. if (drt->dev == parent) {
  108. idx = i;
  109. found[idx] = true;
  110. break;
  111. }
  112. }
  113. ut_assert(idx != -1);
  114. }
  115. device_foreach_child(dev, parent) {
  116. int ret;
  117. ret = find_driver_info(uts, dev, found);
  118. if (ret < 0)
  119. return ret;
  120. }
  121. return 0;
  122. }
  123. /* Check that every device is recorded in its driver_info struct */
  124. static int dm_test_of_plat_dev(struct unit_test_state *uts)
  125. {
  126. const int n_ents = ll_entry_count(struct driver_info, driver_info);
  127. bool found[n_ents];
  128. uint i;
  129. /* Skip this test if there is no platform data */
  130. if (!CONFIG_IS_ENABLED(OF_PLATDATA_DRIVER_RT))
  131. return -EAGAIN;
  132. /* Record the indexes that are found */
  133. memset(found, '\0', sizeof(found));
  134. ut_assertok(find_driver_info(uts, gd->dm_root, found));
  135. /* Make sure that the driver entries without devices have no ->dev */
  136. for (i = 0; i < n_ents; i++) {
  137. const struct driver_rt *drt = gd_dm_driver_rt() + i;
  138. struct udevice *dev;
  139. if (found[i]) {
  140. /* Make sure we can find it */
  141. ut_assertnonnull(drt->dev);
  142. ut_assertok(device_get_by_ofplat_idx(i, &dev));
  143. ut_asserteq_ptr(dev, drt->dev);
  144. } else {
  145. ut_assertnull(drt->dev);
  146. ut_asserteq(-ENOENT, device_get_by_ofplat_idx(i, &dev));
  147. }
  148. }
  149. return 0;
  150. }
  151. DM_TEST(dm_test_of_plat_dev, UT_TESTF_SCAN_PDATA);
  152. /* Test handling of phandles that point to other devices */
  153. static int dm_test_of_plat_phandle(struct unit_test_state *uts)
  154. {
  155. struct dtd_sandbox_clk_test *plat;
  156. struct udevice *dev, *clk;
  157. ut_assertok(uclass_first_device_err(UCLASS_MISC, &dev));
  158. ut_asserteq_str("sandbox_clk_test", dev->name);
  159. plat = dev_get_plat(dev);
  160. ut_assertok(device_get_by_ofplat_idx(plat->clocks[0].idx, &clk));
  161. ut_asserteq_str("sandbox_fixed_clock", clk->name);
  162. ut_assertok(device_get_by_ofplat_idx(plat->clocks[1].idx, &clk));
  163. ut_asserteq_str("sandbox_clk", clk->name);
  164. ut_asserteq(1, plat->clocks[1].arg[0]);
  165. ut_assertok(device_get_by_ofplat_idx(plat->clocks[2].idx, &clk));
  166. ut_asserteq_str("sandbox_clk", clk->name);
  167. ut_asserteq(0, plat->clocks[2].arg[0]);
  168. ut_assertok(device_get_by_ofplat_idx(plat->clocks[3].idx, &clk));
  169. ut_asserteq_str("sandbox_clk", clk->name);
  170. ut_asserteq(3, plat->clocks[3].arg[0]);
  171. ut_assertok(device_get_by_ofplat_idx(plat->clocks[4].idx, &clk));
  172. ut_asserteq_str("sandbox_clk", clk->name);
  173. ut_asserteq(2, plat->clocks[4].arg[0]);
  174. return 0;
  175. }
  176. DM_TEST(dm_test_of_plat_phandle, UT_TESTF_SCAN_PDATA);
  177. #if CONFIG_IS_ENABLED(OF_PLATDATA_PARENT)
  178. /* Test that device parents are correctly set up */
  179. static int dm_test_of_plat_parent(struct unit_test_state *uts)
  180. {
  181. struct udevice *rtc, *i2c;
  182. ut_assertok(uclass_first_device_err(UCLASS_RTC, &rtc));
  183. ut_assertok(uclass_first_device_err(UCLASS_I2C, &i2c));
  184. ut_asserteq_ptr(i2c, dev_get_parent(rtc));
  185. return 0;
  186. }
  187. DM_TEST(dm_test_of_plat_parent, UT_TESTF_SCAN_PDATA);
  188. #endif
  189. /* Test clocks with of-platdata */
  190. static int dm_test_of_plat_clk(struct unit_test_state *uts)
  191. {
  192. struct dtd_sandbox_clk_test *plat;
  193. struct udevice *dev;
  194. struct clk clk;
  195. ut_assertok(uclass_first_device_err(UCLASS_MISC, &dev));
  196. ut_asserteq_str("sandbox_clk_test", dev->name);
  197. plat = dev_get_plat(dev);
  198. ut_assertok(clk_get_by_phandle(dev, &plat->clocks[0], &clk));
  199. ut_asserteq_str("sandbox_fixed_clock", clk.dev->name);
  200. return 0;
  201. }
  202. DM_TEST(dm_test_of_plat_clk, UT_TESTF_SCAN_PDATA);
  203. /* Test irqs with of-platdata */
  204. static int dm_test_of_plat_irq(struct unit_test_state *uts)
  205. {
  206. struct dtd_sandbox_irq_test *plat;
  207. struct udevice *dev;
  208. struct irq irq;
  209. ut_assertok(uclass_get_device_by_name(UCLASS_MISC, "sandbox_irq_test",
  210. &dev));
  211. plat = dev_get_plat(dev);
  212. ut_assertok(irq_get_by_phandle(dev, &plat->interrupts_extended[0],
  213. &irq));
  214. ut_asserteq_str("sandbox_irq", irq.dev->name);
  215. return 0;
  216. }
  217. DM_TEST(dm_test_of_plat_irq, UT_TESTF_SCAN_PDATA);
  218. /* Test GPIOs with of-platdata */
  219. static int dm_test_of_plat_gpio(struct unit_test_state *uts)
  220. {
  221. struct dtd_sandbox_gpio_test *plat;
  222. struct udevice *dev;
  223. struct gpio_desc desc;
  224. ut_assertok(uclass_get_device_by_name(UCLASS_MISC, "sandbox_gpio_test",
  225. &dev));
  226. plat = dev_get_plat(dev);
  227. ut_assertok(gpio_request_by_phandle(dev, &plat->test_gpios[0], &desc,
  228. GPIOD_IS_OUT));
  229. ut_asserteq_str("sandbox_gpio", desc.dev->name);
  230. return 0;
  231. }
  232. DM_TEST(dm_test_of_plat_gpio, UT_TESTF_SCAN_PDATA);