aboutsummaryrefslogtreecommitdiff
path: root/include/stem.h
blob: f4aff54af202bd5c867fefbae62021ac5c650619 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
#ifndef PARSER_H_
#define PARSER_H_
#include <better_string.h>
#include <stdbool.h>
#include <stdlib.h>

typedef struct VALUE_STRUCT value_t;
typedef struct ARRAY_STRUCT array_t;

/*! @brief non-generic array data structure used for stacks and quotes */
/*! array_t holds items of type value_t *; see VALUE_STRUCT for more details*/
struct ARRAY_STRUCT {
  /*! @brief Allowed items */
  value_t **items;
  /*! @brief total number of elements */
  size_t size;
  /*! @brief the buffer size */
  size_t capacity;
};

/*! @brief holds a string, int, float, word, or quote, as well as custom
 * datatypes. */
/*! VCUSTOM allows for definitions of custom structs, where the name of such a
 *custom data structure is to be held in str_word, and the actual custom struct
 *is to be held in custom. custom_t allows for the definition of required
 *functions in order for the builtin functions to remain memory safe. */
struct VALUE_STRUCT {
  /*! @brief Enum that defines different types within the language. */
  enum { VWORD, VINT, VFLOAT, VSTR, VQUOTE, VERR, VCUSTOM } type;
  union {
    /*! @brief floats and ints in this language are to be stored in this
     * variable. */
    long double int_float;
    /*! @brief A quote is an array of more values, which can contain more
     * quotes. */
    array_t *quote;
    /*! @brief this holds the string value of a string, word, or the name of a
     * custom type. */
    string_t *str_word;
  };
  /*! @brief this variable holds the value of a custom type. */
  void *custom;
  /*! @brief only to be used with values of type word */
  /*! This variable tells the evaluator if the word in question needs to be
   * evaluated as a funtion or pushed onto the stack. */
  bool escaped;
};

/*! @brief Parser implementation directly parses without lexer */
/*! the parser data structure parses a string of valid stem code and
 * returns a value until it reaches EOF or end of string. */
typedef struct PARSER_STRUCT {
  /*! @brief The string that contains valid stem code. */
  char *source;
  /*! @brief Index of current character */
  int i;
  /*! @brief The current character */
  char c;
} parser_t;

/*! @brief This structure is to be used in singly linked lists that hold
 * key-value pairs for a hash table. */
typedef struct NODE_STRUCT {
  /*! @brief Stores the key */
  string_t *key;
  /*! @brief Stores the value */
  void *value;
  /*! @brief Stores the next node in the singly linked list */
  struct NODE_STRUCT *next;
} node_t;

/*! @brief Singly Linked List used for separate chaining for a hash table */
typedef struct {
  /*! @brief The first node. Is NULL until value is added to sll. */
  node_t *head;
  /*! @brief current size of hash table */
  size_t size;
} sll_t;

/*! @brief hash table implementation */
/*! this hash table stores a value in a numbered bucket according to what number
 * the key hashes to; each bucket is a singly linked list to handle collisions
 */
typedef struct {
  /*! @brief Stores the key-value pairs */
  sll_t **buckets;
  /*! @brief Total number of buckets in hash table */
  size_t size;
} ht_t;

/*! @brief stores functions for custom structs */
/*! custom_t hold three functions that is to be stored in a hash table named
 *OBJ_TABLE. The key is the custom struct name, and the values are three
 *functions that make the builtin functions memory safe. */
typedef struct {
  /*! @brief Function that prints the custom struct */
  void (*printfunc)(void *);
  /*! @brief deep copy of custom struct */
  void *(*copyfunc)(void *);
  /*! @brief free function for custom struct */
  void (*freefunc)(void *);
} custom_t;

/*! Useless function that is only used in order to be passed into a hash table.
 */
void func_free(void *f);

/*! Allocates memory for new array */
array_t *init_array(size_t size);

/*! appends element to back of array */
void array_append(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);

/*! Deep copy of array and its contents. */
array_t *array_copy(array_t *a);

/*! Concatenate two arrays and put the result in a1. */
void array_extend(array_t *a1, array_t *a2);

/*! Free array and all value_t elements. */
void array_free(array_t *a);

/*! Allocates memory for new value_t. */
value_t *init_value(int type);

/*! deep copy of value struct. */
value_t *value_copy(value_t *v);

/*! Frees value struct */
void value_free(void *v);

/*! Allocates memory for new custom_t */
custom_t *init_custom(void (*)(void *), void (*)(void *), void *(*)(void *));

/*! Frees custom_t *. */
void custom_free(void *);

/*! Adds function to FLIT. */
void add_func(ht_t *h, void (*func)(value_t *), char *key);

/*! Adds object functions to OBJ_TABLE and adds constructor for custom type to
 * FLIT. */
void add_obj(ht_t *h, ht_t *h2, void (*printfunc)(void *),
             void (*freefunc)(void *), void *(*copyfunc)(void *),
             void (*createfunc)(void *), char *key);

/*! Allocates memory for new parser */
parser_t *init_parser(char *source);

/*! Removes comments from string s. */
parser_t *parser_pp(char *s);

/*! Resets state of parser */
void parser_reset(parser_t *p, char *source);

/*! Moves parser by one character if able */
void parser_move(parser_t *p);

/*! skips all whitespace between important characters. */
void parser_skip_whitespace(parser_t *p);

/*! returns VSTR value_t, called when '"' character is seen */
value_t *parse_string(parser_t *p);

/*! returns VQUOTE value_t, called when `[` character is seen */
value_t *parse_quote(parser_t *p);

/*! returns VINT or VFLOAT if the character is a digit. */
value_t *parse_num(parser_t *p);

/*! returns VWORD value_t, called if not any other type. If word starts with
 * `\`, then the word is escaped. */
value_t *parse_word(parser_t *p);

/*! Error in parsing strings can occur if wrong escape code. */
void parser_error(parser_t *p);

/*! Gets the next value_t from the string, returns NULL if EOF. */
value_t *parser_get_next(parser_t *p);

/*! Allocates memory for new node struct. */
node_t *init_node(string_t *key, void *v);

/*! Frees node */
void node_free(node_t *n, void (*freefunc)(void *));

/*! Allocates memory for new singly linked list */
sll_t *init_sll();

/*! Adds value to singly linked list. If key already exists, the value is
 * replaced and the old one is freed according to freefunc. */
void sll_add(sll_t *l, string_t *key, void *v, void (*freefunc)(void *));

/*! Gets value by key from singly linked list. */
void *sll_get(sll_t *l, string_t *key);

/*! deletes item from singly linked list */
void sll_delete(sll_t *l, string_t *k, void (*freefunc)(void *));

/*! Frees singly linked list */
void sll_free(sll_t *l, void (*freefunc)(void *));

/*! Allocates memory for new hash table */
ht_t *init_ht(size_t size);

/*! add key value pair to hash table. If the key already exists, the value is
 *replaced and the old value is freed according to freefunc. */
void ht_add(ht_t *h, string_t *key, void *v, void (*freefunc)(void *));

/*! Gets value from hash table by key */
void *ht_get(ht_t *h, string_t *key);

/*! Deletes item from hash table */
void ht_delete(ht_t *h, string_t *key, void (*freefunc)(void *));

/*! returns true if key exists in hash table. false otherwise */
bool ht_exists(ht_t *h, string_t *key);

/*! Frees hash table */
void ht_free(ht_t *h, void (*freefunc)(void *));

/*! hashes key into integer for hash table */
unsigned long hash(ht_t *h, char *key);

/*! returns true if word found in FLIT table, and then executes that function */
bool eval_builtins(value_t *v);

/*! returns true if word found in WORD_TABLE, and then executes that function */
bool eval_ht(value_t *v);

/*! Evaluates a value returned by the parser. */
void eval(value_t *v);

#endif // PARSER_H_