aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--include/builtins.h2
-rw-r--r--include/stem.h4
-rw-r--r--src/builtins.c74
-rw-r--r--src/stem.c6
4 files changed, 79 insertions, 7 deletions
diff --git a/include/builtins.h b/include/builtins.h
index 38b9965..3b87aff 100644
--- a/include/builtins.h
+++ b/include/builtins.h
@@ -18,6 +18,8 @@ void stemdiv(value_t *v);
* [quote] */
void stemfunc(value_t *v);
+void steminsert(value_t *v);
+
/*! @brief takes first number to the power of the second, pushes result on the
* stack */
void stempow(value_t *v);
diff --git a/include/stem.h b/include/stem.h
index f252694..f4aff54 100644
--- a/include/stem.h
+++ b/include/stem.h
@@ -111,8 +111,8 @@ array_t *init_array(size_t size);
/*! appends element to back of array */
void array_append(array_t *a, value_t *v);
-/*! appends element to front of array */
-void array_curry(array_t *a, value_t *v);
+/*! add element to array at index */
+void array_add(array_t *a, value_t *v, int index);
/*! pops last element off of array */
value_t *array_pop(array_t *a);
diff --git a/src/builtins.c b/src/builtins.c
index 4714c23..f3497b3 100644
--- a/src/builtins.c
+++ b/src/builtins.c
@@ -332,7 +332,6 @@ void stemfloor(value_t *v) {
void stemeval(value_t *v) {
value_t *v1 = array_pop(STACK);
if (v1 == NULL) {
- array_append(STACK, v1);
eval_error("EMPTY STACK");
return;
}
@@ -350,6 +349,28 @@ void stemeval(value_t *v) {
}
}
+void unglue(value_t *v) {
+ value_t *v1 = array_pop(STACK);
+ if (v1 == NULL) {
+ eval_error("EMPTY STACK");
+ return;
+ }
+ if (v1->type != VWORD) {
+ array_append(STACK, v1);
+ eval_error("EMPTY STACK");
+ return;
+ }
+ value_t *value = ht_get(WORD_TABLE, v1->str_word);
+ if (!value) {
+ array_append(STACK, v1);
+ eval_error("UNBOUND WORD");
+ return;
+ }
+ value = value_copy(value);
+ array_append(STACK, value);
+ value_free(v1);
+}
+
void strquote(value_t *v) {
value_t *v1 = array_pop(STACK);
if (v1 == NULL) {
@@ -399,7 +420,54 @@ void curry(value_t *v) {
return;
}
- array_curry(v2->quote, v1);
+ array_add(v2->quote, v1, 0);
+ array_append(STACK, v2);
+}
+
+void steminsert(value_t *v) {
+ value_t *v3 = array_pop(STACK);
+ if (v3 == NULL) {
+ eval_error("EMPTY STACK");
+ return;
+ }
+ value_t *v2 = array_pop(STACK);
+ if (v2 == NULL) {
+ array_append(STACK, v3);
+ eval_error("EMPTY STACK");
+ return;
+ }
+ value_t *v1 = array_pop(STACK);
+ if (v1 == NULL) {
+ array_append(STACK, v2);
+ array_append(STACK, v3);
+ eval_error("EMTPY STACK");
+ return;
+ }
+
+ if (v1->type != VINT) {
+ array_append(STACK, v1);
+ array_append(STACK, v2);
+ array_append(STACK, v3);
+ eval_error("INCORRECT TYPE ARGUMENT");
+ return;
+ }
+
+ if (v3->type != VQUOTE) {
+ array_append(STACK, v1);
+ array_append(STACK, v2);
+ array_append(STACK, v3);
+ eval_error("INCORRECT TYPE ARGUMENT");
+ return;
+ }
+ if (v1->int_float < 0 || v1->int_float >= v3->quote->size) {
+ array_append(STACK, v1);
+ array_append(STACK, v2);
+ array_append(STACK, v3);
+ eval_error("INDEX ERROR");
+ return;
+ }
+
+ array_add(v3->quote, v2, (int)v1->int_float);
array_append(STACK, v2);
}
@@ -1253,4 +1321,6 @@ void add_funcs() {
add_func(FLIT, undef, "undef");
add_func(FLIT, tostr, "tostr");
add_func(FLIT, stemcut, "cut");
+ add_func(FLIT, steminsert, "insert");
+ add_func(FLIT, unglue, "unglue");
}
diff --git a/src/stem.c b/src/stem.c
index 5c2e379..f5a13ad 100644
--- a/src/stem.c
+++ b/src/stem.c
@@ -39,15 +39,15 @@ void array_append(array_t *a, value_t *v) {
a->size++;
}
-void array_curry(array_t *a, value_t *v) {
+void array_add(array_t *a, value_t *v, int index) {
if (a->size >= a->capacity - 3) {
a->capacity = a->capacity * 2;
a->items = realloc(a->items, a->capacity * sizeof(value_t *));
}
- for (int i = a->size - 1; i >= 0; i--) {
+ for (int i = a->size - 1; i >= index; i--) {
a->items[i + 1] = a->items[i];
}
- a->items[0] = v;
+ a->items[index] = v;
a->size++;
}