+ yylex_destroy ();
+}
+
+static
+void at_directive_perform (int at_directive_argc,
+ char *at_directive_argv[],
+ char **outnamep, int *out_linenop)
+{
+ if (0 == strcmp (at_directive_argv[0], "@basename"))
+ {
+ if (at_directive_argc > 2)
+ fail_for_at_directive_too_many_args (at_directive_argv[0]);
+ fputs (last_component (at_directive_argv[1]), yyout);
+ }
+ else if (0 == strcmp (at_directive_argv[0], "@warn")
+ || 0 == strcmp (at_directive_argv[0], "@complain")
+ || 0 == strcmp (at_directive_argv[0], "@fatal"))
+ {
+ void (*func)(char const *, ...);
+ switch (at_directive_argv[0][1])
+ {
+ case 'w': func = warn; break;
+ case 'c': func = complain; break;
+ case 'f': func = fatal; break;
+ default: aver (false); break;
+ }
+ switch (at_directive_argc)
+ {
+ case 2:
+ func (_(at_directive_argv[1]));
+ break;
+ case 3:
+ func (_(at_directive_argv[1]), at_directive_argv[2]);
+ break;
+ case 4:
+ func (_(at_directive_argv[1]), at_directive_argv[2],
+ at_directive_argv[3]);
+ break;
+ case 5:
+ func (_(at_directive_argv[1]), at_directive_argv[2],
+ at_directive_argv[3], at_directive_argv[4]);
+ break;
+ case 6:
+ func (_(at_directive_argv[1]), at_directive_argv[2],
+ at_directive_argv[3], at_directive_argv[4],
+ at_directive_argv[5]);
+ break;
+ default:
+ fail_for_at_directive_too_many_args (at_directive_argv[0]);
+ break;
+ }
+ }
+ else if (0 == strcmp (at_directive_argv[0], "@warn_at")
+ || 0 == strcmp (at_directive_argv[0], "@complain_at")
+ || 0 == strcmp (at_directive_argv[0], "@fatal_at"))
+ {
+ void (*func)(location, char const *, ...);
+ location loc;
+ if (at_directive_argc < 4)
+ fail_for_at_directive_too_few_args (at_directive_argv[0]);
+ switch (at_directive_argv[0][1])
+ {
+ case 'w': func = warn_at; break;
+ case 'c': func = complain_at; break;
+ case 'f': func = fatal_at; break;
+ default: aver (false); break;
+ }
+ boundary_set_from_string (&loc.start, at_directive_argv[1]);
+ boundary_set_from_string (&loc.end, at_directive_argv[2]);
+ switch (at_directive_argc)
+ {
+ case 4:
+ func (loc, _(at_directive_argv[3]));
+ break;
+ case 5:
+ func (loc, _(at_directive_argv[3]), at_directive_argv[4]);
+ break;
+ case 6:
+ func (loc, _(at_directive_argv[3]), at_directive_argv[4],
+ at_directive_argv[5]);
+ break;
+ case 7:
+ func (loc, _(at_directive_argv[3]), at_directive_argv[4],
+ at_directive_argv[5], at_directive_argv[6]);
+ break;
+ case 8:
+ func (loc, _(at_directive_argv[3]), at_directive_argv[4],
+ at_directive_argv[5], at_directive_argv[6],
+ at_directive_argv[7]);
+ break;
+ default:
+ fail_for_at_directive_too_many_args (at_directive_argv[0]);
+ break;
+ }
+ }
+ else if (0 == strcmp (at_directive_argv[0], "@output"))
+ {
+ if (at_directive_argc > 2)
+ fail_for_at_directive_too_many_args (at_directive_argv[0]);
+ if (*outnamep)
+ {
+ free (*outnamep);
+ xfclose (yyout);
+ }
+ *outnamep = xstrdup (at_directive_argv[1]);
+ output_file_name_check (*outnamep);
+ yyout = xfopen (*outnamep, "w");
+ *out_linenop = 1;
+ }
+ else
+ fail_for_invalid_at (at_directive_argv[0]);
+}
+
+static void
+fail_for_at_directive_too_few_args (char const *at_directive_name)
+{
+ fatal (_("too few arguments for %s directive in skeleton"),
+ at_directive_name);
+}
+
+static void
+fail_for_at_directive_too_many_args (char const *at_directive_name)
+{
+ fatal (_("too many arguments for %s directive in skeleton"),
+ at_directive_name);
+}
+
+static void
+fail_for_invalid_at (char const *at)
+{
+ fatal ("invalid @ in skeleton: %s", at);