Mercurial > ~dholland > hg > ag > index.cgi
view anagram/agcore/nckwtr.cpp @ 20:bb115deb6fb2
Improve agfiles rule.
(1) It didn't depend on $(AGCL) and it absolutely should have.
(2) allow AGFORCE=1 to make it rebuild whether or not it looks out of
date.
(3) Document this.
author | David A. Holland |
---|---|
date | Mon, 13 Jun 2022 00:02:15 -0400 |
parents | 13d2b8934445 |
children |
line wrap: on
line source
/* * AnaGram, A System for Syntax Directed Programming * Copyright 1993-2002 Parsifal Software. All Rights Reserved. * See the file COPYING for license and usage terms. * * nckwtr.cpp - Conflict/Keyword Anomaly Trace Utility */ #include "agbaltree.h" #include "arrays.h" #include "assert.h" #include "cd.h" #include "data.h" #include "dict.h" #include "lexeme.h" #include "myalloc.h" #include "nckwtr.h" #include "q1glbl.h" #include "rule.h" #include "stacks.h" #include "token.h" #include "tsd.h" //#define INCLUDE_LOGGING #include "log.h" static int conflict_rule; static int conflict_state; /* * Strategy * * The data given are two state numbers: the conflict state and the * reduction state. In addition, the conflict rule is given. * * The approach is to work backward from the reduction state to a * state that leads to the conflict state. * * If the characteristic token for the reduction state includes the * conflict rule as an expansion rule, then we don't have to worry * about null productions. Otherwise, there has been a shift of some * zero-length production to get us to the reduction state. * * The first routine to build is a function which will call itself * recursively to find the fork state. It will fail if it cannot find * a fork state from its present state. */ static int keyword_switch = 0; static AgBalancedTree<Triple<int> > triples; AgArray<int> new_reduction_stack; int new_reduction_stack_depth; AgArray<int> new_conflict_stack; int new_conflict_stack_depth; tsd *new_rule_derivation; static int simple_path(int from, int to) { unsigned *p = lstptr(map_state_number[to],previous_states); int nt = map_state_number[to].n_previous_states; int i; if (from == to) return 1; for (i = 0; i < nt; i++) { if (xws(p[i])) { continue; } if (simple_path(from, p[i])) { return 1; } fws(); } return 0; } void clear_conflict_expansion(void) { if (conflict_token == 0) { return; } new_rule_derivation = delete_tsd(new_rule_derivation); token_derivation = delete_tsd(token_derivation); new_reduction_stack = AgArray<int>(); new_conflict_stack = AgArray<int>(); conflict_token = 0; } static int new_analysis(int sn, int rn); void analyze_conflict(int csn, int tn, int cfn, int kws) { int flag; LOGSECTION("analyze_conflict"); kws = kws!=0; if (conflict_state == csn && conflict_token == tn && keyword_switch == kws && conflict_rule == cfn) { return; } clear_conflict_expansion(); conflict_state = csn; conflict_token = tn; conflict_rule = cfn; keyword_switch = kws; flag = new_analysis(conflict_state, conflict_rule); LOGV(flag); assert(flag); triples.reset(); } static int equiv_token(Token charToken, Token ruleToken) { if (charToken == ruleToken) { return 1; } AgArray<Rule> expansion = charToken->expansionRuleList; int n = expansion.size(); while (n--) { if (expansion[n]->length() == 0 ) { continue; } if (expansion[n]->length() > 1 && (unsigned) expansion[n].token(1) != disregard_token) { continue; } if (expansion[n].token(0) == ruleToken) { return 1; } } return 0; } static void validate_conflict_stack(void) { int *ctp = list_base; int cfsd = tis(); int sx = 0; int nsn = ctp[sx++]; int kr = 0; int kr_lim = new_rule_derivation->nt; int rn, rx; LOGSECTION("validate_conflict_stack"); LOGV(sx) LCV(cfsd); if (sx >= cfsd) return; #ifdef INCLUDE_LOGGING { int i; for (i = 0; i < cfsd; i++) { LOGV(i) LCV(ctp[i]); } int *sb = new_rule_derivation->sb; for (i = 0; i < kr_lim; i += 2) { LOGV(sb[i]) LCV(sb[i+1]); } } #endif check_tsd(new_rule_derivation); xtxf(new_rule_derivation,kr, &rn, &rx); LOGV(rn) LCV(rx); while (rx == 0) { xtxf(new_rule_derivation, ++kr, &rn, &rx); rx--; LOGV(rn) LCV(rx); } if (sx < cfsd) { while (1) { Token charToken = map_state_number[nsn].char_token; int sn = ctp[sx++]; LOGV(kr) LCV(kr_lim); assert(kr < kr_lim); //assert(equiv_token(tn, lstptr(map_form_number[rn], tokens)[--rx])); Token ruleToken = Rule(rn).token(--rx); LOGV(rn) LCV(Rule(rn)->length()) LCV(rx) LCV(ruleToken); //assert(equiv_token(charToken, Rule(rn).token(--rx))); assert(equiv_token(charToken, ruleToken)); LOGV(nsn) LCV(sx) LCV(sn) LCV(charToken); //int nextState = new_next_state(sn, ruleToken); //LOGV(nextState); //assert(nsn == nextState); assert(equiv_token(charToken, transitionToken(sn,nsn))); LOGV(sx) LCV(cfsd); if (sx >= cfsd) { break; } nsn = sn; while (rx == 0) { xtxf(new_rule_derivation, ++kr, &rn, &rx); rx--; } } } LOGS("Exiting validate_conflict_stack"); } /* static int check_leading_token(unsigned hlt, unsigned llt) { unsigned *ltp = lstptr(map_token_number[hlt],leading_tokens); int nlt = map_token_number[hlt].n_leading_tokens; if (hlt == llt) { return 1; } while (nlt--) { if (ltp[nlt] == llt) { return 1; } } return 0; } */ static int reduction_state_path(int sn) { LOGSECTION("reduction_state_path"); LOGV(sn); int *is = dict_str(isht_dict, sn); int nis = (*is++ - 1)/2; while (nis --) { int stateNumber = sn; Rule rn = *is++; unsigned rx = *is++; LOGV(stateNumber) LCV(rn) LCV(rx); iws(); while (rx < rn->length()) { Token tn = rn.token(rx); //int flag = check_leading_token(tn,conflict_token); int flag = tn.isLeadingToken(conflict_token); aws(stateNumber); LOGV(stateNumber) LCV(tis()); if (flag) { at(new_rule_derivation, (int) rn, rx); return 1; } //if (!map_token_number[tn].zero_length_flag) { // break; //} if (!tn->zero_length_flag) { break; } LOGV(rn) LCV(rx) LCV(sn) LCV(tn); stateNumber = new_next_state(stateNumber,tn); if (stateNumber == 0) { break; } rx++; } rws(); } return 0; } static int ct_accessible(int sn) { LOGSECTION("ct_accessible"); LOGV(sn); int *is = dict_str(isht_dict, sn); int nis = (*is++ - 1)/2; while (nis--) { Rule rn = *is++; unsigned rx = *is++; while (rx < rn->length()) { //int tn = lstptr(map_form_number[rn], tokens)[rx]; Token tn = rn.token(rx); //int flag = check_leading_token(tn,conflict_token); int flag = tn.isLeadingToken(conflict_token); if (flag) { return 1; } //if (!map_token_number[tn].zero_length_flag) { // break; //} if (!tn->zero_length_flag) { break; } LOGV(rn) LCV(rx) LCV(sn) LCV(tn); sn = new_next_state(sn,tn); if (sn == 0) { break; } rx++; } } return 0; } static int anomalous_keyword_path(int sn) { int *is = dict_str(isht_dict, sn); int nis = (*is++ - 1)/2; while (nis--) { Rule rn = *is++; unsigned rx = *is++; if (rx >= rn->non_vanishing_length) { return 0; } } is = dict_str(isht_dict, sn); nis = (*is++ - 1)/2; while (nis--) { Rule rn = *is++; unsigned rx = *is++; if (rx < rn->length()) { sws(sn); at(new_rule_derivation, (int) rn, rx); return 1; } } return 0; } static int hw_reduce(int sn, int rn, int rx); static int nulls_remaining(int sn, int nsn) { LOGSECTION("nulls_remaining"); LOGV(sn) LCV(nsn); int *is = dict_str(isht_dict, nsn); int nis = (*is++ - 1)/2; while (nis --) { Rule rn = *is++; unsigned rx = *is++; if (rx < rn->non_vanishing_length) { continue; } at(new_rule_derivation, (int) rn, rx); if (hw_reduce(sn, rn, rx - 1)) { return 1; } new_rule_derivation->nt--; } return 0; } static int hw_reduce(int sn, int rn, int rx) { LOGSECTION("hw_reduce"); //if (triples.insert(IntegerTriple(sn, rn, rx))) { // return 0; //} if (triples.insert(Triple<int>(sn, rn, rx))) { return 0; } LOGV(sn) LCV(rn) LCV(rx); if (rx) { int i, j; unsigned *p = lstptr(map_state_number[sn],previous_states); int nt = map_state_number[sn].n_previous_states; //int *sorted = local_array(nt, int); LocalArray<int> sorted(nt); for (i = 0; i < nt; i++) { sorted[i] = p[i]; LOGV(p[i]); } for (i = 0; i < nt; i++) { //aws(p[i]); //if (xws(sorted[i])) continue; for (j = i+1; j < nt; j++) { if (sorted[i] > sorted[j]) { int temp = sorted[i]; sorted[i] = sorted[j]; sorted[j] = temp; } } LOGV(sorted[i]); //if (xws(sorted[i])) continue; aws(sorted[i]); #ifdef INCLUDE_LOGGING { for (int index = 0; index < tis(); index++) { LOGV(index) LCV(list_base[index]); } } LOGV(sorted[i]) LCV(tis()); #endif if (hw_reduce(sorted[i], rn, rx-1)) { return 1; } fws(); } return 0; } { int *sp = ibnfs + ibnfb[rn]; int m = ibnfn[rn]; int i; for (i = 0; i < m; i++) { unsigned *gtp = lstptr(map_state_number[sn],gotos); int ngt = map_state_number[sn].n_gotos; unsigned tn = sp[i]; int nsn, crn, crx; int j, k; for (j = k = 0; j < ngt; j++, k+=2) { if (gtp[k] == tn) break; } if (j < ngt) { nsn = gtp[k+1]; if (keyword_switch) { if (ct_accessible(nsn)) { continue; } if (nulls_remaining(sn, nsn)) { return 1; } if (anomalous_keyword_path(nsn)) { return 1; } } else { if (reduction_state_path(nsn)) { return 1; } if (nulls_remaining(sn, nsn)) { return 1; } } continue; } gtp = lstptr(map_state_number[sn], completions); ngt = map_state_number[sn].n_completions; for (j = k = 0; j < ngt; j++, k += 2) { if (gtp[k] == tn) { break; } } if (j == ngt) { continue; } crn = gtp[k+1]; crx = Rule(crn)->length(); int tx = new_rule_derivation->nt; while (tx--) { int trn, trx; xtx(new_rule_derivation, tx, &trn, &trx); if (trn == crn && trx == crx) { break; } } if (tx >= 0) { continue; } at(new_rule_derivation, crn, crx); if (hw_reduce(sn, crn, crx - 1)) { return 1; } new_rule_derivation->nt--; } } return 0; } /* static void trim_ct(void) { int nt = new_rule_derivation->nt; int k = nt - 1; int i, j; LOGSECTION("trim_ct"); AgArray<int> list(buildStaticList()); LOGV(list.size()); int *cs = list.pointer(); int ncs = list.size(); // cs is list of states in reverse order int rn,rx, tn; unsigned *rl; int nrl; check_tsd(new_rule_derivation); xtxf(new_rule_derivation, k, &rn, &rx); tn = lstptr(map_form_number[rn],tokens)[rx-1]; rl = lstptr(map_token_number[tn],forms); nrl = map_token_number[tn].n_forms; iws(); while (--k >= 0) { for (i = 0; i < k; i++) { unsigned srn, srx; int psx; xtxf(new_rule_derivation, i, &srn, &srx); for (j = 0; j < nrl; j++) if (srn == rl[j]) break; if (j == nrl) continue; psx = ncs; for (j = i+1; j < k + 1; j++) { int jrn, jrx; xtxf(new_rule_derivation, j, &jrn, &jrx); psx -= jrx-1; } if (psx <= 1) continue; if (cs[ncs-1] != cs[psx-1]) continue; ncs = psx; break; } if (i < k) { memmove(&new_rule_derivation->sb[2*(i+1)], &new_rule_derivation->sb[2*(k+1)], (nt - k - 1)*2*sizeof(int)); nt -= k-i; new_rule_derivation->nt = nt; k = i; } if (k <= 0) break; xtxf(new_rule_derivation, k, &rn, &rx); tn = lstptr(map_form_number[rn],tokens)[--rx]; rl = lstptr(map_token_number[tn],forms); nrl = map_token_number[tn].n_forms; while (rx--) aws(cs[--ncs]); } while (ncs--) aws(cs[ncs]); list = buildStaticList(); cs = list.pointer(); ncs = list.size(); iws(); while (ncs--) aws(cs[ncs]); } */ static int new_analysis(int sn, int rn) { int length = Rule(rn)->length(); LOGSECTION("new_analysis"); LOGV(sn) LCV(rn); new_rule_derivation = init_tsd(2); at(new_rule_derivation, rn, length); sws(sn); if (hw_reduce(sn,rn,length)) { int fs; new_reduction_stack = buildStaticList(); new_reduction_stack_depth = new_reduction_stack.size(); fs = fws(); //if (tis()) trim_ct(); validate_conflict_stack(); sws(fs); simple_path(0,fs); concat_list(); new_conflict_stack = buildStaticList(); new_conflict_stack_depth = new_conflict_stack.size(); iws(); while(new_reduction_stack_depth--) { aws(new_reduction_stack[new_reduction_stack_depth]); } sws(fs); simple_path(0, fs); concat_list(); new_reduction_stack = buildStaticList(); new_reduction_stack_depth = new_reduction_stack.size(); if (!keyword_switch) { int fn, fx, k; k = new_rule_derivation->nt - 1; xtxf(new_rule_derivation, k, &fn, &fx); token_derivation = init_tsd(2); tried = ZALLOCATE(ntkns+1, char); //memset(tried,0,ntkns+1); //x9x(lstptr(map_form_number[fn], tokens)[fx]); x9x(Rule(fn).token(fx)); at(token_derivation, fn, fx); DEALLOCATE(tried); } return 1; } rws(); return 0; }