Mercurial > ~dholland > hg > ag > index.cgi
view anagram/agcore/bpu.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-1999 Parsifal Software. All Rights Reserved. * See the file COPYING for license and usage terms. * * bpu.cpp - Utility routines for bpe3 */ #include <ctype.h> #include <stdarg.h> #include "port.h" #include "arrays.h" #include "bpe3.h" #include "bpu.h" #include "cd.h" #include "data.h" #include "dict.h" #include "engdef.h" #include "keyword.h" #include "myalloc.h" #include "q1glbl.h" #include "rule.h" #include "stacks.h" #include "symbol.h" #include "token.h" //#define INCLUDE_LOGGING #include "log.h" void template_string(const char *name, const char *tp, const char tc) { ics(); while (*tp) { char c = *tp++; if (c == tc) { ass(name); continue; } acs(c); } tss(); } AgString subs_template(const char *name, const char *tp, const char tc) { template_string(name, tp, tc); return buildAgString(); } unsigned find_token_number(const char *name) { return Symbol(name)->token_number; //return map_token_name[identify_string(name,tkn_dict)].token_number; } int find_completions(int s, const unsigned **q) { state_number_map *sp = &map_state_number[s]; const unsigned *p = lstptr(*sp, chain_completions); int n = sp->n_chain_completions; if (n == 0 && sp->chain_gotos_index == 0) { p = lstptr(*sp, completions); n = sp->n_completions; } *q = p; return n; } void select_actions(int k) { int n = tis()/3; int *lb; list_space(n+1); lb = list_base; iws(); while (n--) { aws(lb[k]); k += 3; } } void find_key_tokens(const int *pl, int n, int tw /*, int ct */) { LOGSECTION("find_key_tokens"); LOGV(n); for (; n--; pl += tw) { //int tn = *pl; Token token = *pl; LOGV(token); //if (map_token_number[tn].key == 0) continue; //if (token->key == 0) continue; if (token->key.isNull()) { continue; } isws(token); } } /* * Note that the text of the defined format must not contain a newline. * Failure to abide by this rule will cause the line count to be * incorrect. */ void define_macro(const char *n, const char *fmt, ...) { va_list ap; fprintf(pe_file, "#define %s ", n); va_start(ap, fmt); vfprintf(pe_file, fmt, ap); va_end(ap); fputc('\n', pe_file); pe_line_count++; } void define_macro_default(const char *n, const char *fmt, ...) { va_list ap; char name_only[80]; char *ptr; strcpy(name_only, n); ptr = strchr(name_only, '('); if (ptr != NULL) { *ptr = 0; } fprintf(pe_file, "#ifndef %s\n#define %s ", name_only, n); va_start(ap, fmt); vfprintf(pe_file, fmt, ap); va_end(ap); fprintf(pe_file, "\n#endif\n"); pe_line_count += 3; } unsigned char *key_string(int tn) { LOGSECTION("key_string"); Keyword key = map_token_number[tn].key; LOGV(tn) LCV(key) LCV(Keyword::count()); if (key.isNull()) { return NULL; } LOGV(key->string); return (unsigned char *) key->string.pointer(); } /* * Function: reducing_token * Arguments: * token number, tn * state number, sn * Returns: 0 if tn is _not_ a reducing token in the given state * Otherwise, returns the number of the rule it reduces. */ #if 0 /* no longer used */ unsigned reducing_token(unsigned tn, unsigned sn) { state_number_map *sp = &map_state_number[sn]; const unsigned *p; unsigned n = sp->n_reductions; /* number of reductions in this state */ if (n) { p = lstptr(*sp, reductions); while (n--) { if (*p++ == tn) { return *p; } p++; } return 0; } if (sp->n_completed_forms != 1) { return 0; } if (shift_token(tn, sn)) { return 0; } return *(lstptr(*sp,completed_forms)); } #endif /* 0 - no longer used */ Rule ruleReducedBy(Token token, unsigned sn) { state_number_map *sp = &map_state_number[sn]; const unsigned *p; unsigned n = sp->n_reductions; /* number of reductions in this state */ if (n) { p = lstptr(*sp, reductions); while (n--) { if (*p++ == (unsigned) token) { return *p; } p++; } return 0; } if (sp->n_completed_forms != 1) { // Exit if more than one completed rule return 0; } if (shift_token(token, sn)) { // Exit if there's a shift return 0; } // Default reduction, return rule # return *(lstptr(*sp, completed_forms)); } void select_write_fragment(const char *name, const char *modes, const char *suffix) { char fragment[80]; strcpy(fragment, name); strcat(fragment, modes); strcat(fragment, suffix); write_code_segment(fragment); } int shift_token(unsigned tn, unsigned sn) { const unsigned *p; unsigned n; n = find_gotos(sn, &p); while (n--) { if (*p == tn) { return 1; } else { p += 2; } } n = find_completions(sn, &p); while (n--) { if (*p == tn) { return 1; } else { p += 2; } } return 0; } void count_pe_line(const char *s, int k) { LOGSECTION("count_pe_line"); while (k--) { if (s[k] == '\n') { pe_line_count++; } } LOGV(pe_line_count); } int wps(const char *s) { LOGSECTION("wps"); int k = strlen(s); fputs(s, pe_file); count_pe_line(s, k); return k; } int wss(void) { count_pe_line(string_base, tis()); fputs(string_base, pe_file); return rcs(); } int wpe(const char *fs, ...) { LOGSECTION("wpe"); va_list ap; char buf[2000]; va_start(ap, fs); vsprintf(buf, fs, ap); va_end(ap); return wps(buf); } void write_code_segment(const char *name) { LOGSECTION("write_code_segment"); LOGV(name); if (name == NULL) { return; } AgString text = code_segment(name); if (!text.exists()) { return; } char *buf = text.pointer(); wps(buf); wps("\n\n"); } static AgString expand_my_macros(const char *s) { LOGSECTION("expand_my_macros"); char *p; int col_no = 0, lb = 0, k; LOGV(s); if (my_macros == NULL) { return AgString(s); } ics(); while (*s) { char *word; int index; while (*s && !isalpha(*s) && *s != '_') { if (*s == '\n') { acs(*s++); col_no = 0; while (*s && *s == ' ') { col_no++, s++; } lb = col_no; continue; } if (lb == col_no) { for (k = lb; k--; ) { acs(' '); } } col_no++; acs(*s++); } if (*s == 0) { break; } ics(); while (*s && (isalpha(*s) || *s == '_')) { acs(*s++); } word = build_string(); index = identify_string(word, my_macros); if (index == 0) { if (lb == col_no) { for (k = lb; k--; ) { acs(' '); } } col_no += strlen(word); ass(word); DEALLOCATE(word); continue; } DEALLOCATE(word); p = my_macros_subs[index]; if (*p == 0 && lb == col_no && (*s == '\n' || strncmp(s,"\\\n",2) == 0)) { if (*s == '\\') { s++; } s++; col_no = 0; while (*s && *s == ' ') { col_no++; s++; } lb = col_no; continue; } if (lb == col_no) for (k = lb; k--;) acs(' '); for (; *p; p++ ) { acs(*p); col_no++; if (*p == '\n') { for (k = lb; k--; ) { acs(' '); } col_no = lb; continue; } } if (*s == '#') { s++; } } LOGV(string_base); return buildAgString(); } AgString code_segment(const char *name) { LOGSECTION("code_segment"); const char *rawtext = engdef_get(name); LOGV(rawtext); AgString mx = expand_my_macros(rawtext); LOGV(mx); return mx; }