From 4ada155b930cb5fb96493dff3a5af8809da3b214 Mon Sep 17 00:00:00 2001 From: Preston Pan Date: Wed, 10 Jan 2024 16:36:53 -0800 Subject: add comments in both stem and also in my c code --- examples/fib.stem | 1 + include/better_string.h | 6 +-- include/macros.h | 2 +- include/stem.h | 102 ++++++++++++++++++++++++++++++++++++++++++++++-- src/builtins.c | 4 +- src/main.c | 4 +- src/stem.c | 19 +++++++++ 7 files changed, 126 insertions(+), 12 deletions(-) diff --git a/examples/fib.stem b/examples/fib.stem index 4104a36..de02b5c 100644 --- a/examples/fib.stem +++ b/examples/fib.stem @@ -1,3 +1,4 @@ +# Author: Andrei Sova 2023 fib [ dup 1 <= [ ] [ dup 1 - fib diff --git a/include/better_string.h b/include/better_string.h index a30b7e2..6726b4d 100644 --- a/include/better_string.h +++ b/include/better_string.h @@ -6,11 +6,11 @@ /*! @brief An array implementation of the string data structure */ typedef struct STRING_STRUCT { - /*! length of string */ + /*! @brief length of string */ size_t length; - /*! Size of current value buffer */ + /*! @brief Size of current value buffer */ size_t bufsize; - /*! String used for the */ + /*! @brief String value of struct */ char *value; } string_t; diff --git a/include/macros.h b/include/macros.h index 0fc94c8..95d3327 100644 --- a/include/macros.h +++ b/include/macros.h @@ -74,7 +74,7 @@ #define CRESET "\e[0m" #define COLOR_RESET "\e[0m" -/* Die in case of catastrophic memory problem */ +/*! Die in case of catastrophic memory problem */ void die(char *message); #endif diff --git a/include/stem.h b/include/stem.h index e8a8587..69bfcb4 100644 --- a/include/stem.h +++ b/include/stem.h @@ -7,131 +7,225 @@ 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); +/*! 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); -value_t *parse_word(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); +/*! 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); +/*! 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); -bool eval_quote(value_t *v); - +/*! Evaluates a value returned by the parser. */ void eval(value_t *v); #endif // PARSER_H_ diff --git a/src/builtins.c b/src/builtins.c index 31981c3..eb4b521 100644 --- a/src/builtins.c +++ b/src/builtins.c @@ -327,7 +327,7 @@ void strquote(value_t *v) { retval->quote = init_array(10); char *s = malloc(strlen(v1->str_word->value) + 1); strcpy(s, v1->str_word->value); - parser_t *p = init_parser(s); + parser_t *p = parser_pp(s); value_t *cur; while (1) { cur = parser_get_next(p); @@ -405,7 +405,7 @@ void stemexit(value_t *v) { ht_free(WORD_TABLE, value_free); ht_free(FLIT, func_free); array_free(STACK); - free(INBUF); + free(PARSER->source); free(PARSER); array_free(EVAL_STACK); ht_free(OBJ_TABLE, func_free); diff --git a/src/main.c b/src/main.c index 7e9f5d9..3b1f53f 100644 --- a/src/main.c +++ b/src/main.c @@ -49,7 +49,7 @@ int main(int argc, char **argv) { ssize_t bytes_read = getdelim(&INBUF, &len, '\0', FP); fclose(FP); - PARSER = init_parser(INBUF); + PARSER = parser_pp(INBUF); STACK = init_array(10); WORD_TABLE = init_ht(500); EVAL_STACK = init_array(10); @@ -65,7 +65,7 @@ int main(int argc, char **argv) { eval(v); } - free(INBUF); + free(PARSER->source); ht_free(WORD_TABLE, value_free); ht_free(FLIT, func_free); ht_free(OBJ_TABLE, custom_free); diff --git a/src/stem.c b/src/stem.c index d2204f9..b9e0a16 100644 --- a/src/stem.c +++ b/src/stem.c @@ -161,6 +161,25 @@ void parser_reset(parser_t *p, char *source) { p->c = source[0]; } +parser_t *parser_pp(char *s) { + parser_t *p = init_parser(s); + string_t *rstr = init_string(NULL); + while (p->c != '\0') { + if (p->c == '#') { + while (p->c != '\n' && p->c != '\0') { + parser_move(p); + } + } else { + string_append(rstr, p->c); + parser_move(p); + } + } + free(p->source); + parser_reset(p, rstr->value); + free(rstr); + return p; +} + void parser_move(parser_t *p) { if (p->i < strlen(p->source) && p->c != '\0') { p->i++; -- cgit