mirror of
https://github.com/radareorg/radare2.git
synced 2025-01-25 07:15:19 +00:00
257 lines
5.1 KiB
C
257 lines
5.1 KiB
C
/* radare - LGPL - Copyright 2010-2015 - pancake */
|
|
|
|
#include <r_search.h>
|
|
|
|
static int ignoreMask(const ut8 *bm, int len) {
|
|
int i;
|
|
for (i=0; i<len; i++) {
|
|
if (bm[i] != 0xff) {
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
R_API RSearchKeyword* r_search_keyword_new(const ut8 *kwbuf, int kwlen, const ut8 *bmbuf, int bmlen, const char *data) {
|
|
RSearchKeyword *kw;
|
|
if (kwlen < 1 || bmlen < 0) {
|
|
return NULL;
|
|
}
|
|
kw = R_NEW0 (RSearchKeyword);
|
|
if (!kw) {
|
|
return NULL;
|
|
}
|
|
kw->type = R_SEARCH_KEYWORD_TYPE_BINARY;
|
|
kw->data = (void *) data;
|
|
kw->keyword_length = kwlen;
|
|
kw->bin_keyword = malloc (kwlen);
|
|
if (!kw->bin_keyword) {
|
|
r_search_keyword_free (kw);
|
|
return NULL;
|
|
}
|
|
memcpy (kw->bin_keyword, kwbuf, kwlen);
|
|
if (bmbuf && bmlen > 0 && !ignoreMask (bmbuf, bmlen)) {
|
|
kw->bin_binmask = malloc (bmlen);
|
|
if (!kw->bin_binmask) {
|
|
r_search_keyword_free (kw);
|
|
return NULL;
|
|
}
|
|
memcpy (kw->bin_binmask, bmbuf, bmlen);
|
|
kw->binmask_length = bmlen;
|
|
} else {
|
|
kw->bin_binmask = NULL;
|
|
kw->binmask_length = 0;
|
|
}
|
|
return kw;
|
|
}
|
|
|
|
R_API void r_search_keyword_free (RSearchKeyword *kw) {
|
|
if (!kw) {
|
|
return;
|
|
}
|
|
free (kw->bin_binmask);
|
|
free (kw->bin_keyword);
|
|
free (kw);
|
|
}
|
|
|
|
R_API RSearchKeyword* r_search_keyword_new_str(const char *kwbuf, const char *bmstr, const char *data, int ignore_case) {
|
|
RSearchKeyword *kw;
|
|
ut8 *bmbuf = NULL;
|
|
int bmlen = 0;
|
|
|
|
if (bmstr) {
|
|
bmbuf = malloc (strlen (bmstr)+1);
|
|
if (!bmbuf) {
|
|
return NULL;
|
|
}
|
|
bmlen = r_hex_str2bin (bmstr, bmbuf);
|
|
if (bmlen < 1) {
|
|
R_FREE (bmbuf);
|
|
}
|
|
}
|
|
kw = r_search_keyword_new ((ut8 *)kwbuf, strlen (kwbuf), bmbuf, bmlen, data);
|
|
if (kw) {
|
|
kw->icase = ignore_case;
|
|
kw->type = R_SEARCH_KEYWORD_TYPE_STRING;
|
|
}
|
|
free (bmbuf);
|
|
return kw;
|
|
}
|
|
|
|
R_API RSearchKeyword* r_search_keyword_new_wide(const char *kwbuf, const char *bmstr, const char *data, int ignore_case) {
|
|
RSearchKeyword *kw;
|
|
int len;
|
|
const char *p2;
|
|
char *p, *str;
|
|
ut8 *bmbuf = NULL;
|
|
int bmlen = 0;
|
|
|
|
if (bmstr) {
|
|
bmbuf = malloc (strlen (bmstr)+1);
|
|
if (!bmbuf) {
|
|
return NULL;
|
|
}
|
|
bmlen = r_hex_str2bin (bmstr, bmbuf);
|
|
if (bmlen < 1) {
|
|
R_FREE (bmbuf);
|
|
}
|
|
}
|
|
|
|
len = strlen(kwbuf);
|
|
str = malloc((len+1)*2);
|
|
for (p2=kwbuf, p=str; *p2; ) {
|
|
RRune ch;
|
|
int num_utf8_bytes = r_utf8_decode ((const ut8 *)p2, kwbuf + len - p2, &ch);
|
|
if (num_utf8_bytes < 1) {
|
|
eprintf ("WARNING: Malformed UTF8 at pos %td\n", p2 - kwbuf);
|
|
p[0] = *p2;
|
|
p[1] = 0;
|
|
p2++;
|
|
p += 2;
|
|
continue;
|
|
}
|
|
if (ignore_case && ch <= 0xff) {
|
|
ch = tolower (ch);
|
|
}
|
|
int num_wide_bytes = r_utf16le_encode ((ut8 *)p, ch);
|
|
r_warn_if_fail (num_wide_bytes != 0);
|
|
p2 += num_utf8_bytes;
|
|
p += num_wide_bytes;
|
|
}
|
|
|
|
kw = r_search_keyword_new ((ut8 *)str, p - str, bmbuf, bmlen, data);
|
|
free(str);
|
|
if (kw) {
|
|
kw->icase = ignore_case;
|
|
}
|
|
free(bmbuf);
|
|
return kw;
|
|
}
|
|
|
|
R_API RSearchKeyword* r_search_keyword_new_hex(const char *kwstr, const char *bmstr, const char *data) {
|
|
RSearchKeyword *kw;
|
|
ut8 *kwbuf, *bmbuf;
|
|
int kwlen, bmlen = 0;
|
|
|
|
if (!kwstr) {
|
|
return NULL;
|
|
}
|
|
|
|
kwbuf = malloc (strlen (kwstr)+1);
|
|
if (!kwbuf) {
|
|
return NULL;
|
|
}
|
|
|
|
kwlen = r_hex_str2bin (kwstr, kwbuf);
|
|
if (kwlen < 1) {
|
|
free (kwbuf);
|
|
return NULL;
|
|
}
|
|
|
|
bmbuf = NULL;
|
|
if (bmstr && *bmstr) {
|
|
bmbuf = malloc (strlen (bmstr)+1);
|
|
if (!bmbuf) {
|
|
free (kwbuf);
|
|
return NULL;
|
|
}
|
|
bmlen = r_hex_str2bin (bmstr, bmbuf);
|
|
if (bmlen < 1) {
|
|
free (bmbuf);
|
|
free (kwbuf);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
kw = r_search_keyword_new (kwbuf, kwlen, bmbuf, bmlen, data);
|
|
free (kwbuf);
|
|
free (bmbuf);
|
|
return kw;
|
|
}
|
|
|
|
R_API RSearchKeyword* r_search_keyword_new_hexmask(const char *kwstr, const char *data) {
|
|
RSearchKeyword *ks = NULL;
|
|
ut8 *kw, *bm;
|
|
if (kwstr != NULL) {
|
|
int len = strlen (kwstr);
|
|
kw = malloc (len + 4);
|
|
bm = malloc (len + 4);
|
|
if (kw != NULL && bm != NULL) {
|
|
len = r_hex_str2binmask (kwstr, (ut8*)kw, (ut8*)bm);
|
|
if (len < 0) {
|
|
len = -len - 1;
|
|
}
|
|
if (len > 0) {
|
|
ks = r_search_keyword_new (kw, len, bm, len, data);
|
|
}
|
|
}
|
|
free (kw);
|
|
free (bm);
|
|
}
|
|
return ks;
|
|
}
|
|
|
|
/* Validate a regexp in the canonical format /<regexp>/<options> */
|
|
R_API RSearchKeyword *r_search_keyword_new_regexp (const char *str, const char *data) {
|
|
RSearchKeyword *kw;
|
|
int i = 0, start, length;
|
|
|
|
while (isspace ((const unsigned char)str[i])) {
|
|
i++;
|
|
}
|
|
|
|
if (str[i++] != '/') {
|
|
return NULL;
|
|
}
|
|
|
|
/* Find the fist non backslash-escaped slash */
|
|
int specials = 0;
|
|
for (start = i; str[i]; i++) {
|
|
if (str[i] == '/' && str[i - 1] != '\\') {
|
|
break;
|
|
} else if (str[i - 1] == '\\' && isalpha (str[i])) {
|
|
specials++;
|
|
}
|
|
}
|
|
|
|
if (str[i++] != '/') {
|
|
return NULL;
|
|
}
|
|
|
|
length = i - start - 1;
|
|
if ((length > 128) || (length < 1)) {
|
|
return NULL;
|
|
}
|
|
|
|
kw = R_NEW0(RSearchKeyword);
|
|
if (!kw) {
|
|
return NULL;
|
|
}
|
|
|
|
kw->bin_keyword = malloc (length+1);
|
|
if (!kw->bin_keyword) {
|
|
r_search_keyword_free (kw);
|
|
return NULL;
|
|
}
|
|
|
|
kw->bin_keyword[length]=0;
|
|
memcpy(kw->bin_keyword, str + start, length);
|
|
kw->keyword_length = length - specials;
|
|
kw->type = R_SEARCH_KEYWORD_TYPE_STRING;
|
|
kw->data = (void *) data;
|
|
|
|
/* Parse the options */
|
|
for (; str[i]; i++) {
|
|
switch (str[i]) {
|
|
case 'i':
|
|
kw->icase = true;
|
|
break;
|
|
default:
|
|
r_search_keyword_free (kw);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
return kw;
|
|
}
|