2007-12-30 16:41:49 +00:00
|
|
|
|
/***************************************************************************
|
|
|
|
|
|
2009-08-17 10:41:51 +00:00
|
|
|
|
gbx_string.c
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2009-08-17 10:41:51 +00:00
|
|
|
|
(c) 2000-2009 Benoît Minisini <gambas@users.sourceforge.net>
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2009-08-17 10:41:51 +00:00
|
|
|
|
This program 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 2, or (at your option)
|
|
|
|
|
any later version.
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2009-08-17 10:41:51 +00:00
|
|
|
|
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.
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2009-08-17 10:41:51 +00:00
|
|
|
|
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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
|
|
#define __STRING_C
|
|
|
|
|
|
|
|
|
|
#include "gb_common.h"
|
|
|
|
|
#include "gb_common_buffer.h"
|
|
|
|
|
#include "gb_common_case.h"
|
2008-01-27 14:00:04 +00:00
|
|
|
|
#include "gb_common_string.h"
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
#include "gb_error.h"
|
|
|
|
|
#include "gbx_value.h"
|
|
|
|
|
#include "gbx_debug.h"
|
|
|
|
|
#include "gbx_local.h"
|
2008-03-19 14:32:30 +00:00
|
|
|
|
#include "gbx_project.h"
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#include "gbx_exec.h"
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <pwd.h>
|
|
|
|
|
#include <ctype.h>
|
|
|
|
|
#include <iconv.h>
|
|
|
|
|
|
|
|
|
|
#include "gbx_string.h"
|
|
|
|
|
|
|
|
|
|
#if DEBUG_STRING
|
|
|
|
|
#define DEBUG_ME
|
|
|
|
|
#endif
|
|
|
|
|
|
2009-02-06 17:40:15 +00:00
|
|
|
|
/*#ifdef DEBUG_ME
|
|
|
|
|
extern FILE *MEMORY_log;
|
|
|
|
|
#undef stderr
|
|
|
|
|
#define stderr MEMORY_log
|
|
|
|
|
static void print_where()
|
|
|
|
|
{
|
|
|
|
|
fprintf(MEMORY_log, "%s: ", DEBUG_get_current_position());
|
|
|
|
|
}
|
|
|
|
|
#define DEBUG_where print_where
|
|
|
|
|
#endif*/
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
#if DEBUG_MEMORY
|
|
|
|
|
|
|
|
|
|
static void *_my_malloc(int size)
|
|
|
|
|
{
|
|
|
|
|
void *ptr;
|
|
|
|
|
ALLOC(&ptr, size, "_my_malloc");
|
|
|
|
|
return ptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void _my_free(void *ptr)
|
|
|
|
|
{
|
|
|
|
|
IFREE(ptr, "_my_free");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void *_my_realloc(void *ptr, int size)
|
|
|
|
|
{
|
|
|
|
|
REALLOC(&ptr, size, "_my_realloc");
|
|
|
|
|
return ptr;
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
#define _my_malloc malloc
|
|
|
|
|
#define _my_free free
|
|
|
|
|
#define _my_realloc realloc
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
#define STRING_last_count 32
|
|
|
|
|
static char *STRING_last[STRING_last_count] = { 0 };
|
|
|
|
|
|
2010-05-22 18:02:34 +00:00
|
|
|
|
const char STRING_char_string[512] =
|
2008-01-17 21:39:26 +00:00
|
|
|
|
"\x00\x00\x01\x00\x02\x00\x03\x00\x04\x00\x05\x00\x06\x00\x07\x00\x08\x00\x09\x00\x0A\x00\x0B\x00\x0C\x00\x0D\x00\x0E\x00\x0F\x00"
|
|
|
|
|
"\x10\x00\x11\x00\x12\x00\x13\x00\x14\x00\x15\x00\x16\x00\x17\x00\x18\x00\x19\x00\x1A\x00\x1B\x00\x1C\x00\x1D\x00\x1E\x00\x1F\x00"
|
|
|
|
|
"\x20\x00\x21\x00\x22\x00\x23\x00\x24\x00\x25\x00\x26\x00\x27\x00\x28\x00\x29\x00\x2A\x00\x2B\x00\x2C\x00\x2D\x00\x2E\x00\x2F\x00"
|
|
|
|
|
"\x30\x00\x31\x00\x32\x00\x33\x00\x34\x00\x35\x00\x36\x00\x37\x00\x38\x00\x39\x00\x3A\x00\x3B\x00\x3C\x00\x3D\x00\x3E\x00\x3F\x00"
|
|
|
|
|
"\x40\x00\x41\x00\x42\x00\x43\x00\x44\x00\x45\x00\x46\x00\x47\x00\x48\x00\x49\x00\x4A\x00\x4B\x00\x4C\x00\x4D\x00\x4E\x00\x4F\x00"
|
|
|
|
|
"\x50\x00\x51\x00\x52\x00\x53\x00\x54\x00\x55\x00\x56\x00\x57\x00\x58\x00\x59\x00\x5A\x00\x5B\x00\x5C\x00\x5D\x00\x5E\x00\x5F\x00"
|
|
|
|
|
"\x60\x00\x61\x00\x62\x00\x63\x00\x64\x00\x65\x00\x66\x00\x67\x00\x68\x00\x69\x00\x6A\x00\x6B\x00\x6C\x00\x6D\x00\x6E\x00\x6F\x00"
|
|
|
|
|
"\x70\x00\x71\x00\x72\x00\x73\x00\x74\x00\x75\x00\x76\x00\x77\x00\x78\x00\x79\x00\x7A\x00\x7B\x00\x7C\x00\x7D\x00\x7E\x00\x7F\x00"
|
|
|
|
|
"\x80\x00\x81\x00\x82\x00\x83\x00\x84\x00\x85\x00\x86\x00\x87\x00\x88\x00\x89\x00\x8A\x00\x8B\x00\x8C\x00\x8D\x00\x8E\x00\x8F\x00"
|
|
|
|
|
"\x90\x00\x91\x00\x92\x00\x93\x00\x94\x00\x95\x00\x96\x00\x97\x00\x98\x00\x99\x00\x9A\x00\x9B\x00\x9C\x00\x9D\x00\x9E\x00\x9F\x00"
|
|
|
|
|
"\xA0\x00\xA1\x00\xA2\x00\xA3\x00\xA4\x00\xA5\x00\xA6\x00\xA7\x00\xA8\x00\xA9\x00\xAA\x00\xAB\x00\xAC\x00\xAD\x00\xAE\x00\xAF\x00"
|
|
|
|
|
"\xB0\x00\xB1\x00\xB2\x00\xB3\x00\xB4\x00\xB5\x00\xB6\x00\xB7\x00\xB8\x00\xB9\x00\xBA\x00\xBB\x00\xBC\x00\xBD\x00\xBE\x00\xBF\x00"
|
|
|
|
|
"\xC0\x00\xC1\x00\xC2\x00\xC3\x00\xC4\x00\xC5\x00\xC6\x00\xC7\x00\xC8\x00\xC9\x00\xCA\x00\xCB\x00\xCC\x00\xCD\x00\xCE\x00\xCF\x00"
|
|
|
|
|
"\xD0\x00\xD1\x00\xD2\x00\xD3\x00\xD4\x00\xD5\x00\xD6\x00\xD7\x00\xD8\x00\xD9\x00\xDA\x00\xDB\x00\xDC\x00\xDD\x00\xDE\x00\xDF\x00"
|
|
|
|
|
"\xE0\x00\xE1\x00\xE2\x00\xE3\x00\xE4\x00\xE5\x00\xE6\x00\xE7\x00\xE8\x00\xE9\x00\xEA\x00\xEB\x00\xEC\x00\xED\x00\xEE\x00\xEF\x00"
|
|
|
|
|
"\xF0\x00\xF1\x00\xF2\x00\xF3\x00\xF4\x00\xF5\x00\xF6\x00\xF7\x00\xF8\x00\xF9\x00\xFA\x00\xFB\x00\xFC\x00\xFD\x00\xFE\x00\xFF\x00";
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
static int _index = 0;
|
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_MAKE STRING_make_buffer;
|
|
|
|
|
#define _make STRING_make_buffer
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
//static HASH_TABLE *_intern = NULL;
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
/****************************************************************************
|
2008-10-21 09:28:34 +00:00
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
String pool management
|
|
|
|
|
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#define SIZE_INC 16
|
|
|
|
|
#define REAL_SIZE(_len) (((_len) + (SIZE_INC - 1)) & ~(SIZE_INC - 1))
|
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
#define POOL_SIZE 16
|
2010-06-08 23:08:04 +00:00
|
|
|
|
#define POOL_MAX 64
|
2008-03-19 14:32:30 +00:00
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
#define POOL_MAX_LEN (POOL_SIZE * SIZE_INC)
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
static STRING *_pool[POOL_SIZE] = { 0 };
|
|
|
|
|
static int _pool_count[POOL_SIZE] = { 0 };
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
#define alloc_string(_len) \
|
|
|
|
|
({ \
|
|
|
|
|
STRING *str; \
|
|
|
|
|
int size = REAL_SIZE((_len) + 1 + sizeof(STRING)); \
|
|
|
|
|
int pool = (size / SIZE_INC) - 1; \
|
|
|
|
|
\
|
|
|
|
|
MEMORY_count++; \
|
|
|
|
|
\
|
|
|
|
|
if (pool < POOL_SIZE && (_pool_count[pool])) \
|
|
|
|
|
{ \
|
|
|
|
|
str = _pool[pool]; \
|
|
|
|
|
_pool[pool] = *((STRING **)str); \
|
|
|
|
|
_pool_count[pool]--; \
|
|
|
|
|
/*fprintf(stderr, "alloc_string: (%p) %d bytes from pool %d\n", str, size, pool);*/ \
|
|
|
|
|
} \
|
|
|
|
|
else \
|
|
|
|
|
{ \
|
|
|
|
|
str = _my_malloc(size); \
|
|
|
|
|
if (!str) \
|
|
|
|
|
THROW_MEMORY(); \
|
|
|
|
|
} \
|
|
|
|
|
str->len = (_len); \
|
|
|
|
|
str->ref = 1; \
|
|
|
|
|
str; \
|
|
|
|
|
})
|
2008-03-19 14:32:30 +00:00
|
|
|
|
|
|
|
|
|
void STRING_free_real(char *ptr)
|
|
|
|
|
{
|
|
|
|
|
STRING *str = STRING_from_ptr(ptr);
|
|
|
|
|
int size = REAL_SIZE(str->len + 1 + sizeof(STRING));
|
|
|
|
|
int pool = (size / SIZE_INC) - 1;
|
|
|
|
|
|
2010-05-28 19:33:05 +00:00
|
|
|
|
MEMORY_count--;
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
//fprintf(stderr, "STRING_free_real: %p\n", ptr);
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
if (pool < POOL_SIZE)
|
|
|
|
|
{
|
|
|
|
|
if (_pool_count[pool] < POOL_MAX)
|
|
|
|
|
{
|
2010-06-04 23:48:53 +00:00
|
|
|
|
//#ifdef DEBUG_ME
|
|
|
|
|
//fprintf(stderr, "STRING_free_real: (%p) %d bytes to pool %d\n", str, size, pool);
|
|
|
|
|
//#endif
|
2008-03-19 14:32:30 +00:00
|
|
|
|
*((STRING **)str) = _pool[pool];
|
|
|
|
|
_pool[pool] = str;
|
|
|
|
|
_pool_count[pool]++;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
_my_free(str);
|
2008-03-19 14:32:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static STRING *realloc_string(STRING *str, int new_len)
|
|
|
|
|
{
|
|
|
|
|
int size;
|
|
|
|
|
int new_size;
|
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (new_len == str->len)
|
|
|
|
|
return str;
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
size = REAL_SIZE(str->len + 1 + sizeof(STRING));
|
|
|
|
|
new_size = REAL_SIZE(new_len + 1 + sizeof(STRING));
|
|
|
|
|
|
|
|
|
|
if (new_size != size)
|
|
|
|
|
{
|
|
|
|
|
if (new_len == 0)
|
|
|
|
|
{
|
|
|
|
|
STRING_free_real(str->data);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2008-11-10 00:18:00 +00:00
|
|
|
|
else if (size > POOL_MAX_LEN && new_size > POOL_MAX_LEN)
|
|
|
|
|
{
|
2010-06-04 23:48:53 +00:00
|
|
|
|
str = _my_realloc(str, new_size);
|
2010-05-28 19:33:05 +00:00
|
|
|
|
//REALLOC(&str, new_size, "realloc_string");
|
2008-11-10 00:18:00 +00:00
|
|
|
|
}
|
2008-03-19 14:32:30 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
STRING *nstr = alloc_string(new_len);
|
|
|
|
|
if (new_len < str->len)
|
|
|
|
|
memcpy(nstr->data, str->data, new_len);
|
|
|
|
|
else
|
|
|
|
|
memcpy(nstr->data, str->data, str->len);
|
2008-05-03 15:07:55 +00:00
|
|
|
|
if (str->ref == 1)
|
|
|
|
|
STRING_free_real(str->data);
|
|
|
|
|
else
|
|
|
|
|
str->ref--;
|
2008-03-19 14:32:30 +00:00
|
|
|
|
str = nstr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
str->len = new_len;
|
|
|
|
|
return str;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void clear_pool(void)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
STRING *str, *next;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < POOL_SIZE; i++)
|
|
|
|
|
{
|
|
|
|
|
str = _pool[i];
|
|
|
|
|
while (str)
|
|
|
|
|
{
|
|
|
|
|
next = *((STRING **)str);
|
2010-06-04 23:48:53 +00:00
|
|
|
|
_my_free(str);
|
2008-03-19 14:32:30 +00:00
|
|
|
|
str = next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/****************************************************************************
|
2008-10-21 09:28:34 +00:00
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
String routines
|
|
|
|
|
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
char *STRING_new(const char *src, int len)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING *str;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2010-05-22 18:02:34 +00:00
|
|
|
|
if (len == 0)
|
2010-06-04 23:48:53 +00:00
|
|
|
|
return NULL;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
//ALLOC(&str, REAL_SIZE(len + 1 + sizeof(STRING)), "STRING_new");
|
|
|
|
|
str = alloc_string(len);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (src)
|
|
|
|
|
memcpy(str->data, src, len);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
str->data[len] = 0;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
DEBUG_where();
|
2010-06-04 23:48:53 +00:00
|
|
|
|
fprintf(stderr, "STRING_new %p ( 0 ) \"%.*s\"\n", str->data, len, src);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
fflush(stderr);
|
|
|
|
|
#endif
|
2010-06-04 23:48:53 +00:00
|
|
|
|
|
|
|
|
|
return str->data;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
char *STRING_free_later(char *ptr)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
/*if (NLast >= MAX_LAST_STRING)
|
|
|
|
|
THROW(E_STRING);*/
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
if (STRING_last[_index])
|
|
|
|
|
{
|
|
|
|
|
DEBUG_where();
|
|
|
|
|
fprintf(stderr, "STRING_free_later: release temp: %p '%s'\n", STRING_last[_index], STRING_last[_index]);
|
|
|
|
|
fflush(stderr);
|
|
|
|
|
}
|
|
|
|
|
#endif
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING_unref(&STRING_last[_index]);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
fprintf(stderr, "STRING_free_later: post temp: %p '%s'\n", ptr, ptr);
|
|
|
|
|
fflush(stderr);
|
|
|
|
|
#endif
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING_last[_index] = ptr;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
_index++;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (_index >= STRING_last_count)
|
|
|
|
|
_index = 0;
|
2010-06-04 23:48:53 +00:00
|
|
|
|
|
|
|
|
|
return ptr;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
int STRING_get_free_index(void)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
return _index;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*void STRING_init(void)
|
|
|
|
|
{
|
2010-02-07 23:10:41 +00:00
|
|
|
|
HASH_TABLE_create(&_intern, 0, GB_COMP_NOCASE);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}*/
|
|
|
|
|
|
|
|
|
|
void STRING_exit(void)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < STRING_last_count; i++)
|
|
|
|
|
{
|
|
|
|
|
/*if (STRING_last[i])
|
|
|
|
|
printf("release temp %p '%s'\n", STRING_last[i], STRING_last[i]);*/
|
|
|
|
|
STRING_unref(&STRING_last[i]);
|
|
|
|
|
STRING_last[i] = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_index = 0;
|
|
|
|
|
|
|
|
|
|
clear_pool();
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void STRING_extend(char **ptr, int new_len)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING *str;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
if (!*ptr)
|
|
|
|
|
str = alloc_string(new_len);
|
|
|
|
|
else
|
|
|
|
|
str = realloc_string(STRING_from_ptr(*ptr), new_len);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
*ptr = str ? str->data : NULL;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void STRING_extend_end(char **ptr)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (*ptr)
|
|
|
|
|
{
|
|
|
|
|
(*ptr)[STRING_length(*ptr)] = 0;
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_free_later(*ptr);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Attention ! Contrairement <20>STRING_new, STRING_new_temp_value cr<63> des
|
2008-10-21 09:28:34 +00:00
|
|
|
|
cha<EFBFBD>es temporaires.
|
2007-12-30 16:41:49 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
void STRING_new_temp_value(VALUE *value, const char *src, int len)
|
|
|
|
|
{
|
2010-06-04 23:48:53 +00:00
|
|
|
|
value->_string.addr = STRING_new_temp(src, len);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
value->_string.len = STRING_length(value->_string.addr);
|
|
|
|
|
value->_string.start = 0;
|
|
|
|
|
value->type = T_STRING;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void STRING_new_constant_value(VALUE *value, const char *src, int len)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
value->_string.addr = (char *)src;
|
|
|
|
|
value->_string.len = ((len < 0) ? strlen(src) : len);
|
|
|
|
|
value->_string.start = 0;
|
|
|
|
|
value->type = T_CSTRING;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void STRING_void_value(VALUE *value)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
value->type = T_CSTRING;
|
|
|
|
|
value->_string.addr = NULL;
|
|
|
|
|
value->_string.start = 0;
|
|
|
|
|
value->_string.len = 0;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#if DEBUG_STRING
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
void STRING_free(char **ptr)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (*ptr == NULL)
|
|
|
|
|
return;
|
2008-03-19 14:32:30 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
DEBUG_where();
|
|
|
|
|
fprintf(stderr, "STRING_free: %p %p\n", *ptr, ptr);
|
|
|
|
|
fflush(stderr);
|
|
|
|
|
#endif
|
2008-03-19 14:32:30 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING_free_real(*ptr);
|
|
|
|
|
*ptr = NULL;
|
2008-03-19 14:32:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
void STRING_ref(char *ptr)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING *str;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (ptr == NULL)
|
|
|
|
|
return;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
str = STRING_from_ptr(ptr);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
DEBUG_where();
|
2010-06-04 23:48:53 +00:00
|
|
|
|
fprintf(stderr, "STRING_ref: %p ( %d -> %d )\n", ptr, str->ref, str->ref + 1);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (str->ref < 0 || str->ref > 10000)
|
|
|
|
|
fprintf(stderr, "*** BAD\n");
|
|
|
|
|
fflush(stderr);
|
|
|
|
|
#endif
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
str->ref++;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void STRING_unref(char **ptr)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING *str;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (*ptr == NULL)
|
|
|
|
|
return;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
str = STRING_from_ptr(*ptr);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
#ifdef DEBUG_ME
|
|
|
|
|
DEBUG_where();
|
2010-06-04 23:48:53 +00:00
|
|
|
|
fprintf(stderr, "STRING_unref: %p ( %d -> %d )\n", *ptr, str->ref, str->ref - 1);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (str->ref < 1 || str->ref > 10000)
|
|
|
|
|
fprintf(stderr, "*** BAD\n");
|
|
|
|
|
fflush(stderr);
|
|
|
|
|
#endif
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if ((--str->ref) <= 0)
|
|
|
|
|
STRING_free(ptr);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
void STRING_unref_keep(char **ptr)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING *str;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (*ptr == NULL)
|
|
|
|
|
return;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
str = STRING_from_ptr(*ptr);
|
|
|
|
|
if (str->ref > 1)
|
|
|
|
|
str->ref--;
|
|
|
|
|
else
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_free_later(*ptr);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-03-26 16:28:18 +00:00
|
|
|
|
/* The get_param argument index starts at 1, not 0! */
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-04 14:07:53 +00:00
|
|
|
|
#define INDEX_AT 0
|
|
|
|
|
#define INDEX_IGNORE (-1)
|
|
|
|
|
#define INDEX_ERROR (-2)
|
|
|
|
|
|
|
|
|
|
static int get_param_index(const char *str, int len, uint *pos, int *len_pattern)
|
|
|
|
|
{
|
|
|
|
|
uint i;
|
|
|
|
|
int index;
|
|
|
|
|
bool err;
|
|
|
|
|
uchar d;
|
|
|
|
|
|
|
|
|
|
i = *pos + 1;
|
|
|
|
|
d = str[i];
|
|
|
|
|
if (d == '&')
|
|
|
|
|
{
|
|
|
|
|
index = INDEX_AT;
|
|
|
|
|
}
|
|
|
|
|
else if (d >= '1' && d <= '9')
|
|
|
|
|
{
|
|
|
|
|
index = d - '0';
|
|
|
|
|
}
|
|
|
|
|
else if (d == '{')
|
|
|
|
|
{
|
|
|
|
|
err = FALSE;
|
|
|
|
|
index = 0;
|
|
|
|
|
for(;;)
|
|
|
|
|
{
|
|
|
|
|
i++;
|
|
|
|
|
if (i >= len)
|
|
|
|
|
break;
|
|
|
|
|
d = str[i];
|
|
|
|
|
if (d == '}')
|
|
|
|
|
break;
|
|
|
|
|
if (d >= '0' && d <= '9')
|
|
|
|
|
index = index * 10 + d - '0';
|
|
|
|
|
else
|
|
|
|
|
err = TRUE;
|
|
|
|
|
}
|
|
|
|
|
if (err || index < 1 || index >= 64)
|
|
|
|
|
index = INDEX_ERROR;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
index = INDEX_IGNORE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (len_pattern)
|
|
|
|
|
*len_pattern = i - *pos + 1;
|
|
|
|
|
|
|
|
|
|
*pos = i;
|
|
|
|
|
return index;
|
|
|
|
|
}
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
char *STRING_subst(const char *str, int len, SUBST_FUNC get_param)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
|
|
|
|
uint i;
|
2008-10-04 14:07:53 +00:00
|
|
|
|
uchar c;
|
|
|
|
|
int np, lenp;
|
2008-03-11 13:29:47 +00:00
|
|
|
|
int len_subst;
|
|
|
|
|
char *subst;
|
|
|
|
|
char *ps;
|
2008-10-04 14:07:53 +00:00
|
|
|
|
char *p[64];
|
|
|
|
|
int lp[64];
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (!str)
|
|
|
|
|
return NULL;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (len <= 0)
|
|
|
|
|
len = strlen(str);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-11 13:29:47 +00:00
|
|
|
|
// Calculate the length
|
|
|
|
|
|
|
|
|
|
len_subst = len;
|
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
|
{
|
|
|
|
|
c = str[i];
|
2008-03-11 13:29:47 +00:00
|
|
|
|
if (c == '&')
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-04 14:07:53 +00:00
|
|
|
|
np = get_param_index(str, len, &i, &lenp);
|
|
|
|
|
len_subst -= lenp;
|
|
|
|
|
|
|
|
|
|
switch (np)
|
|
|
|
|
{
|
|
|
|
|
case INDEX_AT:
|
|
|
|
|
len_subst++;
|
|
|
|
|
break;
|
|
|
|
|
case INDEX_IGNORE:
|
|
|
|
|
len_subst += lenp;
|
|
|
|
|
break;
|
|
|
|
|
case INDEX_ERROR:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
np--;
|
|
|
|
|
(*get_param)(np + 1, &p[np], &lp[np]);
|
|
|
|
|
if (lp[np] < 0)
|
|
|
|
|
lp[np] = strlen(p[np]);
|
|
|
|
|
len_subst += lp[np];
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2008-03-11 13:29:47 +00:00
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
subst = STRING_new(NULL, len_subst);
|
2008-03-11 13:29:47 +00:00
|
|
|
|
ps = subst;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-11 13:29:47 +00:00
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
|
{
|
|
|
|
|
c = str[i];
|
|
|
|
|
if (c == '&')
|
|
|
|
|
{
|
2008-10-04 14:07:53 +00:00
|
|
|
|
np = get_param_index(str, len, &i, &lenp);
|
|
|
|
|
switch (np)
|
|
|
|
|
{
|
|
|
|
|
case INDEX_AT:
|
|
|
|
|
*ps++ = '&';
|
|
|
|
|
break;
|
|
|
|
|
case INDEX_IGNORE:
|
|
|
|
|
*ps++ = '&';
|
|
|
|
|
*ps++ = str[i];
|
|
|
|
|
break;
|
|
|
|
|
case INDEX_ERROR:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
np--;
|
|
|
|
|
memcpy(ps, p[np], lp[np]);
|
|
|
|
|
ps += lp[np];
|
|
|
|
|
}
|
2008-03-11 13:29:47 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
*ps++ = c;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
*ps = 0;
|
2010-06-04 23:48:53 +00:00
|
|
|
|
return STRING_free_later(subst);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-03-26 16:28:18 +00:00
|
|
|
|
char *STRING_subst_add(const char *str, int len, SUBST_ADD_FUNC add_param)
|
|
|
|
|
{
|
|
|
|
|
uint i;
|
2008-11-10 00:18:00 +00:00
|
|
|
|
char c;
|
2008-03-26 16:28:18 +00:00
|
|
|
|
int np;
|
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (!str)
|
|
|
|
|
return NULL;
|
2008-03-26 16:28:18 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (len <= 0)
|
|
|
|
|
len = strlen(str);
|
2008-03-26 16:28:18 +00:00
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_start_len(len);
|
2008-03-26 16:28:18 +00:00
|
|
|
|
|
|
|
|
|
for (i = 0; i < len; i++)
|
|
|
|
|
{
|
|
|
|
|
c = str[i];
|
|
|
|
|
if (c == '&')
|
|
|
|
|
{
|
2008-10-04 14:07:53 +00:00
|
|
|
|
np = get_param_index(str, len, &i, NULL);
|
|
|
|
|
switch (np)
|
|
|
|
|
{
|
|
|
|
|
case INDEX_AT:
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_make_char('&');
|
2008-10-04 14:07:53 +00:00
|
|
|
|
break;
|
|
|
|
|
case INDEX_IGNORE:
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_make_char('&');
|
|
|
|
|
STRING_make_char(str[i]);
|
2008-10-04 14:07:53 +00:00
|
|
|
|
break;
|
|
|
|
|
case INDEX_ERROR:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
(*add_param)(np);
|
|
|
|
|
}
|
2008-03-26 16:28:18 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_make_char(c);
|
2008-03-26 16:28:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
return STRING_end_temp();
|
2008-03-26 16:28:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
void STRING_add(char **ptr, const char *src, int len)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
int old_len;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (len <= 0 && src != NULL)
|
|
|
|
|
len = strlen(src);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (len <= 0)
|
|
|
|
|
return;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
old_len = STRING_length(*ptr);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING_extend(ptr, old_len + len);
|
2009-07-14 23:25:52 +00:00
|
|
|
|
if (src)
|
|
|
|
|
{
|
|
|
|
|
memcpy(&((*ptr)[old_len]), src, len);
|
|
|
|
|
(*ptr)[old_len + len] = 0;
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-27 14:00:04 +00:00
|
|
|
|
void STRING_add_char(char **ptr, char c)
|
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
int len = STRING_length(*ptr);
|
|
|
|
|
char *p;
|
2008-01-27 14:00:04 +00:00
|
|
|
|
|
|
|
|
|
//if (!(len & (SIZE_INC - 1)))
|
2008-10-21 09:28:34 +00:00
|
|
|
|
STRING_extend(ptr, len + 1);
|
|
|
|
|
//else
|
|
|
|
|
//STRING_from_ptr(*ptr)->len++;
|
|
|
|
|
|
|
|
|
|
p = *ptr + len;
|
2008-01-27 14:00:04 +00:00
|
|
|
|
p[0] = c;
|
|
|
|
|
p[1] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
int STRING_conv(char **result, const char *str, int len, const char *src, const char *dst, bool throw)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
iconv_t handle;
|
|
|
|
|
bool err;
|
|
|
|
|
const char *in;
|
|
|
|
|
char *out;
|
|
|
|
|
size_t in_len;
|
|
|
|
|
size_t out_len;
|
|
|
|
|
size_t ret;
|
|
|
|
|
int errcode = 0;
|
|
|
|
|
bool unicode;
|
|
|
|
|
|
|
|
|
|
*result = NULL;
|
|
|
|
|
|
|
|
|
|
in = str;
|
|
|
|
|
in_len = len;
|
|
|
|
|
|
|
|
|
|
if (len == 0)
|
|
|
|
|
return errcode;
|
|
|
|
|
|
|
|
|
|
if (dst == SC_UNICODE)
|
|
|
|
|
{
|
|
|
|
|
dst = EXEC_big_endian ? "UCS-4BE" : "UCS-4LE";
|
|
|
|
|
unicode = TRUE;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
unicode = FALSE;
|
|
|
|
|
|
|
|
|
|
if (!dst || *dst == 0)
|
|
|
|
|
dst = "ASCII";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (src == SC_UNICODE)
|
|
|
|
|
{
|
|
|
|
|
src = EXEC_big_endian ? "UCS-4BE" : "UCS-4LE";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (!src || *src == 0)
|
|
|
|
|
src = "ASCII";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
handle = iconv_open(dst, src);
|
|
|
|
|
if (handle == (iconv_t)(-1))
|
|
|
|
|
{
|
|
|
|
|
if (errno == EINVAL)
|
|
|
|
|
errcode = E_UCONV;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
else
|
|
|
|
|
errcode = E_CONV;
|
2008-10-21 09:28:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2007-12-30 16:41:49 +00:00
|
|
|
|
err = FALSE;
|
|
|
|
|
|
|
|
|
|
for(;;)
|
|
|
|
|
{
|
|
|
|
|
out = COMMON_buffer;
|
|
|
|
|
out_len = COMMON_BUF_MAX;
|
|
|
|
|
|
2008-01-05 01:18:28 +00:00
|
|
|
|
#if defined(OS_SOLARIS) || defined(OS_FREEBSD) || defined(OS_OPENBSD)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
ret = iconv(handle, &in, &in_len, &out, &out_len);
|
|
|
|
|
#else
|
|
|
|
|
ret = iconv(handle, (char **)&in, &in_len, &out, &out_len);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (ret != (size_t)(-1) || errno == E2BIG)
|
|
|
|
|
STRING_add(result, COMMON_buffer, COMMON_BUF_MAX - out_len);
|
|
|
|
|
|
|
|
|
|
if (ret != (size_t)(-1))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (errno != E2BIG)
|
|
|
|
|
{
|
|
|
|
|
err = TRUE;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2010-02-07 23:10:41 +00:00
|
|
|
|
|
2007-12-30 16:41:49 +00:00
|
|
|
|
iconv_close(handle);
|
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (unicode)
|
|
|
|
|
STRING_add(result, "\0\0\0", sizeof(wchar_t) - 1);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
STRING_extend_end(result);
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
errcode = E_CONV;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (throw && errcode)
|
|
|
|
|
THROW(errcode);
|
|
|
|
|
|
|
|
|
|
return errcode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
char *STRING_conv_to_UTF8(const char *name, int len)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
char *result = NULL;
|
|
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
|
|
if (LOCAL_is_UTF8)
|
|
|
|
|
{
|
|
|
|
|
if (len <= 0)
|
|
|
|
|
result = (char *)name;
|
|
|
|
|
else
|
2010-06-04 23:48:53 +00:00
|
|
|
|
result = STRING_new_temp(name, len);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (len <= 0)
|
|
|
|
|
len = strlen(name);
|
|
|
|
|
|
|
|
|
|
STRING_conv(&result, name, len, LOCAL_encoding, "UTF-8", TRUE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result)
|
|
|
|
|
return result;
|
|
|
|
|
else
|
|
|
|
|
return "";
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
char *STRING_conv_file_name(const char *name, int len)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
char *result = NULL;
|
|
|
|
|
int pos;
|
|
|
|
|
struct passwd *info;
|
|
|
|
|
char *dir;
|
|
|
|
|
char *user;
|
|
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
|
return "";
|
|
|
|
|
|
|
|
|
|
if (len <= 0)
|
|
|
|
|
len = strlen(name);
|
|
|
|
|
|
|
|
|
|
if (len > 0 && *name == '~')
|
|
|
|
|
{
|
|
|
|
|
for (pos = 0; pos < len; pos++)
|
|
|
|
|
{
|
|
|
|
|
if (name[pos] == '/')
|
|
|
|
|
break;
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
if (pos <= 1)
|
2008-03-19 14:32:30 +00:00
|
|
|
|
dir = PROJECT_get_home();
|
2007-12-30 16:41:49 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2010-06-04 23:48:53 +00:00
|
|
|
|
user = STRING_new_temp(&name[1], pos - 1);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
info = getpwnam(user);
|
2008-03-19 14:32:30 +00:00
|
|
|
|
if (info)
|
|
|
|
|
dir = info->pw_dir;
|
|
|
|
|
else
|
|
|
|
|
dir = NULL;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-03-19 14:32:30 +00:00
|
|
|
|
if (dir)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2010-06-04 23:48:53 +00:00
|
|
|
|
user = STRING_new_zero(dir);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
if (pos < len)
|
|
|
|
|
STRING_add(&user, &name[pos], len - pos);
|
|
|
|
|
name = user;
|
|
|
|
|
len = STRING_length(name);
|
2008-11-10 00:18:00 +00:00
|
|
|
|
STRING_free_later(user);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
2008-10-21 09:28:34 +00:00
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (LOCAL_is_UTF8)
|
2010-06-04 23:48:53 +00:00
|
|
|
|
result = STRING_new_temp(name, len);
|
2008-10-21 09:28:34 +00:00
|
|
|
|
else
|
|
|
|
|
STRING_conv(&result, name, len, "UTF-8", LOCAL_encoding, TRUE);
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
//fprintf(stderr, "STRING_conv_file_name: %s\n", result);
|
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
if (result)
|
|
|
|
|
return result;
|
|
|
|
|
else
|
|
|
|
|
return "";
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2008-01-17 21:39:26 +00:00
|
|
|
|
int STRING_search(const char *ps, int ls, const char *pp, int lp, int is, bool right, bool nocase)
|
2007-12-30 16:41:49 +00:00
|
|
|
|
{
|
2008-10-21 09:28:34 +00:00
|
|
|
|
int pos, ip;
|
|
|
|
|
|
|
|
|
|
if (lp > ls)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
if (is < 0)
|
|
|
|
|
is += ls;
|
|
|
|
|
else if (is == 0)
|
|
|
|
|
is = right ? ls : 1;
|
|
|
|
|
|
|
|
|
|
ls = ls - lp + 1; /* Longueur du début du texte où effectuer la recherche */
|
|
|
|
|
|
|
|
|
|
if (is > ls)
|
|
|
|
|
{
|
|
|
|
|
if (!right)
|
|
|
|
|
return 0;
|
|
|
|
|
else
|
|
|
|
|
is = ls;
|
|
|
|
|
}
|
|
|
|
|
else if (is < lp)
|
|
|
|
|
{
|
|
|
|
|
if (right)
|
|
|
|
|
return 0;
|
|
|
|
|
else if (is < 1)
|
|
|
|
|
is = 1;
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-19 23:50:34 +00:00
|
|
|
|
is--;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-03-19 23:50:34 +00:00
|
|
|
|
if (lp == 1)
|
|
|
|
|
{
|
2010-02-14 01:16:27 +00:00
|
|
|
|
char cp = *pp;
|
2008-03-19 23:50:34 +00:00
|
|
|
|
|
|
|
|
|
if (nocase)
|
|
|
|
|
{
|
|
|
|
|
cp = tolower(cp);
|
|
|
|
|
|
|
|
|
|
if (right)
|
|
|
|
|
{
|
|
|
|
|
for (; is >= 0; is--)
|
|
|
|
|
{
|
|
|
|
|
if (tolower(ps[is]) == cp)
|
|
|
|
|
return is + 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (; is < ls; is++)
|
|
|
|
|
{
|
|
|
|
|
if (tolower(ps[is]) == cp)
|
|
|
|
|
return is + 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (right)
|
|
|
|
|
{
|
|
|
|
|
for (; is >= 0; is--)
|
|
|
|
|
{
|
|
|
|
|
if (ps[is] == cp)
|
|
|
|
|
return is + 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (; is < ls; is++)
|
|
|
|
|
{
|
|
|
|
|
if (ps[is] == cp)
|
|
|
|
|
return is + 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pos = 0;
|
|
|
|
|
ps += is;
|
|
|
|
|
|
|
|
|
|
if (right)
|
|
|
|
|
{
|
|
|
|
|
if (nocase)
|
|
|
|
|
{
|
|
|
|
|
for (; is >= 0; is--, ps--)
|
|
|
|
|
{
|
|
|
|
|
for (ip = 0; ip < lp; ip++)
|
|
|
|
|
{
|
|
|
|
|
if (tolower(ps[ip]) != tolower(pp[ip]))
|
|
|
|
|
goto __NEXT_RN;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pos = is + 1;
|
|
|
|
|
goto __FOUND;
|
|
|
|
|
|
|
|
|
|
__NEXT_RN:
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (; is >= 0; is--, ps--)
|
|
|
|
|
{
|
|
|
|
|
for (ip = 0; ip < lp; ip++)
|
|
|
|
|
{
|
|
|
|
|
if (ps[ip] != pp[ip])
|
|
|
|
|
goto __NEXT_R;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pos = is + 1;
|
|
|
|
|
goto __FOUND;
|
|
|
|
|
|
|
|
|
|
__NEXT_R:
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (nocase)
|
|
|
|
|
{
|
|
|
|
|
for (; is < ls; is++, ps++)
|
|
|
|
|
{
|
|
|
|
|
for (ip = 0; ip < lp; ip++)
|
|
|
|
|
{
|
|
|
|
|
if (tolower(ps[ip]) != tolower(pp[ip]))
|
|
|
|
|
goto __NEXT_LN;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pos = is + 1;
|
|
|
|
|
goto __FOUND;
|
|
|
|
|
|
|
|
|
|
__NEXT_LN:
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (; is < ls; is++, ps++)
|
|
|
|
|
{
|
|
|
|
|
for (ip = 0; ip < lp; ip++)
|
|
|
|
|
{
|
|
|
|
|
if (ps[ip] != pp[ip])
|
|
|
|
|
goto __NEXT_L;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pos = is + 1;
|
|
|
|
|
goto __FOUND;
|
|
|
|
|
|
|
|
|
|
__NEXT_L:
|
|
|
|
|
;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
|
|
|
|
__FOUND:
|
|
|
|
|
|
2008-10-21 09:28:34 +00:00
|
|
|
|
return pos;
|
2007-12-30 16:41:49 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-11-10 00:18:00 +00:00
|
|
|
|
void STRING_start_len(int len)
|
|
|
|
|
{
|
|
|
|
|
_make.inc = 32;
|
|
|
|
|
|
|
|
|
|
if (len == 0)
|
|
|
|
|
len = 32;
|
|
|
|
|
|
2010-06-04 23:48:53 +00:00
|
|
|
|
_make.buffer = STRING_new(NULL, len);
|
2008-11-10 00:18:00 +00:00
|
|
|
|
_make.max = len;
|
|
|
|
|
_make.len = 0;
|
|
|
|
|
_make.ptr = _make.buffer;
|
|
|
|
|
_make.ntemp = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void STRING_make_dump()
|
|
|
|
|
{
|
|
|
|
|
int n = _make.ntemp;
|
|
|
|
|
_make.ntemp = 0;
|
|
|
|
|
STRING_make(_make.temp, n);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// len == 0 est possible ! On peut vouloir ajouter une chaîne vide.
|
|
|
|
|
|
|
|
|
|
void STRING_make(const char *src, int len)
|
|
|
|
|
{
|
|
|
|
|
int pos;
|
|
|
|
|
|
|
|
|
|
if (!src)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (len < 0)
|
|
|
|
|
len = strlen(src);
|
|
|
|
|
|
|
|
|
|
if (len <= 0)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (_make.ntemp)
|
|
|
|
|
STRING_make_dump();
|
|
|
|
|
|
|
|
|
|
_make.len += len;
|
|
|
|
|
|
|
|
|
|
if (_make.len >= _make.max)
|
|
|
|
|
{
|
|
|
|
|
pos = (_make.len - _make.max) * 4;
|
|
|
|
|
if (pos < _make.inc)
|
|
|
|
|
pos = _make.inc;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_make.inc = (pos + 31) & ~31;
|
|
|
|
|
if (_make.inc > 1024)
|
|
|
|
|
_make.inc = 1024;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_make.max += pos;
|
|
|
|
|
|
|
|
|
|
//fprintf(stderr, "STRING_extend: %d\n", _max - STRING_length(SUBST_buffer));
|
|
|
|
|
pos = _make.ptr - _make.buffer;
|
|
|
|
|
STRING_extend(&_make.buffer, _make.max);
|
|
|
|
|
_make.ptr = _make.buffer + pos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memcpy(_make.ptr, src, len);
|
|
|
|
|
_make.ptr += len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *STRING_end()
|
|
|
|
|
{
|
|
|
|
|
if (_make.ntemp)
|
|
|
|
|
STRING_make_dump();
|
|
|
|
|
|
|
|
|
|
if (_make.len)
|
|
|
|
|
{
|
|
|
|
|
STRING_extend(&_make.buffer, _make.len);
|
|
|
|
|
_make.buffer[_make.len] = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
STRING_free(&_make.buffer);
|
|
|
|
|
|
|
|
|
|
return _make.buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
char *STRING_end_temp()
|
|
|
|
|
{
|
|
|
|
|
STRING_end();
|
|
|
|
|
|
|
|
|
|
if (_make.buffer)
|
|
|
|
|
STRING_free_later(_make.buffer);
|
|
|
|
|
|
|
|
|
|
return _make.buffer;
|
|
|
|
|
}
|
2007-12-30 16:41:49 +00:00
|
|
|
|
|
2008-01-27 14:00:04 +00:00
|
|
|
|
#include "gb_common_string_temp.h"
|
2008-03-19 14:32:30 +00:00
|
|
|
|
|
|
|
|
|
|