cleanup string.c
This commit is contained in:
parent
86e4eac543
commit
1a19e8f582
27
lib/gc.c
27
lib/gc.c
|
@ -199,8 +199,19 @@ gc_mark_object(pic_state *pic, struct object *obj)
|
||||||
pic->gc_attrs = attr;
|
pic->gc_attrs = attr;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case PIC_TYPE_STRING: {
|
||||||
|
struct string *str = (struct string *) obj;
|
||||||
|
LOOP(str->rope);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case PIC_TYPE_ROPE_NODE: {
|
||||||
|
struct rope_node *node = (struct rope_node *) obj;
|
||||||
|
gc_mark_object(pic, (struct object *) node->s1);
|
||||||
|
LOOP(node->s2);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
case PIC_TYPE_STRING:
|
case PIC_TYPE_ROPE_LEAF:
|
||||||
case PIC_TYPE_BLOB:
|
case PIC_TYPE_BLOB:
|
||||||
case PIC_TYPE_DATA:
|
case PIC_TYPE_DATA:
|
||||||
break;
|
break;
|
||||||
|
@ -224,11 +235,6 @@ gc_finalize_object(pic_state *pic, struct object *obj)
|
||||||
pic_free(pic, blob->data);
|
pic_free(pic, blob->data);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case PIC_TYPE_STRING: {
|
|
||||||
struct string *str = (struct string *) obj;
|
|
||||||
pic_rope_decref(pic, str->rope);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case PIC_TYPE_DATA: {
|
case PIC_TYPE_DATA: {
|
||||||
struct data *data = (struct data *) obj;
|
struct data *data = (struct data *) obj;
|
||||||
if (data->type->dtor) {
|
if (data->type->dtor) {
|
||||||
|
@ -264,7 +270,14 @@ gc_finalize_object(pic_state *pic, struct object *obj)
|
||||||
pic_free(pic, frame->regs);
|
pic_free(pic, frame->regs);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case PIC_TYPE_ROPE_LEAF: {
|
||||||
|
struct rope_leaf *leaf = (struct rope_leaf *) obj;
|
||||||
|
pic_free(pic, (char *) leaf->str);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case PIC_TYPE_STRING:
|
||||||
|
case PIC_TYPE_ROPE_NODE:
|
||||||
case PIC_TYPE_PAIR:
|
case PIC_TYPE_PAIR:
|
||||||
case PIC_TYPE_RECORD:
|
case PIC_TYPE_RECORD:
|
||||||
case PIC_TYPE_PROC_FUNC:
|
case PIC_TYPE_PROC_FUNC:
|
||||||
|
@ -394,6 +407,8 @@ type2size(int type)
|
||||||
case PIC_TYPE_RECORD: return sizeof(struct record);
|
case PIC_TYPE_RECORD: return sizeof(struct record);
|
||||||
case PIC_TYPE_PROC_FUNC: return sizeof(struct proc);
|
case PIC_TYPE_PROC_FUNC: return sizeof(struct proc);
|
||||||
case PIC_TYPE_PROC_IREP: return sizeof(struct proc);
|
case PIC_TYPE_PROC_IREP: return sizeof(struct proc);
|
||||||
|
case PIC_TYPE_ROPE_LEAF: return sizeof(struct rope_leaf);
|
||||||
|
case PIC_TYPE_ROPE_NODE: return sizeof(struct rope_node);
|
||||||
default: PIC_UNREACHABLE();
|
default: PIC_UNREACHABLE();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -135,7 +135,7 @@ pic_value pic_deserialize(pic_state *pic, pic_value blob);
|
||||||
bool pic_str_p(pic_state *, pic_value);
|
bool pic_str_p(pic_state *, pic_value);
|
||||||
pic_value pic_str_value(pic_state *, const char *str, int len);
|
pic_value pic_str_value(pic_state *, const char *str, int len);
|
||||||
pic_value pic_cstr_value(pic_state *, const char *str);
|
pic_value pic_cstr_value(pic_state *, const char *str);
|
||||||
#define pic_lit_value(pic, lit) pic_str_value(pic, "" lit, -((int)sizeof lit - 1))
|
#define pic_lit_value(pic, lit) pic_str_value(pic, "" lit, sizeof lit - 1)
|
||||||
pic_value pic_strf_value(pic_state *, const char *fmt, ...);
|
pic_value pic_strf_value(pic_state *, const char *fmt, ...);
|
||||||
pic_value pic_vstrf_value(pic_state *, const char *fmt, va_list ap);
|
pic_value pic_vstrf_value(pic_state *, const char *fmt, va_list ap);
|
||||||
const char *pic_str(pic_state *, pic_value str, int *len);
|
const char *pic_str(pic_state *, pic_value str, int *len);
|
||||||
|
|
23
lib/object.h
23
lib/object.h
|
@ -28,6 +28,25 @@ struct blob {
|
||||||
int len;
|
int len;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#define ROPE_HEADER \
|
||||||
|
OBJECT_HEADER \
|
||||||
|
int len;
|
||||||
|
|
||||||
|
struct rope {
|
||||||
|
ROPE_HEADER
|
||||||
|
};
|
||||||
|
|
||||||
|
struct rope_leaf {
|
||||||
|
ROPE_HEADER
|
||||||
|
const char *str;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct rope_node {
|
||||||
|
ROPE_HEADER
|
||||||
|
struct rope *s1;
|
||||||
|
struct rope *s2;
|
||||||
|
};
|
||||||
|
|
||||||
struct string {
|
struct string {
|
||||||
OBJECT_HEADER
|
OBJECT_HEADER
|
||||||
struct rope *rope;
|
struct rope *rope;
|
||||||
|
@ -198,10 +217,6 @@ pic_value pic_make_cont(pic_state *pic, pic_value k);
|
||||||
int pic_str_hash(pic_state *pic, pic_value str);
|
int pic_str_hash(pic_state *pic, pic_value str);
|
||||||
int pic_str_cmp(pic_state *pic, pic_value str1, pic_value str2);
|
int pic_str_cmp(pic_state *pic, pic_value str1, pic_value str2);
|
||||||
|
|
||||||
struct rope *pic_rope_incref(struct rope *);
|
|
||||||
void pic_rope_decref(pic_state *, struct rope *);
|
|
||||||
|
|
||||||
|
|
||||||
void pic_warnf(pic_state *pic, const char *fmt, ...); /* deprecated */
|
void pic_warnf(pic_state *pic, const char *fmt, ...); /* deprecated */
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
#if defined(__cplusplus)
|
||||||
|
|
313
lib/string.c
313
lib/string.c
|
@ -6,212 +6,26 @@
|
||||||
#include "value.h"
|
#include "value.h"
|
||||||
#include "object.h"
|
#include "object.h"
|
||||||
|
|
||||||
struct rope {
|
|
||||||
int refcnt;
|
|
||||||
int weight;
|
|
||||||
bool isleaf;
|
|
||||||
union {
|
|
||||||
struct {
|
|
||||||
struct rope *owner;
|
|
||||||
const char *str; /* always points to zero-term'd buf */
|
|
||||||
} leaf;
|
|
||||||
struct {
|
|
||||||
struct rope *left, *right;
|
|
||||||
} node;
|
|
||||||
} u;
|
|
||||||
char buf[1];
|
|
||||||
};
|
|
||||||
|
|
||||||
struct rope *
|
|
||||||
pic_rope_incref(struct rope *rope) {
|
|
||||||
rope->refcnt++;
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
|
||||||
pic_rope_decref(pic_state *pic, struct rope *rope) {
|
|
||||||
if (! --rope->refcnt) {
|
|
||||||
if (rope->isleaf) {
|
|
||||||
if (rope->u.leaf.owner) {
|
|
||||||
pic_rope_decref(pic, rope->u.leaf.owner);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
pic_rope_decref(pic, rope->u.node.left);
|
|
||||||
pic_rope_decref(pic, rope->u.node.right);
|
|
||||||
}
|
|
||||||
pic_free(pic, rope);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
make_rope_leaf(pic_state *pic, const char *str, int len)
|
|
||||||
{
|
|
||||||
struct rope *rope;
|
|
||||||
|
|
||||||
rope = pic_malloc(pic, offsetof(struct rope, buf) + len + 1);
|
|
||||||
rope->refcnt = 1;
|
|
||||||
rope->weight = len;
|
|
||||||
rope->isleaf = true;
|
|
||||||
rope->u.leaf.owner = NULL;
|
|
||||||
rope->u.leaf.str = rope->buf;
|
|
||||||
rope->buf[len] = 0;
|
|
||||||
if (str) {
|
|
||||||
memcpy(rope->buf, str, len);
|
|
||||||
}
|
|
||||||
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
make_rope_lit(pic_state *pic, const char *str, int len)
|
|
||||||
{
|
|
||||||
struct rope *rope;
|
|
||||||
|
|
||||||
rope = pic_malloc(pic, offsetof(struct rope, buf));
|
|
||||||
rope->refcnt = 1;
|
|
||||||
rope->weight = len;
|
|
||||||
rope->isleaf = true;
|
|
||||||
rope->u.leaf.owner = NULL;
|
|
||||||
rope->u.leaf.str = str;
|
|
||||||
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
make_rope_slice(pic_state *pic, struct rope *owner, int i, int j)
|
|
||||||
{
|
|
||||||
struct rope *rope, *real_owner;
|
|
||||||
|
|
||||||
assert(owner->isleaf);
|
|
||||||
|
|
||||||
real_owner = owner->u.leaf.owner == NULL ? owner : owner->u.leaf.owner;
|
|
||||||
|
|
||||||
rope = pic_malloc(pic, offsetof(struct rope, buf));
|
|
||||||
rope->refcnt = 1;
|
|
||||||
rope->weight = j - i;
|
|
||||||
rope->isleaf = true;
|
|
||||||
rope->u.leaf.owner = real_owner;
|
|
||||||
rope->u.leaf.str = owner->u.leaf.str + i;
|
|
||||||
|
|
||||||
pic_rope_incref(real_owner);
|
|
||||||
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
make_rope_node(pic_state *pic, struct rope *left, struct rope *right)
|
|
||||||
{
|
|
||||||
struct rope *rope;
|
|
||||||
|
|
||||||
rope = pic_malloc(pic, sizeof(struct rope));
|
|
||||||
rope->refcnt = 1;
|
|
||||||
rope->weight = left->weight + right->weight;
|
|
||||||
rope->isleaf = false;
|
|
||||||
rope->u.node.left = pic_rope_incref(left);
|
|
||||||
rope->u.node.right = pic_rope_incref(right);
|
|
||||||
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
static pic_value
|
|
||||||
make_str(pic_state *pic, struct rope *rope)
|
|
||||||
{
|
|
||||||
struct string *str;
|
|
||||||
|
|
||||||
str = (struct string *)pic_obj_alloc(pic, PIC_TYPE_STRING);
|
|
||||||
str->rope = rope; /* delegate ownership */
|
|
||||||
|
|
||||||
return obj_value(pic, str);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
merge(pic_state *pic, struct rope *left, struct rope *right)
|
|
||||||
{
|
|
||||||
if (left == 0)
|
|
||||||
return pic_rope_incref(right);
|
|
||||||
if (right == 0)
|
|
||||||
return pic_rope_incref(left);
|
|
||||||
|
|
||||||
return make_rope_node(pic, left, right);
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct rope *
|
|
||||||
slice(pic_state *pic, struct rope *rope, int i, int j)
|
|
||||||
{
|
|
||||||
int lweight;
|
|
||||||
|
|
||||||
if (i == 0 && rope->weight == j) {
|
|
||||||
return pic_rope_incref(rope);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rope->isleaf) {
|
|
||||||
return make_rope_slice(pic, rope, i, j);
|
|
||||||
}
|
|
||||||
|
|
||||||
lweight = rope->u.node.left->weight;
|
|
||||||
|
|
||||||
if (j <= lweight) {
|
|
||||||
return slice(pic, rope->u.node.left, i, j);
|
|
||||||
} else if (lweight <= i) {
|
|
||||||
return slice(pic, rope->u.node.right, i - lweight, j - lweight);
|
|
||||||
} else {
|
|
||||||
struct rope *r, *l;
|
|
||||||
|
|
||||||
l = slice(pic, rope->u.node.left, i, lweight);
|
|
||||||
r = slice(pic, rope->u.node.right, 0, j - lweight);
|
|
||||||
rope = merge(pic, l, r);
|
|
||||||
|
|
||||||
pic_rope_decref(pic, l);
|
|
||||||
pic_rope_decref(pic, r);
|
|
||||||
|
|
||||||
return rope;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
flatten(pic_state *pic, struct rope *rope, struct rope *owner, char *buf)
|
|
||||||
{
|
|
||||||
if (rope->isleaf) {
|
|
||||||
memcpy(buf, rope->u.leaf.str, rope->weight);
|
|
||||||
} else {
|
|
||||||
flatten(pic, rope->u.node.left, owner, buf);
|
|
||||||
flatten(pic, rope->u.node.right, owner, buf + rope->u.node.left->weight);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* path compression */
|
|
||||||
|
|
||||||
if (! rope->isleaf) {
|
|
||||||
pic_rope_incref(owner);
|
|
||||||
pic_rope_decref(pic, rope->u.node.left);
|
|
||||||
pic_rope_decref(pic, rope->u.node.right);
|
|
||||||
rope->isleaf = true;
|
|
||||||
rope->u.leaf.owner = owner;
|
|
||||||
rope->u.leaf.str = buf;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void
|
|
||||||
str_update(pic_state *pic, pic_value dst, pic_value src)
|
|
||||||
{
|
|
||||||
pic_rope_incref(str_ptr(pic, src)->rope);
|
|
||||||
pic_rope_decref(pic, str_ptr(pic, dst)->rope);
|
|
||||||
str_ptr(pic, dst)->rope = str_ptr(pic, src)->rope;
|
|
||||||
}
|
|
||||||
|
|
||||||
pic_value
|
pic_value
|
||||||
pic_str_value(pic_state *pic, const char *str, int len)
|
pic_str_value(pic_state *pic, const char *str, int len)
|
||||||
{
|
{
|
||||||
struct rope *r;
|
char *buf;
|
||||||
|
struct rope_leaf *leaf;
|
||||||
|
struct string *s;
|
||||||
|
|
||||||
if (len > 0) {
|
assert(str != NULL);
|
||||||
r = make_rope_leaf(pic, str, len);
|
|
||||||
} else {
|
buf = pic_malloc(pic, len + 1);
|
||||||
if (len == 0) {
|
buf[len] = 0;
|
||||||
str = "";
|
memcpy(buf, str, len);
|
||||||
}
|
|
||||||
r = make_rope_lit(pic, str, -len);
|
leaf = (struct rope_leaf *) pic_obj_alloc(pic, PIC_TYPE_ROPE_LEAF);
|
||||||
}
|
leaf->len = len;
|
||||||
return make_str(pic, r);
|
leaf->str = buf;
|
||||||
|
|
||||||
|
s = (struct string *) pic_obj_alloc(pic, PIC_TYPE_STRING);
|
||||||
|
s->rope = (struct rope *) leaf;
|
||||||
|
return obj_value(pic, s);
|
||||||
}
|
}
|
||||||
|
|
||||||
pic_value
|
pic_value
|
||||||
|
@ -295,19 +109,58 @@ pic_vstrf_value(pic_state *pic, const char *fmt, va_list ap)
|
||||||
int
|
int
|
||||||
pic_str_len(pic_state *pic, pic_value str)
|
pic_str_len(pic_state *pic, pic_value str)
|
||||||
{
|
{
|
||||||
return str_ptr(pic, str)->rope->weight;
|
return str_ptr(pic, str)->rope->len;
|
||||||
}
|
}
|
||||||
|
|
||||||
pic_value
|
pic_value
|
||||||
pic_str_cat(pic_state *pic, pic_value a, pic_value b)
|
pic_str_cat(pic_state *pic, pic_value a, pic_value b)
|
||||||
{
|
{
|
||||||
return make_str(pic, merge(pic, str_ptr(pic, a)->rope, str_ptr(pic, b)->rope));
|
struct rope *s1 = str_ptr(pic, a)->rope, *s2 = str_ptr(pic, b)->rope;
|
||||||
|
struct rope_node *node;
|
||||||
|
struct string *s;
|
||||||
|
|
||||||
|
node = (struct rope_node *) pic_obj_alloc(pic, PIC_TYPE_ROPE_NODE);
|
||||||
|
node->len = s1->len + s2->len;
|
||||||
|
node->s1 = s1;
|
||||||
|
node->s2 = s2;
|
||||||
|
|
||||||
|
s = (struct string *) pic_obj_alloc(pic, PIC_TYPE_STRING);
|
||||||
|
s->rope = (struct rope *) node;
|
||||||
|
return obj_value(pic, s);
|
||||||
|
}
|
||||||
|
|
||||||
|
static pic_value
|
||||||
|
str_sub(pic_state *pic, struct rope *rope, int i, int j)
|
||||||
|
{
|
||||||
|
int lweight;
|
||||||
|
pic_value s1, s2;
|
||||||
|
|
||||||
|
if (i == 0 && rope->len == j) {
|
||||||
|
return obj_value(pic, rope);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (obj_type(rope) == PIC_TYPE_ROPE_LEAF) {
|
||||||
|
return pic_str_value(pic, ((struct rope_leaf *) rope)->str + i, j - i);
|
||||||
|
}
|
||||||
|
|
||||||
|
lweight = ((struct rope_node *) rope)->s1->len;
|
||||||
|
|
||||||
|
if (j <= lweight) {
|
||||||
|
return str_sub(pic, ((struct rope_node *) rope)->s1, i, j);
|
||||||
|
}
|
||||||
|
if (lweight <= i) {
|
||||||
|
return str_sub(pic, ((struct rope_node *) rope)->s2, i - lweight, j - lweight);
|
||||||
|
}
|
||||||
|
|
||||||
|
s1 = str_sub(pic, ((struct rope_node *) rope)->s1, i, lweight);
|
||||||
|
s2 = str_sub(pic, ((struct rope_node *) rope)->s2, 0, j - lweight);
|
||||||
|
return pic_str_cat(pic, s1, s2);
|
||||||
}
|
}
|
||||||
|
|
||||||
pic_value
|
pic_value
|
||||||
pic_str_sub(pic_state *pic, pic_value str, int s, int e)
|
pic_str_sub(pic_state *pic, pic_value str, int s, int e)
|
||||||
{
|
{
|
||||||
return make_str(pic, slice(pic, str_ptr(pic, str)->rope, s, e));
|
return str_sub(pic, str_ptr(pic, str)->rope, s, e);
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
|
@ -342,24 +195,45 @@ pic_str_cmp(pic_state *pic, pic_value str1, pic_value str2)
|
||||||
return len1 - len2;
|
return len1 - len2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
str_cstr(pic_state *pic, struct rope *rope, char *buf)
|
||||||
|
{
|
||||||
|
if (obj_type(rope) == PIC_TYPE_ROPE_LEAF) {
|
||||||
|
memcpy(buf, ((struct rope_leaf *) rope)->str, rope->len);
|
||||||
|
} else {
|
||||||
|
struct rope_node *r = (struct rope_node *) rope;
|
||||||
|
str_cstr(pic, r->s1, buf);
|
||||||
|
str_cstr(pic, r->s2, buf + r->s1->len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const char *
|
const char *
|
||||||
pic_str(pic_state *pic, pic_value str, int *len)
|
pic_str(pic_state *pic, pic_value str, int *len)
|
||||||
{
|
{
|
||||||
struct rope *rope = str_ptr(pic, str)->rope, *r;
|
struct rope *rope = str_ptr(pic, str)->rope;
|
||||||
|
char *buf;
|
||||||
|
struct rope_leaf *leaf;
|
||||||
|
|
||||||
if (len) {
|
if (len) {
|
||||||
*len = rope->weight;
|
*len = rope->len;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rope->isleaf && rope->u.leaf.str[rope->weight] == '\0') {
|
if (obj_type(rope) == PIC_TYPE_ROPE_LEAF) {
|
||||||
return rope->u.leaf.str;
|
return ((struct rope_leaf *) rope)->str;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = make_rope_leaf(pic, 0, rope->weight);
|
buf = pic_malloc(pic, rope->len + 1);
|
||||||
|
buf[rope->len] = 0;
|
||||||
|
str_cstr(pic, rope, buf);
|
||||||
|
|
||||||
flatten(pic, rope, r, r->buf);
|
leaf = (struct rope_leaf *) pic_obj_alloc(pic, PIC_TYPE_ROPE_LEAF);
|
||||||
|
leaf->len = rope->len;
|
||||||
|
leaf->str = buf;
|
||||||
|
|
||||||
return r->u.leaf.str;
|
/* cache the result */
|
||||||
|
str_ptr(pic, str)->rope = (struct rope *) leaf;
|
||||||
|
|
||||||
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
const char *
|
const char *
|
||||||
|
@ -453,7 +327,7 @@ pic_str_string_ref(pic_state *pic)
|
||||||
static pic_value
|
static pic_value
|
||||||
pic_str_string_set(pic_state *pic)
|
pic_str_string_set(pic_state *pic)
|
||||||
{
|
{
|
||||||
pic_value str, x, y, z;
|
pic_value str, x, y, z, w;
|
||||||
char c;
|
char c;
|
||||||
int k, len;
|
int k, len;
|
||||||
|
|
||||||
|
@ -466,8 +340,9 @@ pic_str_string_set(pic_state *pic)
|
||||||
x = pic_str_sub(pic, str, 0, k);
|
x = pic_str_sub(pic, str, 0, k);
|
||||||
y = pic_str_value(pic, &c, 1);
|
y = pic_str_value(pic, &c, 1);
|
||||||
z = pic_str_sub(pic, str, k + 1, len);
|
z = pic_str_sub(pic, str, k + 1, len);
|
||||||
|
w = pic_str_cat(pic, x, pic_str_cat(pic, y, z));
|
||||||
|
|
||||||
str_update(pic, str, pic_str_cat(pic, x, pic_str_cat(pic, y, z)));
|
str_ptr(pic, str)->rope = str_ptr(pic, w)->rope;
|
||||||
|
|
||||||
return pic_undef_value(pic);
|
return pic_undef_value(pic);
|
||||||
}
|
}
|
||||||
|
@ -527,7 +402,7 @@ pic_str_string_copy(pic_state *pic)
|
||||||
static pic_value
|
static pic_value
|
||||||
pic_str_string_copy_ip(pic_state *pic)
|
pic_str_string_copy_ip(pic_state *pic)
|
||||||
{
|
{
|
||||||
pic_value to, from, x, y, z;
|
pic_value to, from, x, y, z, w;
|
||||||
int n, at, start, end, tolen, fromlen;
|
int n, at, start, end, tolen, fromlen;
|
||||||
|
|
||||||
n = pic_get_args(pic, "sis|ii", &to, &at, &from, &start, &end);
|
n = pic_get_args(pic, "sis|ii", &to, &at, &from, &start, &end);
|
||||||
|
@ -547,8 +422,9 @@ pic_str_string_copy_ip(pic_state *pic)
|
||||||
x = pic_str_sub(pic, to, 0, at);
|
x = pic_str_sub(pic, to, 0, at);
|
||||||
y = pic_str_sub(pic, from, start, end);
|
y = pic_str_sub(pic, from, start, end);
|
||||||
z = pic_str_sub(pic, to, at + end - start, tolen);
|
z = pic_str_sub(pic, to, at + end - start, tolen);
|
||||||
|
w = pic_str_cat(pic, x, pic_str_cat(pic, y, z));
|
||||||
|
|
||||||
str_update(pic, to, pic_str_cat(pic, x, pic_str_cat(pic, y, z)));
|
str_ptr(pic, to)->rope = str_ptr(pic, w)->rope;
|
||||||
|
|
||||||
return pic_undef_value(pic);
|
return pic_undef_value(pic);
|
||||||
}
|
}
|
||||||
|
@ -556,7 +432,7 @@ pic_str_string_copy_ip(pic_state *pic)
|
||||||
static pic_value
|
static pic_value
|
||||||
pic_str_string_fill_ip(pic_state *pic)
|
pic_str_string_fill_ip(pic_state *pic)
|
||||||
{
|
{
|
||||||
pic_value str, x, y, z;
|
pic_value str, x, y, z, w;
|
||||||
char c, *buf;
|
char c, *buf;
|
||||||
int n, start, end, len;
|
int n, start, end, len;
|
||||||
|
|
||||||
|
@ -579,8 +455,9 @@ pic_str_string_fill_ip(pic_state *pic)
|
||||||
x = pic_str_sub(pic, str, 0, start);
|
x = pic_str_sub(pic, str, 0, start);
|
||||||
y = pic_str_value(pic, buf, end - start);
|
y = pic_str_value(pic, buf, end - start);
|
||||||
z = pic_str_sub(pic, str, end, len);
|
z = pic_str_sub(pic, str, end, len);
|
||||||
|
w = pic_str_cat(pic, x, pic_str_cat(pic, y, z));
|
||||||
|
|
||||||
str_update(pic, str, pic_str_cat(pic, x, pic_str_cat(pic, y, z)));
|
str_ptr(pic, str)->rope = str_ptr(pic, w)->rope;
|
||||||
|
|
||||||
return pic_undef_value(pic);
|
return pic_undef_value(pic);
|
||||||
}
|
}
|
||||||
|
|
10
lib/value.h
10
lib/value.h
|
@ -30,10 +30,12 @@ enum {
|
||||||
PIC_TYPE_DICT = 22,
|
PIC_TYPE_DICT = 22,
|
||||||
PIC_TYPE_RECORD = 23,
|
PIC_TYPE_RECORD = 23,
|
||||||
PIC_TYPE_ATTR = 24,
|
PIC_TYPE_ATTR = 24,
|
||||||
PIC_TYPE_IREP = 27,
|
PIC_TYPE_IREP = 25,
|
||||||
PIC_TYPE_FRAME = 28,
|
PIC_TYPE_FRAME = 26,
|
||||||
PIC_TYPE_PROC_FUNC = 29,
|
PIC_TYPE_PROC_FUNC = 27,
|
||||||
PIC_TYPE_PROC_IREP = 30,
|
PIC_TYPE_PROC_IREP = 28,
|
||||||
|
PIC_TYPE_ROPE_LEAF = 29,
|
||||||
|
PIC_TYPE_ROPE_NODE = 30,
|
||||||
PIC_TYPE_MAX = 63
|
PIC_TYPE_MAX = 63
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue