picrin/src/string.c

236 lines
4.6 KiB
C
Raw Normal View History

2014-01-17 06:58:31 -05:00
/**
* See Copyright Notice in picrin.h
*/
2013-10-20 19:48:55 -04:00
#include <string.h>
#include "picrin.h"
struct pic_string *
pic_str_new(pic_state *pic, const char *cstr, size_t len)
2013-10-20 19:48:55 -04:00
{
struct pic_string *str;
str = (struct pic_string *)pic_obj_alloc(pic, sizeof(struct pic_string), PIC_TT_STRING);
str->len = len;
2014-02-01 05:32:39 -05:00
str->str = pic_strdup(pic, cstr);
return str;
2013-10-20 19:48:55 -04:00
}
struct pic_string *
pic_str_new_cstr(pic_state *pic, const char *cstr)
{
size_t len;
len = strlen(cstr);
return pic_str_new(pic, cstr, len);
}
2013-11-17 03:42:52 -05:00
static pic_value
pic_str_string_p(pic_state *pic)
{
pic_value v;
pic_get_args(pic, "o", &v);
return pic_bool_value(pic_str_p(v));
}
2013-11-17 11:09:54 -05:00
static pic_value
pic_str_make_string(pic_state *pic)
{
int k, i;
char c = ' ', *cstr;
pic_get_args(pic, "i|c", &k, &c);
cstr = (char *)pic_alloc(pic, k + 1);
for (i = 0; i < k; ++i) {
cstr[i] = c;
}
cstr[k] = '\0';
return pic_obj_value(pic_str_new(pic, cstr, k));
}
static pic_value
pic_str_string_length(pic_state *pic)
{
char *str;
size_t len;
pic_get_args(pic, "s", &str, &len);
return pic_int_value(len);
}
static pic_value
pic_str_string_ref(pic_state *pic)
{
char *str;
size_t len;
int k;
pic_get_args(pic, "si", &str, &len, &k);
return pic_char_value(str[k]);
}
static pic_value
pic_str_string_set(pic_state *pic)
{
char *str, c;
size_t len;
int k;
pic_get_args(pic, "sic", &str, &len, &k, &c);
str[k] = c;
2014-01-08 01:22:23 -05:00
return pic_none_value();
2013-11-17 11:09:54 -05:00
}
#define DEFINE_STRING_CMP(name, op) \
static pic_value \
pic_str_string_##name(pic_state *pic) \
{ \
size_t argc; \
pic_value *argv; \
size_t i; \
2013-11-17 11:09:54 -05:00
\
pic_get_args(pic, "*", &argc, &argv); \
\
if (argc < 1 || ! pic_str_p(argv[0])) { \
return pic_false_value(); \
} \
\
for (i = 1; i < argc; ++i) { \
if (! pic_str_p(argv[i])) { \
return pic_false_value(); \
} \
if (! (strcmp(pic_str_ptr(argv[i-1])->str, \
pic_str_ptr(argv[i])->str) \
op 0)) { \
return pic_false_value(); \
} \
} \
return pic_true_value(); \
}
DEFINE_STRING_CMP(eq, ==)
DEFINE_STRING_CMP(lt, <)
DEFINE_STRING_CMP(gt, >)
DEFINE_STRING_CMP(le, <=)
DEFINE_STRING_CMP(ge, >=)
2014-02-08 13:10:58 -05:00
static pic_value
pic_str_string_copy(pic_state *pic)
{
size_t len, start, end, i;
char *str;
int n;
struct pic_string *copy;
n = pic_get_args(pic, "s|ii", &str, &len, &start, &end);
switch (n) {
case 1:
start = 0;
case 2:
end = len;
}
copy = pic_str_new(pic, NULL, end - start);
for (i = 0; i < end - start; ++i) {
copy->str[i] = str[start + i];
}
return pic_obj_value(copy);
}
static pic_value
pic_str_string_copy_ip(pic_state *pic)
{
size_t to_len, from_len, at, start, end;
char *to_str, *from_str;
int n;
n = pic_get_args(pic, "sis|ii", &to_str, &to_len, &at, &from_str, &from_len, &start, &end);
switch (n) {
case 3:
start = 0;
case 4:
end = from_len;
}
while (start < end) {
to_str[at++] = from_str[start++];
}
return pic_none_value();
}
2014-02-08 12:42:50 -05:00
static pic_value
pic_str_string_append(pic_state *pic)
{
size_t argc, len, i;
pic_value *argv;
char *buf;
pic_get_args(pic, "*", &argc, &argv);
len = 0;
buf = NULL;
for (i = 0; i < argc; ++i) {
if (! pic_str_p(argv[i])) {
pic_error(pic, "type error");
}
buf = pic_realloc(pic, buf, len + pic_str_ptr(argv[i])->len);
/* copy! */
memcpy(buf + len, pic_str_ptr(argv[i])->str, pic_str_ptr(argv[i])->len);
len += pic_str_ptr(argv[i])->len;
}
return pic_obj_value(pic_str_new(pic, buf, len));
}
2014-02-08 13:10:58 -05:00
static pic_value
pic_str_string_fill_ip(pic_state *pic)
{
size_t len, start, end;
char *str, c;
int n;
n = pic_get_args(pic, "sc|ii", &str, &len, &c, &start, &end);
switch (n) {
case 1:
start = 0;
case 2:
end = len;
}
while (start < end) {
str[start++] = c;
}
return pic_none_value();
}
2013-11-17 03:42:52 -05:00
void
pic_init_str(pic_state *pic)
{
pic_defun(pic, "string?", pic_str_string_p);
2013-11-17 11:09:54 -05:00
pic_defun(pic, "make-string", pic_str_make_string);
pic_defun(pic, "string-length", pic_str_string_length);
pic_defun(pic, "string-ref", pic_str_string_ref);
pic_defun(pic, "string-set!", pic_str_string_set);
2014-02-08 13:10:58 -05:00
2013-11-17 11:09:54 -05:00
pic_defun(pic, "string=?", pic_str_string_eq);
pic_defun(pic, "string<?", pic_str_string_lt);
pic_defun(pic, "string>?", pic_str_string_gt);
pic_defun(pic, "string<=?", pic_str_string_le);
pic_defun(pic, "string>=?", pic_str_string_ge);
2014-02-08 13:10:58 -05:00
pic_defun(pic, "string-copy", pic_str_string_copy);
pic_defun(pic, "string-copy!", pic_str_string_copy_ip);
2014-02-08 12:42:50 -05:00
pic_defun(pic, "string-append", pic_str_string_append);
2014-02-08 13:10:58 -05:00
pic_defun(pic, "string-fill!", pic_str_string_fill_ip);
2013-11-17 03:42:52 -05:00
}