sync code with last improvements from OpenBSD
This commit is contained in:
commit
88965415ff
26235 changed files with 29195616 additions and 0 deletions
223
app/xedit/lisp/regex.c
Normal file
223
app/xedit/lisp/regex.c
Normal file
|
@ -0,0 +1,223 @@
|
|||
/*
|
||||
* Copyright (c) 2002 by The XFree86 Project, Inc.
|
||||
*
|
||||
* Permission is hereby granted, free of charge, to any person obtaining a
|
||||
* copy of this software and associated documentation files (the "Software"),
|
||||
* to deal in the Software without restriction, including without limitation
|
||||
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||||
* and/or sell copies of the Software, and to permit persons to whom the
|
||||
* Software is furnished to do so, subject to the following conditions:
|
||||
*
|
||||
* The above copyright notice and this permission notice shall be included in
|
||||
* all copies or substantial portions of the Software.
|
||||
*
|
||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||||
* THE XFREE86 PROJECT BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||||
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
|
||||
* OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
* SOFTWARE.
|
||||
*
|
||||
* Except as contained in this notice, the name of the XFree86 Project shall
|
||||
* not be used in advertising or otherwise to promote the sale, use or other
|
||||
* dealings in this Software without prior written authorization from the
|
||||
* XFree86 Project.
|
||||
*
|
||||
* Author: Paulo César Pereira de Andrade
|
||||
*/
|
||||
|
||||
/* $XFree86: xc/programs/xedit/lisp/regex.c,v 1.10tsi Exp $ */
|
||||
|
||||
#include "lisp/regex.h"
|
||||
#include "lisp/private.h"
|
||||
#include "lisp/helper.h"
|
||||
|
||||
/*
|
||||
* Prototypes
|
||||
*/
|
||||
static re_cod *LispRecomp(LispBuiltin*, char*, int);
|
||||
|
||||
/*
|
||||
* Initialization
|
||||
*/
|
||||
LispObj *Knomatch;
|
||||
|
||||
/*
|
||||
* Implementation
|
||||
*/
|
||||
static re_cod *
|
||||
LispRecomp(LispBuiltin *builtin, char *pattern, int cflags)
|
||||
{
|
||||
int code;
|
||||
re_cod *regex = LispMalloc(sizeof(re_cod));
|
||||
|
||||
if ((code = recomp(regex, pattern, cflags)) != 0) {
|
||||
char buffer[256];
|
||||
|
||||
reerror(code, regex, buffer, sizeof(buffer));
|
||||
refree(regex);
|
||||
LispFree(regex);
|
||||
LispDestroy("%s: recomp(\"%s\"): %s", STRFUN(builtin), pattern, buffer);
|
||||
}
|
||||
|
||||
return (regex);
|
||||
}
|
||||
|
||||
void
|
||||
LispRegexInit(void)
|
||||
{
|
||||
Knomatch = KEYWORD("NOMATCH");
|
||||
}
|
||||
|
||||
LispObj *
|
||||
Lisp_Recomp(LispBuiltin *builtin)
|
||||
/*
|
||||
re-comp pattern &key nospec icase nosub newline
|
||||
*/
|
||||
{
|
||||
re_cod *regex;
|
||||
int cflags = 0;
|
||||
|
||||
LispObj *result;
|
||||
|
||||
LispObj *pattern, *nospec, *icase, *nosub, *newline;
|
||||
|
||||
newline = ARGUMENT(4);
|
||||
nosub = ARGUMENT(3);
|
||||
icase = ARGUMENT(2);
|
||||
nospec = ARGUMENT(1);
|
||||
pattern = ARGUMENT(0);
|
||||
|
||||
/* Don't generate an error if it is already a compiled regex. */
|
||||
if (REGEXP(pattern))
|
||||
return (pattern);
|
||||
|
||||
CHECK_STRING(pattern);
|
||||
|
||||
if (nospec != UNSPEC && nospec != NIL)
|
||||
cflags |= RE_NOSPEC;
|
||||
if (icase != UNSPEC && icase != NIL)
|
||||
cflags |= RE_ICASE;
|
||||
if (nosub != UNSPEC && nosub != NIL)
|
||||
cflags |= RE_NOSUB;
|
||||
if (newline != UNSPEC && newline != NIL)
|
||||
cflags |= RE_NEWLINE;
|
||||
|
||||
regex = LispRecomp(builtin, THESTR(pattern), cflags);
|
||||
result = LispNew(pattern, NIL);
|
||||
result->type = LispRegex_t;
|
||||
result->data.regex.regex = regex;
|
||||
result->data.regex.pattern = pattern;
|
||||
result->data.regex.options = cflags;
|
||||
LispMused(regex);
|
||||
|
||||
return (result);
|
||||
}
|
||||
|
||||
LispObj *
|
||||
Lisp_Reexec(LispBuiltin *builtin)
|
||||
/*
|
||||
re-exec regex string &key count start end notbol noteol
|
||||
*/
|
||||
{
|
||||
size_t nmatch;
|
||||
re_mat match[10];
|
||||
long start, end, length;
|
||||
int code, cflags, eflags;
|
||||
char *string;
|
||||
LispObj *result;
|
||||
re_cod *regexp;
|
||||
|
||||
LispObj *regex, *ostring, *count, *ostart, *oend, *notbol, *noteol;
|
||||
|
||||
noteol = ARGUMENT(6);
|
||||
notbol = ARGUMENT(5);
|
||||
oend = ARGUMENT(4);
|
||||
ostart = ARGUMENT(3);
|
||||
count = ARGUMENT(2);
|
||||
ostring = ARGUMENT(1);
|
||||
regex = ARGUMENT(0);
|
||||
|
||||
if (STRINGP(regex))
|
||||
regexp = LispRecomp(builtin, THESTR(regex), cflags = 0);
|
||||
else {
|
||||
CHECK_REGEX(regex);
|
||||
regexp = regex->data.regex.regex;
|
||||
cflags = regex->data.regex.options;
|
||||
}
|
||||
|
||||
CHECK_STRING(ostring);
|
||||
|
||||
if (count == UNSPEC)
|
||||
nmatch = 1;
|
||||
else {
|
||||
CHECK_INDEX(count);
|
||||
nmatch = FIXNUM_VALUE(count);
|
||||
if (nmatch > 10)
|
||||
LispDestroy("%s: COUNT cannot be larger than 10", STRFUN(builtin));
|
||||
}
|
||||
if (nmatch && (cflags & RE_NOSUB))
|
||||
nmatch = 1;
|
||||
|
||||
eflags = RE_STARTEND;
|
||||
if (notbol != UNSPEC && notbol != NIL)
|
||||
eflags |= RE_NOTBOL;
|
||||
if (noteol != UNSPEC && noteol != NIL)
|
||||
eflags |= RE_NOTEOL;
|
||||
|
||||
string = THESTR(ostring);
|
||||
LispCheckSequenceStartEnd(builtin, ostring, ostart, oend,
|
||||
&start, &end, &length);
|
||||
|
||||
match[0].rm_so = start;
|
||||
match[0].rm_eo = end;
|
||||
code = reexec(regexp, string, nmatch, &match[0], eflags);
|
||||
|
||||
if (code == 0) {
|
||||
if (nmatch && match[0].rm_eo >= match[0].rm_so) {
|
||||
result = CONS(CONS(FIXNUM(match[0].rm_so),
|
||||
FIXNUM(match[0].rm_eo)), NIL);
|
||||
if (nmatch > 1 && match[1].rm_eo >= match[1].rm_so) {
|
||||
int i;
|
||||
GC_ENTER();
|
||||
LispObj *cons = result;
|
||||
|
||||
GC_PROTECT(result);
|
||||
for (i = 1;
|
||||
i < nmatch && match[i].rm_eo >= match[i].rm_so;
|
||||
i++) {
|
||||
RPLACD(cons, CONS(CONS(FIXNUM(match[i].rm_so),
|
||||
FIXNUM(match[i].rm_eo)), NIL));
|
||||
cons = CDR(cons);
|
||||
}
|
||||
GC_LEAVE();
|
||||
}
|
||||
}
|
||||
else
|
||||
result = NIL;
|
||||
}
|
||||
else
|
||||
result = Knomatch;
|
||||
|
||||
/* Maybe shoud cache compiled regex, but better the caller do it */
|
||||
if (!XREGEXP(regex)) {
|
||||
refree(regexp);
|
||||
LispFree(regexp);
|
||||
}
|
||||
|
||||
return (result);
|
||||
}
|
||||
|
||||
LispObj *
|
||||
Lisp_Rep(LispBuiltin *builtin)
|
||||
/*
|
||||
re-p object
|
||||
*/
|
||||
{
|
||||
LispObj *object;
|
||||
|
||||
object = ARGUMENT(0);
|
||||
|
||||
return (REGEXP(object) ? T : NIL);
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue