Snort_AIPreproc/regex.c

175 lines
4.3 KiB
C

/*
* =====================================================================================
*
* Filename: regex.c
*
* Description: Regex management for the module
*
* Version: 0.1
* Created: 08/08/2010 11:01:01
* Revision: none
* Compiler: gcc
*
* Author: BlackLight (http://0x00.ath.cx), <blacklight@autistici.org>
* Licence: GNU GPL v.3
* Company: DO WHAT YOU WANT CAUSE A PIRATE IS FREE, YOU ARE A PIRATE!
*
* =====================================================================================
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <alloca.h>
#include <regex.h>
/** \defgroup regex Regex management
* @{ */
/**
* \brief Check if a string matches a regular expression
* \param expr Regular expression to be matched
* \param str String to be checked
* \param matches Reference to a char** that will contain the submatches (NULL if you don't need it)
* \param nmatches Reference to a int containing the number of submatches found (NULL if you don't need it)
* \return -1 if the regex is wrong, 0 if no match was found, 1 otherwise
*/
int
preg_match ( const char* expr, char* str, char*** matches, int *nmatches )
{
int i;
regex_t regex;
regmatch_t *m = NULL;
if ( nmatches )
*nmatches = 0;
if ( regcomp ( &regex, expr, REG_EXTENDED | REG_ICASE ) != 0 ) {
return -1;
}
if ( regex.re_nsub > 0 )
{
if ( !(m = (regmatch_t*) alloca ( (regex.re_nsub+1) * sizeof(regmatch_t) )) )
{
regfree ( &regex );
fprintf ( stderr, "\nDynamic memory allocation failure at %s:%d\n", __FILE__, __LINE__ );
exit ( EXIT_FAILURE );
}
if ( matches )
{
if ( !( *matches = (char**) malloc ( (regex.re_nsub+1) * sizeof(char*) )) )
{
regfree ( &regex );
m = NULL;
fprintf ( stderr, "\nDynamic memory allocation failure at %s:%d\n", __FILE__, __LINE__ );
exit ( EXIT_FAILURE );
}
}
if ( regexec ( &regex, str, regex.re_nsub+1, m, 0 ) == REG_NOMATCH ) {
regfree ( &regex );
m = NULL;
return 0;
}
} else {
if ( regexec ( &regex, str, 0, NULL, 0 ) == REG_NOMATCH ) {
regfree ( &regex );
m = NULL;
return 0;
}
}
if ( nmatches )
*nmatches = regex.re_nsub;
if ( matches )
{
for ( i=0; i < regex.re_nsub; i++ ) {
if ( !( (*matches)[i] = (char*) malloc ( m[i+1].rm_eo - m[i+1].rm_so + 1 )) )
{
regfree ( &regex );
free ( m );
m = NULL;
fprintf ( stderr, "\nDynamic memory allocation failure at %s:%d\n", __FILE__, __LINE__ );
exit ( EXIT_FAILURE );
}
memset ( (*matches)[i], 0, m[i+1].rm_eo - m[i+1].rm_so + 1 );
strncpy ( (*matches)[i], str + m[i+1].rm_so, m[i+1].rm_eo - m[i+1].rm_so );
}
}
regfree ( &regex );
m = NULL;
return 1;
} /* ----- end of function preg_match ----- */
/**
* \brief Replace the content of 'orig' in 'str' with 'rep'
* \param str String to work on
* \param orig String to be replaced
* \param rep Replacement for 'orig'
* \return The string with the replacement
*/
char*
str_replace ( char *str, char *orig, char *rep )
{
char *new_s = NULL;
unsigned int new_len = 0;
unsigned int pos = 0;
if ( !( pos = (int) strstr ( str, orig )))
return str;
new_len = strlen(str) - strlen(orig) + strlen(rep) + 1;
if ( !( new_s = (char*) malloc ( new_len )))
return NULL;
memset ( new_s, 0, new_len );
strncpy ( new_s, str, pos - (unsigned int) str );
new_s[ pos - (unsigned int) str] = 0;
if ( rep )
{
if ( strlen ( rep ) != 0 )
sprintf ( new_s + pos - (unsigned int) str, "%s%s", rep, (char*) pos + strlen ( orig ));
else
sprintf ( new_s + pos - (unsigned int) str, "%s", (char*) pos + strlen ( orig ));
} else {
sprintf ( new_s + pos - (unsigned int) str, "%s", (char*) pos + strlen ( orig ));
}
return new_s;
} /* ----- end of function str_replace ----- */
/**
* \brief Replace all of the occurrences of 'orig' in 'str' with 'rep'
* \param str String to work on
* \param orig String to be replaced
* \param rep Replacement for 'orig'
* \return The string with the replacement
*/
char*
str_replace_all ( char *str, char *orig, char *rep )
{
char *buf = strdup ( str );
char *tmp = NULL;
while ( strstr ( buf, orig ))
{
if ( tmp )
free ( tmp );
tmp = buf;
buf = str_replace ( str, orig, rep );
}
return buf;
} /* ----- end of function str_replace_all ----- */
/** @} */