ofnode.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689
  1. /*
  2. * Copyright (c) 2017 Google, Inc
  3. * Written by Simon Glass <sjg@chromium.org>
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <fdtdec.h>
  10. #include <fdt_support.h>
  11. #include <linux/libfdt.h>
  12. #include <dm/of_access.h>
  13. #include <dm/of_addr.h>
  14. #include <dm/ofnode.h>
  15. #include <linux/err.h>
  16. #include <linux/ioport.h>
  17. int ofnode_read_u32(ofnode node, const char *propname, u32 *outp)
  18. {
  19. assert(ofnode_valid(node));
  20. debug("%s: %s: ", __func__, propname);
  21. if (ofnode_is_np(node)) {
  22. return of_read_u32(ofnode_to_np(node), propname, outp);
  23. } else {
  24. const fdt32_t *cell;
  25. int len;
  26. cell = fdt_getprop(gd->fdt_blob, ofnode_to_offset(node),
  27. propname, &len);
  28. if (!cell || len < sizeof(int)) {
  29. debug("(not found)\n");
  30. return -EINVAL;
  31. }
  32. *outp = fdt32_to_cpu(cell[0]);
  33. }
  34. debug("%#x (%d)\n", *outp, *outp);
  35. return 0;
  36. }
  37. int ofnode_read_u32_default(ofnode node, const char *propname, u32 def)
  38. {
  39. assert(ofnode_valid(node));
  40. ofnode_read_u32(node, propname, &def);
  41. return def;
  42. }
  43. int ofnode_read_s32_default(ofnode node, const char *propname, s32 def)
  44. {
  45. assert(ofnode_valid(node));
  46. ofnode_read_u32(node, propname, (u32 *)&def);
  47. return def;
  48. }
  49. bool ofnode_read_bool(ofnode node, const char *propname)
  50. {
  51. const void *prop;
  52. assert(ofnode_valid(node));
  53. debug("%s: %s: ", __func__, propname);
  54. prop = ofnode_get_property(node, propname, NULL);
  55. debug("%s\n", prop ? "true" : "false");
  56. return prop ? true : false;
  57. }
  58. const char *ofnode_read_string(ofnode node, const char *propname)
  59. {
  60. const char *str = NULL;
  61. int len = -1;
  62. assert(ofnode_valid(node));
  63. debug("%s: %s: ", __func__, propname);
  64. if (ofnode_is_np(node)) {
  65. struct property *prop = of_find_property(
  66. ofnode_to_np(node), propname, NULL);
  67. if (prop) {
  68. str = prop->value;
  69. len = prop->length;
  70. }
  71. } else {
  72. str = fdt_getprop(gd->fdt_blob, ofnode_to_offset(node),
  73. propname, &len);
  74. }
  75. if (!str) {
  76. debug("<not found>\n");
  77. return NULL;
  78. }
  79. if (strnlen(str, len) >= len) {
  80. debug("<invalid>\n");
  81. return NULL;
  82. }
  83. debug("%s\n", str);
  84. return str;
  85. }
  86. ofnode ofnode_find_subnode(ofnode node, const char *subnode_name)
  87. {
  88. ofnode subnode;
  89. assert(ofnode_valid(node));
  90. debug("%s: %s: ", __func__, subnode_name);
  91. if (ofnode_is_np(node)) {
  92. const struct device_node *np = ofnode_to_np(node);
  93. for (np = np->child; np; np = np->sibling) {
  94. if (!strcmp(subnode_name, np->name))
  95. break;
  96. }
  97. subnode = np_to_ofnode(np);
  98. } else {
  99. int ooffset = fdt_subnode_offset(gd->fdt_blob,
  100. ofnode_to_offset(node), subnode_name);
  101. subnode = offset_to_ofnode(ooffset);
  102. }
  103. debug("%s\n", ofnode_valid(subnode) ?
  104. ofnode_get_name(subnode) : "<none>");
  105. return subnode;
  106. }
  107. int ofnode_read_u32_array(ofnode node, const char *propname,
  108. u32 *out_values, size_t sz)
  109. {
  110. assert(ofnode_valid(node));
  111. debug("%s: %s: ", __func__, propname);
  112. if (ofnode_is_np(node)) {
  113. return of_read_u32_array(ofnode_to_np(node), propname,
  114. out_values, sz);
  115. } else {
  116. return fdtdec_get_int_array(gd->fdt_blob,
  117. ofnode_to_offset(node), propname,
  118. out_values, sz);
  119. }
  120. }
  121. ofnode ofnode_first_subnode(ofnode node)
  122. {
  123. assert(ofnode_valid(node));
  124. if (ofnode_is_np(node))
  125. return np_to_ofnode(node.np->child);
  126. return offset_to_ofnode(
  127. fdt_first_subnode(gd->fdt_blob, ofnode_to_offset(node)));
  128. }
  129. ofnode ofnode_next_subnode(ofnode node)
  130. {
  131. assert(ofnode_valid(node));
  132. if (ofnode_is_np(node))
  133. return np_to_ofnode(node.np->sibling);
  134. return offset_to_ofnode(
  135. fdt_next_subnode(gd->fdt_blob, ofnode_to_offset(node)));
  136. }
  137. ofnode ofnode_get_parent(ofnode node)
  138. {
  139. ofnode parent;
  140. assert(ofnode_valid(node));
  141. if (ofnode_is_np(node))
  142. parent = np_to_ofnode(of_get_parent(ofnode_to_np(node)));
  143. else
  144. parent.of_offset = fdt_parent_offset(gd->fdt_blob,
  145. ofnode_to_offset(node));
  146. return parent;
  147. }
  148. const char *ofnode_get_name(ofnode node)
  149. {
  150. assert(ofnode_valid(node));
  151. if (ofnode_is_np(node))
  152. return strrchr(node.np->full_name, '/') + 1;
  153. return fdt_get_name(gd->fdt_blob, ofnode_to_offset(node), NULL);
  154. }
  155. ofnode ofnode_get_by_phandle(uint phandle)
  156. {
  157. ofnode node;
  158. if (of_live_active())
  159. node = np_to_ofnode(of_find_node_by_phandle(phandle));
  160. else
  161. node.of_offset = fdt_node_offset_by_phandle(gd->fdt_blob,
  162. phandle);
  163. return node;
  164. }
  165. int ofnode_read_size(ofnode node, const char *propname)
  166. {
  167. int len;
  168. if (ofnode_is_np(node)) {
  169. struct property *prop = of_find_property(
  170. ofnode_to_np(node), propname, NULL);
  171. if (prop)
  172. return prop->length;
  173. } else {
  174. if (fdt_getprop(gd->fdt_blob, ofnode_to_offset(node), propname,
  175. &len))
  176. return len;
  177. }
  178. return -EINVAL;
  179. }
  180. fdt_addr_t ofnode_get_addr_index(ofnode node, int index)
  181. {
  182. if (ofnode_is_np(node)) {
  183. const __be32 *prop_val;
  184. uint flags;
  185. u64 size;
  186. int na;
  187. int ns;
  188. prop_val = of_get_address(ofnode_to_np(node), index, &size,
  189. &flags);
  190. if (!prop_val)
  191. return FDT_ADDR_T_NONE;
  192. ns = of_n_size_cells(ofnode_to_np(node));
  193. if (IS_ENABLED(CONFIG_OF_TRANSLATE) && ns > 0) {
  194. return of_translate_address(ofnode_to_np(node), prop_val);
  195. } else {
  196. na = of_n_addr_cells(ofnode_to_np(node));
  197. return of_read_number(prop_val, na);
  198. }
  199. } else {
  200. return fdt_get_base_address(gd->fdt_blob,
  201. ofnode_to_offset(node));
  202. }
  203. return FDT_ADDR_T_NONE;
  204. }
  205. fdt_addr_t ofnode_get_addr(ofnode node)
  206. {
  207. return ofnode_get_addr_index(node, 0);
  208. }
  209. int ofnode_stringlist_search(ofnode node, const char *property,
  210. const char *string)
  211. {
  212. if (ofnode_is_np(node)) {
  213. return of_property_match_string(ofnode_to_np(node),
  214. property, string);
  215. } else {
  216. int ret;
  217. ret = fdt_stringlist_search(gd->fdt_blob,
  218. ofnode_to_offset(node), property,
  219. string);
  220. if (ret == -FDT_ERR_NOTFOUND)
  221. return -ENODATA;
  222. else if (ret < 0)
  223. return -EINVAL;
  224. return ret;
  225. }
  226. }
  227. int ofnode_read_string_index(ofnode node, const char *property, int index,
  228. const char **outp)
  229. {
  230. if (ofnode_is_np(node)) {
  231. return of_property_read_string_index(ofnode_to_np(node),
  232. property, index, outp);
  233. } else {
  234. int len;
  235. *outp = fdt_stringlist_get(gd->fdt_blob, ofnode_to_offset(node),
  236. property, index, &len);
  237. if (len < 0)
  238. return -EINVAL;
  239. return 0;
  240. }
  241. }
  242. int ofnode_read_string_count(ofnode node, const char *property)
  243. {
  244. if (ofnode_is_np(node)) {
  245. return of_property_count_strings(ofnode_to_np(node), property);
  246. } else {
  247. return fdt_stringlist_count(gd->fdt_blob,
  248. ofnode_to_offset(node), property);
  249. }
  250. }
  251. static void ofnode_from_fdtdec_phandle_args(struct fdtdec_phandle_args *in,
  252. struct ofnode_phandle_args *out)
  253. {
  254. assert(OF_MAX_PHANDLE_ARGS == MAX_PHANDLE_ARGS);
  255. out->node = offset_to_ofnode(in->node);
  256. out->args_count = in->args_count;
  257. memcpy(out->args, in->args, sizeof(out->args));
  258. }
  259. static void ofnode_from_of_phandle_args(struct of_phandle_args *in,
  260. struct ofnode_phandle_args *out)
  261. {
  262. assert(OF_MAX_PHANDLE_ARGS == MAX_PHANDLE_ARGS);
  263. out->node = np_to_ofnode(in->np);
  264. out->args_count = in->args_count;
  265. memcpy(out->args, in->args, sizeof(out->args));
  266. }
  267. int ofnode_parse_phandle_with_args(ofnode node, const char *list_name,
  268. const char *cells_name, int cell_count,
  269. int index,
  270. struct ofnode_phandle_args *out_args)
  271. {
  272. if (ofnode_is_np(node)) {
  273. struct of_phandle_args args;
  274. int ret;
  275. ret = of_parse_phandle_with_args(ofnode_to_np(node),
  276. list_name, cells_name, index,
  277. &args);
  278. if (ret)
  279. return ret;
  280. ofnode_from_of_phandle_args(&args, out_args);
  281. } else {
  282. struct fdtdec_phandle_args args;
  283. int ret;
  284. ret = fdtdec_parse_phandle_with_args(gd->fdt_blob,
  285. ofnode_to_offset(node),
  286. list_name, cells_name,
  287. cell_count, index, &args);
  288. if (ret)
  289. return ret;
  290. ofnode_from_fdtdec_phandle_args(&args, out_args);
  291. }
  292. return 0;
  293. }
  294. int ofnode_count_phandle_with_args(ofnode node, const char *list_name,
  295. const char *cells_name)
  296. {
  297. if (ofnode_is_np(node))
  298. return of_count_phandle_with_args(ofnode_to_np(node),
  299. list_name, cells_name);
  300. else
  301. return fdtdec_parse_phandle_with_args(gd->fdt_blob,
  302. ofnode_to_offset(node), list_name, cells_name,
  303. 0, -1, NULL);
  304. }
  305. ofnode ofnode_path(const char *path)
  306. {
  307. if (of_live_active())
  308. return np_to_ofnode(of_find_node_by_path(path));
  309. else
  310. return offset_to_ofnode(fdt_path_offset(gd->fdt_blob, path));
  311. }
  312. const char *ofnode_get_chosen_prop(const char *name)
  313. {
  314. ofnode chosen_node;
  315. chosen_node = ofnode_path("/chosen");
  316. return ofnode_read_string(chosen_node, name);
  317. }
  318. ofnode ofnode_get_chosen_node(const char *name)
  319. {
  320. const char *prop;
  321. prop = ofnode_get_chosen_prop(name);
  322. if (!prop)
  323. return ofnode_null();
  324. return ofnode_path(prop);
  325. }
  326. static int decode_timing_property(ofnode node, const char *name,
  327. struct timing_entry *result)
  328. {
  329. int length, ret = 0;
  330. length = ofnode_read_size(node, name);
  331. if (length < 0) {
  332. debug("%s: could not find property %s\n",
  333. ofnode_get_name(node), name);
  334. return length;
  335. }
  336. if (length == sizeof(u32)) {
  337. result->typ = ofnode_read_u32_default(node, name, 0);
  338. result->min = result->typ;
  339. result->max = result->typ;
  340. } else {
  341. ret = ofnode_read_u32_array(node, name, &result->min, 3);
  342. }
  343. return ret;
  344. }
  345. int ofnode_decode_display_timing(ofnode parent, int index,
  346. struct display_timing *dt)
  347. {
  348. int i;
  349. ofnode timings, node;
  350. u32 val = 0;
  351. int ret = 0;
  352. timings = ofnode_find_subnode(parent, "display-timings");
  353. if (!ofnode_valid(timings))
  354. return -EINVAL;
  355. i = 0;
  356. ofnode_for_each_subnode(node, timings) {
  357. if (i++ == index)
  358. break;
  359. }
  360. if (!ofnode_valid(node))
  361. return -EINVAL;
  362. memset(dt, 0, sizeof(*dt));
  363. ret |= decode_timing_property(node, "hback-porch", &dt->hback_porch);
  364. ret |= decode_timing_property(node, "hfront-porch", &dt->hfront_porch);
  365. ret |= decode_timing_property(node, "hactive", &dt->hactive);
  366. ret |= decode_timing_property(node, "hsync-len", &dt->hsync_len);
  367. ret |= decode_timing_property(node, "vback-porch", &dt->vback_porch);
  368. ret |= decode_timing_property(node, "vfront-porch", &dt->vfront_porch);
  369. ret |= decode_timing_property(node, "vactive", &dt->vactive);
  370. ret |= decode_timing_property(node, "vsync-len", &dt->vsync_len);
  371. ret |= decode_timing_property(node, "clock-frequency", &dt->pixelclock);
  372. dt->flags = 0;
  373. val = ofnode_read_u32_default(node, "vsync-active", -1);
  374. if (val != -1) {
  375. dt->flags |= val ? DISPLAY_FLAGS_VSYNC_HIGH :
  376. DISPLAY_FLAGS_VSYNC_LOW;
  377. }
  378. val = ofnode_read_u32_default(node, "hsync-active", -1);
  379. if (val != -1) {
  380. dt->flags |= val ? DISPLAY_FLAGS_HSYNC_HIGH :
  381. DISPLAY_FLAGS_HSYNC_LOW;
  382. }
  383. val = ofnode_read_u32_default(node, "de-active", -1);
  384. if (val != -1) {
  385. dt->flags |= val ? DISPLAY_FLAGS_DE_HIGH :
  386. DISPLAY_FLAGS_DE_LOW;
  387. }
  388. val = ofnode_read_u32_default(node, "pixelclk-active", -1);
  389. if (val != -1) {
  390. dt->flags |= val ? DISPLAY_FLAGS_PIXDATA_POSEDGE :
  391. DISPLAY_FLAGS_PIXDATA_NEGEDGE;
  392. }
  393. if (ofnode_read_bool(node, "interlaced"))
  394. dt->flags |= DISPLAY_FLAGS_INTERLACED;
  395. if (ofnode_read_bool(node, "doublescan"))
  396. dt->flags |= DISPLAY_FLAGS_DOUBLESCAN;
  397. if (ofnode_read_bool(node, "doubleclk"))
  398. dt->flags |= DISPLAY_FLAGS_DOUBLECLK;
  399. return ret;
  400. }
  401. const void *ofnode_get_property(ofnode node, const char *propname, int *lenp)
  402. {
  403. if (ofnode_is_np(node))
  404. return of_get_property(ofnode_to_np(node), propname, lenp);
  405. else
  406. return fdt_getprop(gd->fdt_blob, ofnode_to_offset(node),
  407. propname, lenp);
  408. }
  409. bool ofnode_is_available(ofnode node)
  410. {
  411. if (ofnode_is_np(node))
  412. return of_device_is_available(ofnode_to_np(node));
  413. else
  414. return fdtdec_get_is_enabled(gd->fdt_blob,
  415. ofnode_to_offset(node));
  416. }
  417. fdt_addr_t ofnode_get_addr_size(ofnode node, const char *property,
  418. fdt_size_t *sizep)
  419. {
  420. if (ofnode_is_np(node)) {
  421. int na, ns;
  422. int psize;
  423. const struct device_node *np = ofnode_to_np(node);
  424. const __be32 *prop = of_get_property(np, property, &psize);
  425. if (!prop)
  426. return FDT_ADDR_T_NONE;
  427. na = of_n_addr_cells(np);
  428. ns = of_n_addr_cells(np);
  429. *sizep = of_read_number(prop + na, ns);
  430. return of_read_number(prop, na);
  431. } else {
  432. return fdtdec_get_addr_size(gd->fdt_blob,
  433. ofnode_to_offset(node), property,
  434. sizep);
  435. }
  436. }
  437. const uint8_t *ofnode_read_u8_array_ptr(ofnode node, const char *propname,
  438. size_t sz)
  439. {
  440. if (ofnode_is_np(node)) {
  441. const struct device_node *np = ofnode_to_np(node);
  442. int psize;
  443. const __be32 *prop = of_get_property(np, propname, &psize);
  444. if (!prop || sz != psize)
  445. return NULL;
  446. return (uint8_t *)prop;
  447. } else {
  448. return fdtdec_locate_byte_array(gd->fdt_blob,
  449. ofnode_to_offset(node), propname, sz);
  450. }
  451. }
  452. int ofnode_read_pci_addr(ofnode node, enum fdt_pci_space type,
  453. const char *propname, struct fdt_pci_addr *addr)
  454. {
  455. const fdt32_t *cell;
  456. int len;
  457. int ret = -ENOENT;
  458. debug("%s: %s: ", __func__, propname);
  459. /*
  460. * If we follow the pci bus bindings strictly, we should check
  461. * the value of the node's parent node's #address-cells and
  462. * #size-cells. They need to be 3 and 2 accordingly. However,
  463. * for simplicity we skip the check here.
  464. */
  465. cell = ofnode_get_property(node, propname, &len);
  466. if (!cell)
  467. goto fail;
  468. if ((len % FDT_PCI_REG_SIZE) == 0) {
  469. int num = len / FDT_PCI_REG_SIZE;
  470. int i;
  471. for (i = 0; i < num; i++) {
  472. debug("pci address #%d: %08lx %08lx %08lx\n", i,
  473. (ulong)fdt32_to_cpu(cell[0]),
  474. (ulong)fdt32_to_cpu(cell[1]),
  475. (ulong)fdt32_to_cpu(cell[2]));
  476. if ((fdt32_to_cpu(*cell) & type) == type) {
  477. addr->phys_hi = fdt32_to_cpu(cell[0]);
  478. addr->phys_mid = fdt32_to_cpu(cell[1]);
  479. addr->phys_lo = fdt32_to_cpu(cell[1]);
  480. break;
  481. }
  482. cell += (FDT_PCI_ADDR_CELLS +
  483. FDT_PCI_SIZE_CELLS);
  484. }
  485. if (i == num) {
  486. ret = -ENXIO;
  487. goto fail;
  488. }
  489. return 0;
  490. }
  491. ret = -EINVAL;
  492. fail:
  493. debug("(not found)\n");
  494. return ret;
  495. }
  496. int ofnode_read_addr_cells(ofnode node)
  497. {
  498. if (ofnode_is_np(node))
  499. return of_n_addr_cells(ofnode_to_np(node));
  500. else /* NOTE: this call should walk up the parent stack */
  501. return fdt_address_cells(gd->fdt_blob, ofnode_to_offset(node));
  502. }
  503. int ofnode_read_size_cells(ofnode node)
  504. {
  505. if (ofnode_is_np(node))
  506. return of_n_size_cells(ofnode_to_np(node));
  507. else /* NOTE: this call should walk up the parent stack */
  508. return fdt_size_cells(gd->fdt_blob, ofnode_to_offset(node));
  509. }
  510. int ofnode_read_simple_addr_cells(ofnode node)
  511. {
  512. if (ofnode_is_np(node))
  513. return of_simple_addr_cells(ofnode_to_np(node));
  514. else
  515. return fdt_address_cells(gd->fdt_blob, ofnode_to_offset(node));
  516. }
  517. int ofnode_read_simple_size_cells(ofnode node)
  518. {
  519. if (ofnode_is_np(node))
  520. return of_simple_size_cells(ofnode_to_np(node));
  521. else
  522. return fdt_size_cells(gd->fdt_blob, ofnode_to_offset(node));
  523. }
  524. bool ofnode_pre_reloc(ofnode node)
  525. {
  526. if (ofnode_read_bool(node, "u-boot,dm-pre-reloc"))
  527. return true;
  528. #ifdef CONFIG_TPL_BUILD
  529. if (ofnode_read_bool(node, "u-boot,dm-tpl"))
  530. return true;
  531. #elif defined(CONFIG_SPL_BUILD)
  532. if (ofnode_read_bool(node, "u-boot,dm-spl"))
  533. return true;
  534. #else
  535. /*
  536. * In regular builds individual spl and tpl handling both
  537. * count as handled pre-relocation for later second init.
  538. */
  539. if (ofnode_read_bool(node, "u-boot,dm-spl") ||
  540. ofnode_read_bool(node, "u-boot,dm-tpl"))
  541. return true;
  542. #endif
  543. return false;
  544. }
  545. int ofnode_read_resource(ofnode node, uint index, struct resource *res)
  546. {
  547. if (ofnode_is_np(node)) {
  548. return of_address_to_resource(ofnode_to_np(node), index, res);
  549. } else {
  550. struct fdt_resource fres;
  551. int ret;
  552. ret = fdt_get_resource(gd->fdt_blob, ofnode_to_offset(node),
  553. "reg", index, &fres);
  554. if (ret < 0)
  555. return -EINVAL;
  556. memset(res, '\0', sizeof(*res));
  557. res->start = fres.start;
  558. res->end = fres.end;
  559. return 0;
  560. }
  561. }
  562. int ofnode_read_resource_byname(ofnode node, const char *name,
  563. struct resource *res)
  564. {
  565. int index;
  566. index = ofnode_stringlist_search(node, "reg-names", name);
  567. if (index < 0)
  568. return index;
  569. return ofnode_read_resource(node, index, res);
  570. }
  571. u64 ofnode_translate_address(ofnode node, const fdt32_t *in_addr)
  572. {
  573. if (ofnode_is_np(node))
  574. return of_translate_address(ofnode_to_np(node), in_addr);
  575. else
  576. return fdt_translate_address(gd->fdt_blob, ofnode_to_offset(node), in_addr);
  577. }