summaryrefslogtreecommitdiff
path: root/lib/predicate.c
diff options
context:
space:
mode:
Diffstat (limited to 'lib/predicate.c')
-rw-r--r--lib/predicate.c241
1 files changed, 241 insertions, 0 deletions
diff --git a/lib/predicate.c b/lib/predicate.c
new file mode 100644
index 0000000..67614cc
--- /dev/null
+++ b/lib/predicate.c
@@ -0,0 +1,241 @@
+/* dctrl-tools - Debian control file inspection tools
+ Copyright (C) 2003, 2004 Antti-Juhani Kaijanaho
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
+*/
+
+#include <ctype.h>
+#include <stdlib.h>
+#include <regex.h>
+#include <string.h>
+#include "fsaf.h"
+#include "msg.h"
+#include "util.h"
+#include "predicate.h"
+#include "strutil.h"
+#include "version.h"
+
+void init_predicate(struct predicate * p)
+{
+ p->num_atoms = 0;
+ p->proglen = 0;
+}
+
+void addinsn(struct predicate * p, int insn)
+{
+ if (insn == I_NOP) return;
+ if (p->proglen >= MAX_OPS) {
+ message(L_FATAL, _("predicate is too complex"), 0);
+ fail();
+ }
+ p->program[p->proglen++] = insn;
+}
+
+void predicate_finish_atom(struct predicate * p)
+{
+ struct atom * atom = get_current_atom(p);
+ debug_message("predicate_finish_atom", 0);
+ if (atom->field_name != 0) {
+ char * repl = strchr(atom->field_name, ':');
+ if (repl != NULL) {
+ *repl++ = '\0';
+ atom->repl_inx = fieldtrie_insert(repl);
+ } else {
+ atom->repl_inx = -1;
+ }
+ atom->field_inx = fieldtrie_insert(atom->field_name);
+ }
+
+ if (atom->mode == M_REGEX || atom->mode == M_EREGEX) {
+ debug_message("compiling:", 0);
+ debug_message(atom->pat, 0);
+ int rerr = regcomp(&atom->regex, atom->pat,
+ (atom->mode == M_EREGEX ? REG_EXTENDED : 0)
+ | REG_NOSUB
+ | (atom->ignore_case ? REG_ICASE : 0));
+ if (rerr != 0) {
+ char * s;
+ s = get_regerror(rerr, &atom->regex);
+ if (s == 0) fatal_enomem(0);
+ message(L_FATAL, s, 0);
+ free(s);
+ fail();
+ }
+ }
+
+}
+
+static bool verify_atom(struct atom * atom, para_t * para)
+{
+ size_t start, end;
+ if (atom->field_inx == -1) {
+ /* Take the full paragraph */
+ start = para->start;
+ end = para->end;
+ } else {
+ /* Take the field */
+ struct field_data * fd = &para->fields[atom->field_inx];
+ if (fd->start == fd->end && atom->repl_inx != (size_t)(-1)) {
+ fd = &para->fields[atom->repl_inx];
+ }
+ start = fd->start;
+ end = fd->end;
+ }
+ size_t len = end - start;
+ struct fsaf_read_rv r = fsaf_read(para->common->fp, start, len);
+ assert(r.len == len);
+ switch (atom->mode) {
+ case M_EXACT:
+ if (len != atom->patlen) return false;
+ if (atom->ignore_case) {
+ return strncasecmp(atom->pat, r.b, len) == 0;
+ } else {
+ return strncmp(atom->pat, r.b, len) == 0;
+ }
+ case M_SUBSTR: {
+#if 0
+ if (atom->ignore_case) {
+ return strncasestr(r.b, atom->pat, len);
+ } else {
+ return strnstr(r.b, atom->pat, len);
+ }
+#else
+ bool rv;
+ char * s = strndup(r.b, len);
+ if (s == 0) fatal_enomem(0);
+ if (atom->ignore_case) {
+ rv = strcasestr(s, atom->pat) != 0;
+ } else {
+ rv = strstr(s, atom->pat) != 0;
+ }
+ free(s);
+ return rv;
+#endif
+ }
+ case M_REGEX: case M_EREGEX: {
+ char * s = strndup(r.b, len);
+ if (s == 0) fatal_enomem(0);
+ int regex_errcode = regexec(&atom->regex, s, 0, 0, 0);
+ free(s);
+ if (regex_errcode == 0 || regex_errcode == REG_NOMATCH) {
+ return (regex_errcode == 0);
+ }
+ /* Error handling be here. */
+ assert(regex_errcode != 0 && regex_errcode != REG_NOMATCH);
+ s = get_regerror (regex_errcode, &atom->regex);
+ if (s == 0) { enomem (0); return false; }
+ message(L_IMPORTANT, s, 0);
+ free(s);
+ return false;
+ }
+ case M_VER_EQ:case M_VER_LT:case M_VER_LE:case M_VER_GT:case M_VER_GE:
+ ;
+ char *pats = strndup(atom->pat, atom->patlen);
+ char *cands = strndup(r.b, len);
+ struct versionrevision pat, cand;
+ if (!parse_version(&pat, pats, atom->patlen)) {
+ free(pats);
+ free(cands);
+ return false;
+ }
+ if (!parse_version(&cand, cands, len)) {
+ free(pats);
+ free(cands);
+ return false;
+ }
+ int res = versioncompare(&cand, &pat);
+ free(pats);
+ free(cands);
+ switch (atom->mode) {
+ case M_VER_EQ:
+ return res == 0;
+ case M_VER_LT:
+ return res < 0;
+ case M_VER_LE:
+ return res <= 0;
+ case M_VER_GT:
+ return res > 0;
+ case M_VER_GE:
+ return res >= 0;
+ default:
+ assert(0);
+ }
+ }
+ assert(0);
+}
+
+bool check_predicate(struct predicate * p)
+{
+ size_t sp = 0;
+ /* Simulate the program. */
+ for (size_t i = 0; i < p->proglen; i++) {
+ switch (p->program[i]) {
+ case I_NOP: break;
+ case I_NEG:
+ if (sp == 0) return false;
+ break;
+ case I_AND: case I_OR:
+ if (sp < 2) return false;
+ --sp;
+ break;
+ default:
+ ++sp;
+ }
+ }
+ if (sp != 1) return false;
+ return true;
+}
+
+bool does_para_satisfy(struct predicate * p, para_t * para)
+{
+ bool sat_atom[MAX_ATOMS];
+ bool stack[MAX_OPS];
+ size_t sp = 0;
+
+ /* Verify atoms. */
+ for (size_t i = 0; i < p->num_atoms; i++) {
+ sat_atom[i] = verify_atom(&p->atoms[i], para);
+ }
+
+ /* Run the program. */
+ for (size_t i = 0; i < p->proglen; i++) {
+ switch (p->program[i]) {
+ case I_NOP: break;
+ case I_NEG:
+ assert(sp >= 1);
+ stack[sp-1] = !stack[sp-1];
+ break;
+ case I_AND:
+ assert(sp >= 2);
+ stack[sp-2] = stack[sp-2] && stack[sp-1];
+ --sp;
+ break;
+ case I_OR:
+ assert(sp >= 2);
+ stack[sp-2] = stack[sp-2] || stack[sp-1];
+ --sp;
+ break;
+ default:
+ {
+ int atom = p->program[i] - I_PUSH(0);
+ assert(atom <= p->num_atoms);
+ stack[sp] = sat_atom[atom];
+ ++sp;
+ }
+ }
+ }
+ assert(sp == 1);
+ return stack[0];
+}