/*
Common strings utilities
Copyright (C) 2007-2025
Free Software Foundation, Inc.
Written by:
Rostislav Benes, 2007
This file is part of the Midnight Commander.
The Midnight Commander is free software: you can redistribute it
and/or modify it under the terms of the GNU General Public License as
published by the Free Software Foundation, either version 3 of the License,
or (at your option) any later version.
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.
You should have received a copy of the GNU General Public License
along with this program. If not, see .
*/
#include
#include
#include
#include
#include
#include "lib/global.h"
#include "lib/util.h" /* MC_PTR_FREE */
#include "lib/strutil.h"
/*** global variables ****************************************************************************/
GIConv str_cnv_to_term;
GIConv str_cnv_from_term;
GIConv str_cnv_not_convert = INVALID_CONV;
/*** file scope macro definitions ****************************************************************/
/*** file scope type declarations ****************************************************************/
/*** forward declarations (file scope functions) *************************************************/
/*** file scope variables ************************************************************************/
/* names, that are used for utf-8 */
static const char *const str_utf8_encodings[] = {
"utf-8",
"utf8",
NULL
};
/* standard 8bit encodings, no wide or multibytes characters */
static const char *const str_8bit_encodings[] = {
"cp-1251",
"cp1251",
/* solaris */
"ansi-1251",
"ansi1251",
"cp-1250",
"cp1250",
"cp-866",
"cp866",
/* glibc */
"ibm-866",
"ibm866",
"cp-850",
"cp850",
"cp-852",
"cp852",
"iso-8859",
"iso8859",
"koi8",
NULL
};
/* terminal encoding */
static char *codeset = NULL;
static char *term_encoding = NULL;
/* function for encoding specific operations */
static struct str_class used_class;
/* --------------------------------------------------------------------------------------------- */
/*** file scope functions ************************************************************************/
/* --------------------------------------------------------------------------------------------- */
/* if enc is same encoding like on terminal */
static int
str_test_not_convert (const char *enc)
{
return g_ascii_strcasecmp (enc, codeset) == 0;
}
/* --------------------------------------------------------------------------------------------- */
static estr_t
_str_convert (GIConv coder, const char *string, int size, GString *buffer)
{
estr_t state = ESTR_SUCCESS;
gssize left;
gsize bytes_read = 0;
gsize bytes_written = 0;
errno = 0; /* FIXME: is it really needed? */
if (coder == INVALID_CONV)
return ESTR_FAILURE;
if (string == NULL || buffer == NULL)
return ESTR_FAILURE;
/*
if (! used_class.is_valid_string (string))
{
return ESTR_FAILURE;
}
*/
if (size < 0)
size = strlen (string);
else
{
left = strlen (string);
if (left < size)
size = left;
}
left = size;
g_iconv (coder, NULL, NULL, NULL, NULL);
while (left != 0)
{
gchar *tmp_buff;
GError *mcerror = NULL;
tmp_buff = g_convert_with_iconv ((const gchar *) string,
left, coder, &bytes_read, &bytes_written, &mcerror);
if (mcerror != NULL)
{
int code = mcerror->code;
g_error_free (mcerror);
mcerror = NULL;
switch (code)
{
case G_CONVERT_ERROR_NO_CONVERSION:
/* Conversion between the requested character sets is not supported. */
g_free (tmp_buff);
mc_g_string_append_c_len (buffer, '?', strlen (string));
return ESTR_FAILURE;
case G_CONVERT_ERROR_ILLEGAL_SEQUENCE:
/* Invalid byte sequence in conversion input. */
if ((tmp_buff == NULL) && (bytes_read != 0))
/* recode valid byte sequence */
tmp_buff = g_convert_with_iconv ((const gchar *) string,
bytes_read, coder, NULL, NULL, NULL);
if (tmp_buff != NULL)
{
g_string_append (buffer, tmp_buff);
g_free (tmp_buff);
}
if ((int) bytes_read >= left)
return ESTR_PROBLEM;
string += bytes_read + 1;
size -= (bytes_read + 1);
left -= (bytes_read + 1);
g_string_append_c (buffer, *(string - 1));
state = ESTR_PROBLEM;
break;
case G_CONVERT_ERROR_PARTIAL_INPUT:
/* Partial character sequence at end of input. */
g_string_append (buffer, tmp_buff);
g_free (tmp_buff);
if ((int) bytes_read < left)
mc_g_string_append_c_len (buffer, '?', left - bytes_read);
return ESTR_PROBLEM;
case G_CONVERT_ERROR_BAD_URI: /* Don't know how handle this error :( */
case G_CONVERT_ERROR_NOT_ABSOLUTE_PATH: /* Don't know how handle this error :( */
case G_CONVERT_ERROR_FAILED: /* Conversion failed for some reason. */
default:
g_free (tmp_buff);
return ESTR_FAILURE;
}
}
else if (tmp_buff == NULL)
{
g_string_append (buffer, string);
return ESTR_PROBLEM;
}
else if (*tmp_buff == '\0')
{
g_free (tmp_buff);
g_string_append (buffer, string);
return state;
}
else
{
g_string_append (buffer, tmp_buff);
g_free (tmp_buff);
string += bytes_read;
left -= bytes_read;
}
}
return state;
}
/* --------------------------------------------------------------------------------------------- */
static int
str_test_encoding_class (const char *encoding, const char *const *table)
{
int result = 0;
if (encoding != NULL)
{
int t;
for (t = 0; table[t] != NULL; t++)
if (g_ascii_strncasecmp (encoding, table[t], strlen (table[t])) == 0)
result++;
}
return result;
}
/* --------------------------------------------------------------------------------------------- */
static void
str_choose_str_functions (void)
{
if (str_test_encoding_class (codeset, str_utf8_encodings))
used_class = str_utf8_init ();
else if (str_test_encoding_class (codeset, str_8bit_encodings))
used_class = str_8bit_init ();
else
used_class = str_ascii_init ();
}
/* --------------------------------------------------------------------------------------------- */
/*** public functions ****************************************************************************/
/* --------------------------------------------------------------------------------------------- */
GIConv
str_crt_conv_to (const char *to_enc)
{
return (!str_test_not_convert (to_enc)) ? g_iconv_open (to_enc, codeset) : str_cnv_not_convert;
}
/* --------------------------------------------------------------------------------------------- */
GIConv
str_crt_conv_from (const char *from_enc)
{
return (!str_test_not_convert (from_enc))
? g_iconv_open (codeset, from_enc) : str_cnv_not_convert;
}
/* --------------------------------------------------------------------------------------------- */
void
str_close_conv (GIConv conv)
{
if (conv != INVALID_CONV && conv != str_cnv_not_convert)
g_iconv_close (conv);
}
/* --------------------------------------------------------------------------------------------- */
estr_t
str_convert (GIConv coder, const char *string, GString *buffer)
{
return _str_convert (coder, string, -1, buffer);
}
/* --------------------------------------------------------------------------------------------- */
estr_t
str_nconvert (GIConv coder, const char *string, int size, GString *buffer)
{
return _str_convert (coder, string, size, buffer);
}
/* --------------------------------------------------------------------------------------------- */
gchar *
str_conv_gerror_message (GError *mcerror, const char *def_msg)
{
return used_class.conv_gerror_message (mcerror, def_msg);
}
/* --------------------------------------------------------------------------------------------- */
estr_t
str_vfs_convert_from (GIConv coder, const char *string, GString *buffer)
{
estr_t result = ESTR_SUCCESS;
if (coder == str_cnv_not_convert)
g_string_append (buffer, string != NULL ? string : "");
else
result = _str_convert (coder, string, -1, buffer);
return result;
}
/* --------------------------------------------------------------------------------------------- */
estr_t
str_vfs_convert_to (GIConv coder, const char *string, int size, GString *buffer)
{
return used_class.vfs_convert_to (coder, string, size, buffer);
}
/* --------------------------------------------------------------------------------------------- */
void
str_printf (GString *buffer, const char *format, ...)
{
va_list ap;
va_start (ap, format);
g_string_append_vprintf (buffer, format, ap);
va_end (ap);
}
/* --------------------------------------------------------------------------------------------- */
void
str_insert_replace_char (GString *buffer)
{
used_class.insert_replace_char (buffer);
}
/* --------------------------------------------------------------------------------------------- */
estr_t
str_translate_char (GIConv conv, const char *keys, size_t ch_size, char *output, size_t out_size)
{
size_t left;
size_t cnv;
g_iconv (conv, NULL, NULL, NULL, NULL);
left = (ch_size == (size_t) (-1)) ? strlen (keys) : ch_size;
cnv = g_iconv (conv, (gchar **) & keys, &left, &output, &out_size);
if (cnv == (size_t) (-1))
return (errno == EINVAL) ? ESTR_PROBLEM : ESTR_FAILURE;
output[0] = '\0';
return ESTR_SUCCESS;
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_detect_termencoding (void)
{
if (term_encoding == NULL)
{
/* On Linux, nl_langinfo (CODESET) returns upper case UTF-8 whether the LANG is set
to utf-8 or UTF-8.
On Mac OS X, it returns the same case as the LANG input.
So let transform result of nl_langinfo (CODESET) to upper case unconditionally. */
term_encoding = g_ascii_strup (nl_langinfo (CODESET), -1);
}
return term_encoding;
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_isutf8 (const char *codeset_name)
{
return (str_test_encoding_class (codeset_name, str_utf8_encodings) != 0);
}
/* --------------------------------------------------------------------------------------------- */
void
str_init_strings (const char *termenc)
{
codeset = termenc != NULL ? g_ascii_strup (termenc, -1) : g_strdup (str_detect_termencoding ());
str_cnv_not_convert = g_iconv_open (codeset, codeset);
if (str_cnv_not_convert == INVALID_CONV)
{
if (termenc != NULL)
{
g_free (codeset);
codeset = g_strdup (str_detect_termencoding ());
str_cnv_not_convert = g_iconv_open (codeset, codeset);
}
if (str_cnv_not_convert == INVALID_CONV)
{
g_free (codeset);
codeset = g_strdup (DEFAULT_CHARSET);
str_cnv_not_convert = g_iconv_open (codeset, codeset);
}
}
str_cnv_to_term = str_cnv_not_convert;
str_cnv_from_term = str_cnv_not_convert;
str_choose_str_functions ();
}
/* --------------------------------------------------------------------------------------------- */
void
str_uninit_strings (void)
{
if (str_cnv_not_convert != INVALID_CONV)
g_iconv_close (str_cnv_not_convert);
/* NULL-ize pointers to avoid double free in unit tests */
MC_PTR_FREE (term_encoding);
MC_PTR_FREE (codeset);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_term_form (const char *text)
{
return used_class.term_form (text);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_fit_to_term (const char *text, int width, align_crt_t just_mode)
{
return used_class.fit_to_term (text, width, just_mode);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_term_trim (const char *text, int width)
{
return used_class.term_trim (text, width);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_term_substring (const char *text, int start, int width)
{
return used_class.term_substring (text, start, width);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_get_next_char (char *text)
{
used_class.cnext_char ((const char **) &text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_cget_next_char (const char *text)
{
used_class.cnext_char (&text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
void
str_next_char (char **text)
{
used_class.cnext_char ((const char **) text);
}
/* --------------------------------------------------------------------------------------------- */
void
str_cnext_char (const char **text)
{
used_class.cnext_char (text);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_get_prev_char (char *text)
{
used_class.cprev_char ((const char **) &text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_cget_prev_char (const char *text)
{
used_class.cprev_char (&text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
void
str_prev_char (char **text)
{
used_class.cprev_char ((const char **) text);
}
/* --------------------------------------------------------------------------------------------- */
void
str_cprev_char (const char **text)
{
used_class.cprev_char (text);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_get_next_char_safe (char *text)
{
used_class.cnext_char_safe ((const char **) &text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_cget_next_char_safe (const char *text)
{
used_class.cnext_char_safe (&text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
void
str_next_char_safe (char **text)
{
used_class.cnext_char_safe ((const char **) text);
}
/* --------------------------------------------------------------------------------------------- */
void
str_cnext_char_safe (const char **text)
{
used_class.cnext_char_safe (text);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_get_prev_char_safe (char *text)
{
used_class.cprev_char_safe ((const char **) &text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_cget_prev_char_safe (const char *text)
{
used_class.cprev_char_safe (&text);
return text;
}
/* --------------------------------------------------------------------------------------------- */
void
str_prev_char_safe (char **text)
{
used_class.cprev_char_safe ((const char **) text);
}
/* --------------------------------------------------------------------------------------------- */
void
str_cprev_char_safe (const char **text)
{
used_class.cprev_char_safe (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_next_noncomb_char (char **text)
{
return used_class.cnext_noncomb_char ((const char **) text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_cnext_noncomb_char (const char **text)
{
return used_class.cnext_noncomb_char (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_prev_noncomb_char (char **text, const char *begin)
{
return used_class.cprev_noncomb_char ((const char **) text, begin);
}
/* --------------------------------------------------------------------------------------------- */
int
str_cprev_noncomb_char (const char **text, const char *begin)
{
return used_class.cprev_noncomb_char (text, begin);
}
/* --------------------------------------------------------------------------------------------- */
int
str_is_valid_char (const char *ch, size_t size)
{
return used_class.is_valid_char (ch, size);
}
/* --------------------------------------------------------------------------------------------- */
int
str_term_width1 (const char *text)
{
return used_class.term_width1 (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_term_width2 (const char *text, size_t length)
{
return used_class.term_width2 (text, length);
}
/* --------------------------------------------------------------------------------------------- */
int
str_term_char_width (const char *text)
{
return used_class.term_char_width (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_offset_to_pos (const char *text, size_t length)
{
return used_class.offset_to_pos (text, length);
}
/* --------------------------------------------------------------------------------------------- */
int
str_length (const char *text)
{
return used_class.length (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_length_char (const char *text)
{
return str_cget_next_char_safe (text) - text;
}
/* --------------------------------------------------------------------------------------------- */
int
str_length2 (const char *text, int size)
{
return used_class.length2 (text, size);
}
/* --------------------------------------------------------------------------------------------- */
int
str_length_noncomb (const char *text)
{
return used_class.length_noncomb (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_column_to_pos (const char *text, size_t pos)
{
return used_class.column_to_pos (text, pos);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_isspace (const char *ch)
{
return used_class.char_isspace (ch);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_ispunct (const char *ch)
{
return used_class.char_ispunct (ch);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_isalnum (const char *ch)
{
return used_class.char_isalnum (ch);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_isdigit (const char *ch)
{
return used_class.char_isdigit (ch);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_toupper (const char *ch, char **out, size_t *remain)
{
return used_class.char_toupper (ch, out, remain);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_tolower (const char *ch, char **out, size_t *remain)
{
return used_class.char_tolower (ch, out, remain);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_isprint (const char *ch)
{
return used_class.char_isprint (ch);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_iscombiningmark (const char *ch)
{
return used_class.char_iscombiningmark (ch);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_trunc (const char *text, int width)
{
return used_class.trunc (text, width);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_create_search_needle (const char *needle, gboolean case_sen)
{
return used_class.create_search_needle (needle, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
void
str_release_search_needle (char *needle, gboolean case_sen)
{
used_class.release_search_needle (needle, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_search_first (const char *text, const char *search, gboolean case_sen)
{
return used_class.search_first (text, search, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
const char *
str_search_last (const char *text, const char *search, gboolean case_sen)
{
return used_class.search_last (text, search, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
gboolean
str_is_valid_string (const char *text)
{
return used_class.is_valid_string (text);
}
/* --------------------------------------------------------------------------------------------- */
int
str_compare (const char *t1, const char *t2)
{
return used_class.compare (t1, t2);
}
/* --------------------------------------------------------------------------------------------- */
int
str_ncompare (const char *t1, const char *t2)
{
return used_class.ncompare (t1, t2);
}
/* --------------------------------------------------------------------------------------------- */
int
str_casecmp (const char *t1, const char *t2)
{
return used_class.casecmp (t1, t2);
}
/* --------------------------------------------------------------------------------------------- */
int
str_ncasecmp (const char *t1, const char *t2)
{
return used_class.ncasecmp (t1, t2);
}
/* --------------------------------------------------------------------------------------------- */
int
str_prefix (const char *text, const char *prefix)
{
return used_class.prefix (text, prefix);
}
/* --------------------------------------------------------------------------------------------- */
int
str_caseprefix (const char *text, const char *prefix)
{
return used_class.caseprefix (text, prefix);
}
/* --------------------------------------------------------------------------------------------- */
void
str_fix_string (char *text)
{
used_class.fix_string (text);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_create_key (const char *text, gboolean case_sen)
{
return used_class.create_key (text, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
char *
str_create_key_for_filename (const char *text, gboolean case_sen)
{
return used_class.create_key_for_filename (text, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
int
str_key_collate (const char *t1, const char *t2, gboolean case_sen)
{
return used_class.key_collate (t1, t2, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
void
str_release_key (char *key, gboolean case_sen)
{
used_class.release_key (key, case_sen);
}
/* --------------------------------------------------------------------------------------------- */
void
str_msg_term_size (const char *text, int *lines, int *columns)
{
char *p, *tmp;
char *q;
char c = '\0';
*lines = 1;
*columns = 0;
tmp = g_strdup (text);
p = tmp;
while (TRUE)
{
int width;
q = strchr (p, '\n');
if (q != NULL)
{
c = q[0];
q[0] = '\0';
}
width = str_term_width1 (p);
if (width > *columns)
*columns = width;
if (q == NULL)
break;
q[0] = c;
p = q + 1;
(*lines)++;
}
g_free (tmp);
}
/* --------------------------------------------------------------------------------------------- */
char *
strrstr_skip_count (const char *haystack, const char *needle, size_t skip_count)
{
char *semi;
ssize_t len;
len = strlen (haystack);
do
{
semi = g_strrstr_len (haystack, len, needle);
if (semi == NULL)
return NULL;
len = semi - haystack - 1;
}
while (skip_count-- != 0);
return semi;
}
/* --------------------------------------------------------------------------------------------- */
/* Interpret string as a non-negative decimal integer, optionally multiplied by various values.
*
* @param str input value
* @param invalid set to TRUE if "str" does not represent a number in this format
*
* @return non-negative integer representation of "str", 0 in case of error.
*/
uintmax_t
parse_integer (const char *str, gboolean *invalid)
{
uintmax_t n;
char *suffix;
strtol_error_t e;
e = xstrtoumax (str, &suffix, 10, &n, "bcEGkKMPTwYZ0");
if (e == LONGINT_INVALID_SUFFIX_CHAR && *suffix == 'x')
{
uintmax_t multiplier;
multiplier = parse_integer (suffix + 1, invalid);
if (multiplier != 0 && n * multiplier / multiplier != n)
{
*invalid = TRUE;
return 0;
}
n *= multiplier;
}
else if (e != LONGINT_OK)
{
*invalid = TRUE;
n = 0;
}
return n;
}
/* --------------------------------------------------------------------------------------------- */