📄 hsregex.c
字号:
/***************************************************************** * HMMER - Biological sequence analysis with profile HMMs * Copyright (C) 1992-1999 Washington University School of Medicine * All Rights Reserved * * This source code is distributed under the terms of the * GNU General Public License. See the files COPYING and LICENSE * for details. *****************************************************************//***************************************************************** * This code is an altered version of Henry Spencer's * regex library. Alterations are limited to minor streamlining, * and some name changes to protect the SQUID namespace. * Henry's copyright notice appears below. * You can obtain the original from * ftp://ftp.zoo.toronto.edu/pub/bookregex.tar.Z * Thanks, Henry! * * SRE, Fri Aug 28 11:10:17 1998 * RCS $Id: hsregex.c,v 1.6 2001/08/04 21:46:47 eddy Exp $ *****************************************************************/ #include <stdio.h>#include <stdlib.h>#include <string.h>#include <ctype.h>#include "squid.h"/* global sqd_parse[] are managed by Strparse(). * WARNING: TODO: this code is not threadsafe, and needs to be revised. */char *sqd_parse[10];/* Function: Strparse() * * Purpose: Match a regexp to a string. Returns 1 if pattern matches, * else 0. * * Much like Perl, Strparse() makes copies of the matching * substrings available via globals, sqd_parse[]. * sqd_parse[0] contains a copy of the complete matched * text. sqd_parse[1-9] contain copies of up to nine * different substrings matched within parentheses. * The memory for these strings is internally managed and * volatile; the next call to Strparse() may destroy them. * If the caller needs the matched substrings to persist * beyond a new Strparse() call, it must make its own * copies. * * A minor drawback of the memory management is that * there will be a small amount of unfree'd memory being * managed by Strparse() when a program exits; this may * confuse memory debugging (Purify, dbmalloc). The * general cleanup function SqdClean() is provided; * you can call this before exiting. * * Uses an extended POSIX regular expression interface. * A copylefted GNU implementation is included in the squid * implementation (gnuregex.c) for use on non-POSIX compliant * systems. POSIX 1003.2-compliant systems (all UNIX, * some WinNT, I believe) can omit the GNU code if necessary. * * I built this for ease of use, not speed nor efficiency. * * Example: Strparse("foo-...-baz", "foo-bar-baz") returns 0 * Strparse("foo-(...)-baz", "foo-bar-baz") * returns 0; sqd_parse[0] is "foo-bar-baz"; * sqd_parse[1] is "bar". * * Args: rexp - regular expression, extended POSIX form * s - string to match against * ntok - number of () substrings we will save (maximum NSUBEXP-1) * * Return: 1 on match, 0 if no match */intStrparse(char *rexp, char *s, int ntok){ sqd_regexp *pat; int code; int len; int i; /* sanity check */ if (ntok >= NSUBEXP ) Die("Strparse(): ntok must be <= %d", NSUBEXP-1); /* Free previous global substring buffers */ for (i = 0; i <= ntok; i++) if (sqd_parse[i] != NULL) { free(sqd_parse[i]); sqd_parse[i] = NULL; } /* Compile and match the pattern, using our modified * copy of Henry Spencer's regexp library */ if ((pat = sqd_regcomp(rexp)) == NULL) Die("regexp compilation failed."); code = sqd_regexec(pat, s); /* Fill the global substring buffers */ if (code == 1) for (i = 0; i <= ntok; i++) if (pat->startp[i] != NULL && pat->endp[i] != NULL) { len = pat->endp[i] - pat->startp[i]; sqd_parse[i] = (char *) MallocOrDie(sizeof(char) * (len+1)); strncpy(sqd_parse[i], pat->startp[i], len); sqd_parse[i][len] = '\0'; } free(pat); return code;}/* Function: SqdClean() * Date: SRE, Wed Oct 29 12:52:08 1997 [TWA 721] * * Purpose: Clean up any squid library allocations before exiting * a program, so we don't leave unfree'd memory around * and confuse a malloc debugger like Purify or dbmalloc. */voidSqdClean(void){ int i; /* Free global substring buffers that Strparse() uses */ for (i = 0; i <= 9; i++) if (sqd_parse[i] != NULL) { free(sqd_parse[i]); sqd_parse[i] = NULL; }}/* all code below is: * Copyright (c) 1986, 1993, 1995 by University of Toronto. * Written by Henry Spencer. Not derived from licensed software. * * Permission is granted to anyone to use this software for any * purpose on any computer system, and to redistribute it in any way, * subject to the following restrictions: * * 1. The author is not responsible for the consequences of use of * this software, no matter how awful, even if they arise * from defects in it. * * 2. The origin of this software must not be misrepresented, either * by explicit claim or by omission. * * 3. Altered versions must be plainly marked as such, and must not * be misrepresented (by explicit claim or omission) as being * the original software. * * 4. This notice must not be removed or altered. *//* * sqd_regcomp and sqd_regexec -- sqd_regsub and sqd_regerror are elsewhere *//* * The first byte of the regexp internal "program" is actually this magic * number; the start node begins in the second byte. */#define SQD_REGMAGIC 0234/* * The "internal use only" fields in regexp.h are present to pass info from * compile to execute that permits the execute phase to run lots faster on * simple cases. They are: * * regstart char that must begin a match; '\0' if none obvious * reganch is the match anchored (at beginning-of-line only)? * regmust string (pointer into program) that match must include, or NULL * regmlen length of regmust string * * Regstart and reganch permit very fast decisions on suitable starting points * for a match, cutting down the work a lot. Regmust permits fast rejection * of lines that cannot possibly match. The regmust tests are costly enough * that sqd_regcomp() supplies a regmust only if the r.e. contains something * potentially expensive (at present, the only such thing detected is * or + * at the start of the r.e., which can involve a lot of backup). Regmlen is * supplied because the test in sqd_regexec() needs it and sqd_regcomp() is computing * it anyway. *//* * Structure for regexp "program". This is essentially a linear encoding * of a nondeterministic finite-state machine (aka syntax charts or * "railroad normal form" in parsing technology). Each node is an opcode * plus a "next" pointer, possibly plus an operand. "Next" pointers of * all nodes except BRANCH implement concatenation; a "next" pointer with * a BRANCH on both ends of it is connecting two alternatives. (Here we * have one of the subtle syntax dependencies: an individual BRANCH (as * opposed to a collection of them) is never concatenated with anything * because of operator precedence.) The operand of some types of node is * a literal string; for others, it is a node leading into a sub-FSM. In * particular, the operand of a BRANCH node is the first node of the branch. * (NB this is *not* a tree structure: the tail of the branch connects * to the thing following the set of BRANCHes.) The opcodes are: *//* definition number opnd? meaning */#define END 0 /* no End of program. */#define BOL 1 /* no Match beginning of line. */#define EOL 2 /* no Match end of line. */#define ANY 3 /* no Match any character. */#define ANYOF 4 /* str Match any of these. */#define ANYBUT 5 /* str Match any but one of these. */#define BRANCH 6 /* node Match this, or the next..\&. */#define BACK 7 /* no "next" ptr points backward. */#define EXACTLY 8 /* str Match this string. */#define NOTHING 9 /* no Match empty string. */#define STAR 10 /* node Match this 0 or more times. */#define PLUS 11 /* node Match this 1 or more times. */#define OPEN 20 /* no Sub-RE starts here. */ /* OPEN+1 is number 1, etc. */#define CLOSE 30 /* no Analogous to OPEN. *//* * Opcode notes: * * BRANCH The set of branches constituting a single choice are hooked * together with their "next" pointers, since precedence prevents * anything being concatenated to any individual branch. The * "next" pointer of the last BRANCH in a choice points to the * thing following the whole choice. This is also where the * final "next" pointer of each individual branch points; each * branch starts with the operand node of a BRANCH node. * * BACK Normal "next" pointers all implicitly point forward; BACK * exists to make loop structures possible. * * STAR,PLUS '?', and complex '*' and '+', are implemented as circular * BRANCH structures using BACK. Simple cases (one character * per match) are implemented with STAR and PLUS for speed * and to minimize recursive plunges. * * OPEN,CLOSE ...are numbered at compile time. *//* * A node is one char of opcode followed by two chars of "next" pointer. * "Next" pointers are stored as two 8-bit pieces, high order first. The * value is a positive offset from the opcode of the node containing it. * An operand, if any, simply follows the node. (Note that much of the * code generation knows about this implicit relationship.) * * Using two bytes for the "next" pointer is vast overkill for most things, * but allows patterns to get big without disasters. */#define OP(p) (*(p))#define NEXT(p) (((*((p)+1)&0177)<<8) + (*((p)+2)&0377))#define OPERAND(p) ((p) + 3)/* * Utility definitions. */#define FAIL(m) { sqd_regerror(m); return(NULL); }#define ISREPN(c) ((c) == '*' || (c) == '+' || (c) == '?')#define META "^$.[()|?+*\\"/* * Flags to be passed up and down. */#define HASWIDTH 01 /* Known never to match null string. */#define SIMPLE 02 /* Simple enough to be STAR/PLUS operand. */#define SPSTART 04 /* Starts with * or +. */#define WORST 0 /* Worst case. *//* * Work-variable struct for sqd_regcomp(). */struct comp { char *regparse; /* Input-scan pointer. */ int regnpar; /* () count. */ char *regcode; /* Code-emit pointer; ®dummy = don't. */ char regdummy[3]; /* NOTHING, 0 next ptr */ long regsize; /* Code size. */};#define EMITTING(cp) ((cp)->regcode != (cp)->regdummy)/* * Forward declarations for sqd_regcomp()'s friends. */static char *reg(struct comp *cp, int paren, int *flagp);static char *regbranch(struct comp *cp, int *flagp);static char *regpiece(struct comp *cp, int *flagp);static char *regatom(struct comp *cp, int *flagp);static char *regnode(struct comp *cp, int op);static char *regnext(char *node);static void regc(struct comp *cp, int c);static void reginsert(struct comp *cp, int op, char *opnd);static void regtail(struct comp *cp, char *p, char *val);static void regoptail(struct comp *cp, char *p, char *val);/* - sqd_regcomp - compile a regular expression into internal code * * We can't allocate space until we know how big the compiled form will be, * but we can't compile it (and thus know how big it is) until we've got a * place to put the code. So we cheat: we compile it twice, once with code * generation turned off and size counting turned on, and once "for real". * This also means that we don't allocate space until we are sure that the * thing really will compile successfully, and we never have to move the * code and thus invalidate pointers into it. (Note that it has to be in * one piece because free() must be able to free it all.) * * Beware that the optimization-preparation code in here knows about some * of the structure of the compiled regexp. */sqd_regexp *sqd_regcomp(exp)const char *exp;{ register sqd_regexp *r; register char *scan; int flags; struct comp co; if (exp == NULL) FAIL("NULL argument to sqd_regcomp"); /* First pass: determine size, legality. */ co.regparse = (char *)exp; co.regnpar = 1; co.regsize = 0L; co.regdummy[0] = NOTHING; co.regdummy[1] = co.regdummy[2] = 0; co.regcode = co.regdummy; regc(&co, SQD_REGMAGIC); if (reg(&co, 0, &flags) == NULL) return(NULL); /* Small enough for pointer-storage convention? */ if (co.regsize >= 0x7fffL) /* Probably could be 0xffffL. */ FAIL("regexp too big"); /* Allocate space. */ r = (sqd_regexp *)malloc(sizeof(sqd_regexp) + (size_t)co.regsize); if (r == NULL) FAIL("out of space"); /* Second pass: emit code. */ co.regparse = (char *)exp; co.regnpar = 1; co.regcode = r->program; regc(&co, SQD_REGMAGIC); if (reg(&co, 0, &flags) == NULL) return(NULL); /* Dig out information for optimizations. */ r->regstart = '\0'; /* Worst-case defaults. */ r->reganch = 0; r->regmust = NULL; r->regmlen = 0; scan = r->program+1; /* First BRANCH. */ if (OP(regnext(scan)) == END) { /* Only one top-level choice. */ scan = OPERAND(scan); /* Starting-point info. */ if (OP(scan) == EXACTLY) r->regstart = *OPERAND(scan); else if (OP(scan) == BOL) r->reganch = 1; /* * If there's something expensive in the r.e., find the * longest literal string that must appear and make it the * regmust. Resolve ties in favor of later strings, since * the regstart check works with the beginning of the r.e. * and avoiding duplication strengthens checking. Not a * strong reason, but sufficient in the absence of others. */ if (flags&SPSTART) { register char *longest = NULL; register size_t len = 0; for (; scan != NULL; scan = regnext(scan)) if (OP(scan) == EXACTLY && strlen(OPERAND(scan)) >= len) { longest = OPERAND(scan); len = strlen(OPERAND(scan)); } r->regmust = longest; r->regmlen = (int)len; } } return(r);}/* - reg - regular expression, i.e. main body or parenthesized thing * * Caller must absorb opening parenthesis. * * Combining parenthesis handling with the base level of regular expression * is a trifle forced, but the need to tie the tails of the branches to what * follows makes it hard to avoid. */static char *reg(cp, paren, flagp)register struct comp *cp;int paren; /* Parenthesized? */int *flagp;{ register char *ret = NULL; /* SRE: NULL init added to silence gcc */ register char *br; register char *ender; register int parno = 0; /* SRE: init added to silence gcc */ int flags; *flagp = HASWIDTH; /* Tentatively. */ if (paren) { /* Make an OPEN node. */ if (cp->regnpar >= NSUBEXP) FAIL("too many ()"); parno = cp->regnpar; cp->regnpar++; ret = regnode(cp, OPEN+parno); } /* Pick up the branches, linking them together. */ br = regbranch(cp, &flags); if (br == NULL) return(NULL); if (paren) regtail(cp, ret, br); /* OPEN -> first. */ else ret = br; *flagp &= ~(~flags&HASWIDTH); /* Clear bit if bit 0. */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -