dtc-parser.y 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519
  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. ERROR(&@2, "Label or path %s not found", $2);
  169. $$ = $1;
  170. }
  171. | devicetree DT_DEL_NODE DT_REF ';'
  172. {
  173. struct node *target = get_node_by_ref($1, $3);
  174. if (target)
  175. delete_node(target);
  176. else
  177. ERROR(&@3, "Label or path %s not found", $3);
  178. $$ = $1;
  179. }
  180. ;
  181. nodedef:
  182. '{' proplist subnodes '}' ';'
  183. {
  184. $$ = build_node($2, $3);
  185. }
  186. ;
  187. proplist:
  188. /* empty */
  189. {
  190. $$ = NULL;
  191. }
  192. | proplist propdef
  193. {
  194. $$ = chain_property($2, $1);
  195. }
  196. ;
  197. propdef:
  198. DT_PROPNODENAME '=' propdata ';'
  199. {
  200. $$ = build_property($1, $3);
  201. }
  202. | DT_PROPNODENAME ';'
  203. {
  204. $$ = build_property($1, empty_data);
  205. }
  206. | DT_DEL_PROP DT_PROPNODENAME ';'
  207. {
  208. $$ = build_property_delete($2);
  209. }
  210. | DT_LABEL propdef
  211. {
  212. add_label(&$2->labels, $1);
  213. $$ = $2;
  214. }
  215. ;
  216. propdata:
  217. propdataprefix DT_STRING
  218. {
  219. $$ = data_merge($1, $2);
  220. }
  221. | propdataprefix arrayprefix '>'
  222. {
  223. $$ = data_merge($1, $2.data);
  224. }
  225. | propdataprefix '[' bytestring ']'
  226. {
  227. $$ = data_merge($1, $3);
  228. }
  229. | propdataprefix DT_REF
  230. {
  231. $$ = data_add_marker($1, REF_PATH, $2);
  232. }
  233. | propdataprefix DT_INCBIN '(' DT_STRING ',' integer_prim ',' integer_prim ')'
  234. {
  235. FILE *f = srcfile_relative_open($4.val, NULL);
  236. struct data d;
  237. if ($6 != 0)
  238. if (fseek(f, $6, SEEK_SET) != 0)
  239. die("Couldn't seek to offset %llu in \"%s\": %s",
  240. (unsigned long long)$6, $4.val,
  241. strerror(errno));
  242. d = data_copy_file(f, $8);
  243. $$ = data_merge($1, d);
  244. fclose(f);
  245. }
  246. | propdataprefix DT_INCBIN '(' DT_STRING ')'
  247. {
  248. FILE *f = srcfile_relative_open($4.val, NULL);
  249. struct data d = empty_data;
  250. d = data_copy_file(f, -1);
  251. $$ = data_merge($1, d);
  252. fclose(f);
  253. }
  254. | propdata DT_LABEL
  255. {
  256. $$ = data_add_marker($1, LABEL, $2);
  257. }
  258. ;
  259. propdataprefix:
  260. /* empty */
  261. {
  262. $$ = empty_data;
  263. }
  264. | propdata ','
  265. {
  266. $$ = $1;
  267. }
  268. | propdataprefix DT_LABEL
  269. {
  270. $$ = data_add_marker($1, LABEL, $2);
  271. }
  272. ;
  273. arrayprefix:
  274. DT_BITS DT_LITERAL '<'
  275. {
  276. unsigned long long bits;
  277. bits = $2;
  278. if ((bits != 8) && (bits != 16) &&
  279. (bits != 32) && (bits != 64)) {
  280. ERROR(&@2, "Array elements must be"
  281. " 8, 16, 32 or 64-bits");
  282. bits = 32;
  283. }
  284. $$.data = empty_data;
  285. $$.bits = bits;
  286. }
  287. | '<'
  288. {
  289. $$.data = empty_data;
  290. $$.bits = 32;
  291. }
  292. | arrayprefix integer_prim
  293. {
  294. if ($1.bits < 64) {
  295. uint64_t mask = (1ULL << $1.bits) - 1;
  296. /*
  297. * Bits above mask must either be all zero
  298. * (positive within range of mask) or all one
  299. * (negative and sign-extended). The second
  300. * condition is true if when we set all bits
  301. * within the mask to one (i.e. | in the
  302. * mask), all bits are one.
  303. */
  304. if (($2 > mask) && (($2 | mask) != -1ULL))
  305. ERROR(&@2, "Value out of range for"
  306. " %d-bit array element", $1.bits);
  307. }
  308. $$.data = data_append_integer($1.data, $2, $1.bits);
  309. }
  310. | arrayprefix DT_REF
  311. {
  312. uint64_t val = ~0ULL >> (64 - $1.bits);
  313. if ($1.bits == 32)
  314. $1.data = data_add_marker($1.data,
  315. REF_PHANDLE,
  316. $2);
  317. else
  318. ERROR(&@2, "References are only allowed in "
  319. "arrays with 32-bit elements.");
  320. $$.data = data_append_integer($1.data, val, $1.bits);
  321. }
  322. | arrayprefix DT_LABEL
  323. {
  324. $$.data = data_add_marker($1.data, LABEL, $2);
  325. }
  326. ;
  327. integer_prim:
  328. DT_LITERAL
  329. | DT_CHAR_LITERAL
  330. | '(' integer_expr ')'
  331. {
  332. $$ = $2;
  333. }
  334. ;
  335. integer_expr:
  336. integer_trinary
  337. ;
  338. integer_trinary:
  339. integer_or
  340. | integer_or '?' integer_expr ':' integer_trinary { $$ = $1 ? $3 : $5; }
  341. ;
  342. integer_or:
  343. integer_and
  344. | integer_or DT_OR integer_and { $$ = $1 || $3; }
  345. ;
  346. integer_and:
  347. integer_bitor
  348. | integer_and DT_AND integer_bitor { $$ = $1 && $3; }
  349. ;
  350. integer_bitor:
  351. integer_bitxor
  352. | integer_bitor '|' integer_bitxor { $$ = $1 | $3; }
  353. ;
  354. integer_bitxor:
  355. integer_bitand
  356. | integer_bitxor '^' integer_bitand { $$ = $1 ^ $3; }
  357. ;
  358. integer_bitand:
  359. integer_eq
  360. | integer_bitand '&' integer_eq { $$ = $1 & $3; }
  361. ;
  362. integer_eq:
  363. integer_rela
  364. | integer_eq DT_EQ integer_rela { $$ = $1 == $3; }
  365. | integer_eq DT_NE integer_rela { $$ = $1 != $3; }
  366. ;
  367. integer_rela:
  368. integer_shift
  369. | integer_rela '<' integer_shift { $$ = $1 < $3; }
  370. | integer_rela '>' integer_shift { $$ = $1 > $3; }
  371. | integer_rela DT_LE integer_shift { $$ = $1 <= $3; }
  372. | integer_rela DT_GE integer_shift { $$ = $1 >= $3; }
  373. ;
  374. integer_shift:
  375. integer_shift DT_LSHIFT integer_add { $$ = $1 << $3; }
  376. | integer_shift DT_RSHIFT integer_add { $$ = $1 >> $3; }
  377. | integer_add
  378. ;
  379. integer_add:
  380. integer_add '+' integer_mul { $$ = $1 + $3; }
  381. | integer_add '-' integer_mul { $$ = $1 - $3; }
  382. | integer_mul
  383. ;
  384. integer_mul:
  385. integer_mul '*' integer_unary { $$ = $1 * $3; }
  386. | integer_mul '/' integer_unary
  387. {
  388. if ($3 != 0) {
  389. $$ = $1 / $3;
  390. } else {
  391. ERROR(&@$, "Division by zero");
  392. $$ = 0;
  393. }
  394. }
  395. | integer_mul '%' integer_unary
  396. {
  397. if ($3 != 0) {
  398. $$ = $1 % $3;
  399. } else {
  400. ERROR(&@$, "Division by zero");
  401. $$ = 0;
  402. }
  403. }
  404. | integer_unary
  405. ;
  406. integer_unary:
  407. integer_prim
  408. | '-' integer_unary { $$ = -$2; }
  409. | '~' integer_unary { $$ = ~$2; }
  410. | '!' integer_unary { $$ = !$2; }
  411. ;
  412. bytestring:
  413. /* empty */
  414. {
  415. $$ = empty_data;
  416. }
  417. | bytestring DT_BYTE
  418. {
  419. $$ = data_append_byte($1, $2);
  420. }
  421. | bytestring DT_LABEL
  422. {
  423. $$ = data_add_marker($1, LABEL, $2);
  424. }
  425. ;
  426. subnodes:
  427. /* empty */
  428. {
  429. $$ = NULL;
  430. }
  431. | subnode subnodes
  432. {
  433. $$ = chain_node($1, $2);
  434. }
  435. | subnode propdef
  436. {
  437. ERROR(&@2, "Properties must precede subnodes");
  438. YYERROR;
  439. }
  440. ;
  441. subnode:
  442. DT_PROPNODENAME nodedef
  443. {
  444. $$ = name_node($2, $1);
  445. }
  446. | DT_DEL_NODE DT_PROPNODENAME ';'
  447. {
  448. $$ = name_node(build_node_delete(), $2);
  449. }
  450. | DT_LABEL subnode
  451. {
  452. add_label(&$2->labels, $1);
  453. $$ = $2;
  454. }
  455. ;
  456. %%
  457. void yyerror(char const *s)
  458. {
  459. ERROR(&yylloc, "%s", s);
  460. }