+| "%code" braceless
+ {
+ muscle_code_grow ("percent_code", $2, @2);
+ code_scanner_last_string_free ();
+ }
+| "%code" STRING braceless
+ {
+ /* FIXME: Special characters in $2 may break %code.
+ For example: `['. */
+ char const name_prefix[] = "percent_code_";
+ char *name = xmalloc (sizeof name_prefix + strlen ($2));
+ strcpy (name, name_prefix);
+ strcpy (name + sizeof name_prefix - 1, $2);
+ muscle_code_grow (uniqstr_new (name), $3, @3);
+ free (name);
+ code_scanner_last_string_free ();
+ muscle_grow_user_name_list ("user_percent_code_qualifiers", $2, @2);
+ }
+;
+
+
+/*----------*
+ | %union. |
+ *----------*/
+
+%token PERCENT_UNION "%union";
+
+union_name:
+ /* Nothing. */ {}
+| ID { muscle_code_grow ("union_name", $1, @1); }
+;
+
+grammar_declaration:
+ "%union" union_name "{...}"
+ {
+ char const *body = $3;
+
+ /* Concatenate the %union bodies. If this is the first %union, make sure
+ the synchronization line appears after the opening '{' so as not to
+ confuse Doxygen. Otherwise, turn the previous %union's trailing '}'
+ into '\n', and omit the new %union's leading '{'. */
+ if (!union_seen)
+ {
+ muscle_grow ("stype", "{", "");
+ }
+ else
+ {
+ char *code = muscle_find ("stype");
+ code[strlen (code) - 1] = '\n';
+ }
+ body++;
+
+ union_seen = true;
+ muscle_code_grow ("stype", body, @3);
+ gram_scanner_last_string_free ();
+ }