parser.y 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716
  1. /* SPDX-License-Identifier: GPL-2.0 */
  2. /*
  3. * Copyright (C) 2002 Roman Zippel <[email protected]>
  4. */
  5. %{
  6. #include <ctype.h>
  7. #include <stdarg.h>
  8. #include <stdio.h>
  9. #include <stdlib.h>
  10. #include <string.h>
  11. #include <stdbool.h>
  12. #include "lkc.h"
  13. #include "internal.h"
  14. #define printd(mask, fmt...) if (cdebug & (mask)) printf(fmt)
  15. #define PRINTD 0x0001
  16. #define DEBUG_PARSE 0x0002
  17. int cdebug = PRINTD;
  18. static void yyerror(const char *err);
  19. static void zconfprint(const char *err, ...);
  20. static void zconf_error(const char *err, ...);
  21. static bool zconf_endtoken(const char *tokenname,
  22. const char *expected_tokenname);
  23. struct symbol *symbol_hash[SYMBOL_HASHSIZE];
  24. struct menu *current_menu, *current_entry;
  25. %}
  26. %union
  27. {
  28. char *string;
  29. struct symbol *symbol;
  30. struct expr *expr;
  31. struct menu *menu;
  32. enum symbol_type type;
  33. enum variable_flavor flavor;
  34. }
  35. %token <string> T_HELPTEXT
  36. %token <string> T_WORD
  37. %token <string> T_WORD_QUOTE
  38. %token T_BOOL
  39. %token T_CHOICE
  40. %token T_CLOSE_PAREN
  41. %token T_COLON_EQUAL
  42. %token T_COMMENT
  43. %token T_CONFIG
  44. %token T_DEFAULT
  45. %token T_DEF_BOOL
  46. %token T_DEF_TRISTATE
  47. %token T_DEPENDS
  48. %token T_ENDCHOICE
  49. %token T_ENDIF
  50. %token T_ENDMENU
  51. %token T_HELP
  52. %token T_HEX
  53. %token T_IF
  54. %token T_IMPLY
  55. %token T_INT
  56. %token T_MAINMENU
  57. %token T_MENU
  58. %token T_MENUCONFIG
  59. %token T_MODULES
  60. %token T_ON
  61. %token T_OPEN_PAREN
  62. %token T_OPTIONAL
  63. %token T_PLUS_EQUAL
  64. %token T_PROMPT
  65. %token T_RANGE
  66. %token T_SELECT
  67. %token T_SOURCE
  68. %token T_STRING
  69. %token T_TRISTATE
  70. %token T_VISIBLE
  71. %token T_EOL
  72. %token <string> T_ASSIGN_VAL
  73. %left T_OR
  74. %left T_AND
  75. %left T_EQUAL T_UNEQUAL
  76. %left T_LESS T_LESS_EQUAL T_GREATER T_GREATER_EQUAL
  77. %nonassoc T_NOT
  78. %type <symbol> nonconst_symbol
  79. %type <symbol> symbol
  80. %type <type> type logic_type default
  81. %type <expr> expr
  82. %type <expr> if_expr
  83. %type <string> end
  84. %type <menu> if_entry menu_entry choice_entry
  85. %type <string> word_opt assign_val
  86. %type <flavor> assign_op
  87. %destructor {
  88. fprintf(stderr, "%s:%d: missing end statement for this entry\n",
  89. $$->file->name, $$->lineno);
  90. if (current_menu == $$)
  91. menu_end_menu();
  92. } if_entry menu_entry choice_entry
  93. %%
  94. input: mainmenu_stmt stmt_list | stmt_list;
  95. /* mainmenu entry */
  96. mainmenu_stmt: T_MAINMENU T_WORD_QUOTE T_EOL
  97. {
  98. menu_add_prompt(P_MENU, $2, NULL);
  99. };
  100. stmt_list:
  101. /* empty */
  102. | stmt_list assignment_stmt
  103. | stmt_list choice_stmt
  104. | stmt_list comment_stmt
  105. | stmt_list config_stmt
  106. | stmt_list if_stmt
  107. | stmt_list menu_stmt
  108. | stmt_list menuconfig_stmt
  109. | stmt_list source_stmt
  110. | stmt_list T_WORD error T_EOL { zconf_error("unknown statement \"%s\"", $2); }
  111. | stmt_list error T_EOL { zconf_error("invalid statement"); }
  112. ;
  113. stmt_list_in_choice:
  114. /* empty */
  115. | stmt_list_in_choice comment_stmt
  116. | stmt_list_in_choice config_stmt
  117. | stmt_list_in_choice if_stmt_in_choice
  118. | stmt_list_in_choice error T_EOL { zconf_error("invalid statement"); }
  119. ;
  120. /* config/menuconfig entry */
  121. config_entry_start: T_CONFIG nonconst_symbol T_EOL
  122. {
  123. $2->flags |= SYMBOL_OPTIONAL;
  124. menu_add_entry($2);
  125. printd(DEBUG_PARSE, "%s:%d:config %s\n", zconf_curname(), zconf_lineno(), $2->name);
  126. };
  127. config_stmt: config_entry_start config_option_list
  128. {
  129. printd(DEBUG_PARSE, "%s:%d:endconfig\n", zconf_curname(), zconf_lineno());
  130. };
  131. menuconfig_entry_start: T_MENUCONFIG nonconst_symbol T_EOL
  132. {
  133. $2->flags |= SYMBOL_OPTIONAL;
  134. menu_add_entry($2);
  135. printd(DEBUG_PARSE, "%s:%d:menuconfig %s\n", zconf_curname(), zconf_lineno(), $2->name);
  136. };
  137. menuconfig_stmt: menuconfig_entry_start config_option_list
  138. {
  139. if (current_entry->prompt)
  140. current_entry->prompt->type = P_MENU;
  141. else
  142. zconfprint("warning: menuconfig statement without prompt");
  143. printd(DEBUG_PARSE, "%s:%d:endconfig\n", zconf_curname(), zconf_lineno());
  144. };
  145. config_option_list:
  146. /* empty */
  147. | config_option_list config_option
  148. | config_option_list depends
  149. | config_option_list help
  150. ;
  151. config_option: type prompt_stmt_opt T_EOL
  152. {
  153. menu_set_type($1);
  154. printd(DEBUG_PARSE, "%s:%d:type(%u)\n",
  155. zconf_curname(), zconf_lineno(),
  156. $1);
  157. };
  158. config_option: T_PROMPT T_WORD_QUOTE if_expr T_EOL
  159. {
  160. menu_add_prompt(P_PROMPT, $2, $3);
  161. printd(DEBUG_PARSE, "%s:%d:prompt\n", zconf_curname(), zconf_lineno());
  162. };
  163. config_option: default expr if_expr T_EOL
  164. {
  165. menu_add_expr(P_DEFAULT, $2, $3);
  166. if ($1 != S_UNKNOWN)
  167. menu_set_type($1);
  168. printd(DEBUG_PARSE, "%s:%d:default(%u)\n",
  169. zconf_curname(), zconf_lineno(),
  170. $1);
  171. };
  172. config_option: T_SELECT nonconst_symbol if_expr T_EOL
  173. {
  174. menu_add_symbol(P_SELECT, $2, $3);
  175. printd(DEBUG_PARSE, "%s:%d:select\n", zconf_curname(), zconf_lineno());
  176. };
  177. config_option: T_IMPLY nonconst_symbol if_expr T_EOL
  178. {
  179. menu_add_symbol(P_IMPLY, $2, $3);
  180. printd(DEBUG_PARSE, "%s:%d:imply\n", zconf_curname(), zconf_lineno());
  181. };
  182. config_option: T_RANGE symbol symbol if_expr T_EOL
  183. {
  184. menu_add_expr(P_RANGE, expr_alloc_comp(E_RANGE,$2, $3), $4);
  185. printd(DEBUG_PARSE, "%s:%d:range\n", zconf_curname(), zconf_lineno());
  186. };
  187. config_option: T_MODULES T_EOL
  188. {
  189. if (modules_sym)
  190. zconf_error("symbol '%s' redefines option 'modules' already defined by symbol '%s'",
  191. current_entry->sym->name, modules_sym->name);
  192. modules_sym = current_entry->sym;
  193. };
  194. /* choice entry */
  195. choice: T_CHOICE word_opt T_EOL
  196. {
  197. struct symbol *sym = sym_lookup($2, SYMBOL_CHOICE);
  198. sym->flags |= SYMBOL_NO_WRITE;
  199. menu_add_entry(sym);
  200. menu_add_expr(P_CHOICE, NULL, NULL);
  201. free($2);
  202. printd(DEBUG_PARSE, "%s:%d:choice\n", zconf_curname(), zconf_lineno());
  203. };
  204. choice_entry: choice choice_option_list
  205. {
  206. $$ = menu_add_menu();
  207. };
  208. choice_end: end
  209. {
  210. if (zconf_endtoken($1, "choice")) {
  211. menu_end_menu();
  212. printd(DEBUG_PARSE, "%s:%d:endchoice\n", zconf_curname(), zconf_lineno());
  213. }
  214. };
  215. choice_stmt: choice_entry stmt_list_in_choice choice_end
  216. ;
  217. choice_option_list:
  218. /* empty */
  219. | choice_option_list choice_option
  220. | choice_option_list depends
  221. | choice_option_list help
  222. ;
  223. choice_option: T_PROMPT T_WORD_QUOTE if_expr T_EOL
  224. {
  225. menu_add_prompt(P_PROMPT, $2, $3);
  226. printd(DEBUG_PARSE, "%s:%d:prompt\n", zconf_curname(), zconf_lineno());
  227. };
  228. choice_option: logic_type prompt_stmt_opt T_EOL
  229. {
  230. menu_set_type($1);
  231. printd(DEBUG_PARSE, "%s:%d:type(%u)\n",
  232. zconf_curname(), zconf_lineno(), $1);
  233. };
  234. choice_option: T_OPTIONAL T_EOL
  235. {
  236. current_entry->sym->flags |= SYMBOL_OPTIONAL;
  237. printd(DEBUG_PARSE, "%s:%d:optional\n", zconf_curname(), zconf_lineno());
  238. };
  239. choice_option: T_DEFAULT nonconst_symbol if_expr T_EOL
  240. {
  241. menu_add_symbol(P_DEFAULT, $2, $3);
  242. printd(DEBUG_PARSE, "%s:%d:default\n",
  243. zconf_curname(), zconf_lineno());
  244. };
  245. type:
  246. logic_type
  247. | T_INT { $$ = S_INT; }
  248. | T_HEX { $$ = S_HEX; }
  249. | T_STRING { $$ = S_STRING; }
  250. logic_type:
  251. T_BOOL { $$ = S_BOOLEAN; }
  252. | T_TRISTATE { $$ = S_TRISTATE; }
  253. default:
  254. T_DEFAULT { $$ = S_UNKNOWN; }
  255. | T_DEF_BOOL { $$ = S_BOOLEAN; }
  256. | T_DEF_TRISTATE { $$ = S_TRISTATE; }
  257. /* if entry */
  258. if_entry: T_IF expr T_EOL
  259. {
  260. printd(DEBUG_PARSE, "%s:%d:if\n", zconf_curname(), zconf_lineno());
  261. menu_add_entry(NULL);
  262. menu_add_dep($2);
  263. $$ = menu_add_menu();
  264. };
  265. if_end: end
  266. {
  267. if (zconf_endtoken($1, "if")) {
  268. menu_end_menu();
  269. printd(DEBUG_PARSE, "%s:%d:endif\n", zconf_curname(), zconf_lineno());
  270. }
  271. };
  272. if_stmt: if_entry stmt_list if_end
  273. ;
  274. if_stmt_in_choice: if_entry stmt_list_in_choice if_end
  275. ;
  276. /* menu entry */
  277. menu: T_MENU T_WORD_QUOTE T_EOL
  278. {
  279. menu_add_entry(NULL);
  280. menu_add_prompt(P_MENU, $2, NULL);
  281. printd(DEBUG_PARSE, "%s:%d:menu\n", zconf_curname(), zconf_lineno());
  282. };
  283. menu_entry: menu menu_option_list
  284. {
  285. $$ = menu_add_menu();
  286. };
  287. menu_end: end
  288. {
  289. if (zconf_endtoken($1, "menu")) {
  290. menu_end_menu();
  291. printd(DEBUG_PARSE, "%s:%d:endmenu\n", zconf_curname(), zconf_lineno());
  292. }
  293. };
  294. menu_stmt: menu_entry stmt_list menu_end
  295. ;
  296. menu_option_list:
  297. /* empty */
  298. | menu_option_list visible
  299. | menu_option_list depends
  300. ;
  301. source_stmt: T_SOURCE T_WORD_QUOTE T_EOL
  302. {
  303. printd(DEBUG_PARSE, "%s:%d:source %s\n", zconf_curname(), zconf_lineno(), $2);
  304. zconf_nextfile($2);
  305. free($2);
  306. };
  307. /* comment entry */
  308. comment: T_COMMENT T_WORD_QUOTE T_EOL
  309. {
  310. menu_add_entry(NULL);
  311. menu_add_prompt(P_COMMENT, $2, NULL);
  312. printd(DEBUG_PARSE, "%s:%d:comment\n", zconf_curname(), zconf_lineno());
  313. };
  314. comment_stmt: comment comment_option_list
  315. ;
  316. comment_option_list:
  317. /* empty */
  318. | comment_option_list depends
  319. ;
  320. /* help option */
  321. help_start: T_HELP T_EOL
  322. {
  323. printd(DEBUG_PARSE, "%s:%d:help\n", zconf_curname(), zconf_lineno());
  324. zconf_starthelp();
  325. };
  326. help: help_start T_HELPTEXT
  327. {
  328. if (current_entry->help) {
  329. free(current_entry->help);
  330. zconfprint("warning: '%s' defined with more than one help text -- only the last one will be used",
  331. current_entry->sym->name ?: "<choice>");
  332. }
  333. /* Is the help text empty or all whitespace? */
  334. if ($2[strspn($2, " \f\n\r\t\v")] == '\0')
  335. zconfprint("warning: '%s' defined with blank help text",
  336. current_entry->sym->name ?: "<choice>");
  337. current_entry->help = $2;
  338. };
  339. /* depends option */
  340. depends: T_DEPENDS T_ON expr T_EOL
  341. {
  342. menu_add_dep($3);
  343. printd(DEBUG_PARSE, "%s:%d:depends on\n", zconf_curname(), zconf_lineno());
  344. };
  345. /* visibility option */
  346. visible: T_VISIBLE if_expr T_EOL
  347. {
  348. menu_add_visibility($2);
  349. };
  350. /* prompt statement */
  351. prompt_stmt_opt:
  352. /* empty */
  353. | T_WORD_QUOTE if_expr
  354. {
  355. menu_add_prompt(P_PROMPT, $1, $2);
  356. };
  357. end: T_ENDMENU T_EOL { $$ = "menu"; }
  358. | T_ENDCHOICE T_EOL { $$ = "choice"; }
  359. | T_ENDIF T_EOL { $$ = "if"; }
  360. ;
  361. if_expr: /* empty */ { $$ = NULL; }
  362. | T_IF expr { $$ = $2; }
  363. ;
  364. expr: symbol { $$ = expr_alloc_symbol($1); }
  365. | symbol T_LESS symbol { $$ = expr_alloc_comp(E_LTH, $1, $3); }
  366. | symbol T_LESS_EQUAL symbol { $$ = expr_alloc_comp(E_LEQ, $1, $3); }
  367. | symbol T_GREATER symbol { $$ = expr_alloc_comp(E_GTH, $1, $3); }
  368. | symbol T_GREATER_EQUAL symbol { $$ = expr_alloc_comp(E_GEQ, $1, $3); }
  369. | symbol T_EQUAL symbol { $$ = expr_alloc_comp(E_EQUAL, $1, $3); }
  370. | symbol T_UNEQUAL symbol { $$ = expr_alloc_comp(E_UNEQUAL, $1, $3); }
  371. | T_OPEN_PAREN expr T_CLOSE_PAREN { $$ = $2; }
  372. | T_NOT expr { $$ = expr_alloc_one(E_NOT, $2); }
  373. | expr T_OR expr { $$ = expr_alloc_two(E_OR, $1, $3); }
  374. | expr T_AND expr { $$ = expr_alloc_two(E_AND, $1, $3); }
  375. ;
  376. /* For symbol definitions, selects, etc., where quotes are not accepted */
  377. nonconst_symbol: T_WORD { $$ = sym_lookup($1, 0); free($1); };
  378. symbol: nonconst_symbol
  379. | T_WORD_QUOTE { $$ = sym_lookup($1, SYMBOL_CONST); free($1); }
  380. ;
  381. word_opt: /* empty */ { $$ = NULL; }
  382. | T_WORD
  383. /* assignment statement */
  384. assignment_stmt: T_WORD assign_op assign_val T_EOL { variable_add($1, $3, $2); free($1); free($3); }
  385. assign_op:
  386. T_EQUAL { $$ = VAR_RECURSIVE; }
  387. | T_COLON_EQUAL { $$ = VAR_SIMPLE; }
  388. | T_PLUS_EQUAL { $$ = VAR_APPEND; }
  389. ;
  390. assign_val:
  391. /* empty */ { $$ = xstrdup(""); };
  392. | T_ASSIGN_VAL
  393. ;
  394. %%
  395. void conf_parse(const char *name)
  396. {
  397. struct symbol *sym;
  398. int i;
  399. zconf_initscan(name);
  400. _menu_init();
  401. if (getenv("ZCONF_DEBUG"))
  402. yydebug = 1;
  403. yyparse();
  404. /* Variables are expanded in the parse phase. We can free them here. */
  405. variable_all_del();
  406. if (yynerrs)
  407. exit(1);
  408. if (!modules_sym)
  409. modules_sym = sym_find( "n" );
  410. if (!menu_has_prompt(&rootmenu)) {
  411. current_entry = &rootmenu;
  412. menu_add_prompt(P_MENU, "Main menu", NULL);
  413. }
  414. menu_finalize(&rootmenu);
  415. for_all_symbols(i, sym) {
  416. if (sym_check_deps(sym))
  417. yynerrs++;
  418. }
  419. if (yynerrs)
  420. exit(1);
  421. conf_set_changed(true);
  422. }
  423. static bool zconf_endtoken(const char *tokenname,
  424. const char *expected_tokenname)
  425. {
  426. if (strcmp(tokenname, expected_tokenname)) {
  427. zconf_error("unexpected '%s' within %s block",
  428. tokenname, expected_tokenname);
  429. yynerrs++;
  430. return false;
  431. }
  432. if (current_menu->file != current_file) {
  433. zconf_error("'%s' in different file than '%s'",
  434. tokenname, expected_tokenname);
  435. fprintf(stderr, "%s:%d: location of the '%s'\n",
  436. current_menu->file->name, current_menu->lineno,
  437. expected_tokenname);
  438. yynerrs++;
  439. return false;
  440. }
  441. return true;
  442. }
  443. static void zconfprint(const char *err, ...)
  444. {
  445. va_list ap;
  446. fprintf(stderr, "%s:%d: ", zconf_curname(), zconf_lineno());
  447. va_start(ap, err);
  448. vfprintf(stderr, err, ap);
  449. va_end(ap);
  450. fprintf(stderr, "\n");
  451. }
  452. static void zconf_error(const char *err, ...)
  453. {
  454. va_list ap;
  455. yynerrs++;
  456. fprintf(stderr, "%s:%d: ", zconf_curname(), zconf_lineno());
  457. va_start(ap, err);
  458. vfprintf(stderr, err, ap);
  459. va_end(ap);
  460. fprintf(stderr, "\n");
  461. }
  462. static void yyerror(const char *err)
  463. {
  464. fprintf(stderr, "%s:%d: %s\n", zconf_curname(), zconf_lineno() + 1, err);
  465. }
  466. static void print_quoted_string(FILE *out, const char *str)
  467. {
  468. const char *p;
  469. int len;
  470. putc('"', out);
  471. while ((p = strchr(str, '"'))) {
  472. len = p - str;
  473. if (len)
  474. fprintf(out, "%.*s", len, str);
  475. fputs("\\\"", out);
  476. str = p + 1;
  477. }
  478. fputs(str, out);
  479. putc('"', out);
  480. }
  481. static void print_symbol(FILE *out, struct menu *menu)
  482. {
  483. struct symbol *sym = menu->sym;
  484. struct property *prop;
  485. if (sym_is_choice(sym))
  486. fprintf(out, "\nchoice\n");
  487. else
  488. fprintf(out, "\nconfig %s\n", sym->name);
  489. switch (sym->type) {
  490. case S_BOOLEAN:
  491. fputs(" bool\n", out);
  492. break;
  493. case S_TRISTATE:
  494. fputs(" tristate\n", out);
  495. break;
  496. case S_STRING:
  497. fputs(" string\n", out);
  498. break;
  499. case S_INT:
  500. fputs(" integer\n", out);
  501. break;
  502. case S_HEX:
  503. fputs(" hex\n", out);
  504. break;
  505. default:
  506. fputs(" ???\n", out);
  507. break;
  508. }
  509. for (prop = sym->prop; prop; prop = prop->next) {
  510. if (prop->menu != menu)
  511. continue;
  512. switch (prop->type) {
  513. case P_PROMPT:
  514. fputs(" prompt ", out);
  515. print_quoted_string(out, prop->text);
  516. if (!expr_is_yes(prop->visible.expr)) {
  517. fputs(" if ", out);
  518. expr_fprint(prop->visible.expr, out);
  519. }
  520. fputc('\n', out);
  521. break;
  522. case P_DEFAULT:
  523. fputs( " default ", out);
  524. expr_fprint(prop->expr, out);
  525. if (!expr_is_yes(prop->visible.expr)) {
  526. fputs(" if ", out);
  527. expr_fprint(prop->visible.expr, out);
  528. }
  529. fputc('\n', out);
  530. break;
  531. case P_CHOICE:
  532. fputs(" #choice value\n", out);
  533. break;
  534. case P_SELECT:
  535. fputs( " select ", out);
  536. expr_fprint(prop->expr, out);
  537. fputc('\n', out);
  538. break;
  539. case P_IMPLY:
  540. fputs( " imply ", out);
  541. expr_fprint(prop->expr, out);
  542. fputc('\n', out);
  543. break;
  544. case P_RANGE:
  545. fputs( " range ", out);
  546. expr_fprint(prop->expr, out);
  547. fputc('\n', out);
  548. break;
  549. case P_MENU:
  550. fputs( " menu ", out);
  551. print_quoted_string(out, prop->text);
  552. fputc('\n', out);
  553. break;
  554. case P_SYMBOL:
  555. fputs( " symbol ", out);
  556. fprintf(out, "%s\n", prop->menu->sym->name);
  557. break;
  558. default:
  559. fprintf(out, " unknown prop %d!\n", prop->type);
  560. break;
  561. }
  562. }
  563. if (menu->help) {
  564. int len = strlen(menu->help);
  565. while (menu->help[--len] == '\n')
  566. menu->help[len] = 0;
  567. fprintf(out, " help\n%s\n", menu->help);
  568. }
  569. }
  570. void zconfdump(FILE *out)
  571. {
  572. struct property *prop;
  573. struct symbol *sym;
  574. struct menu *menu;
  575. menu = rootmenu.list;
  576. while (menu) {
  577. if ((sym = menu->sym))
  578. print_symbol(out, menu);
  579. else if ((prop = menu->prompt)) {
  580. switch (prop->type) {
  581. case P_COMMENT:
  582. fputs("\ncomment ", out);
  583. print_quoted_string(out, prop->text);
  584. fputs("\n", out);
  585. break;
  586. case P_MENU:
  587. fputs("\nmenu ", out);
  588. print_quoted_string(out, prop->text);
  589. fputs("\n", out);
  590. break;
  591. default:
  592. ;
  593. }
  594. if (!expr_is_yes(prop->visible.expr)) {
  595. fputs(" depends ", out);
  596. expr_fprint(prop->visible.expr, out);
  597. fputc('\n', out);
  598. }
  599. }
  600. if (menu->list)
  601. menu = menu->list;
  602. else if (menu->next)
  603. menu = menu->next;
  604. else while ((menu = menu->parent)) {
  605. if (menu->prompt && menu->prompt->type == P_MENU)
  606. fputs("\nendmenu\n", out);
  607. if (menu->next) {
  608. menu = menu->next;
  609. break;
  610. }
  611. }
  612. }
  613. }