2014-08-22 14:39:17 -04:00
|
|
|
/*
|
|
|
|
* 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_CONFIG_H
|
|
|
|
#include "clamav-config.h"
|
|
|
|
#endif
|
|
|
|
|
2014-08-25 19:11:12 -04:00
|
|
|
#if HAVE_PCRE
|
2014-08-22 14:39:17 -04:00
|
|
|
#include <pcre.h>
|
|
|
|
|
|
|
|
#include "clamav.h"
|
|
|
|
#include "cltypes.h"
|
|
|
|
#include "others.h"
|
|
|
|
#include "regex_pcre.h"
|
|
|
|
|
|
|
|
/* TODO: redefine pcre_malloc and pcre_free */
|
|
|
|
|
2014-08-25 19:11:12 -04:00
|
|
|
int cli_pcre_parse(struct cli_pcre_data *pd, const char *pattern)
|
2014-08-22 14:39:17 -04:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2014-08-25 19:11:12 -04:00
|
|
|
return CL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
int cli_pcre_compile(struct cli_pcre_data *pd, long long unsigned match_limit, long long unsigned match_limit_recursion, unsigned int options)
|
|
|
|
{
|
|
|
|
const char *error;
|
|
|
|
int erroffset;
|
|
|
|
|
|
|
|
if (!pd || !pd->expression) {
|
|
|
|
cli_errmsg("cli_pcre_compile: NULL pd or NULL pd->expression\n");
|
|
|
|
return CL_ENULLARG;
|
|
|
|
}
|
|
|
|
|
2014-08-22 14:39:17 -04:00
|
|
|
/* compile the pcre regex last arg is charset */
|
2014-08-25 19:11:12 -04:00
|
|
|
pd->re = pcre_compile(pd->expression, pd->options, &error, &erroffset, NULL); /* pd->re handled by libpcre -> call pcre_free() */
|
2014-08-22 14:39:17 -04:00
|
|
|
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) {
|
2014-08-25 19:11:12 -04:00
|
|
|
pd->ex->match_limit = match_limit;
|
2014-08-22 14:39:17 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
pd->ex->flags |= PCRE_EXTRA_MATCH_LIMIT;
|
2014-08-25 19:11:12 -04:00
|
|
|
pd->ex->match_limit = match_limit;
|
2014-08-22 14:39:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 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) {
|
2014-08-25 19:11:12 -04:00
|
|
|
pd->ex->match_limit_recursion = match_limit_recursion;
|
2014-08-22 14:39:17 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
pd->ex->flags |= PCRE_EXTRA_MATCH_LIMIT_RECURSION;
|
2014-08-25 19:11:12 -04:00
|
|
|
pd->ex->match_limit_recursion = match_limit_recursion;
|
2014-08-22 14:39:17 -04:00
|
|
|
}
|
|
|
|
#endif /* PCRE_EXTRA_MATCH_LIMIT_RECURSION */
|
|
|
|
|
|
|
|
/* non-dynamic allocated fields set by caller */
|
|
|
|
return CL_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO: fix this function */
|
2014-08-26 12:31:11 -04:00
|
|
|
int cli_pcre_match(struct cli_pcre_data *pd, const unsigned char *buffer, uint32_t buflen, int override_offset, int *ovector, size_t ovlen)
|
2014-08-22 14:39:17 -04:00
|
|
|
{
|
2014-08-26 12:31:11 -04:00
|
|
|
int rc, startoffset;
|
2014-08-22 17:29:40 -04:00
|
|
|
|
|
|
|
if (ovlen % 3) {
|
|
|
|
cli_dbgmsg("cli_pcre_match: ovector length is not a multiple of 3\n");
|
|
|
|
return CL_EARG;
|
|
|
|
}
|
|
|
|
|
2014-08-26 12:31:11 -04:00
|
|
|
startoffset = pd->search_offset;
|
|
|
|
if (override_offset >= 0)
|
|
|
|
startoffset = override_offset;
|
|
|
|
|
|
|
|
rc = pcre_exec(pd->re, pd->ex, buffer, buflen, startoffset, pd->options, ovector, ovlen);
|
2014-08-22 17:29:40 -04:00
|
|
|
|
|
|
|
return rc;
|
2014-08-22 14:39:17 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
2014-08-25 15:07:30 -04:00
|
|
|
#endif /* HAVE_PCRE */
|