pcreposix.c
上传用户:yhdzpy8989
上传日期:2007-06-13
资源大小:13604k
文件大小:9k
源码类别:

生物技术

开发平台:

C/C++

  1. /*
  2.  * ===========================================================================
  3.  * PRODUCTION $Log: pcreposix.c,v $
  4.  * PRODUCTION Revision 1000.0  2003/10/29 15:56:12  gouriano
  5.  * PRODUCTION PRODUCTION: IMPORTED [ORIGINAL] Dev-tree R1.2
  6.  * PRODUCTION
  7.  * ===========================================================================
  8.  */
  9. /*************************************************
  10. *      Perl-Compatible Regular Expressions       *
  11. *************************************************/
  12. /*
  13. This is a library of functions to support regular expressions whose syntax
  14. and semantics are as close as possible to those of the Perl 5 language. See
  15. the file Tech.Notes for some information on the internals.
  16. This module is a wrapper that provides a POSIX API to the underlying PCRE
  17. functions.
  18. Written by: Philip Hazel <ph10@cam.ac.uk>
  19.            Copyright (c) 1997-2001 University of Cambridge
  20. -----------------------------------------------------------------------------
  21. Permission is granted to anyone to use this software for any purpose on any
  22. computer system, and to redistribute it freely, subject to the following
  23. restrictions:
  24. 1. This software is distributed in the hope that it will be useful,
  25.    but WITHOUT ANY WARRANTY; without even the implied warranty of
  26.    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  27. 2. The origin of this software must not be misrepresented, either by
  28.    explicit claim or by omission.
  29. 3. Altered versions must be plainly marked as such, and must not be
  30.    misrepresented as being the original software.
  31. 4. If PCRE is embedded in any software that is released under the GNU
  32.    General Purpose Licence (GPL), then the terms of that licence shall
  33.    supersede any condition above with which it is incompatible.
  34. -----------------------------------------------------------------------------
  35. */
  36. #include "pcre_internal.h"
  37. #include <util/regexp/pcreposix.h>
  38. #include "stdlib.h"
  39. /* Corresponding tables of PCRE error messages and POSIX error codes. */
  40. static const char *estring[] = {
  41.   ERR1,  ERR2,  ERR3,  ERR4,  ERR5,  ERR6,  ERR7,  ERR8,  ERR9,  ERR10,
  42.   ERR11, ERR12, ERR13, ERR14, ERR15, ERR16, ERR17, ERR18, ERR19, ERR20,
  43.   ERR21, ERR22, ERR23, ERR24, ERR25, ERR26, ERR27, ERR29, ERR29, ERR30,
  44.   ERR31 };
  45. static int eint[] = {
  46.   REG_EESCAPE, /* "\ at end of pattern" */
  47.   REG_EESCAPE, /* "\c at end of pattern" */
  48.   REG_EESCAPE, /* "unrecognized character follows \" */
  49.   REG_BADBR,   /* "numbers out of order in {} quantifier" */
  50.   REG_BADBR,   /* "number too big in {} quantifier" */
  51.   REG_EBRACK,  /* "missing terminating ] for character class" */
  52.   REG_ECTYPE,  /* "invalid escape sequence in character class" */
  53.   REG_ERANGE,  /* "range out of order in character class" */
  54.   REG_BADRPT,  /* "nothing to repeat" */
  55.   REG_BADRPT,  /* "operand of unlimited repeat could match the empty string" */
  56.   REG_ASSERT,  /* "internal error: unexpected repeat" */
  57.   REG_BADPAT,  /* "unrecognized character after (?" */
  58.   REG_ASSERT,  /* "unused error" */
  59.   REG_EPAREN,  /* "missing )" */
  60.   REG_ESUBREG, /* "back reference to non-existent subpattern" */
  61.   REG_INVARG,  /* "erroffset passed as NULL" */
  62.   REG_INVARG,  /* "unknown option bit(s) set" */
  63.   REG_EPAREN,  /* "missing ) after comment" */
  64.   REG_ESIZE,   /* "parentheses nested too deeply" */
  65.   REG_ESIZE,   /* "regular expression too large" */
  66.   REG_ESPACE,  /* "failed to get memory" */
  67.   REG_EPAREN,  /* "unmatched brackets" */
  68.   REG_ASSERT,  /* "internal error: code overflow" */
  69.   REG_BADPAT,  /* "unrecognized character after (?<" */
  70.   REG_BADPAT,  /* "lookbehind assertion is not fixed length" */
  71.   REG_BADPAT,  /* "malformed number after (?(" */
  72.   REG_BADPAT,  /* "conditional group containe more than two branches" */
  73.   REG_BADPAT,  /* "assertion expected after (?(" */
  74.   REG_BADPAT,  /* "(?p must be followed by )" */
  75.   REG_ECTYPE,  /* "unknown POSIX class name" */
  76.   REG_BADPAT,  /* "POSIX collating elements are not supported" */
  77.   REG_INVARG,  /* "this version of PCRE is not compiled with PCRE_UTF8 support" */
  78.   REG_BADPAT,  /* "characters with values > 255 are not yet supported in classes" */
  79.   REG_BADPAT,  /* "character value in x{...} sequence is too large" */
  80.   REG_BADPAT   /* "invalid condition (?(0)" */
  81. };
  82. /* Table of texts corresponding to POSIX error codes */
  83. static const char *pstring[] = {
  84.   "",                                /* Dummy for value 0 */
  85.   "internal error",                  /* REG_ASSERT */
  86.   "invalid repeat counts in {}",     /* BADBR      */
  87.   "pattern error",                   /* BADPAT     */
  88.   "? * + invalid",                   /* BADRPT     */
  89.   "unbalanced {}",                   /* EBRACE     */
  90.   "unbalanced []",                   /* EBRACK     */
  91.   "collation error - not relevant",  /* ECOLLATE   */
  92.   "bad class",                       /* ECTYPE     */
  93.   "bad escape sequence",             /* EESCAPE    */
  94.   "empty expression",                /* EMPTY      */
  95.   "unbalanced ()",                   /* EPAREN     */
  96.   "bad range inside []",             /* ERANGE     */
  97.   "expression too big",              /* ESIZE      */
  98.   "failed to get memory",            /* ESPACE     */
  99.   "bad back reference",              /* ESUBREG    */
  100.   "bad argument",                    /* INVARG     */
  101.   "match failed"                     /* NOMATCH    */
  102. };
  103. /*************************************************
  104. *          Translate PCRE text code to int       *
  105. *************************************************/
  106. /* PCRE compile-time errors are given as strings defined as macros. We can just
  107. look them up in a table to turn them into POSIX-style error codes. */
  108. static int
  109. pcre_posix_error_code(const char *s)
  110. {
  111. size_t i;
  112. for (i = 0; i < sizeof(estring)/sizeof(char *); i++)
  113.   if (strcmp(s, estring[i]) == 0) return eint[i];
  114. return REG_ASSERT;
  115. }
  116. /*************************************************
  117. *          Translate error code to string        *
  118. *************************************************/
  119. size_t
  120. regerror(int errcode, const regex_t *preg, char *errbuf, size_t errbuf_size)
  121. {
  122. const char *message, *addmessage;
  123. size_t length, addlength;
  124. message = (errcode >= (int)(sizeof(pstring)/sizeof(char *)))?
  125.   "unknown error code" : pstring[errcode];
  126. length = strlen(message) + 1;
  127. addmessage = " at offset ";
  128. addlength = (preg != NULL && (int)preg->re_erroffset != -1)?
  129.   strlen(addmessage) + 6 : 0;
  130. if (errbuf_size > 0)
  131.   {
  132.   if (addlength > 0 && errbuf_size >= length + addlength)
  133.     sprintf(errbuf, "%s%s%-6d", message, addmessage, (int)preg->re_erroffset);
  134.   else
  135.     {
  136.     strncpy(errbuf, message, errbuf_size - 1);
  137.     errbuf[errbuf_size-1] = 0;
  138.     }
  139.   }
  140. return length + addlength;
  141. }
  142. /*************************************************
  143. *           Free store held by a regex           *
  144. *************************************************/
  145. void
  146. regfree(regex_t *preg)
  147. {
  148. (pcre_free)(preg->re_pcre);
  149. }
  150. /*************************************************
  151. *            Compile a regular expression        *
  152. *************************************************/
  153. /*
  154. Arguments:
  155.   preg        points to a structure for recording the compiled expression
  156.   pattern     the pattern to compile
  157.   cflags      compilation flags
  158. Returns:      0 on success
  159.               various non-zero codes on failure
  160. */
  161. int
  162. regcomp(regex_t *preg, const char *pattern, int cflags)
  163. {
  164. const char *errorptr;
  165. int erroffset;
  166. int options = 0;
  167. if ((cflags & REG_ICASE) != 0) options |= PCRE_CASELESS;
  168. if ((cflags & REG_NEWLINE) != 0) options |= PCRE_MULTILINE;
  169. preg->re_pcre = pcre_compile(pattern, options, &errorptr, &erroffset, NULL);
  170. preg->re_erroffset = erroffset;
  171. if (preg->re_pcre == NULL) return pcre_posix_error_code(errorptr);
  172. preg->re_nsub = pcre_info(preg->re_pcre, NULL, NULL);
  173. return 0;
  174. }
  175. /*************************************************
  176. *              Match a regular expression        *
  177. *************************************************/
  178. /* Unfortunately, PCRE requires 3 ints of working space for each captured
  179. substring, so we have to get and release working store instead of just using
  180. the POSIX structures as was done in earlier releases when PCRE needed only 2
  181. ints. */
  182. int
  183. regexec(regex_t *preg, const char *string, size_t nmatch,
  184.   regmatch_t pmatch[], int eflags)
  185. {
  186. int rc;
  187. int options = 0;
  188. int *ovector = NULL;
  189. if ((eflags & REG_NOTBOL) != 0) options |= PCRE_NOTBOL;
  190. if ((eflags & REG_NOTEOL) != 0) options |= PCRE_NOTEOL;
  191. preg->re_erroffset = (size_t)(-1);   /* Only has meaning after compile */
  192. if (nmatch > 0)
  193.   {
  194.   ovector = (int *)malloc(sizeof(int) * nmatch * 3);
  195.   if (ovector == NULL) return REG_ESPACE;
  196.   }
  197. rc = pcre_exec(preg->re_pcre, NULL, string, (int)strlen(string), 0, options,
  198.   ovector, nmatch * 3);
  199. if (rc == 0) rc = nmatch;    /* All captured slots were filled in */
  200. if (rc >= 0)
  201.   {
  202.   size_t i;
  203.   for (i = 0; i < (size_t)rc; i++)
  204.     {
  205.     pmatch[i].rm_so = ovector[i*2];
  206.     pmatch[i].rm_eo = ovector[i*2+1];
  207.     }
  208.   if (ovector != NULL) free(ovector);
  209.   for (; i < nmatch; i++) pmatch[i].rm_so = pmatch[i].rm_eo = -1;
  210.   return 0;
  211.   }
  212. else
  213.   {
  214.   if (ovector != NULL) free(ovector);
  215.   switch(rc)
  216.     {
  217.     case PCRE_ERROR_NOMATCH: return REG_NOMATCH;
  218.     case PCRE_ERROR_NULL: return REG_INVARG;
  219.     case PCRE_ERROR_BADOPTION: return REG_INVARG;
  220.     case PCRE_ERROR_BADMAGIC: return REG_INVARG;
  221.     case PCRE_ERROR_UNKNOWN_NODE: return REG_ASSERT;
  222.     case PCRE_ERROR_NOMEMORY: return REG_ESPACE;
  223.     default: return REG_ASSERT;
  224.     }
  225.   }
  226. }
  227. /* End of pcreposix.c */