2009-05-05 23:28:27 +04:00
|
|
|
/*
|
|
|
|
Search text engine.
|
|
|
|
HEX-style pattern matching
|
|
|
|
|
2016-01-01 09:53:15 +03:00
|
|
|
Copyright (C) 2009-2016
|
2014-02-12 10:33:10 +04:00
|
|
|
Free Software Foundation, Inc.
|
2009-05-05 23:28:27 +04:00
|
|
|
|
|
|
|
Written by:
|
|
|
|
Slava Zanko <slavazanko@gmail.com>, 2009.
|
|
|
|
|
|
|
|
This file is part of the Midnight Commander.
|
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
The Midnight Commander is free software: you can redistribute it
|
2009-05-05 23:28:27 +04:00
|
|
|
and/or modify it under the terms of the GNU General Public License as
|
2011-10-15 14:56:47 +04:00
|
|
|
published by the Free Software Foundation, either version 3 of the License,
|
|
|
|
or (at your option) any later version.
|
2009-05-05 23:28:27 +04:00
|
|
|
|
2011-10-15 14:56:47 +04:00
|
|
|
The Midnight Commander 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.
|
2009-05-05 23:28:27 +04:00
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2011-10-15 14:56:47 +04:00
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2009-05-05 23:28:27 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2009-05-06 11:27:06 +04:00
|
|
|
#include <stdio.h>
|
2009-05-05 23:28:27 +04:00
|
|
|
|
2010-01-20 18:11:52 +03:00
|
|
|
#include "lib/global.h"
|
2010-01-21 15:17:26 +03:00
|
|
|
#include "lib/strutil.h"
|
2010-01-21 16:06:15 +03:00
|
|
|
#include "lib/search.h"
|
2010-01-21 16:31:29 +03:00
|
|
|
#include "lib/strescape.h"
|
2010-01-21 15:17:26 +03:00
|
|
|
|
|
|
|
#include "internal.h"
|
|
|
|
|
2009-05-05 23:28:27 +04:00
|
|
|
/*** global variables ****************************************************************************/
|
|
|
|
|
|
|
|
/*** file scope macro definitions ****************************************************************/
|
|
|
|
|
2016-09-25 14:09:43 +03:00
|
|
|
typedef enum
|
|
|
|
{
|
|
|
|
MC_SEARCH_HEX_E_OK,
|
2016-09-25 18:30:45 +03:00
|
|
|
MC_SEARCH_HEX_E_NUM_OUT_OF_RANGE,
|
|
|
|
MC_SEARCH_HEX_E_INVALID_CHARACTER
|
2016-09-25 14:09:43 +03:00
|
|
|
} mc_search_hex_parse_error_t;
|
|
|
|
|
2009-05-05 23:28:27 +04:00
|
|
|
/*** file scope type declarations ****************************************************************/
|
|
|
|
|
|
|
|
/*** file scope variables ************************************************************************/
|
|
|
|
|
|
|
|
/*** file scope functions ************************************************************************/
|
|
|
|
|
|
|
|
static GString *
|
2016-09-25 14:09:43 +03:00
|
|
|
mc_search__hex_translate_to_regex (const GString * astr, mc_search_hex_parse_error_t * error_ptr,
|
|
|
|
int *error_pos_ptr)
|
2009-05-05 23:28:27 +04:00
|
|
|
{
|
2010-10-21 17:18:18 +04:00
|
|
|
GString *buff;
|
2012-01-29 19:52:16 +04:00
|
|
|
gchar *tmp_str, *tmp_str2;
|
2011-07-19 12:32:37 +04:00
|
|
|
gsize tmp_str_len;
|
2009-05-06 11:27:06 +04:00
|
|
|
gsize loop = 0;
|
2016-09-25 14:09:43 +03:00
|
|
|
mc_search_hex_parse_error_t error = MC_SEARCH_HEX_E_OK;
|
2009-05-06 11:27:06 +04:00
|
|
|
|
2010-10-21 17:18:18 +04:00
|
|
|
buff = g_string_sized_new (64);
|
2011-07-19 12:43:00 +04:00
|
|
|
tmp_str = g_strndup (astr->str, astr->len);
|
2012-01-29 19:52:16 +04:00
|
|
|
tmp_str2 = tmp_str;
|
|
|
|
|
|
|
|
/* remove 0x prefices */
|
|
|
|
while (TRUE)
|
|
|
|
{
|
|
|
|
tmp_str2 = strstr (tmp_str2, "0x");
|
|
|
|
if (tmp_str2 == NULL)
|
|
|
|
break;
|
|
|
|
|
|
|
|
*tmp_str2++ = ' ';
|
|
|
|
*tmp_str2++ = ' ';
|
|
|
|
}
|
|
|
|
|
2011-07-19 12:32:37 +04:00
|
|
|
tmp_str_len = strlen (tmp_str);
|
2009-05-06 11:27:06 +04:00
|
|
|
|
2016-09-25 14:09:43 +03:00
|
|
|
while (loop < tmp_str_len && error == MC_SEARCH_HEX_E_OK)
|
2010-10-21 17:33:32 +04:00
|
|
|
{
|
2016-09-17 15:04:35 +03:00
|
|
|
unsigned int val;
|
|
|
|
int ptr;
|
2011-07-19 12:43:00 +04:00
|
|
|
|
2016-09-25 18:30:45 +03:00
|
|
|
if (g_ascii_isspace (tmp_str[loop]))
|
|
|
|
{
|
|
|
|
/* Eat-up whitespace between tokens. */
|
|
|
|
while (g_ascii_isspace (tmp_str[loop]))
|
|
|
|
loop++;
|
|
|
|
}
|
2013-10-10 16:21:26 +04:00
|
|
|
/* cppcheck-suppress invalidscanf */
|
2016-09-25 18:30:45 +03:00
|
|
|
else if (sscanf (tmp_str + loop, "%x%n", &val, &ptr) == 1)
|
2010-10-21 17:33:32 +04:00
|
|
|
{
|
2016-09-17 15:04:35 +03:00
|
|
|
if (val > 255)
|
2016-09-25 14:09:43 +03:00
|
|
|
error = MC_SEARCH_HEX_E_NUM_OUT_OF_RANGE;
|
2011-07-19 12:43:00 +04:00
|
|
|
else
|
|
|
|
{
|
2016-09-17 15:04:35 +03:00
|
|
|
g_string_append_printf (buff, "\\x%02X", val);
|
2011-07-19 12:43:00 +04:00
|
|
|
loop += ptr;
|
2009-05-06 11:27:06 +04:00
|
|
|
}
|
|
|
|
}
|
2010-10-21 17:33:32 +04:00
|
|
|
else if (*(tmp_str + loop) == '"')
|
|
|
|
{
|
2009-06-08 14:54:35 +04:00
|
|
|
gsize loop2 = 0;
|
2010-10-21 17:33:32 +04:00
|
|
|
|
2009-07-15 22:01:52 +04:00
|
|
|
loop++;
|
2011-07-19 12:32:37 +04:00
|
|
|
while (loop + loop2 < tmp_str_len)
|
2010-10-21 17:33:32 +04:00
|
|
|
{
|
2009-06-08 14:54:35 +04:00
|
|
|
if (*(tmp_str + loop + loop2) == '"' &&
|
2009-06-09 13:06:45 +04:00
|
|
|
!strutils_is_char_escaped (tmp_str, tmp_str + loop + loop2))
|
2009-06-08 14:54:35 +04:00
|
|
|
break;
|
|
|
|
loop2++;
|
|
|
|
}
|
2010-10-21 17:33:32 +04:00
|
|
|
|
2009-06-08 14:54:35 +04:00
|
|
|
g_string_append_len (buff, tmp_str + loop, loop2 - 1);
|
|
|
|
loop += loop2;
|
2009-05-06 11:27:06 +04:00
|
|
|
}
|
2010-10-21 17:33:32 +04:00
|
|
|
else
|
2016-09-25 18:30:45 +03:00
|
|
|
error = MC_SEARCH_HEX_E_INVALID_CHARACTER;
|
2009-05-06 11:27:06 +04:00
|
|
|
}
|
2010-02-17 16:47:28 +03:00
|
|
|
|
|
|
|
g_free (tmp_str);
|
|
|
|
|
2016-09-25 14:09:43 +03:00
|
|
|
if (error != MC_SEARCH_HEX_E_OK)
|
|
|
|
{
|
|
|
|
g_string_free (buff, TRUE);
|
|
|
|
if (error_ptr != NULL)
|
|
|
|
*error_ptr = error;
|
|
|
|
if (error_pos_ptr != NULL)
|
|
|
|
*error_pos_ptr = loop;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2009-05-05 23:28:27 +04:00
|
|
|
return buff;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*** public functions ****************************************************************************/
|
|
|
|
|
|
|
|
void
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search__cond_struct_new_init_hex (const char *charset, mc_search_t * lc_mc_search,
|
2009-05-06 11:27:06 +04:00
|
|
|
mc_search_cond_t * mc_search_cond)
|
2009-05-05 23:28:27 +04:00
|
|
|
{
|
2010-10-21 17:07:06 +04:00
|
|
|
GString *tmp;
|
2016-09-25 14:09:43 +03:00
|
|
|
mc_search_hex_parse_error_t error = MC_SEARCH_HEX_E_OK;
|
|
|
|
int error_pos = 0;
|
2009-05-05 23:28:27 +04:00
|
|
|
|
2012-01-29 19:52:16 +04:00
|
|
|
g_string_ascii_down (mc_search_cond->str);
|
2016-09-25 14:09:43 +03:00
|
|
|
tmp = mc_search__hex_translate_to_regex (mc_search_cond->str, &error, &error_pos);
|
|
|
|
if (tmp != NULL)
|
|
|
|
{
|
|
|
|
g_string_free (mc_search_cond->str, TRUE);
|
|
|
|
mc_search_cond->str = tmp;
|
|
|
|
mc_search__cond_struct_new_init_regex (charset, lc_mc_search, mc_search_cond);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const char *desc;
|
|
|
|
|
|
|
|
switch (error)
|
|
|
|
{
|
|
|
|
case MC_SEARCH_HEX_E_NUM_OUT_OF_RANGE:
|
|
|
|
desc =
|
|
|
|
_
|
|
|
|
("Number out of range (should be in byte range, 0 <= n <= 0xFF, expressed in hex)");
|
|
|
|
break;
|
2016-09-25 18:30:45 +03:00
|
|
|
case MC_SEARCH_HEX_E_INVALID_CHARACTER:
|
|
|
|
desc = _("Invalid character");
|
|
|
|
break;
|
2016-09-25 14:09:43 +03:00
|
|
|
default:
|
|
|
|
desc = "";
|
|
|
|
}
|
2009-05-05 23:28:27 +04:00
|
|
|
|
2016-09-25 14:09:43 +03:00
|
|
|
lc_mc_search->error = MC_SEARCH_E_INPUT;
|
|
|
|
lc_mc_search->error_str =
|
|
|
|
g_strdup_printf (_("Hex pattern error at position %d:\n%s."), error_pos + 1, desc);
|
|
|
|
}
|
2009-05-05 23:28:27 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
gboolean
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search__run_hex (mc_search_t * lc_mc_search, const void *user_data,
|
2009-05-06 11:27:06 +04:00
|
|
|
gsize start_search, gsize end_search, gsize * found_len)
|
2009-05-05 23:28:27 +04:00
|
|
|
{
|
2009-10-30 04:12:04 +03:00
|
|
|
return mc_search__run_regex (lc_mc_search, user_data, start_search, end_search, found_len);
|
2009-05-05 23:28:27 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|
|
|
|
|
|
|
|
GString *
|
2009-10-30 04:12:04 +03:00
|
|
|
mc_search_hex_prepare_replace_str (mc_search_t * lc_mc_search, GString * replace_str)
|
2009-05-05 23:28:27 +04:00
|
|
|
{
|
2009-10-30 04:12:04 +03:00
|
|
|
(void) lc_mc_search;
|
2009-06-08 14:54:35 +04:00
|
|
|
return g_string_new_len (replace_str->str, replace_str->len);
|
2009-05-05 23:28:27 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* --------------------------------------------------------------------------------------------- */
|