clamav/libclamav/regex_pcre.c

126 lines
3.7 KiB
C
Raw Normal View History

/*
* Support for PCRE regex variant
*
* Copyright (C) 2007-2013 Sourcefire, Inc.
* Copyright (C) 2014 Cisco Systems, Inc.
* All Rights Reserved.
*
* Authors: Kevin Lin
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation.
*
* 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., 51 Franklin Street, Fifth Floor, Boston,
* MA 02110-1301, USA.
*/
#if HAVE_PCRE
#if HAVE_CONFIG_H
#include "clamav-config.h"
#endif
#include <pcre.h>
#include "clamav.h"
#include "cltypes.h"
#include "others.h"
#include "regex_pcre.h"
/* TODO: redefine pcre_malloc and pcre_free */
int cli_pcre_parse(struct cli_pcre_data *pd, const char *pattern, unsigned int options)
{
const char *error;
int erroffset;
if (!pd || !pattern) {
cli_errmsg("cli_pcre_parse: NULL pd or NULL pattern\n");
return CL_ENULLARG;
}
/* copy expression to struct cli_pcre_data */
pd->expression = cli_strdup(pattern);
if (!(pd->expression)) {
cli_errmsg("cli_pcre_parse: Unable to allocate memory\n");
return CL_EMEM;
}
/* compile the pcre regex last arg is charset */
pd->re = pcre_compile(pattern, pd->options, &error, &erroffset, NULL); /* pd->re handled by libpcre -> call pcre_free() */
if (pd->re == NULL) {
cli_errmsg("cli_pcre_parse: PCRE compilation failed at offset %d: %s\n", erroffset, error);
return CL_EPARSE; /* TODO - change ERRORCODE */
}
/* now study it... (section totally not from snort) */
pd->ex = pcre_study(pd->re, 0, &error);
if (!(pd->ex)) {
/* TODO: this is complicated because pcre will use system malloc */
pd->ex = (pcre_extra *)cli_calloc(1, sizeof(*(pd->ex)));
if (!(pd->ex)) {
cli_errmsg("cli_pcre_parse: Unable to allocate memory\n");
return CL_EMEM;
}
}
/* set the match limits -> TODO: engine options; hard coded for now */
if (pd->ex->flags & PCRE_EXTRA_MATCH_LIMIT) {
pd->ex->match_limit = CLAMAV_PCRE_MATCH_LIMIT;
}
else {
pd->ex->flags |= PCRE_EXTRA_MATCH_LIMIT;
pd->ex->match_limit = CLAMAV_PCRE_MATCH_LIMIT;
}
/* set the recursion match limits -> TODO: engine options; hard coded for now */
#ifdef PCRE_EXTRA_MATCH_LIMIT_RECURSION
if (pd->ex->flags & PCRE_EXTRA_MATCH_LIMIT_RECURSION) {
pd->ex->match_limit_recursion = CLAMAV_PCRE_REC_MATCH_LIMIT;
}
else {
pd->ex->flags |= PCRE_EXTRA_MATCH_LIMIT_RECURSION;
pd->ex->match_limit_recursion = CLAMAV_PCRE_REC_MATCH_LIMIT;
}
#endif /* PCRE_EXTRA_MATCH_LIMIT_RECURSION */
/* non-dynamic allocated fields set by caller */
return CL_SUCCESS;
}
/* TODO: fix this function */
int cli_pcre_match(struct cli_pcre_data *pd, const unsigned char *buffer, uint32_t buflen, int *ovector, size_t ovlen)
{
int rc;
if (ovlen % 3) {
cli_dbgmsg("cli_pcre_match: ovector length is not a multiple of 3\n");
return CL_EARG;
}
rc = pcre_exec(pd->re, pd->ex, buffer, buflen, pd->search_offset, pd->options, ovector, ovlen);
return rc;
}
void cli_pcre_free_single(struct cli_pcre_data *pd)
{
if (pd->re) {
pcre_free(pd->re);
}
if (pd->ex) {
free(pd->ex);
}
if (pd->expression) {
free(pd->expression);
}
}
#endif /* HAVE_PCRE */