gambas 4e50dec11d Work continues on new JIT system. Translation is now done at runtime.
[INTERPRETER]
* OPT: String routines are now compiled with -O3.
* NEW: Don't display JIT debugging message unless GB_JIT_DEBUG is set to something different from zero.
* NEW: String whose length is greater than 256 now have a growth step of 256 bytes instead of 16.

[GB.JIT]
* NEW: Do many global optimizations as now the class metadata is fully available.
* NEW: Support for optional argument. Still buggy at the moment.
2018-06-09 22:42:35 +02:00

228 lines
3.5 KiB
C

/***************************************************************************
gb_str.c
(c) 2000-2017 Benoît Minisini <g4mba5@gmail.com>
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.
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.
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., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
***************************************************************************/
#define __GB_STR_C
#include "gb_common.h"
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <fcntl.h>
#include <stdarg.h>
#include <ctype.h>
#include "main.h"
#include "gb_str.h"
//#define DEBUG
static char *_free_later = NULL;
static char *_last_str = NULL;
static int _last_len = 0;
#ifdef DEBUG
static int _count = 0;
#endif
void STR_free(char *str)
{
if (!str)
return;
#ifdef DEBUG
_count--;
fprintf(stderr, "free %p -> %d\n", str, _count);
#endif
GB.Free((void **)&(str));
}
void STR_vadd(char **str, const char *fmt, va_list args)
{
va_list copy;
int len, add;
char *new;
va_copy(copy, args);
add = vsnprintf(NULL, 0, fmt, args);
if (*str)
len = (*str == _last_str ? _last_len : strlen(*str));
else
len = 0;
GB.Alloc((void **)&new, len + add + 1);
if (*str) strcpy(new, *str);
vsprintf(&new[len], fmt, copy);
va_end(copy);
#ifdef DEBUG
if (*str) {
_count--;
fprintf(stderr, "free %p -> %d\n", *str, _count);
}
_count++;
fprintf(stderr, "alloc %p -> %d\n", new, _count);
#endif
if (*str) GB.Free((void **)str);
*str = new;
_last_str = new;
_last_len = len + add;
}
void STR_add(char **str, const char *fmt, ...)
{
va_list args;
va_start(args, fmt);
STR_vadd(str, fmt, args);
va_end(args);
}
char *STR_copy_len(const char *str, int len)
{
char *cpy;
GB.Alloc((void **)&cpy, len + 1);
memcpy(cpy, str, len + 1);
#ifdef DEBUG
_count++;
fprintf(stderr, "alloc %p -> %d\n", cpy, _count);
#endif
return cpy;
}
char *STR_copy(const char *str)
{
return STR_copy_len(str, strlen(str));
}
/*static char *str_add(char *d, const char *s)
{
for(;;)
{
if ((*d = *s) == 0)
break;
d++;
s++;
}
return d;
}
char *STR_cat(const char *str, ...)
{
va_list args;
char *cpy;
char *p;
int len = 0;
va_start(args, str);
p = (char *)str;
while (p)
{
len += strlen(p);
p = va_arg(args, char *);
}
va_end(args);
GB.Alloc((void **)&cpy, len + 1);
p = cpy;
va_start(args, str);
while (str)
{
p = str_add(p, str);
str = va_arg(args, char *);
}
va_end(args);
return cpy;
}*/
char *STR_upper(const char *str)
{
char *s;
char *p;
p = s = STR_copy(str);
while (*p)
{
*p = toupper(*p);
p++;
}
return s;
}
char *STR_lower(const char *str)
{
char *s;
char *p;
p = s = STR_copy(str);
while (*p)
{
*p = tolower(*p);
p++;
}
return s;
}
char *STR_free_later(char *str)
{
if (_free_later)
STR_free(_free_later);
_free_later = str;
return str;
}
char *STR_print(const char *fmt, ...)
{
va_list args;
char *str = NULL;
va_start(args, fmt);
STR_vadd(&str, fmt, args);
va_end(args);
return str;
}