summaryrefslogtreecommitdiffstats
path: root/match.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2017-01-22 06:12:24 -0800
committerKaz Kylheku <kaz@kylheku.com>2017-01-22 06:12:24 -0800
commita928f32d7f83e7cb0cbf7a7251d3edd5ca5487ce (patch)
tree1aabfe9013641554fee6f11f43412780c06550a0 /match.c
parent1468f3fc7fba26345de8a67b66d4af5367c13037 (diff)
downloadtxr-a928f32d7f83e7cb0cbf7a7251d3edd5ca5487ce.tar.gz
txr-a928f32d7f83e7cb0cbf7a7251d3edd5ca5487ce.tar.bz2
txr-a928f32d7f83e7cb0cbf7a7251d3edd5ca5487ce.zip
Enable unbound warnings when expanding TXR code.
With this change, Lisp expansion-time warnings are no longer suppressed during the parsing of the TXR pattern language. Embedded Lisp expressions can refer to TXR pattern variables, which generates spurious warnings that must be suppressed. Since TXR pattern variables are dynamically introduced in a very flexible way, it's hard to do an exact job of this. We take the crude approach that warnings are suppressed for all pattern variables that appear anywhere in the TXR code. To do that, we identify, at parse time, all directives which can bind new variables, and register those variables as if they were tentative global defs, purging all pending warnings for them. * match.c (binding_directive_table): New static hash table. (match_reg_var, match_reg_params, match_reg_elem): New functions. (match_reg_var_rec): New static function. (dir_tables_init): gc-protect binding_directive_table, and populate its entries. * match.h (into_k, named_k): Declared. (match_reg_var, match_reg_params, match_reg_elem): Declared. * parser.y (process_catch_exprs): New static function. (elem): Call match_reg_elem for each basic directive, to process the variables in that directive according to its operator symbol. Do this for each compound form elem and variable elem. Te horizontal @(define) eleme has its own grammar production here, and we handle its parameter list in that rule. (define_clause): Handle the parameters of a vertical @(define). It binds pattern variables, and so we must suppress unbound warnings for those. (catch_clauses_opt): Process the parameters bound by @(catch) clauses. (output_clause): Suppress warnings for the variables nominated by any :into or :named argument. (expand_repeat_rep_args): Suppress warnings for :counter variable, and for :vars variables. (parse_once): Remove the warning-muffling handler frame set up around the yyparse call. * txr.c (txr_main): Suppress warnings for TXR variables defined using -D syntax on the command line. Dump deferred warnings after parsing a .txr file.
Diffstat (limited to 'match.c')
-rw-r--r--match.c68
1 files changed, 64 insertions, 4 deletions
diff --git a/match.c b/match.c
index d3c780ee..6ad9ac6a 100644
--- a/match.c
+++ b/match.c
@@ -71,7 +71,7 @@ val filter_s;
val noval_s;
static val h_directive_table, v_directive_table;
-static val non_matching_directive_table;
+static val non_matching_directive_table, binding_directive_table;
static void debuglf(val form, val fmt, ...)
{
@@ -4401,6 +4401,55 @@ val extract(val spec, val files, val predefined_bindings)
return result;
}
+void match_reg_var(val sym)
+{
+ if (bindable(sym) && !uw_tentative_def_exists(sym)) {
+ val tag = cons(var_s, sym);
+ uw_purge_deferred_warning(tag);
+ uw_register_tentative_def(tag);
+ }
+}
+
+static void match_reg_var_rec(val sym)
+{
+ if (consp(sym)) {
+ match_reg_var_rec(car(sym));
+ match_reg_var_rec(cdr(sym));
+ return;
+ }
+ match_reg_var(sym);
+}
+
+void match_reg_params(val params)
+{
+ for (; params; params = cdr(params)) {
+ val var = car(params);
+ if (atom(var))
+ match_reg_var(var);
+ else
+ match_reg_var(car(var));
+ }
+}
+
+void match_reg_elem(val elem)
+{
+ if (consp(elem)) {
+ val sym = car(elem);
+ val vpos = gethash(binding_directive_table, sym);
+ if (vpos) {
+ val var = ref(elem, vpos);
+ match_reg_var_rec(var);
+ } else if (!gethash(h_directive_table, sym) &&
+ !gethash(v_directive_table, sym))
+ {
+ elem = cdr(elem);
+ for (; consp(elem); elem = cdr(elem))
+ match_reg_var(car(elem));
+ match_reg_var(elem);
+ }
+ }
+}
+
static void syms_init(void)
{
decline_k = intern(lit("decline"), keyword_package);
@@ -4452,12 +4501,14 @@ static void syms_init(void)
static void dir_tables_init(void)
{
+ protect(&h_directive_table, &v_directive_table,
+ &non_matching_directive_table, &binding_directive_table,
+ convert(val *, 0));
+
h_directive_table = make_hash(nil, nil, nil);
v_directive_table = make_hash(nil, nil, nil);
non_matching_directive_table = make_hash(nil, nil, nil);
-
- protect(&h_directive_table, &v_directive_table,
- &non_matching_directive_table, convert(val *, 0));
+ binding_directive_table = make_hash(nil, nil, nil);
sethash(v_directive_table, skip_s, cptr(coerce(mem_t *, v_skip)));
sethash(v_directive_table, fuzz_s, cptr(coerce(mem_t *, v_fuzz)));
@@ -4558,6 +4609,15 @@ static void dir_tables_init(void)
sethash(non_matching_directive_table, do_s, t);
sethash(non_matching_directive_table, load_s, t);
sethash(non_matching_directive_table, close_s, t);
+
+ sethash(binding_directive_table, var_s, one);
+ sethash(binding_directive_table, merge_s, one);
+ sethash(binding_directive_table, bind_s, one);
+ sethash(binding_directive_table, rebind_s, one);
+ sethash(binding_directive_table, line_s, one);
+ sethash(binding_directive_table, chr_s, one);
+ sethash(binding_directive_table, data_s, one);
+ sethash(binding_directive_table, name_s, one);
}
void match_init(void)