dtc-parser.y 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533
  1. /*
  2. * (C) Copyright David Gibson <dwg@au1.ibm.com>, IBM Corporation. 2005.
  3. *
  4. *
  5. * This program is free software; you can redistribute it and/or
  6. * modify it under the terms of the GNU General Public License as
  7. * published by the Free Software Foundation; either version 2 of the
  8. * License, or (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  13. * General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program; if not, write to the Free Software
  17. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
  18. * USA
  19. */
  20. %{
  21. #include <stdio.h>
  22. #include <inttypes.h>
  23. #include "dtc.h"
  24. #include "srcpos.h"
  25. extern int yylex(void);
  26. extern void yyerror(char const *s);
  27. #define ERROR(loc, ...) \
  28. do { \
  29. srcpos_error((loc), "Error", __VA_ARGS__); \
  30. treesource_error = true; \
  31. } while (0)
  32. extern struct dt_info *parser_output;
  33. extern bool treesource_error;
  34. %}
  35. %union {
  36. char *propnodename;
  37. char *labelref;
  38. uint8_t byte;
  39. struct data data;
  40. struct {
  41. struct data data;
  42. int bits;
  43. } array;
  44. struct property *prop;
  45. struct property *proplist;
  46. struct node *node;
  47. struct node *nodelist;
  48. struct reserve_info *re;
  49. uint64_t integer;
  50. unsigned int flags;
  51. }
  52. %token DT_V1
  53. %token DT_PLUGIN
  54. %token DT_MEMRESERVE
  55. %token DT_LSHIFT DT_RSHIFT DT_LE DT_GE DT_EQ DT_NE DT_AND DT_OR
  56. %token DT_BITS
  57. %token DT_DEL_PROP
  58. %token DT_DEL_NODE
  59. %token <propnodename> DT_PROPNODENAME
  60. %token <integer> DT_LITERAL
  61. %token <integer> DT_CHAR_LITERAL
  62. %token <byte> DT_BYTE
  63. %token <data> DT_STRING
  64. %token <labelref> DT_LABEL
  65. %token <labelref> DT_REF
  66. %token DT_INCBIN
  67. %type <data> propdata
  68. %type <data> propdataprefix
  69. %type <flags> header
  70. %type <flags> headers
  71. %type <re> memreserve
  72. %type <re> memreserves
  73. %type <array> arrayprefix
  74. %type <data> bytestring
  75. %type <prop> propdef
  76. %type <proplist> proplist
  77. %type <node> devicetree
  78. %type <node> nodedef
  79. %type <node> subnode
  80. %type <nodelist> subnodes
  81. %type <integer> integer_prim
  82. %type <integer> integer_unary
  83. %type <integer> integer_mul
  84. %type <integer> integer_add
  85. %type <integer> integer_shift
  86. %type <integer> integer_rela
  87. %type <integer> integer_eq
  88. %type <integer> integer_bitand
  89. %type <integer> integer_bitxor
  90. %type <integer> integer_bitor
  91. %type <integer> integer_and
  92. %type <integer> integer_or
  93. %type <integer> integer_trinary
  94. %type <integer> integer_expr
  95. %%
  96. sourcefile:
  97. headers memreserves devicetree
  98. {
  99. parser_output = build_dt_info($1, $2, $3,
  100. guess_boot_cpuid($3));
  101. }
  102. ;
  103. header:
  104. DT_V1 ';'
  105. {
  106. $$ = DTSF_V1;
  107. }
  108. | DT_V1 ';' DT_PLUGIN ';'
  109. {
  110. $$ = DTSF_V1 | DTSF_PLUGIN;
  111. }
  112. ;
  113. headers:
  114. header
  115. | header headers
  116. {
  117. if ($2 != $1)
  118. ERROR(&@2, "Header flags don't match earlier ones");
  119. $$ = $1;
  120. }
  121. ;
  122. memreserves:
  123. /* empty */
  124. {
  125. $$ = NULL;
  126. }
  127. | memreserve memreserves
  128. {
  129. $$ = chain_reserve_entry($1, $2);
  130. }
  131. ;
  132. memreserve:
  133. DT_MEMRESERVE integer_prim integer_prim ';'
  134. {
  135. $$ = build_reserve_entry($2, $3);
  136. }
  137. | DT_LABEL memreserve
  138. {
  139. add_label(&$2->labels, $1);
  140. $$ = $2;
  141. }
  142. ;
  143. devicetree:
  144. '/' nodedef
  145. {
  146. $$ = name_node($2, "");
  147. }
  148. | devicetree '/' nodedef
  149. {
  150. $$ = merge_nodes($1, $3);
  151. }
  152. | devicetree DT_LABEL DT_REF nodedef
  153. {
  154. struct node *target = get_node_by_ref($1, $3);
  155. if (target) {
  156. add_label(&target->labels, $2);
  157. merge_nodes(target, $4);
  158. } else
  159. ERROR(&@3, "Label or path %s not found", $3);
  160. $$ = $1;
  161. }
  162. | devicetree DT_REF nodedef
  163. {
  164. struct node *target = get_node_by_ref($1, $2);
  165. if (target) {
  166. merge_nodes(target, $3);
  167. } else {
  168. /*
  169. * We rely on the rule being always:
  170. * versioninfo plugindecl memreserves devicetree
  171. * so $-1 is what we want (plugindecl)
  172. */
  173. if ($<flags>-1 & DTSF_PLUGIN)
  174. add_orphan_node($1, $3, $2);
  175. else
  176. ERROR(&@2, "Label or path %s not found", $2);
  177. }
  178. $$ = $1;
  179. }
  180. | devicetree DT_DEL_NODE DT_REF ';'
  181. {
  182. struct node *target = get_node_by_ref($1, $3);
  183. if (target)
  184. delete_node(target);
  185. else
  186. ERROR(&@3, "Label or path %s not found", $3);
  187. $$ = $1;
  188. }
  189. | /* empty */
  190. {
  191. /* build empty node */
  192. $$ = name_node(build_node(NULL, NULL), "");
  193. }
  194. ;
  195. nodedef:
  196. '{' proplist subnodes '}' ';'
  197. {
  198. $$ = build_node($2, $3);
  199. }
  200. ;
  201. proplist:
  202. /* empty */
  203. {
  204. $$ = NULL;
  205. }
  206. | proplist propdef
  207. {
  208. $$ = chain_property($2, $1);
  209. }
  210. ;
  211. propdef:
  212. DT_PROPNODENAME '=' propdata ';'
  213. {
  214. $$ = build_property($1, $3);
  215. }
  216. | DT_PROPNODENAME ';'
  217. {
  218. $$ = build_property($1, empty_data);
  219. }
  220. | DT_DEL_PROP DT_PROPNODENAME ';'
  221. {
  222. $$ = build_property_delete($2);
  223. }
  224. | DT_LABEL propdef
  225. {
  226. add_label(&$2->labels, $1);
  227. $$ = $2;
  228. }
  229. ;
  230. propdata:
  231. propdataprefix DT_STRING
  232. {
  233. $$ = data_merge($1, $2);
  234. }
  235. | propdataprefix arrayprefix '>'
  236. {
  237. $$ = data_merge($1, $2.data);
  238. }
  239. | propdataprefix '[' bytestring ']'
  240. {
  241. $$ = data_merge($1, $3);
  242. }
  243. | propdataprefix DT_REF
  244. {
  245. $$ = data_add_marker($1, REF_PATH, $2);
  246. }
  247. | propdataprefix DT_INCBIN '(' DT_STRING ',' integer_prim ',' integer_prim ')'
  248. {
  249. FILE *f = srcfile_relative_open($4.val, NULL);
  250. struct data d;
  251. if ($6 != 0)
  252. if (fseek(f, $6, SEEK_SET) != 0)
  253. die("Couldn't seek to offset %llu in \"%s\": %s",
  254. (unsigned long long)$6, $4.val,
  255. strerror(errno));
  256. d = data_copy_file(f, $8);
  257. $$ = data_merge($1, d);
  258. fclose(f);
  259. }
  260. | propdataprefix DT_INCBIN '(' DT_STRING ')'
  261. {
  262. FILE *f = srcfile_relative_open($4.val, NULL);
  263. struct data d = empty_data;
  264. d = data_copy_file(f, -1);
  265. $$ = data_merge($1, d);
  266. fclose(f);
  267. }
  268. | propdata DT_LABEL
  269. {
  270. $$ = data_add_marker($1, LABEL, $2);
  271. }
  272. ;
  273. propdataprefix:
  274. /* empty */
  275. {
  276. $$ = empty_data;
  277. }
  278. | propdata ','
  279. {
  280. $$ = $1;
  281. }
  282. | propdataprefix DT_LABEL
  283. {
  284. $$ = data_add_marker($1, LABEL, $2);
  285. }
  286. ;
  287. arrayprefix:
  288. DT_BITS DT_LITERAL '<'
  289. {
  290. unsigned long long bits;
  291. bits = $2;
  292. if ((bits != 8) && (bits != 16) &&
  293. (bits != 32) && (bits != 64)) {
  294. ERROR(&@2, "Array elements must be"
  295. " 8, 16, 32 or 64-bits");
  296. bits = 32;
  297. }
  298. $$.data = empty_data;
  299. $$.bits = bits;
  300. }
  301. | '<'
  302. {
  303. $$.data = empty_data;
  304. $$.bits = 32;
  305. }
  306. | arrayprefix integer_prim
  307. {
  308. if ($1.bits < 64) {
  309. uint64_t mask = (1ULL << $1.bits) - 1;
  310. /*
  311. * Bits above mask must either be all zero
  312. * (positive within range of mask) or all one
  313. * (negative and sign-extended). The second
  314. * condition is true if when we set all bits
  315. * within the mask to one (i.e. | in the
  316. * mask), all bits are one.
  317. */
  318. if (($2 > mask) && (($2 | mask) != -1ULL))
  319. ERROR(&@2, "Value out of range for"
  320. " %d-bit array element", $1.bits);
  321. }
  322. $$.data = data_append_integer($1.data, $2, $1.bits);
  323. }
  324. | arrayprefix DT_REF
  325. {
  326. uint64_t val = ~0ULL >> (64 - $1.bits);
  327. if ($1.bits == 32)
  328. $1.data = data_add_marker($1.data,
  329. REF_PHANDLE,
  330. $2);
  331. else
  332. ERROR(&@2, "References are only allowed in "
  333. "arrays with 32-bit elements.");
  334. $$.data = data_append_integer($1.data, val, $1.bits);
  335. }
  336. | arrayprefix DT_LABEL
  337. {
  338. $$.data = data_add_marker($1.data, LABEL, $2);
  339. }
  340. ;
  341. integer_prim:
  342. DT_LITERAL
  343. | DT_CHAR_LITERAL
  344. | '(' integer_expr ')'
  345. {
  346. $$ = $2;
  347. }
  348. ;
  349. integer_expr:
  350. integer_trinary
  351. ;
  352. integer_trinary:
  353. integer_or
  354. | integer_or '?' integer_expr ':' integer_trinary { $$ = $1 ? $3 : $5; }
  355. ;
  356. integer_or:
  357. integer_and
  358. | integer_or DT_OR integer_and { $$ = $1 || $3; }
  359. ;
  360. integer_and:
  361. integer_bitor
  362. | integer_and DT_AND integer_bitor { $$ = $1 && $3; }
  363. ;
  364. integer_bitor:
  365. integer_bitxor
  366. | integer_bitor '|' integer_bitxor { $$ = $1 | $3; }
  367. ;
  368. integer_bitxor:
  369. integer_bitand
  370. | integer_bitxor '^' integer_bitand { $$ = $1 ^ $3; }
  371. ;
  372. integer_bitand:
  373. integer_eq
  374. | integer_bitand '&' integer_eq { $$ = $1 & $3; }
  375. ;
  376. integer_eq:
  377. integer_rela
  378. | integer_eq DT_EQ integer_rela { $$ = $1 == $3; }
  379. | integer_eq DT_NE integer_rela { $$ = $1 != $3; }
  380. ;
  381. integer_rela:
  382. integer_shift
  383. | integer_rela '<' integer_shift { $$ = $1 < $3; }
  384. | integer_rela '>' integer_shift { $$ = $1 > $3; }
  385. | integer_rela DT_LE integer_shift { $$ = $1 <= $3; }
  386. | integer_rela DT_GE integer_shift { $$ = $1 >= $3; }
  387. ;
  388. integer_shift:
  389. integer_shift DT_LSHIFT integer_add { $$ = $1 << $3; }
  390. | integer_shift DT_RSHIFT integer_add { $$ = $1 >> $3; }
  391. | integer_add
  392. ;
  393. integer_add:
  394. integer_add '+' integer_mul { $$ = $1 + $3; }
  395. | integer_add '-' integer_mul { $$ = $1 - $3; }
  396. | integer_mul
  397. ;
  398. integer_mul:
  399. integer_mul '*' integer_unary { $$ = $1 * $3; }
  400. | integer_mul '/' integer_unary
  401. {
  402. if ($3 != 0) {
  403. $$ = $1 / $3;
  404. } else {
  405. ERROR(&@$, "Division by zero");
  406. $$ = 0;
  407. }
  408. }
  409. | integer_mul '%' integer_unary
  410. {
  411. if ($3 != 0) {
  412. $$ = $1 % $3;
  413. } else {
  414. ERROR(&@$, "Division by zero");
  415. $$ = 0;
  416. }
  417. }
  418. | integer_unary
  419. ;
  420. integer_unary:
  421. integer_prim
  422. | '-' integer_unary { $$ = -$2; }
  423. | '~' integer_unary { $$ = ~$2; }
  424. | '!' integer_unary { $$ = !$2; }
  425. ;
  426. bytestring:
  427. /* empty */
  428. {
  429. $$ = empty_data;
  430. }
  431. | bytestring DT_BYTE
  432. {
  433. $$ = data_append_byte($1, $2);
  434. }
  435. | bytestring DT_LABEL
  436. {
  437. $$ = data_add_marker($1, LABEL, $2);
  438. }
  439. ;
  440. subnodes:
  441. /* empty */
  442. {
  443. $$ = NULL;
  444. }
  445. | subnode subnodes
  446. {
  447. $$ = chain_node($1, $2);
  448. }
  449. | subnode propdef
  450. {
  451. ERROR(&@2, "Properties must precede subnodes");
  452. YYERROR;
  453. }
  454. ;
  455. subnode:
  456. DT_PROPNODENAME nodedef
  457. {
  458. $$ = name_node($2, $1);
  459. }
  460. | DT_DEL_NODE DT_PROPNODENAME ';'
  461. {
  462. $$ = name_node(build_node_delete(), $2);
  463. }
  464. | DT_LABEL subnode
  465. {
  466. add_label(&$2->labels, $1);
  467. $$ = $2;
  468. }
  469. ;
  470. %%
  471. void yyerror(char const *s)
  472. {
  473. ERROR(&yylloc, "%s", s);
  474. }