1
0

fdt.c 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194
  1. /*
  2. * Copyright (c) 2006-2024, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2022-08-25 GuEe-GUI first version
  9. */
  10. #include <rthw.h>
  11. #include <rtthread.h>
  12. #include <drivers/ofw_fdt.h>
  13. #include <drivers/ofw_raw.h>
  14. #include <drivers/core/dm.h>
  15. #define DBG_TAG "rtdm.ofw"
  16. #define DBG_LVL DBG_INFO
  17. #include <rtdbg.h>
  18. #include "ofw_internal.h"
  19. struct rt_fdt_earlycon fdt_earlycon rt_section(".bss.noclean.earlycon");
  20. RT_OFW_SYMBOL_TYPE_RANGE(earlycon, struct rt_fdt_earlycon_id, _earlycon_start = {}, _earlycon_end = {});
  21. #ifndef ARCH_INIT_MEMREGION_NR
  22. #define ARCH_INIT_MEMREGION_NR 128
  23. #endif
  24. static rt_region_t _memregion[ARCH_INIT_MEMREGION_NR] rt_section(".bss.noclean.memregion");
  25. static int _memregion_front_idx = 0, _memregion_last_idx = RT_ARRAY_SIZE(_memregion) - 1;
  26. static void *_fdt = RT_NULL;
  27. static rt_phandle _phandle_min;
  28. static rt_phandle _phandle_max;
  29. static rt_size_t _root_size_cells;
  30. static rt_size_t _root_addr_cells;
  31. const char *rt_fdt_node_name(const char *full_name)
  32. {
  33. const char *node_name = strrchr(full_name, '/');
  34. return node_name ? node_name + 1 : full_name;
  35. }
  36. rt_uint64_t rt_fdt_read_number(const fdt32_t *cell, int size)
  37. {
  38. rt_uint64_t val = 0;
  39. for (; size--; ++cell)
  40. {
  41. val = (val << 32) | fdt32_to_cpu(*cell);
  42. }
  43. return val;
  44. }
  45. rt_uint64_t rt_fdt_next_cell(const fdt32_t **cellptr, int size)
  46. {
  47. const fdt32_t *ptr = *cellptr;
  48. *cellptr = ptr + size;
  49. return rt_fdt_read_number(ptr, size);
  50. }
  51. rt_uint64_t rt_fdt_translate_address(void *fdt, int nodeoffset, rt_uint64_t address)
  52. {
  53. rt_uint64_t ret = address;
  54. if (fdt && nodeoffset >= 0)
  55. {
  56. struct
  57. {
  58. rt_uint64_t addr;
  59. rt_size_t size;
  60. int addr_cells;
  61. int size_cells;
  62. } local, cpu;
  63. int parent, length = 0, group_len;
  64. const fdt32_t *ranges = RT_NULL;
  65. parent = fdt_parent_offset(fdt, nodeoffset);
  66. if (parent >= 0)
  67. {
  68. ranges = fdt_getprop(fdt, nodeoffset, "ranges", &length);
  69. }
  70. if (ranges && length > 0)
  71. {
  72. local.addr_cells = fdt_address_cells(fdt, nodeoffset);
  73. local.size_cells = fdt_size_cells(fdt, nodeoffset);
  74. cpu.addr_cells = fdt_io_addr_cells(fdt, nodeoffset);
  75. cpu.size_cells = fdt_io_size_cells(fdt, nodeoffset);
  76. group_len = local.addr_cells + cpu.addr_cells + local.size_cells;
  77. while (length > 0)
  78. {
  79. local.addr = rt_fdt_next_cell(&ranges, local.addr_cells);
  80. cpu.addr = rt_fdt_next_cell(&ranges, cpu.addr_cells);
  81. local.size = rt_fdt_next_cell(&ranges, local.size_cells);
  82. if (local.addr <= address && local.addr + local.size > address)
  83. {
  84. ret += address - cpu.addr;
  85. break;
  86. }
  87. length -= group_len;
  88. }
  89. }
  90. }
  91. return ret;
  92. }
  93. rt_bool_t rt_fdt_device_is_available(void *fdt, int nodeoffset)
  94. {
  95. rt_bool_t ret;
  96. const char *status = fdt_getprop(fdt, nodeoffset, "status", RT_NULL);
  97. if (!status)
  98. {
  99. ret = RT_TRUE;
  100. }
  101. else if (!rt_strcmp(status, "ok") || !rt_strcmp(status, "okay"))
  102. {
  103. ret = RT_TRUE;
  104. }
  105. else
  106. {
  107. ret = RT_FALSE;
  108. }
  109. return ret;
  110. }
  111. rt_err_t rt_fdt_commit_memregion_early(rt_region_t *region, rt_bool_t is_reserved)
  112. {
  113. rt_err_t err = RT_EOK;
  114. if (region && region->name)
  115. {
  116. if (_memregion_front_idx < _memregion_last_idx)
  117. {
  118. int idx;
  119. if (!_memregion_front_idx && _memregion_last_idx == RT_ARRAY_SIZE(_memregion) - 1)
  120. {
  121. for (int i = 0; i < RT_ARRAY_SIZE(_memregion); ++i)
  122. {
  123. _memregion[i].name = RT_NULL;
  124. }
  125. }
  126. idx = is_reserved ? _memregion_last_idx-- : _memregion_front_idx++;
  127. rt_memcpy(&_memregion[idx], region, sizeof(*region));
  128. }
  129. else
  130. {
  131. err = -RT_EEMPTY;
  132. }
  133. }
  134. else
  135. {
  136. err = -RT_EINVAL;
  137. }
  138. return err;
  139. }
  140. rt_err_t rt_fdt_commit_memregion_request(rt_region_t **out_region, rt_size_t *out_nr, rt_bool_t is_reserved)
  141. {
  142. rt_err_t err = RT_EOK;
  143. if (out_region && out_nr)
  144. {
  145. if (is_reserved)
  146. {
  147. *out_region = &_memregion[_memregion_last_idx + 1];
  148. *out_nr = RT_ARRAY_SIZE(_memregion) - 1 - _memregion_last_idx;
  149. }
  150. else
  151. {
  152. *out_region = &_memregion[0];
  153. *out_nr = _memregion_front_idx;
  154. }
  155. if (*out_nr == 0)
  156. {
  157. err = -RT_EEMPTY;
  158. }
  159. }
  160. else
  161. {
  162. err = -RT_EINVAL;
  163. }
  164. return err;
  165. }
  166. rt_err_t rt_fdt_prefetch(void *fdt)
  167. {
  168. rt_err_t err = -RT_ERROR;
  169. if (fdt)
  170. {
  171. _fdt = fdt;
  172. if (!fdt_check_header(_fdt))
  173. {
  174. err = rt_fdt_scan_root();
  175. }
  176. else
  177. {
  178. err = -RT_EINVAL;
  179. }
  180. }
  181. return err;
  182. }
  183. rt_err_t rt_fdt_scan_root(void)
  184. {
  185. rt_err_t err = RT_EOK;
  186. int root = fdt_path_offset(_fdt, "/");
  187. if (root >= 0)
  188. {
  189. const fdt32_t *prop;
  190. _root_addr_cells = OFW_ROOT_NODE_ADDR_CELLS_DEFAULT;
  191. _root_size_cells = OFW_ROOT_NODE_SIZE_CELLS_DEFAULT;
  192. if ((prop = fdt_getprop(_fdt, root, "#address-cells", RT_NULL)))
  193. {
  194. _root_addr_cells = fdt32_to_cpu(*prop);
  195. }
  196. if ((prop = fdt_getprop(_fdt, root, "#size-cells", RT_NULL)))
  197. {
  198. _root_size_cells = fdt32_to_cpu(*prop);
  199. }
  200. }
  201. else
  202. {
  203. err = -RT_EEMPTY;
  204. }
  205. return err;
  206. }
  207. rt_inline rt_err_t commit_memregion(const char *name, rt_uint64_t base, rt_uint64_t size, rt_bool_t is_reserved)
  208. {
  209. return rt_fdt_commit_memregion_early(&(rt_region_t)
  210. {
  211. .name = name,
  212. .start = (rt_size_t)base,
  213. .end = (rt_size_t)(base + size),
  214. }, is_reserved);
  215. }
  216. static rt_err_t reserve_memregion(const char *name, rt_uint64_t base, rt_uint64_t size)
  217. {
  218. if (commit_memregion(name, base, size, RT_TRUE) == -RT_EEMPTY)
  219. {
  220. LOG_W("Reserved memory: %p - %p%s", base, base + size, " unable to record");
  221. }
  222. return RT_EOK;
  223. }
  224. static rt_err_t fdt_reserved_mem_check_root(int nodeoffset)
  225. {
  226. rt_err_t err = RT_EOK;
  227. const fdt32_t *prop = fdt_getprop(_fdt, nodeoffset, "#size-cells", RT_NULL);
  228. if (!prop || fdt32_to_cpu(*prop) != _root_size_cells)
  229. {
  230. err = -RT_EINVAL;
  231. }
  232. if (!err)
  233. {
  234. prop = fdt_getprop(_fdt, nodeoffset, "#address-cells", RT_NULL);
  235. if (!prop || fdt32_to_cpu(*prop) != _root_addr_cells)
  236. {
  237. err = -RT_EINVAL;
  238. }
  239. }
  240. if (!err && !(prop = fdt_getprop(_fdt, nodeoffset, "ranges", RT_NULL)))
  241. {
  242. err = -RT_EINVAL;
  243. }
  244. return err;
  245. }
  246. static rt_err_t fdt_reserved_memory_reg(int nodeoffset, const char *uname)
  247. {
  248. rt_err_t err = RT_EOK;
  249. rt_ubase_t base, size;
  250. const fdt32_t *prop;
  251. int len, t_len = (_root_addr_cells + _root_size_cells) * sizeof(fdt32_t);
  252. if ((prop = fdt_getprop(_fdt, nodeoffset, "reg", &len)))
  253. {
  254. if (len && len % t_len != 0)
  255. {
  256. LOG_E("Reserved memory: invalid reg property in '%s', skipping node", uname);
  257. err = -RT_EINVAL;
  258. }
  259. else
  260. {
  261. for (; len >= t_len; len -= t_len)
  262. {
  263. base = rt_fdt_next_cell(&prop, _root_addr_cells);
  264. size = rt_fdt_next_cell(&prop, _root_size_cells);
  265. if (!size)
  266. {
  267. continue;
  268. }
  269. base = rt_fdt_translate_address(_fdt, nodeoffset, base);
  270. reserve_memregion(fdt_get_name(_fdt, nodeoffset, RT_NULL), base, size);
  271. len -= t_len;
  272. }
  273. }
  274. }
  275. else
  276. {
  277. err = -RT_EEMPTY;
  278. }
  279. return err;
  280. }
  281. static void fdt_scan_reserved_memory(void)
  282. {
  283. int nodeoffset, child;
  284. nodeoffset = fdt_path_offset(_fdt, "/reserved-memory");
  285. if (nodeoffset >= 0)
  286. {
  287. if (!fdt_reserved_mem_check_root(nodeoffset))
  288. {
  289. fdt_for_each_subnode(child, _fdt, nodeoffset)
  290. {
  291. rt_err_t err;
  292. const char *uname;
  293. if (!rt_fdt_device_is_available(_fdt, child))
  294. {
  295. continue;
  296. }
  297. uname = fdt_get_name(_fdt, child, RT_NULL);
  298. err = fdt_reserved_memory_reg(child, uname);
  299. if (err == -RT_EEMPTY && fdt_getprop(_fdt, child, "size", RT_NULL))
  300. {
  301. reserve_memregion(fdt_get_name(_fdt, child, RT_NULL), 0, 0);
  302. }
  303. }
  304. }
  305. else
  306. {
  307. LOG_E("Reserved memory: unsupported node format, ignoring");
  308. }
  309. }
  310. }
  311. static rt_err_t fdt_scan_memory(void)
  312. {
  313. int nodeoffset, no;
  314. rt_region_t *region;
  315. rt_uint64_t base, size;
  316. rt_err_t err = -RT_EEMPTY;
  317. /* Process header /memreserve/ fields */
  318. for (no = 0; ; ++no)
  319. {
  320. fdt_get_mem_rsv(_fdt, no, &base, &size);
  321. if (!size)
  322. {
  323. break;
  324. }
  325. reserve_memregion("memreserve", base, size);
  326. }
  327. no = 0;
  328. fdt_for_each_subnode(nodeoffset, _fdt, 0)
  329. {
  330. int len;
  331. const fdt32_t *reg, *endptr;
  332. const char *name = fdt_get_name(_fdt, nodeoffset, RT_NULL);
  333. const char *type = fdt_getprop(_fdt, nodeoffset, "device_type", RT_NULL);
  334. if (!type || rt_strcmp(type, "memory"))
  335. {
  336. continue;
  337. }
  338. if (!rt_fdt_device_is_available(_fdt, nodeoffset))
  339. {
  340. continue;
  341. }
  342. reg = fdt_getprop(_fdt, nodeoffset, "reg", &len);
  343. if (!reg)
  344. {
  345. continue;
  346. }
  347. endptr = reg + (len / sizeof(fdt32_t));
  348. name = name ? name : "memory";
  349. while ((endptr - reg) >= (_root_addr_cells + _root_size_cells))
  350. {
  351. base = rt_fdt_next_cell(&reg, _root_addr_cells);
  352. size = rt_fdt_next_cell(&reg, _root_size_cells);
  353. if (!size)
  354. {
  355. continue;
  356. }
  357. err = commit_memregion(name, base, size, RT_FALSE);
  358. if (!err)
  359. {
  360. LOG_I("Memory node(%d) ranges: %p - %p%s", no, base, base + size, "");
  361. }
  362. else
  363. {
  364. LOG_W("Memory node(%d) ranges: %p - %p%s", no, base, base + size, " unable to record");
  365. }
  366. ++no;
  367. }
  368. }
  369. if (!err)
  370. {
  371. fdt_scan_reserved_memory();
  372. }
  373. region = &_memregion[0];
  374. for (no = 0; region->name; ++region)
  375. {
  376. /* We need check the memory region now. */
  377. for (int i = RT_ARRAY_SIZE(_memregion) - 1; i > no; --i)
  378. {
  379. rt_region_t *res_region = &_memregion[i];
  380. if (!res_region->name)
  381. {
  382. break;
  383. }
  384. /*
  385. * +--------+ +--------+
  386. * | memory | | memory |
  387. * +--------+ +----------+ +----------+ +--------+
  388. * | reserved | | reserved |
  389. * +----------+ +----------+
  390. */
  391. if (res_region->start >= region->end || res_region->end <= region->start)
  392. {
  393. /* No adjustments needed */
  394. continue;
  395. }
  396. /*
  397. * case 0: case 1:
  398. * +------------------+ +----------+
  399. * | memory | | memory |
  400. * +---+----------+---+ +---+----------+---+
  401. * | reserved | | reserved |
  402. * +----------+ +---+----------+---+
  403. *
  404. * case 2: case 3:
  405. * +------------------+ +------------------+
  406. * | memory | | memory |
  407. * +--------------+---+------+ +------+---+--------------+
  408. * | reserved | | reserved |
  409. * +----------+ +----------+
  410. */
  411. if (res_region->start > region->start)
  412. {
  413. if (res_region->end < region->end)
  414. {
  415. /* case 0 */
  416. rt_size_t new_size = region->end - res_region->end;
  417. region->end = res_region->start;
  418. /* Commit part next block */
  419. err = commit_memregion(region->name, res_region->end, new_size, RT_FALSE);
  420. if (!err)
  421. {
  422. ++no;
  423. /* Scan again */
  424. region = &_memregion[0];
  425. --region;
  426. break;
  427. }
  428. }
  429. else
  430. {
  431. /* case 2 */
  432. region->end = res_region->start;
  433. }
  434. }
  435. else
  436. {
  437. if (res_region->end < region->end)
  438. {
  439. /* case 3 */
  440. region->start = res_region->end;
  441. }
  442. else
  443. {
  444. /* case 1 */
  445. region->name = RT_NULL;
  446. break;
  447. }
  448. }
  449. }
  450. }
  451. return err;
  452. }
  453. rt_err_t rt_fdt_scan_memory(void)
  454. {
  455. rt_err_t err = -RT_EEMPTY;
  456. if (_fdt)
  457. {
  458. err = fdt_scan_memory();
  459. }
  460. return err;
  461. }
  462. static rt_err_t fdt_scan_initrd(rt_uint64_t *ranges, const char *name, const char *oem)
  463. {
  464. char tmp_name[32];
  465. rt_err_t err = -RT_EEMPTY;
  466. if (_fdt && ranges)
  467. {
  468. int offset = fdt_path_offset(_fdt, "/chosen");
  469. if (offset >= 0)
  470. {
  471. int s_len, e_len;
  472. const fdt32_t *start = RT_NULL, *end = RT_NULL;
  473. rt_snprintf(tmp_name, sizeof(tmp_name), "%s,%d-start", oem, name);
  474. start = fdt_getprop(_fdt, offset, tmp_name, &s_len);
  475. rt_snprintf(tmp_name, sizeof(tmp_name), "%s,%d-end", oem, name);
  476. end = fdt_getprop(_fdt, offset, tmp_name, &e_len);
  477. if (start && end)
  478. {
  479. s_len /= sizeof(*start);
  480. e_len /= sizeof(*end);
  481. ranges[0] = rt_fdt_read_number(start, s_len);
  482. ranges[1] = rt_fdt_read_number(end, e_len);
  483. err = RT_EOK;
  484. }
  485. }
  486. if (err)
  487. {
  488. int len;
  489. const char *options, *bootargs = fdt_getprop(_fdt, offset, "bootargs", &len);
  490. rt_snprintf(tmp_name, sizeof(tmp_name), "%s=", name);
  491. if (bootargs && (options = rt_strstr(bootargs, tmp_name)))
  492. {
  493. rt_uint64_t value;
  494. options += rt_strlen(tmp_name) + sizeof("0x") - 1;
  495. err = RT_EOK;
  496. for (int i = 0; i < 2 && !err; ++i)
  497. {
  498. value = 0;
  499. while (*options && *options != ',' && *options != ' ')
  500. {
  501. /* To lowercase or keep number */
  502. char ch = *options | ' ';
  503. value *= 16;
  504. if (ch >= '0' && ch <= '9')
  505. {
  506. value += ch - '0';
  507. }
  508. else if (ch >= 'a' && ch <= 'f')
  509. {
  510. value += ch - 'a' + 10;
  511. }
  512. else
  513. {
  514. err = -RT_EINVAL;
  515. break;
  516. }
  517. ++options;
  518. }
  519. ranges[i] = value;
  520. options += sizeof(",0x") - 1;
  521. }
  522. /* This is initrd's size, convert to initrd's end */
  523. ranges[1] += ranges[0];
  524. }
  525. }
  526. if (!err)
  527. {
  528. commit_memregion("initrd", ranges[0], ranges[1] - ranges[0], RT_TRUE);
  529. }
  530. }
  531. else if (!ranges)
  532. {
  533. err = -RT_EINVAL;
  534. }
  535. return err;
  536. }
  537. rt_err_t rt_fdt_scan_initrd(rt_uint64_t *ranges)
  538. {
  539. rt_err_t err;
  540. err = fdt_scan_initrd(ranges, "cromfs", "rt-thread");
  541. if (err && err == -RT_EEMPTY)
  542. {
  543. err = fdt_scan_initrd(ranges, "initrd", "linux");
  544. }
  545. return err;
  546. }
  547. rt_err_t rt_fdt_model_dump(void)
  548. {
  549. rt_err_t err = RT_EOK;
  550. int root = fdt_path_offset(_fdt, "/");
  551. if (root >= 0)
  552. {
  553. const char *mach_model = fdt_getprop(_fdt, root, "model", RT_NULL);
  554. if (!mach_model)
  555. {
  556. mach_model = fdt_getprop(_fdt, root, "compatible", RT_NULL);
  557. }
  558. LOG_I("Machine model: %s", mach_model ? mach_model : "<undefined>");
  559. }
  560. else
  561. {
  562. err = -RT_EEMPTY;
  563. }
  564. return err;
  565. }
  566. rt_weak rt_err_t rt_fdt_boot_dump(void)
  567. {
  568. LOG_I("Booting RT-Thread on physical CPU 0x%x", rt_hw_cpu_id());
  569. return RT_EOK;
  570. }
  571. void rt_fdt_earlycon_output(const char *str)
  572. {
  573. if (fdt_earlycon.console_putc)
  574. {
  575. while (*str)
  576. {
  577. fdt_earlycon.console_putc(fdt_earlycon.data, *str);
  578. if (*str == '\n')
  579. {
  580. /* Make sure return */
  581. fdt_earlycon.console_putc(fdt_earlycon.data, '\r');
  582. }
  583. ++str;
  584. }
  585. }
  586. else
  587. {
  588. /* We need a byte to save '\0' */
  589. while (*str && fdt_earlycon.msg_idx < sizeof(fdt_earlycon.msg) - 1)
  590. {
  591. fdt_earlycon.msg[fdt_earlycon.msg_idx++] = *str;
  592. ++str;
  593. }
  594. fdt_earlycon.msg[fdt_earlycon.msg_idx] = '\0';
  595. }
  596. }
  597. void rt_fdt_earlycon_kick(int why)
  598. {
  599. if (fdt_earlycon.console_kick)
  600. {
  601. fdt_earlycon.console_kick(&fdt_earlycon, why);
  602. }
  603. if (why == FDT_EARLYCON_KICK_COMPLETED)
  604. {
  605. fdt_earlycon.console_putc = RT_NULL;
  606. if (fdt_earlycon.msg_idx)
  607. {
  608. fdt_earlycon.msg_idx = 0;
  609. /* Dump old messages */
  610. rt_kputs(fdt_earlycon.msg);
  611. }
  612. }
  613. }
  614. rt_err_t rt_fdt_scan_chosen_stdout(void)
  615. {
  616. rt_err_t err = RT_EOK;
  617. int offset;
  618. int len, options_len = 0;
  619. const char *options = RT_NULL, *con_type = RT_NULL;
  620. rt_memset(&fdt_earlycon, 0, sizeof(fdt_earlycon) - sizeof(fdt_earlycon.msg));
  621. fdt_earlycon.nodeoffset = -1;
  622. offset = fdt_path_offset(_fdt, "/chosen");
  623. if (offset >= 0)
  624. {
  625. const char *stdout_path = RT_NULL;
  626. const char *bootargs = fdt_getprop(_fdt, offset, "bootargs", &len);
  627. if (bootargs && (options = rt_strstr(bootargs, "earlycon")))
  628. {
  629. options += sizeof("earlycon") - 1;
  630. if (*options == '\0' || *options == ' ')
  631. {
  632. stdout_path = fdt_getprop(_fdt, offset, "stdout-path", &len);
  633. if (stdout_path && len)
  634. {
  635. const char *path_split = strchrnul(stdout_path, ':');
  636. if (*path_split != '\0')
  637. {
  638. options = path_split + 1;
  639. }
  640. len = path_split - stdout_path;
  641. /*
  642. * Will try 2 styles:
  643. * 1: stdout-path = "serialN:bbbbpnf";
  644. * 2: stdout-path = "/serial-path";
  645. */
  646. offset = fdt_path_offset_namelen(_fdt, stdout_path, len);
  647. if (offset < 0)
  648. {
  649. stdout_path = RT_NULL;
  650. }
  651. }
  652. else if (*options == '=')
  653. {
  654. ++options;
  655. }
  656. else
  657. {
  658. /* Maybe is error in bootargs or it is a new arg */
  659. options = RT_NULL;
  660. }
  661. if (!stdout_path)
  662. {
  663. /* We couldn't know how to setup the earlycon */
  664. options = RT_NULL;
  665. }
  666. }
  667. else
  668. {
  669. offset = -1;
  670. }
  671. if (options)
  672. {
  673. int type_len = 0;
  674. struct rt_fdt_earlycon_id *earlycon_id, *earlycon_id_end, *best_earlycon_id = RT_NULL;
  675. earlycon_id = (struct rt_fdt_earlycon_id *)&_earlycon_start;
  676. earlycon_id_end = (struct rt_fdt_earlycon_id *)&_earlycon_end;
  677. err = -RT_ENOSYS;
  678. /* Only "earlycon" in bootargs */
  679. if (stdout_path)
  680. {
  681. const fdt32_t *reg;
  682. options = RT_NULL;
  683. if ((reg = fdt_getprop(_fdt, offset, "reg", RT_NULL)))
  684. {
  685. rt_uint64_t address;
  686. int addr_cells = fdt_io_addr_cells(_fdt, offset);
  687. int size_cells = fdt_io_size_cells(_fdt, offset);
  688. address = rt_fdt_read_number(reg, addr_cells);
  689. fdt_earlycon.mmio = rt_fdt_translate_address(_fdt, offset, address);
  690. fdt_earlycon.size = rt_fdt_read_number(reg + addr_cells, size_cells);
  691. }
  692. }
  693. else
  694. {
  695. /* Pass split */
  696. while (*options && (*options == '=' || *options == ' '))
  697. {
  698. ++options;
  699. }
  700. if (*options)
  701. {
  702. type_len = strchrnul(options, ',') - options;
  703. }
  704. }
  705. if (options && *options && *options != ' ')
  706. {
  707. options_len = strchrnul(options, ' ') - options;
  708. rt_strncpy(fdt_earlycon.options, options, options_len);
  709. }
  710. /* console > stdout-path */
  711. for (int max_score = 0; earlycon_id < earlycon_id_end; ++earlycon_id)
  712. {
  713. int score = 0;
  714. if (type_len && earlycon_id->type)
  715. {
  716. if (!rt_strncmp(earlycon_id->type, options, type_len))
  717. {
  718. score += 1;
  719. }
  720. }
  721. if (stdout_path && earlycon_id->compatible)
  722. {
  723. if (!fdt_node_check_compatible(_fdt, offset, earlycon_id->compatible))
  724. {
  725. score += 2;
  726. }
  727. }
  728. if (score > max_score)
  729. {
  730. max_score = score;
  731. best_earlycon_id = earlycon_id;
  732. if (score == 3)
  733. {
  734. break;
  735. }
  736. }
  737. }
  738. if (best_earlycon_id && best_earlycon_id->setup)
  739. {
  740. const char earlycon_magic[] = { 'O', 'F', 'W', '\0' };
  741. if (!con_type)
  742. {
  743. con_type = best_earlycon_id->type;
  744. }
  745. fdt_earlycon.fdt = _fdt;
  746. fdt_earlycon.nodeoffset = offset;
  747. options = &fdt_earlycon.options[options_len + 1];
  748. rt_strncpy((void *)options, earlycon_magic, RT_ARRAY_SIZE(earlycon_magic));
  749. err = best_earlycon_id->setup(&fdt_earlycon, fdt_earlycon.options);
  750. if (rt_strncmp(options, earlycon_magic, RT_ARRAY_SIZE(earlycon_magic)))
  751. {
  752. const char *option_start = options - 1;
  753. while (option_start[-1] != '\0')
  754. {
  755. --option_start;
  756. }
  757. rt_memmove(fdt_earlycon.options, option_start, options - option_start);
  758. }
  759. else
  760. {
  761. fdt_earlycon.options[0] = '\0';
  762. }
  763. }
  764. }
  765. }
  766. else
  767. {
  768. err = -RT_EEMPTY;
  769. }
  770. }
  771. else
  772. {
  773. err = -RT_EEMPTY;
  774. }
  775. if (fdt_earlycon.msg_idx)
  776. {
  777. fdt_earlycon.msg_idx = 0;
  778. rt_kputs(fdt_earlycon.msg);
  779. }
  780. rt_fdt_boot_dump();
  781. rt_fdt_model_dump();
  782. if (fdt_earlycon.mmio)
  783. {
  784. LOG_I("Earlycon: %s at MMIO/PIO %p (options '%s')",
  785. con_type, fdt_earlycon.mmio, fdt_earlycon.options);
  786. }
  787. return err;
  788. }
  789. static void system_node_init_flag(struct rt_ofw_node *np)
  790. {
  791. if (np)
  792. {
  793. rt_ofw_node_set_flag(np, RT_OFW_F_READLY);
  794. rt_ofw_node_set_flag(np, RT_OFW_F_SYSTEM);
  795. }
  796. }
  797. rt_err_t rt_fdt_unflatten(void)
  798. {
  799. rt_err_t err = RT_EOK;
  800. if (_fdt)
  801. {
  802. _phandle_min = OFW_PHANDLE_MAX;
  803. _phandle_max = OFW_PHANDLE_MIN;
  804. ofw_node_root = rt_fdt_unflatten_single(_fdt);
  805. if (ofw_node_root)
  806. {
  807. ofw_node_cpus = rt_ofw_find_node_by_path("/cpus");
  808. ofw_node_chosen = rt_ofw_find_node_by_path("/chosen");
  809. ofw_node_aliases = rt_ofw_find_node_by_path("/aliases");
  810. ofw_node_reserved_memory = rt_ofw_find_node_by_path("/reserved-memory");
  811. RT_ASSERT(ofw_node_cpus != RT_NULL);
  812. system_node_init_flag(ofw_node_root);
  813. system_node_init_flag(ofw_node_cpus);
  814. system_node_init_flag(ofw_node_chosen);
  815. system_node_init_flag(ofw_node_aliases);
  816. system_node_init_flag(ofw_node_reserved_memory);
  817. if (ofw_node_aliases)
  818. {
  819. err = ofw_alias_scan();
  820. }
  821. err = err ? : ofw_phandle_hash_reset(_phandle_min, _phandle_max);
  822. }
  823. }
  824. else
  825. {
  826. err = -RT_ERROR;
  827. }
  828. return err;
  829. }
  830. static rt_err_t fdt_unflatten_props(struct rt_ofw_node *np, int node_off)
  831. {
  832. rt_err_t err = RT_EOK;
  833. struct rt_ofw_prop *prop;
  834. int prop_off = fdt_first_property_offset(_fdt, node_off);
  835. if (prop_off >= 0)
  836. {
  837. np->props = rt_malloc(sizeof(struct rt_ofw_prop));
  838. }
  839. prop = np->props;
  840. while (prop_off >= 0)
  841. {
  842. if (!prop)
  843. {
  844. err = -RT_ENOMEM;
  845. break;
  846. }
  847. prop->value = (void *)fdt_getprop_by_offset(_fdt, prop_off, &prop->name, &prop->length);
  848. if (prop->name && !rt_strcmp(prop->name, "name"))
  849. {
  850. np->name = prop->value;
  851. }
  852. prop_off = fdt_next_property_offset(_fdt, prop_off);
  853. if (prop_off < 0)
  854. {
  855. prop->next = RT_NULL;
  856. break;
  857. }
  858. prop->next = rt_malloc(sizeof(struct rt_ofw_prop));
  859. prop = prop->next;
  860. }
  861. return err;
  862. }
  863. static rt_err_t fdt_unflatten_single(struct rt_ofw_node *np, int node_off)
  864. {
  865. int depth = 0;
  866. rt_err_t err = RT_EOK;
  867. struct rt_ofw_node *np_stack[OFW_NODE_MAX_DEPTH], *parent = RT_NULL;
  868. do {
  869. if (!np)
  870. {
  871. err = -RT_ENOMEM;
  872. break;
  873. }
  874. np->name = "<NULL>";
  875. np->full_name = fdt_get_name(_fdt, node_off, RT_NULL);
  876. np->phandle = fdt_get_phandle(_fdt, node_off);
  877. if (np->phandle >= OFW_PHANDLE_MIN)
  878. {
  879. if (np->phandle < _phandle_min)
  880. {
  881. _phandle_min = np->phandle;
  882. }
  883. if (np->phandle > _phandle_max)
  884. {
  885. _phandle_max = np->phandle;
  886. }
  887. }
  888. if ((err = fdt_unflatten_props(np, node_off)))
  889. {
  890. break;
  891. }
  892. np->parent = parent;
  893. rt_ref_init(&np->ref);
  894. np->flags = 0;
  895. if (!np->child)
  896. {
  897. /* Save node offset temp */
  898. rt_ofw_data(np) = (void *)(rt_ubase_t)node_off;
  899. /* Check children */
  900. node_off = fdt_first_subnode(_fdt, node_off);
  901. if (node_off >= 0)
  902. {
  903. parent = np;
  904. np_stack[depth++] = np;
  905. np->child = rt_calloc(1, sizeof(struct rt_ofw_node));
  906. np = np->child;
  907. continue;
  908. }
  909. }
  910. while (depth >= 0)
  911. {
  912. /* Restore node offset temp */
  913. node_off = (long)rt_ofw_data(np);
  914. rt_ofw_data(np) = RT_NULL;
  915. /* Next step */
  916. node_off = fdt_next_subnode(_fdt, node_off);
  917. if (node_off < 0)
  918. {
  919. np->sibling = RT_NULL;
  920. np = np_stack[--depth];
  921. }
  922. else
  923. {
  924. parent = np->parent;
  925. np->sibling = rt_calloc(1, sizeof(struct rt_ofw_node));
  926. np = np->sibling;
  927. break;
  928. }
  929. }
  930. } while (depth >= 0);
  931. return err;
  932. }
  933. struct rt_ofw_node *rt_fdt_unflatten_single(void *fdt)
  934. {
  935. int root_off;
  936. struct fdt_info *header;
  937. struct rt_ofw_node *root = RT_NULL;
  938. if (fdt && (root_off = fdt_path_offset(fdt, "/")) >= 0)
  939. {
  940. root = rt_calloc(1, sizeof(struct fdt_info) + sizeof(struct rt_ofw_node));
  941. }
  942. if (root)
  943. {
  944. header = (void *)root + sizeof(struct rt_ofw_node);
  945. rt_strncpy(header->name, "/", sizeof("/"));
  946. header->fdt = fdt;
  947. header->rsvmap = (struct fdt_reserve_entry *)((void *)fdt + fdt_off_mem_rsvmap(fdt));
  948. header->rsvmap_nr = fdt_num_mem_rsv(fdt);
  949. if (!fdt_unflatten_single(root, root_off))
  950. {
  951. root->name = (const char *)header;
  952. }
  953. else
  954. {
  955. rt_ofw_node_destroy(root);
  956. root = RT_NULL;
  957. }
  958. }
  959. return root;
  960. }