all repos — hex @ c76da17c2ee93a3b4a20cdf33d840314003165ca

A tiny, minimalist, slightly-esoteric concatenative programming lannguage.

Split source into separate file + added amalgamation.
h3rald h3rald@h3rald.com
Mon, 16 Dec 2024 07:12:45 +0100
commit

c76da17c2ee93a3b4a20cdf33d840314003165ca

parent

61083ef3f40146918d1bdf59d19b53b55209ba2e

M .gitignore.gitignore

@@ -54,6 +54,7 @@ # Exe names

hex # Other +src/hex.c example.hex web/out/ web/assets/hex.js
M MakefileMakefile

@@ -2,8 +2,13 @@ CC = gcc

CFLAGS = -Wall -Wextra -g LDFLAGS = +.PHONY: wasm, playground, clean, test, web + hex: src/hex.c $(CC) $(CFLAGS) $(LDFLAGS) $< -o hex + +src/hex.c: + sh scripts/amalgamate.sh web/assets/hex.wasm: src/hex.c emcc -O2 -sASYNCIFY -DBROWSER -sEXPORTED_RUNTIME_METHODS=stringToUTF8 src/hex.c -o web/assets/hex.js --pre-js web/assets/hex-playground.js

@@ -14,23 +19,19 @@

ape: src/hex.c cosmocc $(CFLAGS) $(LDFLAGS) $< -o hex -.PHONY: wasm wasm: hex.wasm -.PHONY: playground playground: web/assets/hex.wasm -.PHONY: clean clean: + rm -f src/hex.c rm -f hex rm -f hex.exe rm -f hex.js rm -f hex.wasm -.PHONY: test test: ./hex scripts/test.hex -.PHONY: web web: playground ./hex scripts/web.hex
A scripts/amalgamate.sh

@@ -0,0 +1,26 @@

+#!/bin/bash + +# Files to combine +header_file="src/hex.h" +source_files=( "src/stack.c" "src/registry.c" "src/error.c" "src/help.c" "src/stacktrace.c" + "src/parser.c" "src/interpreter.c" "src/helpers.c" "src/symbols.c" "src/main.c") +output_file="src/hex.c" + +# Start with a clean output file +echo "/* *** hex amalgamation *** */" > "$output_file" + +# Add the header file with a #line directive +echo "/* File: $header_file */" >> "$output_file" +echo "#line 1 \"$header_file\"" >> "$output_file" +cat "$header_file" >> "$output_file" +echo "" >> "$output_file" + +# Add each source file with #line directives +for file in "${source_files[@]}"; do + echo "/* File: $file */" >> "$output_file" + echo "#line 1 \"$file\"" >> "$output_file" + cat "$file" >> "$output_file" + echo "" >> "$output_file" +done + +echo "Amalgamation file created: $output_file"
A src/error.c

@@ -0,0 +1,64 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Error & Debugging // +//////////////////////////////////////// + +void hex_error(hex_context_t *ctx, const char *format, ...) +{ + va_list args; + va_start(args, format); + vsnprintf(ctx->error, sizeof(ctx->error), format, args); + if (ctx->settings.errors_enabled) /// FC + { + fprintf(stderr, "[error] "); + fprintf(stderr, "%s\n", ctx->error); + } + va_end(args); +} + +void hex_debug(hex_context_t *ctx, const char *format, ...) +{ + if (ctx->settings.debugging_enabled) + { + va_list args; + va_start(args, format); + fprintf(stdout, "*** "); + vfprintf(stdout, format, args); + fprintf(stdout, "\n"); + va_end(args); + } +} + +char *hex_type(hex_item_type_t type) +{ + switch (type) + { + case HEX_TYPE_INTEGER: + return "integer"; + case HEX_TYPE_STRING: + return "string"; + case HEX_TYPE_QUOTATION: + return "quotation"; + case HEX_TYPE_NATIVE_SYMBOL: + return "native-symbol"; + case HEX_TYPE_USER_SYMBOL: + return "user-symbol"; + case HEX_TYPE_INVALID: + return "invalid"; + default: + return "unknown"; + } +} + +void hex_debug_item(hex_context_t *ctx, const char *message, hex_item_t item) +{ + if (ctx->settings.debugging_enabled) + { + fprintf(stdout, "*** %s: ", message); + hex_print_item(stdout, item); + fprintf(stdout, "\n"); + } +}
A src/help.c

@@ -0,0 +1,125 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Help System // +//////////////////////////////////////// + +void hex_doc(hex_doc_dictionary_t *dict, const char *name, const char *input, const char *output, const char *description) +{ + hex_doc_entry_t doc = {.name = name, .description = description, .input = input, .output = output}; + // No overflow check as the dictionary is fixed size + dict->entries[dict->size] = doc; + dict->size++; +} + +int hex_get_doc(hex_doc_dictionary_t *docs, const char *key, hex_doc_entry_t *result) +{ + for (int i = 0; i < docs->size; i++) + { + if (strcmp(docs->entries[i].name, key) == 0) + { + *result = docs->entries[i]; + return 1; + } + } + return 0; +} + +void hex_create_docs(hex_doc_dictionary_t *docs) +{ + // Memory + hex_doc(docs, ":", "a s", "", "Stores 'a' as symbol 's'."); + hex_doc(docs, "#", "s", "", "Deletes user symbol 's'."); + + // Control flow + hex_doc(docs, "if", "q q q", "*", "If 'q1' is not 0x0, executes 'q2', else 'q3'."); + hex_doc(docs, "when", "q1 q2", "*", "If 'q1' is not 0x0, executes 'q2'."); + hex_doc(docs, "while", "q1 q2", "*", "While 'q1' is not 0x0, executes 'q2'."); + hex_doc(docs, "error", "", "s", "Returns the last error message."); + hex_doc(docs, "try", "q1 q2", "*", "If 'q1' fails, executes 'q2'."); + + // Stack + hex_doc(docs, "dup", "a", "a a", "Duplicates 'a'."); + hex_doc(docs, "pop", "a", "", "Removes the top item from the stack."); + hex_doc(docs, "swap", "a1 a2", "a2 a1", "Swaps 'a2' with 'a1'."); + hex_doc(docs, "stack", "", "q", "Returns the contents of the stack."); + hex_doc(docs, "clear", "", "", "Clears the stack."); + + // Evaluation + hex_doc(docs, ".", "q", "*", "Pushes each item of 'q' on the stack."); + hex_doc(docs, "!", "s", "", "Evaluates 's' as a hex program."); + hex_doc(docs, "'", "a", "q", "Wraps 'a' in a quotation."); + + // Arithmetic + hex_doc(docs, "+", "i1 12", "i", "Adds two integers."); + hex_doc(docs, "-", "i1 12", "i", "Subtracts 'i2' from 'i1'."); + hex_doc(docs, "*", "i1 12", "i", "Multiplies two integers."); + hex_doc(docs, "/", "i1 12", "i", "Divides 'i1' by 'i2'."); + hex_doc(docs, "%", "i1 12", "i", "Calculates the modulo of 'i1' divided by 'i2'."); + + // Bitwise + hex_doc(docs, "&", "i1 12", "i", "Calculates the bitwise AND of two integers."); + hex_doc(docs, "|", "i1 12", "i", "Calculates the bitwise OR of two integers."); + hex_doc(docs, "^", "i1 12", "i", "Calculates the bitwise XOR of two integers."); + hex_doc(docs, "~", "i", "i", "Calculates the bitwise NOT of an integer."); + hex_doc(docs, "<<", "i1 12", "i", "Shifts 'i1' by 'i2' bytes to the left."); + hex_doc(docs, ">>", "i1 12", "i", "Shifts 'i1' by 'i2' bytes to the right."); + + // Comparison + hex_doc(docs, "==", "a1 a2", "i", "Returns 0x1 if 'a1' == 'a2', 0x0 otherwise."); + hex_doc(docs, "!=", "a1 a2", "i", "Returns 0x1 if 'a1' != 'a2', 0x0 otherwise."); + hex_doc(docs, ">", "a1 a2", "i", "Returns 0x1 if 'a1' > 'a2', 0x0 otherwise."); + hex_doc(docs, "<", "a1 a2", "i", "Returns 0x1 if 'a1' < 'a2', 0x0 otherwise."); + hex_doc(docs, ">=", "a1 a2", "i", "Returns 0x1 if 'a1' >= 'a2', 0x0 otherwise."); + hex_doc(docs, "<=", "a1 a2", "i", "Returns 0x1 if 'a1' <= 'a2', 0x0 otherwise."); + + // Logical + hex_doc(docs, "and", "i1 i2", "i", "Returns 0x1 if both 'i1' and 'i2' are not 0x0."); + hex_doc(docs, "or", "i1 i2", "i", "Returns 0x1 if either 'i1' or 'i2' are not 0x0."); + hex_doc(docs, "not", "i", "i", "Returns 0x1 if 'i' is 0x0, 0x0 otherwise."); + hex_doc(docs, "xor", "i1 i2", "i", "Returns 0x1 if only one item is not 0x0."); + + // Type + hex_doc(docs, "int", "s", "i", "Converts a string to a hex integer."); + hex_doc(docs, "str", "i", "s", "Converts a hex integer to a string."); + hex_doc(docs, "dec", "i", "s", "Converts a hex integer to a decimal string."); + hex_doc(docs, "hex", "s", "i", "Converter a decimal string to a hex integer."); + hex_doc(docs, "chr", "i", "s", "Converts an integer to a single-character."); + hex_doc(docs, "ord", "s", "i", "Converts a single-character to an integer."); + hex_doc(docs, "type", "a", "s", "Pushes the data type of 'a' on the stack."); + + // List + hex_doc(docs, "cat", "(s1 s2|q1 q2) ", "(s3|q3)", "Concatenates two quotations or two strings."); + hex_doc(docs, "len", "(s|q)", "i ", "Returns the length of 's' or 'q'."); + hex_doc(docs, "get", "(s|q)", "a", "Gets the item at position 'i' in 's' or 'q'."); + hex_doc(docs, "index", "(s a|q a)", "i", "Returns the index of 'a' within 's' or 'q'."); + hex_doc(docs, "join", "q s", "s", "Joins the strings in 'q' using separator 's'."); + + // String + hex_doc(docs, "split", "s1 s2", "q", "Splits 's1' using separator 's2'."); + hex_doc(docs, "replace", "s1 s2 s3", "s", "Replaces 's2' with 's3' within 's1'."); + + // Quotation + hex_doc(docs, "each", "q1 q2", "*", "Executes 'q2' for each item of 'q1'."); + hex_doc(docs, "map", "q1 q2", "q3", "Applies 'q2' to 'q1' items and returns results."); + hex_doc(docs, "filter", "q1 q2", "q3", "Filters 'q2' by applying 'q1'."); + + // I/O + hex_doc(docs, "puts", "a", "", "Prints 'a' and a new line to standard output."); + hex_doc(docs, "warn", "a", "", "Prints 'a' and a new line to standard error."); + hex_doc(docs, "print", "a", "", "Prints 'a' to standard output."); + hex_doc(docs, "gets", "", "s", "Gets a string from standard input."); + + // File + hex_doc(docs, "read", "s1", "s2", "Returns the contents of the specified file."); + hex_doc(docs, "write", "s1 s2", "", "Writes 's2' to the file 's1'."); + hex_doc(docs, "append", "s1 s2", "", "Appends 's2' to the file 's1'."); + + // Shell + hex_doc(docs, "args", "", "q", "Returns the program arguments."); + hex_doc(docs, "exit", "i", "", "Exits the program with exit code 'i'."); + hex_doc(docs, "exec", "s", "", "Executes the command 's'."); + hex_doc(docs, "run", "s", "q", "Executes 's' and returns code, stdout, stderr."); +}
A src/helpers.c

@@ -0,0 +1,225 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Helper Functions // +//////////////////////////////////////// + +#if defined(__EMSCRIPTEN__) +#include <emscripten.h> + +EM_ASYNC_JS(char *, em_fgets, (const char *buf, size_t bufsize), { + return await new Promise(function(resolve, reject) { + if (Module.pending_lines.length > 0) + { + resolve(Module.pending_lines.shift()); + } + else + { + Module.pending_fgets.push(resolve); + } + }) + .then(function(s) { + // convert JS string to WASM string + let l = s.length + 1; + if (l >= bufsize) + { + // truncate + l = bufsize - 1; + } + Module.stringToUTF8(s.slice(0, l), buf, l); + return buf; + }); +}); + +#endif + +static void hex_rpad(const char *str, int total_length) +{ + int len = strlen(str); + printf("%s", str); + for (int i = len; i < total_length; i++) + { + printf(" "); + } +} +static void hex_lpad(const char *str, int total_length) +{ + int len = strlen(str); + for (int i = len; i < total_length; i++) + { + printf(" "); + } + printf("%s", str); +} + +char *hex_itoa(int num, int base) +{ + static char str[20]; + int i = 0; + + // Handle 0 explicitly, otherwise empty string is printed + if (num == 0) + { + str[i++] = '0'; + str[i] = '\0'; + return str; + } + + // Process each digit + while (num != 0) + { + int rem = num % base; + str[i++] = (rem > 9) ? (rem - 10) + 'a' : rem + '0'; + num = num / base; + } + + str[i] = '\0'; // Null-terminate string + + // Reverse the string + int start = 0; + int end = i - 1; + while (start < end) + { + char temp = str[start]; + str[start] = str[end]; + str[end] = temp; + start++; + end--; + } + + return str; +} + +char *hex_itoa_dec(int num) +{ + return hex_itoa(num, 10); +} + +char *hex_itoa_hex(int num) +{ + return hex_itoa(num, 16); +} + +void hex_raw_print_item(FILE *stream, hex_item_t item) +{ + switch (item.type) + { + case HEX_TYPE_INTEGER: + fprintf(stream, "0x%x", item.data.int_value); + break; + case HEX_TYPE_STRING: + fprintf(stream, "%s", item.data.str_value); + break; + case HEX_TYPE_USER_SYMBOL: + case HEX_TYPE_NATIVE_SYMBOL: + fprintf(stream, "%s", item.token->value); + break; + case HEX_TYPE_QUOTATION: + fprintf(stream, "("); + for (int i = 0; i < item.quotation_size; i++) + { + if (i > 0) + { + fprintf(stream, " "); + } + hex_print_item(stream, *item.data.quotation_value[i]); + } + fprintf(stream, ")"); + break; + + case HEX_TYPE_INVALID: + fprintf(stream, "<invalid>"); + break; + default: + fprintf(stream, "<unknown>"); + break; + } +} + +void hex_print_item(FILE *stream, hex_item_t item) +{ + switch (item.type) + { + case HEX_TYPE_INTEGER: + fprintf(stream, "0x%x", item.data.int_value); + break; + + case HEX_TYPE_STRING: + fprintf(stream, "\""); + for (char *c = item.data.str_value; *c != '\0'; c++) + { + switch (*c) + { + case '\n': + fprintf(stream, "\\n"); + break; + case '\t': + fprintf(stream, "\\t"); + break; + case '\r': + fprintf(stream, "\\r"); + break; + case '\b': + fprintf(stream, "\\b"); + break; + case '\f': + fprintf(stream, "\\f"); + break; + case '\v': + fprintf(stream, "\\v"); + break; + case '\\': + fprintf(stream, "\\\\"); + break; + case '\"': + fprintf(stream, "\\\""); + break; + default: + if ((unsigned char)*c < 32 || (unsigned char)*c > 126) + { + // Escape non-printable characters as hex (e.g., \x1F) + fprintf(stream, "\\x%02x", (unsigned char)*c); + } + else + { + fputc(*c, stream); + } + break; + } + } + fprintf(stream, "\""); + break; + + case HEX_TYPE_USER_SYMBOL: + case HEX_TYPE_NATIVE_SYMBOL: + fprintf(stream, "%s", item.token->value); + break; + + case HEX_TYPE_QUOTATION: + fprintf(stream, "("); + for (int i = 0; i < item.quotation_size; i++) + { + if (i > 0) + { + fprintf(stream, " "); + } + hex_print_item(stream, *item.data.quotation_value[i]); + } + fprintf(stream, ")"); + break; + + case HEX_TYPE_INVALID: + fprintf(stream, "<invalid>"); + break; + default: + fprintf(stream, "<unknown>"); + break; + } +} + +int hex_is_symbol(hex_token_t *token, char *value) +{ + return strcmp(token->value, value) == 0; +}
D src/hex.c

@@ -1,4088 +0,0 @@

-#include "hex.h" - -#if defined(__EMSCRIPTEN__) -#include <emscripten.h> - -EM_ASYNC_JS(char *, em_fgets, (const char *buf, size_t bufsize), { - return await new Promise(function(resolve, reject) { - if (Module.pending_lines.length > 0) - { - resolve(Module.pending_lines.shift()); - } - else - { - Module.pending_fgets.push(resolve); - } - }) - .then(function(s) { - // convert JS string to WASM string - let l = s.length + 1; - if (l >= bufsize) - { - // truncate - l = bufsize - 1; - } - Module.stringToUTF8(s.slice(0, l), buf, l); - return buf; - }); -}); - -#endif - -// Common operations -#define POP(ctx, x) hex_item_t x = hex_pop(ctx) -#define FREE(ctx, x) hex_free_item(ctx, x) -#define PUSH(ctx, x) hex_push(ctx, x) - -//////////////////////////////////////// -// Registry Implementation // -//////////////////////////////////////// - -int hex_valid_user_symbol(hex_context_t *ctx, const char *symbol) -{ - // Check that key starts with a letter, or underscore - // and subsequent characters (if any) are letters, numbers, or underscores - if (strlen(symbol) == 0) - { - hex_error(ctx, "Symbol name cannot be an empty string"); - return 0; - } - if (!isalpha(symbol[0]) && symbol[0] != '_') - { - hex_error(ctx, "Invalid symbol: %s", symbol); - return 0; - } - for (size_t j = 1; j < strlen(symbol); j++) - { - if (!isalnum(symbol[j]) && symbol[j] != '_' && symbol[j] != '-') - { - hex_error(ctx, "Invalid symbol: %s", symbol); - return 0; - } - } - return 1; -} - -// Add a symbol to the registry -int hex_set_symbol(hex_context_t *ctx, const char *key, hex_item_t value, int native) -{ - - if (!native && hex_valid_user_symbol(ctx, key) == 0) - { - return 1; - } - for (int i = 0; i < ctx->registry.size; i++) - { - if (strcmp(ctx->registry.entries[i].key, key) == 0) - { - if (ctx->registry.entries[i].value.type == HEX_TYPE_NATIVE_SYMBOL) - { - hex_error(ctx, "Cannot overwrite native symbol '%s'", key); - return 1; - } - free(ctx->registry.entries[i].key); - ctx->registry.entries[i].key = strdup(key); - ctx->registry.entries[i].value = value; - return 0; - } - } - - if (ctx->registry.size >= HEX_REGISTRY_SIZE) - { - hex_error(ctx, "Registry overflow"); - hex_free_token(value.token); - return 1; - } - - ctx->registry.entries[ctx->registry.size].key = strdup(key); - ctx->registry.entries[ctx->registry.size].value = value; - ctx->registry.size++; - return 0; -} - -// Register a native symbol -void hex_set_native_symbol(hex_context_t *ctx, const char *name, int (*func)()) -{ - hex_item_t func_item; - func_item.type = HEX_TYPE_NATIVE_SYMBOL; - func_item.data.fn_value = func; - - if (hex_set_symbol(ctx, name, func_item, 1) != 0) - { - hex_error(ctx, "Error: Failed to register native symbol '%s'", name); - } -} - -// Get a symbol value from the registry -int hex_get_symbol(hex_context_t *ctx, const char *key, hex_item_t *result) -{ - - for (int i = 0; i < ctx->registry.size; i++) - { - if (strcmp(ctx->registry.entries[i].key, key) == 0) - { - *result = ctx->registry.entries[i].value; - return 1; - } - } - return 0; -} - -//////////////////////////////////////// -// Stack Implementation // -//////////////////////////////////////// - -// Push functions -int hex_push(hex_context_t *ctx, hex_item_t item) -{ - - if (ctx->stack.top >= HEX_STACK_SIZE - 1) - { - hex_error(ctx, "Stack overflow"); - return 1; - } - hex_debug_item(ctx, "PUSH", item); - int result = 0; - if (item.type == HEX_TYPE_USER_SYMBOL) - { - hex_item_t value; - if (hex_get_symbol(ctx, item.token->value, &value)) - { - result = PUSH(ctx, value); - } - else - { - hex_error(ctx, "Undefined user symbol: %s", item.token->value); - FREE(ctx, value); - result = 1; - } - } - else if (item.type == HEX_TYPE_NATIVE_SYMBOL) - { - hex_debug_item(ctx, "CALL", item); - result = item.data.fn_value(ctx); - } - else - { - ctx->stack.entries[++ctx->stack.top] = item; - } - if (result == 0) - { - hex_debug_item(ctx, "DONE", item); - } - else - { - hex_debug_item(ctx, "FAIL", item); - } - return result; -} - -char *hex_process_string(hex_context_t *ctx, const char *value) -{ - int len = strlen(value); - char *processed_str = (char *)malloc(len + 1); - if (!processed_str) - { - hex_error(ctx, "Memory allocation failed"); - return NULL; - } - - char *dst = processed_str; - const char *src = value; - while (*src) - { - if (*src == '\\' && *(src + 1)) - { - src++; - switch (*src) - { - case 'n': - *dst++ = '\n'; - break; - case 't': - *dst++ = '\t'; - break; - case 'r': - *dst++ = '\r'; - break; - case 'b': - *dst++ = '\b'; - break; - case 'f': - *dst++ = '\f'; - break; - case 'v': - *dst++ = '\v'; - break; - // case '\\': - // *dst++ = '\\'; - // break; - case '\"': - *dst++ = '\"'; - break; - default: - *dst++ = '\\'; - *dst++ = *src; - break; - } - } - else - { - *dst++ = *src; - } - src++; - } - *dst = '\0'; - return processed_str; -} - -hex_item_t hex_string_item(hex_context_t *ctx, const char *value) -{ - hex_item_t item = {.type = HEX_TYPE_STRING, .data.str_value = hex_process_string(ctx, value)}; - return item; -} - -hex_item_t hex_integer_item(hex_context_t *ctx, int value) -{ - (void)(ctx); - hex_item_t item = {.type = HEX_TYPE_INTEGER, .data.int_value = value}; - return item; -} - -hex_item_t hex_quotation_item(hex_context_t *ctx, hex_item_t **quotation, int size) -{ - (void)(ctx); - hex_item_t item = {.type = HEX_TYPE_QUOTATION, .data.quotation_value = quotation, .quotation_size = size}; - return item; -} - -int hex_push_string(hex_context_t *ctx, const char *value) -{ - return PUSH(ctx, hex_string_item(ctx, value)); -} - -int hex_push_integer(hex_context_t *ctx, int value) -{ - return PUSH(ctx, hex_integer_item(ctx, value)); -} - -int hex_push_quotation(hex_context_t *ctx, hex_item_t **quotation, int size) -{ - return PUSH(ctx, hex_quotation_item(ctx, quotation, size)); -} - -int hex_push_symbol(hex_context_t *ctx, hex_token_t *token) -{ - add_to_stack_trace(ctx, token); - hex_item_t value; - if (hex_get_symbol(ctx, token->value, &value)) - { - value.token = token; - return PUSH(ctx, value); - } - else - { - hex_error(ctx, "Undefined symbol: %s", token->value); - return 1; - } -} - -// Pop function -hex_item_t hex_pop(hex_context_t *ctx) -{ - if (ctx->stack.top < 0) - { - hex_error(ctx, "Insufficient items on the stack"); - return (hex_item_t){.type = HEX_TYPE_INVALID}; - } - hex_debug_item(ctx, " POP", ctx->stack.entries[ctx->stack.top]); - return ctx->stack.entries[ctx->stack.top--]; -} - -// Free a stack item -void hex_free_item(hex_context_t *ctx, hex_item_t item) -{ - hex_debug_item(ctx, "FREE", item); - if (item.type == HEX_TYPE_STRING && item.data.str_value != NULL) - { - hex_debug(ctx, "FREE: ** string start"); - item.data.str_value = NULL; - free(item.data.str_value); - hex_debug(ctx, "FREE: ** string end"); - } - - else if (item.type == HEX_TYPE_QUOTATION && item.data.quotation_value != NULL) - { - hex_debug(ctx, "FREE: ** quotation start"); - hex_free_list(ctx, item.data.quotation_value, item.quotation_size); - item.data.quotation_value = NULL; - hex_debug(ctx, "FREE: ** quotation end"); - } - else if (item.type == HEX_TYPE_NATIVE_SYMBOL && item.token->value != NULL) - { - hex_debug(ctx, "FREE: ** native symbol start"); - item.token = NULL; - hex_free_token(item.token); - hex_debug(ctx, "FREE: ** native symbol end"); - } - else if (item.type == HEX_TYPE_USER_SYMBOL && item.token->value != NULL) - { - hex_debug(ctx, "FREE: ** user symbol start"); - item.token = NULL; - hex_free_token(item.token); - hex_debug(ctx, "FREE: ** user symbol end"); - } - else - { - hex_debug(ctx, "FREE: ** nothing to free"); - } -} - -void hex_free_list(hex_context_t *ctx, hex_item_t **quotation, int size) -{ - for (int i = 0; i < size; i++) - { - FREE(ctx, *quotation[i]); - } -} - -//////////////////////////////////////// -// Error & Debugging // -//////////////////////////////////////// - -void hex_error(hex_context_t *ctx, const char *format, ...) -{ - va_list args; - va_start(args, format); - vsnprintf(ctx->error, sizeof(ctx->error), format, args); - if (ctx->settings.errors_enabled) /// FC - { - fprintf(stderr, "[error] "); - fprintf(stderr, "%s\n", ctx->error); - } - va_end(args); -} - -void hex_debug(hex_context_t *ctx, const char *format, ...) -{ - if (ctx->settings.debugging_enabled) - { - va_list args; - va_start(args, format); - fprintf(stdout, "*** "); - vfprintf(stdout, format, args); - fprintf(stdout, "\n"); - va_end(args); - } -} - -char *hex_type(hex_item_type_t type) -{ - switch (type) - { - case HEX_TYPE_INTEGER: - return "integer"; - case HEX_TYPE_STRING: - return "string"; - case HEX_TYPE_QUOTATION: - return "quotation"; - case HEX_TYPE_NATIVE_SYMBOL: - return "native-symbol"; - case HEX_TYPE_USER_SYMBOL: - return "user-symbol"; - case HEX_TYPE_INVALID: - return "invalid"; - default: - return "unknown"; - } -} - -void hex_debug_item(hex_context_t *ctx, const char *message, hex_item_t item) -{ - if (ctx->settings.debugging_enabled) - { - fprintf(stdout, "*** %s: ", message); - hex_print_item(stdout, item); - fprintf(stdout, "\n"); - } -} - -//////////////////////////////////////// -// Help System // -//////////////////////////////////////// - -void hex_doc(hex_doc_dictionary_t *dict, const char *name, const char *input, const char *output, const char *description) -{ - hex_doc_entry_t doc = {.name = name, .description = description, .input = input, .output = output}; - // No overflow check as the dictionary is fixed size - dict->entries[dict->size] = doc; - dict->size++; -} - -int hex_get_doc(hex_doc_dictionary_t *docs, const char *key, hex_doc_entry_t *result) -{ - for (int i = 0; i < docs->size; i++) - { - if (strcmp(docs->entries[i].name, key) == 0) - { - *result = docs->entries[i]; - return 1; - } - } - return 0; -} - -void hex_create_docs(hex_doc_dictionary_t *docs) -{ - // Memory - hex_doc(docs, ":", "a s", "", "Stores 'a' as symbol 's'."); - hex_doc(docs, "#", "s", "", "Deletes user symbol 's'."); - - // Control flow - hex_doc(docs, "if", "q q q", "*", "If 'q1' is not 0x0, executes 'q2', else 'q3'."); - hex_doc(docs, "when", "q1 q2", "*", "If 'q1' is not 0x0, executes 'q2'."); - hex_doc(docs, "while", "q1 q2", "*", "While 'q1' is not 0x0, executes 'q2'."); - hex_doc(docs, "error", "", "s", "Returns the last error message."); - hex_doc(docs, "try", "q1 q2", "*", "If 'q1' fails, executes 'q2'."); - - // Stack - hex_doc(docs, "dup", "a", "a a", "Duplicates 'a'."); - hex_doc(docs, "pop", "a", "", "Removes the top item from the stack."); - hex_doc(docs, "swap", "a1 a2", "a2 a1", "Swaps 'a2' with 'a1'."); - hex_doc(docs, "stack", "", "q", "Returns the contents of the stack."); - hex_doc(docs, "clear", "", "", "Clears the stack."); - - // Evaluation - hex_doc(docs, ".", "q", "*", "Pushes each item of 'q' on the stack."); - hex_doc(docs, "!", "s", "", "Evaluates 's' as a hex program."); - hex_doc(docs, "'", "a", "q", "Wraps 'a' in a quotation."); - - // Arithmetic - hex_doc(docs, "+", "i1 12", "i", "Adds two integers."); - hex_doc(docs, "-", "i1 12", "i", "Subtracts 'i2' from 'i1'."); - hex_doc(docs, "*", "i1 12", "i", "Multiplies two integers."); - hex_doc(docs, "/", "i1 12", "i", "Divides 'i1' by 'i2'."); - hex_doc(docs, "%", "i1 12", "i", "Calculates the modulo of 'i1' divided by 'i2'."); - - // Bitwise - hex_doc(docs, "&", "i1 12", "i", "Calculates the bitwise AND of two integers."); - hex_doc(docs, "|", "i1 12", "i", "Calculates the bitwise OR of two integers."); - hex_doc(docs, "^", "i1 12", "i", "Calculates the bitwise XOR of two integers."); - hex_doc(docs, "~", "i", "i", "Calculates the bitwise NOT of an integer."); - hex_doc(docs, "<<", "i1 12", "i", "Shifts 'i1' by 'i2' bytes to the left."); - hex_doc(docs, ">>", "i1 12", "i", "Shifts 'i1' by 'i2' bytes to the right."); - - // Comparison - hex_doc(docs, "==", "a1 a2", "i", "Returns 0x1 if 'a1' == 'a2', 0x0 otherwise."); - hex_doc(docs, "!=", "a1 a2", "i", "Returns 0x1 if 'a1' != 'a2', 0x0 otherwise."); - hex_doc(docs, ">", "a1 a2", "i", "Returns 0x1 if 'a1' > 'a2', 0x0 otherwise."); - hex_doc(docs, "<", "a1 a2", "i", "Returns 0x1 if 'a1' < 'a2', 0x0 otherwise."); - hex_doc(docs, ">=", "a1 a2", "i", "Returns 0x1 if 'a1' >= 'a2', 0x0 otherwise."); - hex_doc(docs, "<=", "a1 a2", "i", "Returns 0x1 if 'a1' <= 'a2', 0x0 otherwise."); - - // Logical - hex_doc(docs, "and", "i1 i2", "i", "Returns 0x1 if both 'i1' and 'i2' are not 0x0."); - hex_doc(docs, "or", "i1 i2", "i", "Returns 0x1 if either 'i1' or 'i2' are not 0x0."); - hex_doc(docs, "not", "i", "i", "Returns 0x1 if 'i' is 0x0, 0x0 otherwise."); - hex_doc(docs, "xor", "i1 i2", "i", "Returns 0x1 if only one item is not 0x0."); - - // Type - hex_doc(docs, "int", "s", "i", "Converts a string to a hex integer."); - hex_doc(docs, "str", "i", "s", "Converts a hex integer to a string."); - hex_doc(docs, "dec", "i", "s", "Converts a hex integer to a decimal string."); - hex_doc(docs, "hex", "s", "i", "Converter a decimal string to a hex integer."); - hex_doc(docs, "chr", "i", "s", "Converts an integer to a single-character."); - hex_doc(docs, "ord", "s", "i", "Converts a single-character to an integer."); - hex_doc(docs, "type", "a", "s", "Pushes the data type of 'a' on the stack."); - - // List - hex_doc(docs, "cat", "(s1 s2|q1 q2) ", "(s3|q3)", "Concatenates two quotations or two strings."); - hex_doc(docs, "len", "(s|q)", "i ", "Returns the length of 's' or 'q'."); - hex_doc(docs, "get", "(s|q)", "a", "Gets the item at position 'i' in 's' or 'q'."); - hex_doc(docs, "index", "(s a|q a)", "i", "Returns the index of 'a' within 's' or 'q'."); - hex_doc(docs, "join", "q s", "s", "Joins the strings in 'q' using separator 's'."); - - // String - hex_doc(docs, "split", "s1 s2", "q", "Splits 's1' using separator 's2'."); - hex_doc(docs, "replace", "s1 s2 s3", "s", "Replaces 's2' with 's3' within 's1'."); - - // Quotation - hex_doc(docs, "each", "q1 q2", "*", "Executes 'q2' for each item of 'q1'."); - hex_doc(docs, "map", "q1 q2", "q3", "Applies 'q2' to 'q1' items and returns results."); - hex_doc(docs, "filter", "q1 q2", "q3", "Filters 'q2' by applying 'q1'."); - - // I/O - hex_doc(docs, "puts", "a", "", "Prints 'a' and a new line to standard output."); - hex_doc(docs, "warn", "a", "", "Prints 'a' and a new line to standard error."); - hex_doc(docs, "print", "a", "", "Prints 'a' to standard output."); - hex_doc(docs, "gets", "", "s", "Gets a string from standard input."); - - // File - hex_doc(docs, "read", "s1", "s2", "Returns the contents of the specified file."); - hex_doc(docs, "write", "s1 s2", "", "Writes 's2' to the file 's1'."); - hex_doc(docs, "append", "s1 s2", "", "Appends 's2' to the file 's1'."); - - // Shell - hex_doc(docs, "args", "", "q", "Returns the program arguments."); - hex_doc(docs, "exit", "i", "", "Exits the program with exit code 'i'."); - hex_doc(docs, "exec", "s", "", "Executes the command 's'."); - hex_doc(docs, "run", "s", "q", "Executes 's' and returns code, stdout, stderr."); -} - -//////////////////////////////////////// -// Tokenizer Implementation // -//////////////////////////////////////// - -// Process a token from the input -hex_token_t *hex_next_token(hex_context_t *ctx, const char **input, hex_file_position_t *position) -{ - const char *ptr = *input; - - // Skip whitespace - while (isspace(*ptr)) - { - if (*ptr == '\n') - { - position->line++; - position->column = 1; - } - else - { - position->column++; - } - ptr++; - } - - if (*ptr == '\0') - { - return NULL; // End of input - } - - hex_token_t *token = (hex_token_t *)malloc(sizeof(hex_token_t)); - token->value = NULL; - token->position.line = position->line; - token->position.column = position->column; - - if (*ptr == ';') - { - // Comment token - const char *start = ptr; - while (*ptr != '\0' && *ptr != '\n') - { - ptr++; - position->column++; - } - int len = ptr - start; - token->value = (char *)malloc(len + 1); - strncpy(token->value, start, len); - token->value[len] = '\0'; - token->type = HEX_TOKEN_COMMENT; - } - else if (strncmp(ptr, "#|", 2) == 0) - { - // Block comment token - const char *start = ptr; - ptr += 2; // Skip the "#|" prefix - position->column += 2; - while (*ptr != '\0' && strncmp(ptr, "|#", 2) != 0) - { - if (*ptr == '\n') - { - position->line++; - position->column = 1; - } - else - { - position->column++; - } - ptr++; - } - if (*ptr == '\0') - { - token->type = HEX_TOKEN_INVALID; - token->position.line = position->line; - token->position.column = position->column; - hex_error(ctx, "(%d,%d) Unterminated block comment", position->line, position->column); - return token; - } - ptr += 2; // Skip the "|#" suffix - position->column += 2; - int len = ptr - start; - token->value = (char *)malloc(len + 1); - strncpy(token->value, start, len); - token->value[len] = '\0'; - token->type = HEX_TOKEN_COMMENT; - } - else if (*ptr == '"') - { - // String token - ptr++; - const char *start = ptr; - int len = 0; - - while (*ptr != '\0') - { - if (*ptr == '\\' && *(ptr + 1) == '"') - { - ptr += 2; - len++; - position->column += 2; - } - if (*ptr == '\\' && *(ptr + 1) == '\\') - { - ptr += 2; - len++; - position->column += 2; - } - else if (*ptr == '"') - { - break; - } - else if (*ptr == '\n') - { - token->type = HEX_TOKEN_INVALID; - token->position.line = position->line; - token->position.column = position->column; - hex_error(ctx, "(%d,%d) Unescaped new line in string", position->line, position->column); - return token; - } - else - { - ptr++; - len++; - position->column++; - } - } - - if (*ptr != '"') - { - token->type = HEX_TOKEN_INVALID; - token->position.line = position->line; - token->position.column = position->column; - hex_error(ctx, "(%d,%d) Unterminated string", position->line, position->column); - return token; - } - - token->value = (char *)malloc(len + 1); - char *dst = token->value; - - ptr = start; - while (*ptr != '\0' && *ptr != '"') - { - if (*ptr == '\\' && *(ptr + 1) == '\\') - { - *dst++ = '\\'; - *dst++ = '\\'; - ptr += 2; - } - else if (*ptr == '\\' && *(ptr + 1) == '"') - { - *dst++ = '"'; - ptr += 2; - } - else - { - *dst++ = *ptr++; - } - } - *dst = '\0'; - ptr++; - position->column++; - token->type = HEX_TOKEN_STRING; - } - else if (strncmp(ptr, "0x", 2) == 0 || strncmp(ptr, "0X", 2) == 0) - { - // Hexadecimal integer token - const char *start = ptr; - ptr += 2; // Skip the "0x" prefix - position->column += 2; - while (isxdigit(*ptr)) - { - ptr++; - position->column++; - } - int len = ptr - start; - token->value = (char *)malloc(len + 1); - strncpy(token->value, start, len); - token->value[len] = '\0'; - token->type = HEX_TOKEN_INTEGER; - } - else if (*ptr == '(') - { - token->type = HEX_TOKEN_QUOTATION_START; - ptr++; - position->column++; - } - else if (*ptr == ')') - { - token->type = HEX_TOKEN_QUOTATION_END; - ptr++; - position->column++; - } - else - { - // Symbol token - const char *start = ptr; - while (*ptr != '\0' && !isspace(*ptr) && *ptr != ';' && *ptr != '(' && *ptr != ')' && *ptr != '"') - { - ptr++; - position->column++; - } - - int len = ptr - start; - token->value = (char *)malloc(len + 1); - strncpy(token->value, start, len); - token->value[len] = '\0'; - if (hex_valid_native_symbol(ctx, token->value) || hex_valid_user_symbol(ctx, token->value)) - { - token->type = HEX_TOKEN_SYMBOL; - } - else - { - token->type = HEX_TOKEN_INVALID; - token->position.line = position->line; - token->position.column = position->column; - } - } - - *input = ptr; - return token; -} - -// Free a token -void hex_free_token(hex_token_t *token) -{ - if (token) - { - free(token->value); - free(token); - } -} - -int hex_valid_native_symbol(hex_context_t *ctx, const char *symbol) -{ - hex_doc_entry_t doc; - for (size_t i = 0; i < HEX_NATIVE_SYMBOLS; i++) - { - if (hex_get_doc(&ctx->docs, symbol, &doc)) - { - return 1; - } - } - return 0; -} - -int32_t hex_parse_integer(const char *hex_str) -{ - // Parse the hexadecimal string as an unsigned 32-bit integer - uint32_t unsigned_value = (uint32_t)strtoul(hex_str, NULL, 16); - - // Cast the unsigned value to a signed 32-bit integer - return (int32_t)unsigned_value; -} - -int hex_parse_quotation(hex_context_t *ctx, const char **input, hex_item_t *result, hex_file_position_t *position) -{ - hex_item_t **quotation = NULL; - int capacity = 2; - int size = 0; - int balanced = 1; - - quotation = (hex_item_t **)malloc(capacity * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - return 1; - } - - hex_token_t *token; - while ((token = hex_next_token(ctx, input, position)) != NULL) - { - if (token->type == HEX_TOKEN_QUOTATION_END) - { - balanced--; - break; - } - - if (size >= capacity) - { - capacity *= 2; - quotation = (hex_item_t **)realloc(quotation, capacity * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "(%d,%d), Memory allocation failed", position->line, position->column); - return 1; - } - } - - hex_item_t *item = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (token->type == HEX_TOKEN_INTEGER) - { - - *item = hex_integer_item(ctx, hex_parse_integer(token->value)); - quotation[size] = item; - size++; - } - else if (token->type == HEX_TOKEN_STRING) - { - *item = hex_string_item(ctx, token->value); - quotation[size] = item; - size++; - } - else if (token->type == HEX_TOKEN_SYMBOL) - { - if (hex_valid_native_symbol(ctx, token->value)) - { - item->type = HEX_TYPE_NATIVE_SYMBOL; - hex_item_t value; - if (hex_get_symbol(ctx, token->value, &value)) - { - item->token = token; - item->type = HEX_TYPE_NATIVE_SYMBOL; - item->data.fn_value = value.data.fn_value; - } - else - { - hex_error(ctx, "(%d,%d) Unable to reference native symbol: %s", position->line, position->column, token->value); - hex_free_token(token); - hex_free_list(ctx, quotation, size); - return 1; - } - } - else - { - item->type = HEX_TYPE_USER_SYMBOL; - } - token->position.filename = strdup(position->filename); - item->token = token; - quotation[size] = item; - size++; - } - else if (token->type == HEX_TOKEN_QUOTATION_START) - { - item->type = HEX_TYPE_QUOTATION; - if (hex_parse_quotation(ctx, input, item, position) != 0) - { - hex_free_token(token); - hex_free_list(ctx, quotation, size); - return 1; - } - quotation[size] = item; - size++; - } - else if (token->type == HEX_TOKEN_COMMENT) - { - // Ignore comments - } - else - { - hex_error(ctx, "(%d,%d) Unexpected token in quotation: %d", position->line, position->column, token->value); - hex_free_token(token); - hex_free_list(ctx, quotation, size); - return 1; - } - } - - if (balanced != 0) - { - hex_error(ctx, "(%d,%d) Unterminated quotation", position->line, position->column); - hex_free_token(token); - hex_free_list(ctx, quotation, size); - return 1; - } - - result->type = HEX_TYPE_QUOTATION; - result->data.quotation_value = quotation; - result->quotation_size = size; - hex_free_token(token); - return 0; -} - -//////////////////////////////////////// -// Stack trace implementation // -//////////////////////////////////////// - -// Add an entry to the circular stack trace -void add_to_stack_trace(hex_context_t *ctx, hex_token_t *token) -{ - int index = (ctx->stack_trace.start + ctx->stack_trace.size) % HEX_STACK_TRACE_SIZE; - - if (ctx->stack_trace.size < HEX_STACK_TRACE_SIZE) - { - // Buffer is not full; add item - ctx->stack_trace.entries[index] = *token; - ctx->stack_trace.size++; - } - else - { - // Buffer is full; overwrite the oldest item - ctx->stack_trace.entries[index] = *token; - ctx->stack_trace.start = (ctx->stack_trace.start + 1) % HEX_STACK_TRACE_SIZE; - } -} - -// Print the stack trace -void print_stack_trace(hex_context_t *ctx) -{ - if (!ctx->settings.stack_trace_enabled || !ctx->settings.errors_enabled || ctx->stack_trace.size <= 0) - { - return; - } - fprintf(stderr, "[stack trace] (most recent symbol first):\n"); - - for (int i = 0; i < ctx->stack_trace.size; i++) - { - int index = (ctx->stack_trace.start + ctx->stack_trace.size - 1 - i) % HEX_STACK_TRACE_SIZE; - hex_token_t token = ctx->stack_trace.entries[index]; - fprintf(stderr, " %s (%s:%d:%d)\n", token.value, token.position.filename, token.position.line, token.position.column); - } -} - -//////////////////////////////////////// -// Helper Functions // -//////////////////////////////////////// - -static void hex_rpad(const char *str, int total_length) -{ - int len = strlen(str); - printf("%s", str); - for (int i = len; i < total_length; i++) - { - printf(" "); - } -} -static void hex_lpad(const char *str, int total_length) -{ - int len = strlen(str); - for (int i = len; i < total_length; i++) - { - printf(" "); - } - printf("%s", str); -} - -char *hex_itoa(int num, int base) -{ - static char str[20]; - int i = 0; - - // Handle 0 explicitly, otherwise empty string is printed - if (num == 0) - { - str[i++] = '0'; - str[i] = '\0'; - return str; - } - - // Process each digit - while (num != 0) - { - int rem = num % base; - str[i++] = (rem > 9) ? (rem - 10) + 'a' : rem + '0'; - num = num / base; - } - - str[i] = '\0'; // Null-terminate string - - // Reverse the string - int start = 0; - int end = i - 1; - while (start < end) - { - char temp = str[start]; - str[start] = str[end]; - str[end] = temp; - start++; - end--; - } - - return str; -} - -char *hex_itoa_dec(int num) -{ - return hex_itoa(num, 10); -} - -char *hex_itoa_hex(int num) -{ - return hex_itoa(num, 16); -} - -void hex_raw_print_item(FILE *stream, hex_item_t item) -{ - switch (item.type) - { - case HEX_TYPE_INTEGER: - fprintf(stream, "0x%x", item.data.int_value); - break; - case HEX_TYPE_STRING: - fprintf(stream, "%s", item.data.str_value); - break; - case HEX_TYPE_USER_SYMBOL: - case HEX_TYPE_NATIVE_SYMBOL: - fprintf(stream, "%s", item.token->value); - break; - case HEX_TYPE_QUOTATION: - fprintf(stream, "("); - for (int i = 0; i < item.quotation_size; i++) - { - if (i > 0) - { - fprintf(stream, " "); - } - hex_print_item(stream, *item.data.quotation_value[i]); - } - fprintf(stream, ")"); - break; - - case HEX_TYPE_INVALID: - fprintf(stream, "<invalid>"); - break; - default: - fprintf(stream, "<unknown>"); - break; - } -} - -void hex_print_item(FILE *stream, hex_item_t item) -{ - switch (item.type) - { - case HEX_TYPE_INTEGER: - fprintf(stream, "0x%x", item.data.int_value); - break; - - case HEX_TYPE_STRING: - fprintf(stream, "\""); - for (char *c = item.data.str_value; *c != '\0'; c++) - { - switch (*c) - { - case '\n': - fprintf(stream, "\\n"); - break; - case '\t': - fprintf(stream, "\\t"); - break; - case '\r': - fprintf(stream, "\\r"); - break; - case '\b': - fprintf(stream, "\\b"); - break; - case '\f': - fprintf(stream, "\\f"); - break; - case '\v': - fprintf(stream, "\\v"); - break; - case '\\': - fprintf(stream, "\\\\"); - break; - case '\"': - fprintf(stream, "\\\""); - break; - default: - if ((unsigned char)*c < 32 || (unsigned char)*c > 126) - { - // Escape non-printable characters as hex (e.g., \x1F) - fprintf(stream, "\\x%02x", (unsigned char)*c); - } - else - { - fputc(*c, stream); - } - break; - } - } - fprintf(stream, "\""); - break; - - case HEX_TYPE_USER_SYMBOL: - case HEX_TYPE_NATIVE_SYMBOL: - fprintf(stream, "%s", item.token->value); - break; - - case HEX_TYPE_QUOTATION: - fprintf(stream, "("); - for (int i = 0; i < item.quotation_size; i++) - { - if (i > 0) - { - fprintf(stream, " "); - } - hex_print_item(stream, *item.data.quotation_value[i]); - } - fprintf(stream, ")"); - break; - - case HEX_TYPE_INVALID: - fprintf(stream, "<invalid>"); - break; - default: - fprintf(stream, "<unknown>"); - break; - } -} - -int hex_is_symbol(hex_token_t *token, char *value) -{ - return strcmp(token->value, value) == 0; -} - -//////////////////////////////////////// -// Native Symbol Implementations // -//////////////////////////////////////// - -// Definition symbols - -int hex_symbol_store(hex_context_t *ctx) -{ - - POP(ctx, name); - if (name.type == HEX_TYPE_INVALID) - { - FREE(ctx, name); - return 1; - } - POP(ctx, value); - if (value.type == HEX_TYPE_INVALID) - { - FREE(ctx, name); - FREE(ctx, value); - return 1; - } - if (name.type != HEX_TYPE_STRING) - { - hex_error(ctx, "Symbol name must be a string"); - FREE(ctx, name); - FREE(ctx, value); - return 1; - } - if (hex_set_symbol(ctx, name.data.str_value, value, 0) != 0) - { - hex_error(ctx, "Failed to store symbol '%s'", name.data.str_value); - FREE(ctx, name); - FREE(ctx, value); - return 1; - } - return 0; -} - -int hex_symbol_free(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type != HEX_TYPE_STRING) - { - FREE(ctx, item); - hex_error(ctx, "Symbol name must be a string"); - return 1; - } - if (hex_valid_native_symbol(ctx, item.data.str_value)) - { - hex_error(ctx, "Cannot free native symbol '%s'", item.data.str_value); - FREE(ctx, item); - return 1; - } - for (int i = 0; i < ctx->registry.size; i++) - { - if (strcmp(ctx->registry.entries[i].key, item.data.str_value) == 0) - { - free(ctx->registry.entries[i].key); - FREE(ctx, ctx->registry.entries[i].value); - for (int j = i; j < ctx->registry.size - 1; j++) - { - ctx->registry.entries[j] = ctx->registry.entries[j + 1]; - } - ctx->registry.size--; - FREE(ctx, item); - return 0; - } - } - FREE(ctx, item); - return 0; -} - -int hex_symbol_type(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - return hex_push_string(ctx, hex_type(item.type)); -} - -// Evaluation symbols - -int hex_symbol_i(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "Symbol '.' requires a quotation"); - FREE(ctx, item); - return 1; - } - for (int i = 0; i < item.quotation_size; i++) - { - if (hex_push(ctx, *item.data.quotation_value[i]) != 0) - { - FREE(ctx, item); - return 1; - } - } - return 0; -} - -// evaluate a string -int hex_symbol_eval(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type != HEX_TYPE_STRING) - { - hex_error(ctx, "Symbol '!' requires a string"); - FREE(ctx, item); - return 1; - } - return hex_interpret(ctx, item.data.str_value, "<!>", 1, 1); -} - -// IO Symbols - -int hex_symbol_puts(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - hex_raw_print_item(stdout, item); - printf("\n"); - return 0; -} - -int hex_symbol_warn(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - hex_raw_print_item(stderr, item); - printf("\n"); - return 0; -} - -int hex_symbol_print(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - hex_raw_print_item(stdout, item); - fflush(stdout); - return 0; -} - -int hex_symbol_gets(hex_context_t *ctx) -{ - - char input[HEX_STDIN_BUFFER_SIZE]; // Buffer to store the input (adjust size if needed) - char *p = input; -#if defined(__EMSCRIPTEN__) - p = em_fgets(input, 1024); -#else - p = fgets(input, sizeof(input), stdin); -#endif - - if (p != NULL) - { - // Strip the newline character at the end of the string - input[strcspn(input, "\n")] = '\0'; - - // Push the input string onto the stack - return hex_push_string(ctx, input); - } - else - { - hex_error(ctx, "Failed to read input"); - return 1; - } -} - -// Mathematical symbols -int hex_symbol_add(hex_context_t *ctx) -{ - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value + b.data.int_value); - } - hex_error(ctx, "Symbol '+' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_subtract(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value - b.data.int_value); - } - hex_error(ctx, "Symbol '-' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_multiply(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value * b.data.int_value); - } - hex_error(ctx, "'*' symbol requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_divide(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - if (b.data.int_value == 0) - { - hex_error(ctx, "Division by zero"); - return 1; - } - return hex_push_integer(ctx, a.data.int_value / b.data.int_value); - } - hex_error(ctx, "Symbol '/' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_modulo(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - if (b.data.int_value == 0) - { - hex_error(ctx, "Division by zero"); - return 1; - } - return hex_push_integer(ctx, a.data.int_value % b.data.int_value); - } - hex_error(ctx, "Symbol '%%' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -// Bit symbols - -int hex_symbol_bitand(hex_context_t *ctx) -{ - - POP(ctx, right); - if (right.type == HEX_TYPE_INVALID) - { - FREE(ctx, right); - return 1; - } - POP(ctx, left); - if (left.type == HEX_TYPE_INVALID) - { - FREE(ctx, left); - FREE(ctx, right); - return 1; - } - if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, left.data.int_value & right.data.int_value); - } - hex_error(ctx, "Symbol '&' requires two integers"); - FREE(ctx, left); - FREE(ctx, right); - return 1; -} - -int hex_symbol_bitor(hex_context_t *ctx) -{ - - POP(ctx, right); - if (right.type == HEX_TYPE_INVALID) - { - FREE(ctx, right); - return 1; - } - POP(ctx, left); - if (left.type == HEX_TYPE_INVALID) - { - FREE(ctx, left); - FREE(ctx, right); - return 1; - } - if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, left.data.int_value | right.data.int_value); - } - hex_error(ctx, "Symbol '|' requires two integers"); - FREE(ctx, left); - FREE(ctx, right); - return 1; -} - -int hex_symbol_bitxor(hex_context_t *ctx) -{ - - POP(ctx, right); - if (right.type == HEX_TYPE_INVALID) - { - FREE(ctx, right); - return 1; - } - POP(ctx, left); - if (left.type == HEX_TYPE_INVALID) - { - FREE(ctx, left); - FREE(ctx, right); - return 1; - } - if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, left.data.int_value ^ right.data.int_value); - } - hex_error(ctx, "Symbol '^' requires two integers"); - FREE(ctx, left); - FREE(ctx, right); - return 1; -} - -int hex_symbol_shiftleft(hex_context_t *ctx) -{ - - POP(ctx, right); - if (right.type == HEX_TYPE_INVALID) - { - FREE(ctx, right); - return 1; - } - POP(ctx, left); - if (left.type == HEX_TYPE_INVALID) - { - FREE(ctx, left); - FREE(ctx, right); - return 1; - } - if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, left.data.int_value << right.data.int_value); - } - hex_error(ctx, "Symbol '<<' requires two integers"); - FREE(ctx, left); - FREE(ctx, right); - return 1; -} - -int hex_symbol_shiftright(hex_context_t *ctx) -{ - - POP(ctx, right); - if (right.type == HEX_TYPE_INVALID) - { - FREE(ctx, right); - return 1; - } - POP(ctx, left); - if (left.type == HEX_TYPE_INVALID) - { - FREE(ctx, left); - FREE(ctx, right); - return 1; - } - if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, left.data.int_value >> right.data.int_value); - } - hex_error(ctx, "Symbol '>>' requires two integers"); - FREE(ctx, left); - FREE(ctx, right); - return 1; -} - -int hex_symbol_bitnot(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, ~item.data.int_value); - } - hex_error(ctx, "Symbol '~' requires an integer"); - FREE(ctx, item); - return 1; -} - -// Conversion symbols - -int hex_symbol_int(hex_context_t *ctx) -{ - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - return 1; - } - if (a.type == HEX_TYPE_STRING) - { - return hex_push_integer(ctx, strtol(a.data.str_value, NULL, 16)); - } - hex_error(ctx, "Symbol 'int' requires a string"); - FREE(ctx, a); - return 1; -} - -int hex_symbol_str(hex_context_t *ctx) -{ - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - return 1; - } - if (a.type == HEX_TYPE_INTEGER) - { - return hex_push_string(ctx, hex_itoa_hex(a.data.int_value)); - } - hex_error(ctx, "Symbol 'str' requires an integer"); - FREE(ctx, a); - return 1; -} - -int hex_symbol_dec(hex_context_t *ctx) -{ - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - return 1; - } - if (a.type == HEX_TYPE_INTEGER) - { - return hex_push_string(ctx, hex_itoa_dec(a.data.int_value)); - } - hex_error(ctx, "Symbol 'dec' requires an integer"); - FREE(ctx, a); - return 1; -} - -int hex_symbol_hex(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type == HEX_TYPE_STRING) - { - return hex_push_integer(ctx, strtol(item.data.str_value, NULL, 10)); - } - hex_error(ctx, "Symbol 'hex' requires a string representing a decimal integer"); - FREE(ctx, item); - return 1; -} - -int hex_symbol_ord(hex_context_t *ctx) -{ - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type == HEX_TYPE_STRING) - { - if (strlen(item.data.str_value) > 1) - { - return hex_push_integer(ctx, -1); - } - unsigned char *str = (unsigned char *)item.data.str_value; - if (str[0] < 128) - { - return hex_push_integer(ctx, str[0]); - } - else - { - return hex_push_integer(ctx, -1); - } - } - hex_error(ctx, "Symbol 'ord' requires a string"); - FREE(ctx, item); - return 1; -} - -int hex_symbol_chr(hex_context_t *ctx) -{ - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type == HEX_TYPE_INTEGER) - { - if (item.data.int_value >= 0 && item.data.int_value < 128) - { - char str[2] = {(char)item.data.int_value, '\0'}; - return hex_push_string(ctx, str); - } - else - { - return hex_push_string(ctx, ""); - } - } - hex_error(ctx, "Symbol 'chr' requires an integer"); - FREE(ctx, item); - return 1; -} - -// Comparison symbols - -static int hex_equal(hex_item_t a, hex_item_t b) -{ - if (a.type == HEX_TYPE_INVALID || b.type == HEX_TYPE_INVALID) - { - return 0; - } - if (a.type == HEX_TYPE_NATIVE_SYMBOL || a.type == HEX_TYPE_USER_SYMBOL) - { - return (strcmp(a.token->value, b.token->value) == 0); - } - if (a.type != b.type) - { - return 0; - } - if (a.type == HEX_TYPE_INTEGER) - { - return a.data.int_value == b.data.int_value; - } - if (a.type == HEX_TYPE_STRING) - { - return (strcmp(a.data.str_value, b.data.str_value) == 0); - } - if (a.type == HEX_TYPE_QUOTATION) - { - if (a.quotation_size != b.quotation_size) - { - return 0; - } - else - { - for (int i = 0; i < a.quotation_size; i++) - { - if (!hex_equal(*a.data.quotation_value[i], *b.data.quotation_value[i])) - { - return 0; - } - } - return 1; - } - } - return 0; -} - -static int hex_is_type_symbol(hex_item_t *item) -{ - if (item->type == HEX_TYPE_USER_SYMBOL || item->type == HEX_TYPE_NATIVE_SYMBOL) - { - return 1; - } - return 0; -} - -static int hex_greater(hex_context_t *ctx, hex_item_t *a, hex_item_t *b, char *symbol) -{ - if (a->type == HEX_TYPE_INTEGER && b->type == HEX_TYPE_INTEGER) - { - return a->data.int_value > b->data.int_value; - } - else if (a->type == HEX_TYPE_STRING && b->type == HEX_TYPE_STRING) - { - return strcmp(a->data.str_value, b->data.str_value) > 0; - } - else if (a->type == HEX_TYPE_QUOTATION && b->type == HEX_TYPE_QUOTATION) - { - // Compare quotations lexicographically - size_t min_size = a->quotation_size < b->quotation_size ? a->quotation_size : b->quotation_size; - int is_greater = 0; - - for (size_t i = 0; i < min_size; i++) - { - hex_item_t *it_a = a->data.quotation_value[i]; - hex_item_t *it_b = b->data.quotation_value[i]; - - // Perform element-wise comparison - if (it_a->type != it_b->type && !(hex_is_type_symbol(it_a) && hex_is_type_symbol(it_b))) - { - // Mismatched types, return false - return 0; - } - - if (it_a->type == HEX_TYPE_INTEGER) - { - if (it_a->data.int_value != it_b->data.int_value) - { - is_greater = it_a->data.int_value > it_b->data.int_value; - break; - } - } - else if (it_a->type == HEX_TYPE_STRING) - { - int cmp = strcmp(it_a->data.str_value, it_b->data.str_value); - if (cmp != 0) - { - is_greater = cmp > 0; - break; - } - } - else if (hex_is_type_symbol(it_a)) - { - int cmp = strcmp(it_a->token->value, it_b->token->value); - if (cmp != 0) - { - is_greater = cmp > 0; - break; - } - } - else - { - // Mismatched types, return false - return 0; - } - } - - if (!is_greater) - { - // If all compared elements are equal, compare sizes - return a->quotation_size > b->quotation_size; - } - return is_greater; - } - else - { - hex_error(ctx, "'%s' symbol requires two integers, two strings, or two quotations", symbol); - return -1; - } -} - -int hex_symbol_equal(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if ((a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) || (a.type == HEX_TYPE_STRING && b.type == HEX_TYPE_STRING) || (a.type == HEX_TYPE_QUOTATION && b.type == HEX_TYPE_QUOTATION)) - { - return hex_push_integer(ctx, hex_equal(a, b)); - } - // Different types => false - return hex_push_integer(ctx, 0); -} - -int hex_symbol_notequal(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if ((a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) || (a.type == HEX_TYPE_STRING && b.type == HEX_TYPE_STRING) || (a.type == HEX_TYPE_QUOTATION && b.type == HEX_TYPE_QUOTATION)) - { - return hex_push_integer(ctx, !hex_equal(a, b)); - } - // Different types => true - return hex_push_integer(ctx, 1); -} - -int hex_symbol_greater(hex_context_t *ctx) -{ - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - hex_item_t *pa = &a; - hex_item_t *pb = &b; - hex_push_integer(ctx, hex_greater(ctx, pa, pb, ">")); - return 0; -} - -int hex_symbol_less(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - hex_item_t *pa = &a; - hex_item_t *pb = &b; - hex_push_integer(ctx, hex_greater(ctx, pb, pa, "<")); - return 0; -} - -int hex_symbol_greaterequal(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - hex_item_t *pa = &a; - hex_item_t *pb = &b; - hex_push_integer(ctx, hex_greater(ctx, pa, pb, ">") || hex_equal(a, b)); - return 0; -} - -int hex_symbol_lessequal(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - hex_item_t *pa = &a; - hex_item_t *pb = &b; - hex_push_integer(ctx, !hex_greater(ctx, pb, pa, "<") || hex_equal(a, b)); - return 0; -} - -// Boolean symbols - -int hex_symbol_and(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value && b.data.int_value); - } - hex_error(ctx, "Symbol 'and' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_or(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value || b.data.int_value); - } - hex_error(ctx, "Symbol 'or' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -int hex_symbol_not(hex_context_t *ctx) -{ - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - return 1; - } - if (a.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, !a.data.int_value); - } - hex_error(ctx, "Symbol 'not' requires an integer"); - FREE(ctx, a); - return 1; -} - -int hex_symbol_xor(hex_context_t *ctx) -{ - - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - return 1; - } - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) - { - return hex_push_integer(ctx, a.data.int_value ^ b.data.int_value); - } - hex_error(ctx, "Symbol 'xor' requires two integers"); - FREE(ctx, a); - FREE(ctx, b); - return 1; -} - -// Quotation and String (List) Symbols - -int hex_symbol_cat(hex_context_t *ctx) -{ - - POP(ctx, value); - if (value.type == HEX_TYPE_INVALID) - { - FREE(ctx, value); - return 1; // Failed to pop value - } - - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, list); - FREE(ctx, value); - return 1; // Failed to pop list - } - - int result = 0; - - if (list.type == HEX_TYPE_QUOTATION && value.type == HEX_TYPE_QUOTATION) - { - // Concatenate two quotations - size_t newSize = list.quotation_size + value.quotation_size; - hex_item_t **newQuotation = (hex_item_t **)realloc( - list.data.quotation_value, newSize * sizeof(hex_item_t *)); - if (!newQuotation) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - // Append items from the second quotation - for (size_t i = 0; i < (size_t)value.quotation_size; i++) - { - newQuotation[list.quotation_size + i] = value.data.quotation_value[i]; - } - - list.data.quotation_value = newQuotation; - list.quotation_size = newSize; - result = hex_push_quotation(ctx, list.data.quotation_value, newSize); - } - } - else if (list.type == HEX_TYPE_STRING && value.type == HEX_TYPE_STRING) - { - // Concatenate two strings - size_t newLength = strlen(list.data.str_value) + strlen(value.data.str_value) + 1; - char *newStr = (char *)malloc(newLength); - if (!newStr) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - strcpy(newStr, list.data.str_value); - strcat(newStr, value.data.str_value); - result = hex_push_string(ctx, newStr); - } - } - else - { - hex_error(ctx, "Symbol 'cat' requires two quotations or two strings"); - result = 1; - } - - // Free resources if the operation fails - if (result != 0) - { - FREE(ctx, list); - FREE(ctx, value); - } - - return result; -} - -int hex_symbol_len(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - int result = 0; - if (item.type == HEX_TYPE_QUOTATION) - { - result = hex_push_integer(ctx, item.quotation_size); - } - else if (item.type == HEX_TYPE_STRING) - { - result = hex_push_integer(ctx, strlen(item.data.str_value)); - } - else - { - hex_error(ctx, "Symbol 'len' requires a quotation or a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, item); - } - return result; -} - -int hex_symbol_get(hex_context_t *ctx) -{ - - POP(ctx, index); - if (index.type == HEX_TYPE_INVALID) - { - FREE(ctx, index); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, list); - FREE(ctx, index); - return 1; - } - int result = 0; - if (list.type == HEX_TYPE_QUOTATION) - { - if (index.type != HEX_TYPE_INTEGER) - { - hex_error(ctx, "Index must be an integer"); - result = 1; - } - else if (index.data.int_value < 0 || index.data.int_value >= list.quotation_size) - { - hex_error(ctx, "Index out of range"); - result = 1; - } - else - { - result = hex_push(ctx, *list.data.quotation_value[index.data.int_value]); - } - } - else if (list.type == HEX_TYPE_STRING) - { - if (index.type != HEX_TYPE_INTEGER) - { - hex_error(ctx, "Index must be an integer"); - result = 1; - } - else if (index.data.int_value < 0 || index.data.int_value >= (int)strlen(list.data.str_value)) - { - hex_error(ctx, "Index out of range"); - result = 1; - } - else - { - char str[2] = {list.data.str_value[index.data.int_value], '\0'}; - result = hex_push_string(ctx, str); - } - } - else - { - hex_error(ctx, "Symbol 'get' requires a quotation or a string"); - result = 1; - } - if (result != 0) - { - - FREE(ctx, list); - FREE(ctx, index); - } - return result; -} - -int hex_symbol_index(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, list); - FREE(ctx, item); - return 1; - } - int result = -1; - if (list.type == HEX_TYPE_QUOTATION) - { - for (int i = 0; i < list.quotation_size; i++) - { - if (hex_equal(*list.data.quotation_value[i], item)) - { - result = i; - break; - } - } - } - else if (list.type == HEX_TYPE_STRING) - { - char *ptr = strstr(list.data.str_value, item.data.str_value); - if (ptr) - { - result = ptr - list.data.str_value; - } - } - else - { - hex_error(ctx, "Symbol 'index' requires a quotation or a string"); - FREE(ctx, list); - FREE(ctx, item); - return 1; - } - return hex_push_integer(ctx, result); -} - -// String symbols - -int hex_symbol_join(hex_context_t *ctx) -{ - - POP(ctx, separator); - if (separator.type == HEX_TYPE_INVALID) - { - FREE(ctx, separator); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, list); - FREE(ctx, separator); - return 1; - } - int result = 0; - if (list.type == HEX_TYPE_QUOTATION && separator.type == HEX_TYPE_STRING) - { - int length = 0; - for (int i = 0; i < list.quotation_size; i++) - { - if (list.data.quotation_value[i]->type == HEX_TYPE_STRING) - { - length += strlen(list.data.quotation_value[i]->data.str_value); - } - else - { - hex_error(ctx, "Quotation must contain only strings"); - FREE(ctx, list); - FREE(ctx, separator); - return 1; - } - } - if (result == 0) - { - length += (list.quotation_size - 1) * strlen(separator.data.str_value); - char *newStr = (char *)malloc(length + 1); - if (!newStr) - { - hex_error(ctx, "Memory allocation failed"); - FREE(ctx, list); - FREE(ctx, separator); - return 1; - } - newStr[0] = '\0'; - for (int i = 0; i < list.quotation_size; i++) - { - strcat(newStr, list.data.quotation_value[i]->data.str_value); - if (i < list.quotation_size - 1) - { - strcat(newStr, separator.data.str_value); - } - } - result = hex_push_string(ctx, newStr); - } - } - else - { - hex_error(ctx, "Symbol 'join' requires a quotation and a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, list); - FREE(ctx, separator); - } - return result; -} - -int hex_symbol_split(hex_context_t *ctx) -{ - POP(ctx, separator); - if (separator.type == HEX_TYPE_INVALID) - { - FREE(ctx, separator); - return 1; - } - POP(ctx, str); - if (str.type == HEX_TYPE_INVALID) - { - FREE(ctx, str); - FREE(ctx, separator); - return 1; - } - int result = 0; - if (str.type == HEX_TYPE_STRING && separator.type == HEX_TYPE_STRING) - { - if (strlen(separator.data.str_value) == 0) - { - // Separator is an empty string: split into individual characters - int size = strlen(str.data.str_value); - hex_item_t **quotation = (hex_item_t **)malloc(size * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - for (int i = 0; i < size; i++) - { - quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (!quotation[i]) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - break; - } - quotation[i]->type = HEX_TYPE_STRING; - quotation[i]->data.str_value = (char *)malloc(2); // Allocate 2 bytes: 1 for the character and 1 for null terminator - if (!quotation[i]->data.str_value) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - break; - } - quotation[i]->data.str_value[0] = str.data.str_value[i]; // Copy the single character - quotation[i]->data.str_value[1] = '\0'; // Null-terminate the string - } - if (result == 0) - { - result = hex_push_quotation(ctx, quotation, size); - } - } - } - else - { - // Separator is not empty: split as usual - char *token = strtok(str.data.str_value, separator.data.str_value); - int capacity = 2; - int size = 0; - hex_item_t **quotation = (hex_item_t **)malloc(capacity * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - while (token) - { - if (size >= capacity) - { - capacity *= 2; - quotation = (hex_item_t **)realloc(quotation, capacity * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - break; - } - } - quotation[size] = (hex_item_t *)malloc(sizeof(hex_item_t)); - quotation[size]->type = HEX_TYPE_STRING; - quotation[size]->data.str_value = strdup(token); - size++; - token = strtok(NULL, separator.data.str_value); - } - if (result == 0) - { - result = hex_push_quotation(ctx, quotation, size); - } - } - } - } - else - { - hex_error(ctx, "Symbol 'split' requires two strings"); - result = 1; - } - if (result != 0) - { - FREE(ctx, str); - FREE(ctx, separator); - } - return result; -} - -int hex_symbol_replace(hex_context_t *ctx) -{ - - POP(ctx, replacement); - if (replacement.type == HEX_TYPE_INVALID) - { - FREE(ctx, replacement); - return 1; - } - POP(ctx, search); - if (search.type == HEX_TYPE_INVALID) - { - FREE(ctx, search); - FREE(ctx, replacement); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, list); - FREE(ctx, search); - FREE(ctx, replacement); - return 1; - } - int result = 0; - if (list.type == HEX_TYPE_STRING && search.type == HEX_TYPE_STRING && replacement.type == HEX_TYPE_STRING) - { - char *str = list.data.str_value; - char *find = search.data.str_value; - char *replace = replacement.data.str_value; - char *ptr = strstr(str, find); - if (ptr) - { - int findLen = strlen(find); - int replaceLen = strlen(replace); - int newLen = strlen(str) - findLen + replaceLen + 1; - char *newStr = (char *)malloc(newLen); - if (!newStr) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - strncpy(newStr, str, ptr - str); - strcpy(newStr + (ptr - str), replace); - strcpy(newStr + (ptr - str) + replaceLen, ptr + findLen); - result = hex_push_string(ctx, newStr); - } - } - else - { - result = hex_push_string(ctx, str); - } - } - else - { - hex_error(ctx, "Symbol 'replace' requires three strings"); - result = 1; - } - if (result != 0) - { - FREE(ctx, list); - FREE(ctx, search); - FREE(ctx, replacement); - } - return result; -} - -// File symbols - -int hex_symbol_read(hex_context_t *ctx) -{ - - POP(ctx, filename); - if (filename.type == HEX_TYPE_INVALID) - { - FREE(ctx, filename); - return 1; - } - int result = 0; - if (filename.type == HEX_TYPE_STRING) - { - FILE *file = fopen(filename.data.str_value, "r"); - if (!file) - { - hex_error(ctx, "Could not open file for reading: %s", filename.data.str_value); - result = 1; - } - else - { - fseek(file, 0, SEEK_END); - long length = ftell(file); - fseek(file, 0, SEEK_SET); - - char *buffer = (char *)malloc(length + 1); - if (!buffer) - { - hex_error(ctx, "Memory allocation failed"); - result = 1; - } - else - { - size_t bytesRead = fread(buffer, 1, length, file); - buffer[bytesRead] = '\0'; - result = hex_push_string(ctx, buffer); - free(buffer); - } - fclose(file); - } - } - else - { - hex_error(ctx, "Symbol 'read' requires a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, filename); - } - return result; -} - -int hex_symbol_write(hex_context_t *ctx) -{ - - POP(ctx, filename); - if (filename.type == HEX_TYPE_INVALID) - { - FREE(ctx, filename); - return 1; - } - POP(ctx, data); - if (data.type == HEX_TYPE_INVALID) - { - FREE(ctx, data); - FREE(ctx, filename); - return 1; - } - int result = 0; - if (filename.type == HEX_TYPE_STRING) - { - if (data.type == HEX_TYPE_STRING) - { - FILE *file = fopen(filename.data.str_value, "w"); - if (file) - { - fputs(data.data.str_value, file); - fclose(file); - result = 0; - } - else - { - hex_error(ctx, "Could not open file for writing: %s", filename.data.str_value); - result = 1; - } - } - else - { - hex_error(ctx, "Symbol 'write' requires a string"); - result = 1; - } - } - else - { - hex_error(ctx, "Symbol 'write' requires a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, data); - FREE(ctx, filename); - } - return result; -} - -int hex_symbol_append(hex_context_t *ctx) -{ - - POP(ctx, filename); - if (filename.type == HEX_TYPE_INVALID) - { - FREE(ctx, filename); - return 1; - } - POP(ctx, data); - if (data.type == HEX_TYPE_INVALID) - { - FREE(ctx, data); - FREE(ctx, filename); - return 1; - } - int result = 0; - if (filename.type == HEX_TYPE_STRING) - { - if (data.type == HEX_TYPE_STRING) - { - FILE *file = fopen(filename.data.str_value, "a"); - if (file) - { - fputs(data.data.str_value, file); - fclose(file); - result = 0; - } - else - { - hex_error(ctx, "Could not open file for appending: %s", filename.data.str_value); - result = 1; - } - } - else - { - hex_error(ctx, "Symbol 'append' requires a string"); - result = 1; - } - } - else - { - hex_error(ctx, "Symbol 'append' requires a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, data); - FREE(ctx, filename); - } - return result; -} - -// Shell symbols - -int hex_symbol_args(hex_context_t *ctx) -{ - - hex_item_t **quotation = (hex_item_t **)malloc(ctx->argc * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - return 1; - } - else - { - for (int i = 0; i < ctx->argc; i++) - { - quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); - quotation[i]->type = HEX_TYPE_STRING; - quotation[i]->data.str_value = ctx->argv[i]; - } - if (hex_push_quotation(ctx, quotation, ctx->argc) != 0) - { - hex_free_list(ctx, quotation, ctx->argc); - return 1; - } - } - return 0; -} - -int hex_symbol_exit(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (item.type != HEX_TYPE_INTEGER) - { - hex_error(ctx, "Exit status must be an integer"); - FREE(ctx, item); - return 1; - } - int exit_status = item.data.int_value; - exit(exit_status); - return 0; // This line will never be reached, but it's here to satisfy the return type -} - -int hex_symbol_exec(hex_context_t *ctx) -{ - - POP(ctx, command); - if (command.type == HEX_TYPE_INVALID) - { - FREE(ctx, command); - return 1; - } - int result = 0; - if (command.type == HEX_TYPE_STRING) - { - int status = system(command.data.str_value); - result = hex_push_integer(ctx, status); - } - else - { - hex_error(ctx, "Symbol 'exec' requires a string"); - result = 1; - } - if (result != 0) - { - FREE(ctx, command); - } - return result; -} - -int hex_symbol_run(hex_context_t *ctx) -{ - - POP(ctx, command); - if (command.type == HEX_TYPE_INVALID) - { - FREE(ctx, command); - return 1; - } - if (command.type != HEX_TYPE_STRING) - { - hex_error(ctx, "Symbol 'run' requires a string"); - FREE(ctx, command); - return 1; - } - - char output[8192] = ""; - char error[8192] = ""; - int return_code = 0; - -#ifdef _WIN32 - // Windows implementation - HANDLE hOutputRead, hOutputWrite; - HANDLE hErrorRead, hErrorWrite; - SECURITY_ATTRIBUTES sa; - PROCESS_INFORMATION pi; - STARTUPINFO si; - DWORD exitCode; - - sa.nLength = sizeof(SECURITY_ATTRIBUTES); - sa.lpSecurityDescriptor = NULL; - sa.bInheritHandle = TRUE; - - // Create pipes for capturing stdout and stderr - if (!CreatePipe(&hOutputRead, &hOutputWrite, &sa, 0) || !CreatePipe(&hErrorRead, &hErrorWrite, &sa, 0)) - { - hex_error(ctx, "Failed to create pipes"); - FREE(ctx, command); - return 1; - } - - // Set up STARTUPINFO structure - ZeroMemory(&si, sizeof(STARTUPINFO)); - si.cb = sizeof(STARTUPINFO); - si.hStdOutput = hOutputWrite; - si.hStdError = hErrorWrite; - si.dwFlags |= STARTF_USESTDHANDLES; - - // Create the child process - if (!CreateProcess(NULL, command.data.str_value, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi)) - { - hex_error(ctx, "Failed to create process"); - FREE(ctx, command); - return 1; - } - - // Close write ends of the pipes - CloseHandle(hOutputWrite); - CloseHandle(hErrorWrite); - - // Read stdout - DWORD bytesRead; - char buffer[1024]; - while (ReadFile(hOutputRead, buffer, sizeof(buffer) - 1, &bytesRead, NULL) && bytesRead > 0) - { - buffer[bytesRead] = '\0'; - strcat(output, buffer); - } - - // Read stderr - while (ReadFile(hErrorRead, buffer, sizeof(buffer) - 1, &bytesRead, NULL) && bytesRead > 0) - { - buffer[bytesRead] = '\0'; - strcat(error, buffer); - } - - // Wait for the child process to finish and get the return code - WaitForSingleObject(pi.hProcess, INFINITE); - GetExitCodeProcess(pi.hProcess, &exitCode); - return_code = exitCode; - - // Close handles - CloseHandle(pi.hProcess); - CloseHandle(pi.hThread); - CloseHandle(hOutputRead); - CloseHandle(hErrorRead); -#else - // POSIX implementation (Linux/macOS) - int stdout_pipe[2]; - int stderr_pipe[2]; - if (pipe(stdout_pipe) != 0 || pipe(stderr_pipe) != 0) - { - hex_error(ctx, "Failed to create pipes"); - FREE(ctx, command); - return 1; - } - - pid_t pid = fork(); - if (pid == -1) - { - hex_error(ctx, "Failed to fork process"); - FREE(ctx, command); - return 1; - } - else if (pid == 0) - { - // Child process - close(stdout_pipe[0]); - close(stderr_pipe[0]); - dup2(stdout_pipe[1], STDOUT_FILENO); - dup2(stderr_pipe[1], STDERR_FILENO); - execl("/bin/sh", "sh", "-c", command.data.str_value, (char *)NULL); - exit(1); - } - else - { - // Parent process - close(stdout_pipe[1]); - close(stderr_pipe[1]); - - // Read stdout - FILE *stdout_fp = fdopen(stdout_pipe[0], "r"); - char path[1035]; - while (fgets(path, sizeof(path), stdout_fp) != NULL) - { - strcat(output, path); - } - fclose(stdout_fp); - - // Read stderr - FILE *stderr_fp = fdopen(stderr_pipe[0], "r"); - while (fgets(path, sizeof(path), stderr_fp) != NULL) - { - strcat(error, path); - } - fclose(stderr_fp); - - // Wait for child process to finish and get the return code - int status; - waitpid(pid, &status, 0); - return_code = WEXITSTATUS(status); - } -#endif - - // Push the return code, output, and error as a quotation - hex_item_t **quotation = (hex_item_t **)malloc(3 * sizeof(hex_item_t *)); - quotation[0] = (hex_item_t *)malloc(sizeof(hex_item_t)); - quotation[0]->type = HEX_TYPE_INTEGER; - quotation[0]->data.int_value = return_code; - - quotation[1] = (hex_item_t *)malloc(sizeof(hex_item_t)); - quotation[1]->type = HEX_TYPE_STRING; - quotation[1]->data.str_value = strdup(output); - - quotation[2] = (hex_item_t *)malloc(sizeof(hex_item_t)); - quotation[2]->type = HEX_TYPE_STRING; - quotation[2]->data.str_value = strdup(error); - - return hex_push_quotation(ctx, quotation, 3); -} - -// Control flow symbols - -int hex_symbol_if(hex_context_t *ctx) -{ - - POP(ctx, elseBlock); - if (elseBlock.type == HEX_TYPE_INVALID) - { - FREE(ctx, elseBlock); - return 1; - } - POP(ctx, thenBlock); - if (thenBlock.type == HEX_TYPE_INVALID) - { - FREE(ctx, thenBlock); - FREE(ctx, elseBlock); - return 1; - } - POP(ctx, condition); - if (condition.type == HEX_TYPE_INVALID) - { - FREE(ctx, condition); - FREE(ctx, thenBlock); - FREE(ctx, elseBlock); - return 1; - } - if (condition.type != HEX_TYPE_QUOTATION || thenBlock.type != HEX_TYPE_QUOTATION || elseBlock.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'if' symbol requires three quotations"); - return 1; - } - else - { - for (int i = 0; i < condition.quotation_size; i++) - { - if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) - { - FREE(ctx, condition); - FREE(ctx, thenBlock); - FREE(ctx, elseBlock); - return 1; - } - } - POP(ctx, evalResult); - if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) - { - for (int i = 0; i < thenBlock.quotation_size; i++) - { - if (hex_push(ctx, *thenBlock.data.quotation_value[i]) != 0) - { - FREE(ctx, condition); - FREE(ctx, thenBlock); - FREE(ctx, elseBlock); - return 1; - } - } - } - else - { - for (int i = 0; i < elseBlock.quotation_size; i++) - { - if (hex_push(ctx, *elseBlock.data.quotation_value[i]) != 0) - { - FREE(ctx, condition); - FREE(ctx, thenBlock); - FREE(ctx, elseBlock); - return 1; - } - } - } - } - return 0; -} - -int hex_symbol_when(hex_context_t *ctx) -{ - - POP(ctx, action); - if (action.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - return 1; - } - POP(ctx, condition); - if (condition.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - FREE(ctx, condition); - return 1; - } - int result = 0; - if (condition.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'when' symbol requires two quotations"); - result = 1; - } - else - { - for (int i = 0; i < condition.quotation_size; i++) - { - if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) - { - result = 1; - break; // Break if pushing the item failed - } - } - POP(ctx, evalResult); - if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) - { - for (int i = 0; i < action.quotation_size; i++) - { - if (hex_push(ctx, *action.data.quotation_value[i]) != 0) - { - result = 1; - break; - } - } - } - } - if (result != 0) - { - FREE(ctx, action); - FREE(ctx, condition); - } - return result; -} - -int hex_symbol_while(hex_context_t *ctx) -{ - - POP(ctx, action); - if (action.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - return 1; - } - POP(ctx, condition); - if (condition.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - FREE(ctx, condition); - return 1; - } - if (condition.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'while' symbol requires two quotations"); - FREE(ctx, action); - FREE(ctx, condition); - return 1; - } - else - { - while (1) - { - for (int i = 0; i < condition.quotation_size; i++) - { - if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) - { - FREE(ctx, action); - FREE(ctx, condition); - return 1; - } - } - POP(ctx, evalResult); - if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value == 0) - { - break; - } - for (int i = 0; i < action.quotation_size; i++) - { - if (hex_push(ctx, *action.data.quotation_value[i]) != 0) - { - FREE(ctx, action); - FREE(ctx, condition); - return 1; - } - } - } - } - return 0; -} - -int hex_symbol_each(hex_context_t *ctx) -{ - - POP(ctx, action); - if (action.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'each' symbol requires two quotations"); - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - else - { - for (int i = 0; i < list.quotation_size; i++) - { - if (hex_push(ctx, *list.data.quotation_value[i]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - for (int j = 0; j < action.quotation_size; j++) - { - if (hex_push(ctx, *action.data.quotation_value[j]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - } - } - } - return 0; -} - -int hex_symbol_error(hex_context_t *ctx) -{ - - char *message = strdup(ctx->error); - ctx->error[0] = '\0'; - return hex_push_string(ctx, message); -} - -int hex_symbol_try(hex_context_t *ctx) -{ - - POP(ctx, catch_block); - if (catch_block.type == HEX_TYPE_INVALID) - { - FREE(ctx, catch_block); - return 1; - } - POP(ctx, try_block); - if (try_block.type == HEX_TYPE_INVALID) - { - FREE(ctx, catch_block); - FREE(ctx, try_block); - return 1; - } - if (try_block.type != HEX_TYPE_QUOTATION || catch_block.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'try' symbol requires two quotations"); - FREE(ctx, catch_block); - FREE(ctx, try_block); - return 1; - } - else - { - char prevError[256]; - strncpy(prevError, ctx->error, sizeof(ctx->error)); - ctx->error[0] = '\0'; - - ctx->settings.errors_enabled = 0; - for (int i = 0; i < try_block.quotation_size; i++) - { - if (hex_push(ctx, *try_block.data.quotation_value[i]) != 0) - { - ctx->settings.errors_enabled = 1; - } - } - ctx->settings.errors_enabled = 1; - - if (strcmp(ctx->error, "")) - { - for (int i = 0; i < catch_block.quotation_size; i++) - { - if (hex_push(ctx, *catch_block.data.quotation_value[i]) != 0) - { - FREE(ctx, catch_block); - FREE(ctx, try_block); - return 1; - } - } - } - - strncpy(ctx->error, prevError, sizeof(ctx->error)); - } - return 0; -} - -// Quotation symbols - -int hex_symbol_q(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - - hex_item_t *quotation = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - FREE(ctx, item); - return 1; - } - - *quotation = item; - - hex_item_t result; - result.type = HEX_TYPE_QUOTATION; - result.data.quotation_value = (hex_item_t **)malloc(sizeof(hex_item_t *)); - if (!result.data.quotation_value) - { - FREE(ctx, item); - free(quotation); - hex_error(ctx, "Memory allocation failed"); - return 1; - } - - result.data.quotation_value[0] = quotation; - result.quotation_size = 1; - - if (PUSH(ctx, result) != 0) - { - FREE(ctx, item); - free(quotation); - free(result.data.quotation_value); - return 1; - } - - return 0; -} - -int hex_symbol_map(hex_context_t *ctx) -{ - - POP(ctx, action); - if (action.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'map' symbol requires two quotations"); - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - else - { - hex_item_t **quotation = (hex_item_t **)malloc(list.quotation_size * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - for (int i = 0; i < list.quotation_size; i++) - { - if (hex_push(ctx, *list.data.quotation_value[i]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, i); - return 1; - } - for (int j = 0; j < action.quotation_size; j++) - { - if (hex_push(ctx, *action.data.quotation_value[j]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, i); - return 1; - } - } - quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); - *quotation[i] = hex_pop(ctx); - } - if (hex_push_quotation(ctx, quotation, list.quotation_size) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, list.quotation_size); - return 1; - } - } - return 0; -} - -int hex_symbol_filter(hex_context_t *ctx) -{ - - POP(ctx, action); - if (action.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - return 1; - } - POP(ctx, list); - if (list.type == HEX_TYPE_INVALID) - { - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) - { - hex_error(ctx, "'filter' symbol requires two quotations"); - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - else - { - hex_item_t **quotation = (hex_item_t **)malloc(list.quotation_size * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - FREE(ctx, action); - FREE(ctx, list); - return 1; - } - int count = 0; - for (int i = 0; i < list.quotation_size; i++) - { - if (hex_push(ctx, *list.data.quotation_value[i]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, count); - return 1; - } - for (int j = 0; j < action.quotation_size; j++) - { - if (hex_push(ctx, *action.data.quotation_value[j]) != 0) - { - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, count); - return 1; - } - } - POP(ctx, evalResult); - if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) - { - quotation[count] = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (!quotation[count]) - { - hex_error(ctx, "Memory allocation failed"); - FREE(ctx, action); - FREE(ctx, list); - hex_free_list(ctx, quotation, count); - return 1; - } - *quotation[count] = *list.data.quotation_value[i]; - count++; - } - } - if (hex_push_quotation(ctx, quotation, count) != 0) - { - hex_error(ctx, "An error occurred while filtering the list"); - FREE(ctx, action); - FREE(ctx, list); - for (int i = 0; i < count; i++) - { - FREE(ctx, *quotation[i]); - } - return 1; - } - } - return 0; -} - -// Stack manipulation symbols - -int hex_symbol_swap(hex_context_t *ctx) -{ - - POP(ctx, a); - if (a.type == HEX_TYPE_INVALID) - { - FREE(ctx, a); - return 1; - } - POP(ctx, b); - if (b.type == HEX_TYPE_INVALID) - { - FREE(ctx, b); - FREE(ctx, a); - return 1; - } - if (PUSH(ctx, a) != 0) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - if (PUSH(ctx, b) != 0) - { - FREE(ctx, a); - FREE(ctx, b); - return 1; - } - return 0; -} - -int hex_symbol_dup(hex_context_t *ctx) -{ - - POP(ctx, item); - if (item.type == HEX_TYPE_INVALID) - { - FREE(ctx, item); - return 1; - } - if (PUSH(ctx, item) == 0 && PUSH(ctx, item) == 0) - { - return 0; - } - FREE(ctx, item); - return 1; -} - -int hex_symbol_stack(hex_context_t *ctx) -{ - - hex_item_t **quotation = (hex_item_t **)malloc((ctx->stack.top + 1) * sizeof(hex_item_t *)); - if (!quotation) - { - hex_error(ctx, "Memory allocation failed"); - return 1; - } - int count = 0; - for (int i = 0; i <= ctx->stack.top; i++) - { - quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (!quotation[i]) - { - hex_error(ctx, "Memory allocation failed"); - hex_free_list(ctx, quotation, count); - return 1; - } - *quotation[i] = ctx->stack.entries[i]; - //*quotation[i] = HEX_STACK[i]; - count++; - } - - if (hex_push_quotation(ctx, quotation, ctx->stack.top + 1) != 0) - { - hex_error(ctx, "An error occurred while fetching stack contents"); - hex_free_list(ctx, quotation, count); - return 1; - } - return 0; -} - -int hex_symbol_clear(hex_context_t *ctx) -{ - - while (ctx->stack.top >= 0) - { - FREE(ctx, ctx->stack.entries[ctx->stack.top--]); - // FREE(ctx, HEX_STACK[HEX_TOP--]); - } - ctx->stack.top = -1; - return 0; -} - -int hex_symbol_pop(hex_context_t *ctx) -{ - - POP(ctx, item); - FREE(ctx, item); - return 0; -} - -//////////////////////////////////////// -// Native Symbol Registration // -//////////////////////////////////////// - -void hex_register_symbols(hex_context_t *ctx) -{ - hex_set_native_symbol(ctx, ":", hex_symbol_store); - hex_set_native_symbol(ctx, "#", hex_symbol_free); - hex_set_native_symbol(ctx, "type", hex_symbol_type); - hex_set_native_symbol(ctx, ".", hex_symbol_i); - hex_set_native_symbol(ctx, "!", hex_symbol_eval); - hex_set_native_symbol(ctx, "puts", hex_symbol_puts); - hex_set_native_symbol(ctx, "warn", hex_symbol_warn); - hex_set_native_symbol(ctx, "print", hex_symbol_print); - hex_set_native_symbol(ctx, "gets", hex_symbol_gets); - hex_set_native_symbol(ctx, "+", hex_symbol_add); - hex_set_native_symbol(ctx, "-", hex_symbol_subtract); - hex_set_native_symbol(ctx, "*", hex_symbol_multiply); - hex_set_native_symbol(ctx, "/", hex_symbol_divide); - hex_set_native_symbol(ctx, "%", hex_symbol_modulo); - hex_set_native_symbol(ctx, "&", hex_symbol_bitand); - hex_set_native_symbol(ctx, "|", hex_symbol_bitor); - hex_set_native_symbol(ctx, "^", hex_symbol_bitxor); - hex_set_native_symbol(ctx, "~", hex_symbol_bitnot); - hex_set_native_symbol(ctx, "<<", hex_symbol_shiftleft); - hex_set_native_symbol(ctx, ">>", hex_symbol_shiftright); - hex_set_native_symbol(ctx, "int", hex_symbol_int); - hex_set_native_symbol(ctx, "str", hex_symbol_str); - hex_set_native_symbol(ctx, "dec", hex_symbol_dec); - hex_set_native_symbol(ctx, "hex", hex_symbol_hex); - hex_set_native_symbol(ctx, "chr", hex_symbol_chr); - hex_set_native_symbol(ctx, "ord", hex_symbol_ord); - hex_set_native_symbol(ctx, "==", hex_symbol_equal); - hex_set_native_symbol(ctx, "!=", hex_symbol_notequal); - hex_set_native_symbol(ctx, ">", hex_symbol_greater); - hex_set_native_symbol(ctx, "<", hex_symbol_less); - hex_set_native_symbol(ctx, ">=", hex_symbol_greaterequal); - hex_set_native_symbol(ctx, "<=", hex_symbol_lessequal); - hex_set_native_symbol(ctx, "and", hex_symbol_and); - hex_set_native_symbol(ctx, "or", hex_symbol_or); - hex_set_native_symbol(ctx, "not", hex_symbol_not); - hex_set_native_symbol(ctx, "xor", hex_symbol_xor); - hex_set_native_symbol(ctx, "cat", hex_symbol_cat); - hex_set_native_symbol(ctx, "len", hex_symbol_len); - hex_set_native_symbol(ctx, "get", hex_symbol_get); - hex_set_native_symbol(ctx, "index", hex_symbol_index); - hex_set_native_symbol(ctx, "join", hex_symbol_join); - hex_set_native_symbol(ctx, "split", hex_symbol_split); - hex_set_native_symbol(ctx, "replace", hex_symbol_replace); - hex_set_native_symbol(ctx, "read", hex_symbol_read); - hex_set_native_symbol(ctx, "write", hex_symbol_write); - hex_set_native_symbol(ctx, "append", hex_symbol_append); - hex_set_native_symbol(ctx, "args", hex_symbol_args); - hex_set_native_symbol(ctx, "exit", hex_symbol_exit); - hex_set_native_symbol(ctx, "exec", hex_symbol_exec); - hex_set_native_symbol(ctx, "run", hex_symbol_run); - hex_set_native_symbol(ctx, "if", hex_symbol_if); - hex_set_native_symbol(ctx, "when", hex_symbol_when); - hex_set_native_symbol(ctx, "while", hex_symbol_while); - hex_set_native_symbol(ctx, "each", hex_symbol_each); - hex_set_native_symbol(ctx, "error", hex_symbol_error); - hex_set_native_symbol(ctx, "try", hex_symbol_try); - hex_set_native_symbol(ctx, "'", hex_symbol_q); - hex_set_native_symbol(ctx, "map", hex_symbol_map); - hex_set_native_symbol(ctx, "filter", hex_symbol_filter); - hex_set_native_symbol(ctx, "swap", hex_symbol_swap); - hex_set_native_symbol(ctx, "dup", hex_symbol_dup); - hex_set_native_symbol(ctx, "stack", hex_symbol_stack); - hex_set_native_symbol(ctx, "clear", hex_symbol_clear); - hex_set_native_symbol(ctx, "pop", hex_symbol_pop); -} - -//////////////////////////////////////// -// Hex Interpreter Implementation // -//////////////////////////////////////// - -hex_context_t hex_init() -{ - hex_context_t context; - context.argc = 0; - context.argv = NULL; - context.registry.size = 0; - context.docs.size = 0; - context.stack.top = -1; - context.stack_trace.start = 0; - context.stack_trace.size = 0; - context.settings.debugging_enabled = 0; - context.settings.errors_enabled = 1; - context.settings.stack_trace_enabled = 1; - return context; -} - -int hex_interpret(hex_context_t *ctx, const char *code, const char *filename, int line, int column) -{ - - const char *input = code; - hex_file_position_t position = {filename, line, column}; - hex_token_t *token = hex_next_token(ctx, &input, &position); - - while (token != NULL && token->type != HEX_TOKEN_INVALID) - { - int result = 0; - if (token->type == HEX_TOKEN_INTEGER) - { - result = hex_push_integer(ctx, hex_parse_integer(token->value)); - } - else if (token->type == HEX_TOKEN_STRING) - { - result = hex_push_string(ctx, token->value); - } - else if (token->type == HEX_TOKEN_SYMBOL) - { - token->position.filename = strdup(filename); - result = hex_push_symbol(ctx, token); - } - else if (token->type == HEX_TOKEN_QUOTATION_END) - { - hex_error(ctx, "(%d,%d) Unexpected end of quotation", position.line, position.column); - result = 1; - } - else if (token->type == HEX_TOKEN_QUOTATION_START) - { - hex_item_t *quotationItem = (hex_item_t *)malloc(sizeof(hex_item_t)); - if (hex_parse_quotation(ctx, &input, quotationItem, &position) != 0) - { - hex_error(ctx, "(%d,%d) Failed to parse quotation", position.line, position.column); - result = 1; - } - else - { - hex_item_t **quotation = quotationItem->data.quotation_value; - int quotation_size = quotationItem->quotation_size; - result = hex_push_quotation(ctx, quotation, quotation_size); - } - free(quotationItem); - } - - if (result != 0) - { - hex_free_token(token); - print_stack_trace(ctx); - return result; - } - - token = hex_next_token(ctx, &input, &position); - } - if (token != NULL && token->type == HEX_TOKEN_INVALID) - { - token->position.filename = strdup(filename); - add_to_stack_trace(ctx, token); - print_stack_trace(ctx); - return 1; - } - hex_free_token(token); - return 0; -} - -// Read a file into a buffer -char *hex_read_file(hex_context_t *ctx, const char *filename) -{ - FILE *file = fopen(filename, "r"); - if (file == NULL) - { - hex_error(ctx, "Failed to open file: %s", filename); - return NULL; - } - - // Allocate an initial buffer - int bufferSize = 1024; // Start with a 1 KB buffer - char *content = (char *)malloc(bufferSize); - if (content == NULL) - { - hex_error(ctx, "Memory allocation failed"); - fclose(file); - return NULL; - } - - int bytesReadTotal = 0; - int bytesRead = 0; - - // Handle hashbang if present - char hashbangLine[1024]; - if (fgets(hashbangLine, sizeof(hashbangLine), file) != NULL) - { - if (strncmp(hashbangLine, "#!", 2) != 0) - { - // Not a hashbang line, reset file pointer to the beginning - fseek(file, 0, SEEK_SET); - ctx->hashbang = 0; - } - else - { - ctx->hashbang = 1; - } - } - - while ((bytesRead = fread(content + bytesReadTotal, 1, bufferSize - bytesReadTotal, file)) > 0) - { - bytesReadTotal += bytesRead; - - // Resize the buffer if necessary - if (bytesReadTotal == bufferSize) - { - bufferSize *= 2; // Double the buffer size - char *temp = (char *)realloc(content, bufferSize); - if (temp == NULL) - { - hex_error(ctx, "Memory reallocation failed"); - free(content); - fclose(file); - return NULL; - } - content = temp; - } - } - - if (ferror(file)) - { - hex_error(ctx, "Error reading the file"); - free(content); - fclose(file); - return NULL; - } - - // Null-terminate the content - char *finalContent = (char *)realloc(content, bytesReadTotal + 1); - if (finalContent == NULL) - { - hex_error(ctx, "Final memory allocation failed"); - free(content); - fclose(file); - return NULL; - } - content = finalContent; - content[bytesReadTotal] = '\0'; - - fclose(file); - return content; -} - -#if defined(__EMSCRIPTEN__) && defined(BROWSER) -static void prompt() -{ - // no prompt needed on browser -} -#elif defined(__EMSCRIPTEN__) && !defined(BROWSER) -static void prompt() -{ - printf(">\n"); -} -#else -static void prompt() -{ - printf("> "); - fflush(stdout); -} -#endif - -#if defined(__EMSCRIPTEN__) -static void do_repl(void *v_ctx) -{ - hex_context_t *ctx = (hex_context_t *)v_ctx; - prompt(); - char line[1024]; - char *p = line; - p = em_fgets(line, 1024); - if (!p) - { - printf("Error reading output"); - return; - } - // Normalize line endings (remove trailing \r\n or \n) - line[strcspn(line, "\r\n")] = '\0'; - - // Tokenize and process the input - hex_interpret(ctx, line, "<repl>", 1, 1); - // Print the top item of the stack - if (ctx->stack.top >= 0) - { - hex_print_item(stdout, ctx->stack.entries[ctx->stack.top]); - // hex_print_item(stdout, HEX_STACK[HEX_TOP]); - printf("\n"); - } - return; -} - -#else - -static int do_repl(void *v_ctx) -{ - hex_context_t *ctx = (hex_context_t *)v_ctx; - char line[1024]; - prompt(); - if (fgets(line, sizeof(line), stdin) == NULL) - { - printf("\n"); // Handle EOF (Ctrl+D) - return 1; - } - // Normalize line endings (remove trailing \r\n or \n) - line[strcspn(line, "\r\n")] = '\0'; - - // Tokenize and process the input - hex_interpret(ctx, line, "<repl>", 1, 1); - // Print the top item of the stack - if (ctx->stack.top >= 0) - { - hex_print_item(stdout, ctx->stack.entries[ctx->stack.top]); - // hex_print_item(stdout, HEX_STACK[HEX_TOP]); - printf("\n"); - } - return 0; -} - -#endif - -// REPL implementation -void hex_repl(hex_context_t *ctx) -{ -#if defined(__EMSCRIPTEN__) - printf(" _*_ _\n"); - printf(" / \\hex\\*\n"); - printf(" *\\_/_/_/ v%s - WASM Build\n", HEX_VERSION); - printf(" *\n"); - int fps = 0; - int simulate_infinite_loop = 1; - emscripten_set_main_loop_arg(do_repl, ctx, fps, simulate_infinite_loop); -#else - - printf(" _*_ _\n"); - printf(" / \\hex\\*\n"); - printf(" *\\_/_/_/ v%s - Press Ctrl+C to exit.\n", HEX_VERSION); - printf(" *\n"); - - while (1) - { - if (do_repl(ctx) != 0) - { - exit(1); - } - } -#endif -} - -void hex_handle_sigint(int sig) -{ - (void)sig; // Suppress unused warning - printf("\n"); - exit(0); -} - -// Process piped input from stdin -void hex_process_stdin(hex_context_t *ctx) -{ - - char buffer[8192]; // Adjust buffer size as needed - int bytesRead = fread(buffer, 1, sizeof(buffer) - 1, stdin); - if (bytesRead == 0) - { - hex_error(ctx, "Error: No input provided via stdin."); - return; - } - - buffer[bytesRead] = '\0'; // Null-terminate the input - hex_interpret(ctx, buffer, "<stdin>", 1, 1); -} - -void hex_print_help() -{ - printf(" _*_ _\n" - " / \\hex\\*\n" - " *\\_/_/_/ v%s - (c) 2024 Fabio Cevasco\n" - " * \n", - HEX_VERSION); - printf("\n" - "USAGE\n" - " hex [options] [file]\n" - "\n" - "ARGUMENTS\n" - " file A .hex file to interpret\n" - "\n" - "OPTIONS\n" - " -d, --debug Enable debug mode.\n" - " -h, --help Display this help message.\n" - " -m, --manual Display the manual.\n" - " -v, --version Display hex version.\n\n"); -} - -void hex_print_docs(hex_doc_dictionary_t *docs) -{ - printf("\n" - " _*_ _\n" - " / \\hex\\*\n" - " *\\_/_/_/ v%s - (c) 2024 Fabio Cevasco\n" - " * \n", - HEX_VERSION); - printf("\n" - "BASICS\n" - " hex is a minimalist, slightly-esoteric, concatenative programming language that supports\n" - " only integers, strings, symbols, and quotations (lists).\n" - "\n" - " It uses a stack-based execution model and provides 64 native symbols for stack\n" - " manipulation, arithmetic operations, control flow, reading and writing\n" - " (standard input/output/error and files), executing external processes, and more.\n" - "\n" - " Symbols and literals are separated by whitespace and can be grouped in quotations using\n" - " parentheses.\n" - "\n" - " Symbols are evaluated only when they are pushed on the stack, therefore, symbols inside\n" - " quotations are not evaluated until the contents of the quotation are pushed on the stack.\n" - " You can define your own symbols using the symbol ':' and execute a quotation with '.'.\n" - "\n" - " Oh, and of course all integers are in hexadecimal format! ;)\n" - "\n" - "SYMBOLS\n" - " +---------+----------------------------+-------------------------------------------------+\n" - " | Symbol | Input -> Output | Description |\n" - " +---------+----------------------------+-------------------------------------------------+\n"); - for (int i = 0; i < docs->size; i++) - { - printf(" | "); - hex_rpad(docs->entries[i].name, 7); - printf(" | "); - hex_lpad(docs->entries[i].input, 15); - printf(" -> "); - hex_rpad(docs->entries[i].output, 7); - printf(" | "); - hex_rpad(docs->entries[i].description, 47); - printf(" |\n"); - } - printf(" +---------+----------------------------+-------------------------------------------------+\n"); -} - -//////////////////////////////////////// -// Main Program // -//////////////////////////////////////// - -int main(int argc, char *argv[]) -{ - // Register SIGINT (Ctrl+C) signal handler - signal(SIGINT, hex_handle_sigint); - - // Initialize the context - hex_context_t ctx = hex_init(); - ctx.argc = argc; - ctx.argv = argv; - - hex_register_symbols(&ctx); - hex_create_docs(&ctx.docs); - - char *file; - - if (argc > 1) - { - for (int i = 1; i < argc; i++) - { - char *arg = strdup(argv[i]); - if ((strcmp(arg, "-v") == 0 || strcmp(arg, "--version") == 0)) - { - printf("%s\n", HEX_VERSION); - return 0; - } - else if ((strcmp(arg, "-h") == 0 || strcmp(arg, "--help") == 0)) - { - hex_print_help(); - return 0; - } - else if ((strcmp(arg, "-m") == 0 || strcmp(arg, "--manual") == 0)) - { - hex_print_docs(&ctx.docs); - return 0; - } - else if ((strcmp(arg, "-d") == 0 || strcmp(arg, "--debug") == 0)) - { - ctx.settings.debugging_enabled = 1; - printf("*** Debug mode enabled ***\n"); - } - else - { - file = arg; - } - } - if (file) - { - char *fileContent = hex_read_file(&ctx, file); - hex_interpret(&ctx, fileContent, file, 1 + ctx.hashbang, 1); - return 0; - } - } -#if !(__EMSCRIPTEN__) - if (!isatty(fileno(stdin))) - { - ctx.settings.stack_trace_enabled = 0; - // Process piped input from stdin - hex_process_stdin(&ctx); - } -#endif - else - { - ctx.settings.stack_trace_enabled = 0; - // Start REPL - hex_repl(&ctx); - } - - return 0; -}
M src/hex.hsrc/hex.h

@@ -267,4 +267,9 @@ void hex_process_stdin(hex_context_t *ctx);

void hex_handle_sigint(int sig); char *hex_read_file(hex_context_t *ctx, const char *filename); +// Common operations +#define POP(ctx, x) hex_item_t x = hex_pop(ctx) +#define FREE(ctx, x) hex_free_item(ctx, x) +#define PUSH(ctx, x) hex_push(ctx, x) + #endif // HEX_H
A src/interpreter.c

@@ -0,0 +1,88 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Hex Interpreter Implementation // +//////////////////////////////////////// + +hex_context_t hex_init() +{ + hex_context_t context; + context.argc = 0; + context.argv = NULL; + context.registry.size = 0; + context.docs.size = 0; + context.stack.top = -1; + context.stack_trace.start = 0; + context.stack_trace.size = 0; + context.settings.debugging_enabled = 0; + context.settings.errors_enabled = 1; + context.settings.stack_trace_enabled = 1; + return context; +} + +int hex_interpret(hex_context_t *ctx, const char *code, const char *filename, int line, int column) +{ + + const char *input = code; + hex_file_position_t position = {filename, line, column}; + hex_token_t *token = hex_next_token(ctx, &input, &position); + + while (token != NULL && token->type != HEX_TOKEN_INVALID) + { + int result = 0; + if (token->type == HEX_TOKEN_INTEGER) + { + result = hex_push_integer(ctx, hex_parse_integer(token->value)); + } + else if (token->type == HEX_TOKEN_STRING) + { + result = hex_push_string(ctx, token->value); + } + else if (token->type == HEX_TOKEN_SYMBOL) + { + token->position.filename = strdup(filename); + result = hex_push_symbol(ctx, token); + } + else if (token->type == HEX_TOKEN_QUOTATION_END) + { + hex_error(ctx, "(%d,%d) Unexpected end of quotation", position.line, position.column); + result = 1; + } + else if (token->type == HEX_TOKEN_QUOTATION_START) + { + hex_item_t *quotationItem = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (hex_parse_quotation(ctx, &input, quotationItem, &position) != 0) + { + hex_error(ctx, "(%d,%d) Failed to parse quotation", position.line, position.column); + result = 1; + } + else + { + hex_item_t **quotation = quotationItem->data.quotation_value; + int quotation_size = quotationItem->quotation_size; + result = hex_push_quotation(ctx, quotation, quotation_size); + } + free(quotationItem); + } + + if (result != 0) + { + hex_free_token(token); + print_stack_trace(ctx); + return result; + } + + token = hex_next_token(ctx, &input, &position); + } + if (token != NULL && token->type == HEX_TOKEN_INVALID) + { + token->position.filename = strdup(filename); + add_to_stack_trace(ctx, token); + print_stack_trace(ctx); + return 1; + } + hex_free_token(token); + return 0; +}
A src/main.c

@@ -0,0 +1,352 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +// Read a file into a buffer +char *hex_read_file(hex_context_t *ctx, const char *filename) +{ + FILE *file = fopen(filename, "r"); + if (file == NULL) + { + hex_error(ctx, "Failed to open file: %s", filename); + return NULL; + } + + // Allocate an initial buffer + int bufferSize = 1024; // Start with a 1 KB buffer + char *content = (char *)malloc(bufferSize); + if (content == NULL) + { + hex_error(ctx, "Memory allocation failed"); + fclose(file); + return NULL; + } + + int bytesReadTotal = 0; + int bytesRead = 0; + + // Handle hashbang if present + char hashbangLine[1024]; + if (fgets(hashbangLine, sizeof(hashbangLine), file) != NULL) + { + if (strncmp(hashbangLine, "#!", 2) != 0) + { + // Not a hashbang line, reset file pointer to the beginning + fseek(file, 0, SEEK_SET); + ctx->hashbang = 0; + } + else + { + ctx->hashbang = 1; + } + } + + while ((bytesRead = fread(content + bytesReadTotal, 1, bufferSize - bytesReadTotal, file)) > 0) + { + bytesReadTotal += bytesRead; + + // Resize the buffer if necessary + if (bytesReadTotal == bufferSize) + { + bufferSize *= 2; // Double the buffer size + char *temp = (char *)realloc(content, bufferSize); + if (temp == NULL) + { + hex_error(ctx, "Memory reallocation failed"); + free(content); + fclose(file); + return NULL; + } + content = temp; + } + } + + if (ferror(file)) + { + hex_error(ctx, "Error reading the file"); + free(content); + fclose(file); + return NULL; + } + + // Null-terminate the content + char *finalContent = (char *)realloc(content, bytesReadTotal + 1); + if (finalContent == NULL) + { + hex_error(ctx, "Final memory allocation failed"); + free(content); + fclose(file); + return NULL; + } + content = finalContent; + content[bytesReadTotal] = '\0'; + + fclose(file); + return content; +} + +#if defined(__EMSCRIPTEN__) && defined(BROWSER) +static void prompt() +{ + // no prompt needed on browser +} +#elif defined(__EMSCRIPTEN__) && !defined(BROWSER) +static void prompt() +{ + printf(">\n"); +} +#else +static void prompt() +{ + printf("> "); + fflush(stdout); +} +#endif + +#if defined(__EMSCRIPTEN__) +static void do_repl(void *v_ctx) +{ + hex_context_t *ctx = (hex_context_t *)v_ctx; + prompt(); + char line[1024]; + char *p = line; + p = em_fgets(line, 1024); + if (!p) + { + printf("Error reading output"); + return; + } + // Normalize line endings (remove trailing \r\n or \n) + line[strcspn(line, "\r\n")] = '\0'; + + // Tokenize and process the input + hex_interpret(ctx, line, "<repl>", 1, 1); + // Print the top item of the stack + if (ctx->stack.top >= 0) + { + hex_print_item(stdout, ctx->stack.entries[ctx->stack.top]); + // hex_print_item(stdout, HEX_STACK[HEX_TOP]); + printf("\n"); + } + return; +} + +#else + +static int do_repl(void *v_ctx) +{ + hex_context_t *ctx = (hex_context_t *)v_ctx; + char line[1024]; + prompt(); + if (fgets(line, sizeof(line), stdin) == NULL) + { + printf("\n"); // Handle EOF (Ctrl+D) + return 1; + } + // Normalize line endings (remove trailing \r\n or \n) + line[strcspn(line, "\r\n")] = '\0'; + + // Tokenize and process the input + hex_interpret(ctx, line, "<repl>", 1, 1); + // Print the top item of the stack + if (ctx->stack.top >= 0) + { + hex_print_item(stdout, ctx->stack.entries[ctx->stack.top]); + // hex_print_item(stdout, HEX_STACK[HEX_TOP]); + printf("\n"); + } + return 0; +} + +#endif + +// REPL implementation +void hex_repl(hex_context_t *ctx) +{ +#if defined(__EMSCRIPTEN__) + printf(" _*_ _\n"); + printf(" / \\hex\\*\n"); + printf(" *\\_/_/_/ v%s - WASM Build\n", HEX_VERSION); + printf(" *\n"); + int fps = 0; + int simulate_infinite_loop = 1; + emscripten_set_main_loop_arg(do_repl, ctx, fps, simulate_infinite_loop); +#else + + printf(" _*_ _\n"); + printf(" / \\hex\\*\n"); + printf(" *\\_/_/_/ v%s - Press Ctrl+C to exit.\n", HEX_VERSION); + printf(" *\n"); + + while (1) + { + if (do_repl(ctx) != 0) + { + exit(1); + } + } +#endif +} + +void hex_handle_sigint(int sig) +{ + (void)sig; // Suppress unused warning + printf("\n"); + exit(0); +} + +// Process piped input from stdin +void hex_process_stdin(hex_context_t *ctx) +{ + + char buffer[8192]; // Adjust buffer size as needed + int bytesRead = fread(buffer, 1, sizeof(buffer) - 1, stdin); + if (bytesRead == 0) + { + hex_error(ctx, "Error: No input provided via stdin."); + return; + } + + buffer[bytesRead] = '\0'; // Null-terminate the input + hex_interpret(ctx, buffer, "<stdin>", 1, 1); +} + +void hex_print_help() +{ + printf(" _*_ _\n" + " / \\hex\\*\n" + " *\\_/_/_/ v%s - (c) 2024 Fabio Cevasco\n" + " * \n", + HEX_VERSION); + printf("\n" + "USAGE\n" + " hex [options] [file]\n" + "\n" + "ARGUMENTS\n" + " file A .hex file to interpret\n" + "\n" + "OPTIONS\n" + " -d, --debug Enable debug mode.\n" + " -h, --help Display this help message.\n" + " -m, --manual Display the manual.\n" + " -v, --version Display hex version.\n\n"); +} + +void hex_print_docs(hex_doc_dictionary_t *docs) +{ + printf("\n" + " _*_ _\n" + " / \\hex\\*\n" + " *\\_/_/_/ v%s - (c) 2024 Fabio Cevasco\n" + " * \n", + HEX_VERSION); + printf("\n" + "BASICS\n" + " hex is a minimalist, slightly-esoteric, concatenative programming language that supports\n" + " only integers, strings, symbols, and quotations (lists).\n" + "\n" + " It uses a stack-based execution model and provides 64 native symbols for stack\n" + " manipulation, arithmetic operations, control flow, reading and writing\n" + " (standard input/output/error and files), executing external processes, and more.\n" + "\n" + " Symbols and literals are separated by whitespace and can be grouped in quotations using\n" + " parentheses.\n" + "\n" + " Symbols are evaluated only when they are pushed on the stack, therefore, symbols inside\n" + " quotations are not evaluated until the contents of the quotation are pushed on the stack.\n" + " You can define your own symbols using the symbol ':' and execute a quotation with '.'.\n" + "\n" + " Oh, and of course all integers are in hexadecimal format! ;)\n" + "\n" + "SYMBOLS\n" + " +---------+----------------------------+-------------------------------------------------+\n" + " | Symbol | Input -> Output | Description |\n" + " +---------+----------------------------+-------------------------------------------------+\n"); + for (int i = 0; i < docs->size; i++) + { + printf(" | "); + hex_rpad(docs->entries[i].name, 7); + printf(" | "); + hex_lpad(docs->entries[i].input, 15); + printf(" -> "); + hex_rpad(docs->entries[i].output, 7); + printf(" | "); + hex_rpad(docs->entries[i].description, 47); + printf(" |\n"); + } + printf(" +---------+----------------------------+-------------------------------------------------+\n"); +} + +//////////////////////////////////////// +// Main Program // +//////////////////////////////////////// + +int main(int argc, char *argv[]) +{ + // Register SIGINT (Ctrl+C) signal handler + signal(SIGINT, hex_handle_sigint); + + // Initialize the context + hex_context_t ctx = hex_init(); + ctx.argc = argc; + ctx.argv = argv; + + hex_register_symbols(&ctx); + hex_create_docs(&ctx.docs); + + char *file; + + if (argc > 1) + { + for (int i = 1; i < argc; i++) + { + char *arg = strdup(argv[i]); + if ((strcmp(arg, "-v") == 0 || strcmp(arg, "--version") == 0)) + { + printf("%s\n", HEX_VERSION); + return 0; + } + else if ((strcmp(arg, "-h") == 0 || strcmp(arg, "--help") == 0)) + { + hex_print_help(); + return 0; + } + else if ((strcmp(arg, "-m") == 0 || strcmp(arg, "--manual") == 0)) + { + hex_print_docs(&ctx.docs); + return 0; + } + else if ((strcmp(arg, "-d") == 0 || strcmp(arg, "--debug") == 0)) + { + ctx.settings.debugging_enabled = 1; + printf("*** Debug mode enabled ***\n"); + } + else + { + file = arg; + } + } + if (file) + { + char *fileContent = hex_read_file(&ctx, file); + hex_interpret(&ctx, fileContent, file, 1 + ctx.hashbang, 1); + return 0; + } + } +#if !(__EMSCRIPTEN__) + if (!isatty(fileno(stdin))) + { + ctx.settings.stack_trace_enabled = 0; + // Process piped input from stdin + hex_process_stdin(&ctx); + } +#endif + else + { + ctx.settings.stack_trace_enabled = 0; + // Start REPL + hex_repl(&ctx); + } + + return 0; +}
A src/parser.c

@@ -0,0 +1,362 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +///////////////////////////////////////// +// Tokenizer and Parser Implementation // +///////////////////////////////////////// + +// Process a token from the input +hex_token_t *hex_next_token(hex_context_t *ctx, const char **input, hex_file_position_t *position) +{ + const char *ptr = *input; + + // Skip whitespace + while (isspace(*ptr)) + { + if (*ptr == '\n') + { + position->line++; + position->column = 1; + } + else + { + position->column++; + } + ptr++; + } + + if (*ptr == '\0') + { + return NULL; // End of input + } + + hex_token_t *token = (hex_token_t *)malloc(sizeof(hex_token_t)); + token->value = NULL; + token->position.line = position->line; + token->position.column = position->column; + + if (*ptr == ';') + { + // Comment token + const char *start = ptr; + while (*ptr != '\0' && *ptr != '\n') + { + ptr++; + position->column++; + } + int len = ptr - start; + token->value = (char *)malloc(len + 1); + strncpy(token->value, start, len); + token->value[len] = '\0'; + token->type = HEX_TOKEN_COMMENT; + } + else if (strncmp(ptr, "#|", 2) == 0) + { + // Block comment token + const char *start = ptr; + ptr += 2; // Skip the "#|" prefix + position->column += 2; + while (*ptr != '\0' && strncmp(ptr, "|#", 2) != 0) + { + if (*ptr == '\n') + { + position->line++; + position->column = 1; + } + else + { + position->column++; + } + ptr++; + } + if (*ptr == '\0') + { + token->type = HEX_TOKEN_INVALID; + token->position.line = position->line; + token->position.column = position->column; + hex_error(ctx, "(%d,%d) Unterminated block comment", position->line, position->column); + return token; + } + ptr += 2; // Skip the "|#" suffix + position->column += 2; + int len = ptr - start; + token->value = (char *)malloc(len + 1); + strncpy(token->value, start, len); + token->value[len] = '\0'; + token->type = HEX_TOKEN_COMMENT; + } + else if (*ptr == '"') + { + // String token + ptr++; + const char *start = ptr; + int len = 0; + + while (*ptr != '\0') + { + if (*ptr == '\\' && *(ptr + 1) == '"') + { + ptr += 2; + len++; + position->column += 2; + } + if (*ptr == '\\' && *(ptr + 1) == '\\') + { + ptr += 2; + len++; + position->column += 2; + } + else if (*ptr == '"') + { + break; + } + else if (*ptr == '\n') + { + token->type = HEX_TOKEN_INVALID; + token->position.line = position->line; + token->position.column = position->column; + hex_error(ctx, "(%d,%d) Unescaped new line in string", position->line, position->column); + return token; + } + else + { + ptr++; + len++; + position->column++; + } + } + + if (*ptr != '"') + { + token->type = HEX_TOKEN_INVALID; + token->position.line = position->line; + token->position.column = position->column; + hex_error(ctx, "(%d,%d) Unterminated string", position->line, position->column); + return token; + } + + token->value = (char *)malloc(len + 1); + char *dst = token->value; + + ptr = start; + while (*ptr != '\0' && *ptr != '"') + { + if (*ptr == '\\' && *(ptr + 1) == '\\') + { + *dst++ = '\\'; + *dst++ = '\\'; + ptr += 2; + } + else if (*ptr == '\\' && *(ptr + 1) == '"') + { + *dst++ = '"'; + ptr += 2; + } + else + { + *dst++ = *ptr++; + } + } + *dst = '\0'; + ptr++; + position->column++; + token->type = HEX_TOKEN_STRING; + } + else if (strncmp(ptr, "0x", 2) == 0 || strncmp(ptr, "0X", 2) == 0) + { + // Hexadecimal integer token + const char *start = ptr; + ptr += 2; // Skip the "0x" prefix + position->column += 2; + while (isxdigit(*ptr)) + { + ptr++; + position->column++; + } + int len = ptr - start; + token->value = (char *)malloc(len + 1); + strncpy(token->value, start, len); + token->value[len] = '\0'; + token->type = HEX_TOKEN_INTEGER; + } + else if (*ptr == '(') + { + token->type = HEX_TOKEN_QUOTATION_START; + ptr++; + position->column++; + } + else if (*ptr == ')') + { + token->type = HEX_TOKEN_QUOTATION_END; + ptr++; + position->column++; + } + else + { + // Symbol token + const char *start = ptr; + while (*ptr != '\0' && !isspace(*ptr) && *ptr != ';' && *ptr != '(' && *ptr != ')' && *ptr != '"') + { + ptr++; + position->column++; + } + + int len = ptr - start; + token->value = (char *)malloc(len + 1); + strncpy(token->value, start, len); + token->value[len] = '\0'; + if (hex_valid_native_symbol(ctx, token->value) || hex_valid_user_symbol(ctx, token->value)) + { + token->type = HEX_TOKEN_SYMBOL; + } + else + { + token->type = HEX_TOKEN_INVALID; + token->position.line = position->line; + token->position.column = position->column; + } + } + + *input = ptr; + return token; +} + +int hex_valid_native_symbol(hex_context_t *ctx, const char *symbol) +{ + hex_doc_entry_t doc; + for (size_t i = 0; i < HEX_NATIVE_SYMBOLS; i++) + { + if (hex_get_doc(&ctx->docs, symbol, &doc)) + { + return 1; + } + } + return 0; +} + +int32_t hex_parse_integer(const char *hex_str) +{ + // Parse the hexadecimal string as an unsigned 32-bit integer + uint32_t unsigned_value = (uint32_t)strtoul(hex_str, NULL, 16); + + // Cast the unsigned value to a signed 32-bit integer + return (int32_t)unsigned_value; +} + +int hex_parse_quotation(hex_context_t *ctx, const char **input, hex_item_t *result, hex_file_position_t *position) +{ + hex_item_t **quotation = NULL; + int capacity = 2; + int size = 0; + int balanced = 1; + + quotation = (hex_item_t **)malloc(capacity * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + return 1; + } + + hex_token_t *token; + while ((token = hex_next_token(ctx, input, position)) != NULL) + { + if (token->type == HEX_TOKEN_QUOTATION_END) + { + balanced--; + break; + } + + if (size >= capacity) + { + capacity *= 2; + quotation = (hex_item_t **)realloc(quotation, capacity * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "(%d,%d), Memory allocation failed", position->line, position->column); + return 1; + } + } + + hex_item_t *item = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (token->type == HEX_TOKEN_INTEGER) + { + + *item = hex_integer_item(ctx, hex_parse_integer(token->value)); + quotation[size] = item; + size++; + } + else if (token->type == HEX_TOKEN_STRING) + { + *item = hex_string_item(ctx, token->value); + quotation[size] = item; + size++; + } + else if (token->type == HEX_TOKEN_SYMBOL) + { + if (hex_valid_native_symbol(ctx, token->value)) + { + item->type = HEX_TYPE_NATIVE_SYMBOL; + hex_item_t value; + if (hex_get_symbol(ctx, token->value, &value)) + { + item->token = token; + item->type = HEX_TYPE_NATIVE_SYMBOL; + item->data.fn_value = value.data.fn_value; + } + else + { + hex_error(ctx, "(%d,%d) Unable to reference native symbol: %s", position->line, position->column, token->value); + hex_free_token(token); + hex_free_list(ctx, quotation, size); + return 1; + } + } + else + { + item->type = HEX_TYPE_USER_SYMBOL; + } + token->position.filename = strdup(position->filename); + item->token = token; + quotation[size] = item; + size++; + } + else if (token->type == HEX_TOKEN_QUOTATION_START) + { + item->type = HEX_TYPE_QUOTATION; + if (hex_parse_quotation(ctx, input, item, position) != 0) + { + hex_free_token(token); + hex_free_list(ctx, quotation, size); + return 1; + } + quotation[size] = item; + size++; + } + else if (token->type == HEX_TOKEN_COMMENT) + { + // Ignore comments + } + else + { + hex_error(ctx, "(%d,%d) Unexpected token in quotation: %d", position->line, position->column, token->value); + hex_free_token(token); + hex_free_list(ctx, quotation, size); + return 1; + } + } + + if (balanced != 0) + { + hex_error(ctx, "(%d,%d) Unterminated quotation", position->line, position->column); + hex_free_token(token); + hex_free_list(ctx, quotation, size); + return 1; + } + + result->type = HEX_TYPE_QUOTATION; + result->data.quotation_value = quotation; + result->quotation_size = size; + hex_free_token(token); + return 0; +}
A src/registry.c

@@ -0,0 +1,97 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Registry Implementation // +//////////////////////////////////////// + +int hex_valid_user_symbol(hex_context_t *ctx, const char *symbol) +{ + // Check that key starts with a letter, or underscore + // and subsequent characters (if any) are letters, numbers, or underscores + if (strlen(symbol) == 0) + { + hex_error(ctx, "Symbol name cannot be an empty string"); + return 0; + } + if (!isalpha(symbol[0]) && symbol[0] != '_') + { + hex_error(ctx, "Invalid symbol: %s", symbol); + return 0; + } + for (size_t j = 1; j < strlen(symbol); j++) + { + if (!isalnum(symbol[j]) && symbol[j] != '_' && symbol[j] != '-') + { + hex_error(ctx, "Invalid symbol: %s", symbol); + return 0; + } + } + return 1; +} + +// Add a symbol to the registry +int hex_set_symbol(hex_context_t *ctx, const char *key, hex_item_t value, int native) +{ + + if (!native && hex_valid_user_symbol(ctx, key) == 0) + { + return 1; + } + for (int i = 0; i < ctx->registry.size; i++) + { + if (strcmp(ctx->registry.entries[i].key, key) == 0) + { + if (ctx->registry.entries[i].value.type == HEX_TYPE_NATIVE_SYMBOL) + { + hex_error(ctx, "Cannot overwrite native symbol '%s'", key); + return 1; + } + free(ctx->registry.entries[i].key); + ctx->registry.entries[i].key = strdup(key); + ctx->registry.entries[i].value = value; + return 0; + } + } + + if (ctx->registry.size >= HEX_REGISTRY_SIZE) + { + hex_error(ctx, "Registry overflow"); + hex_free_token(value.token); + return 1; + } + + ctx->registry.entries[ctx->registry.size].key = strdup(key); + ctx->registry.entries[ctx->registry.size].value = value; + ctx->registry.size++; + return 0; +} + +// Register a native symbol +void hex_set_native_symbol(hex_context_t *ctx, const char *name, int (*func)()) +{ + hex_item_t func_item; + func_item.type = HEX_TYPE_NATIVE_SYMBOL; + func_item.data.fn_value = func; + + if (hex_set_symbol(ctx, name, func_item, 1) != 0) + { + hex_error(ctx, "Error: Failed to register native symbol '%s'", name); + } +} + +// Get a symbol value from the registry +int hex_get_symbol(hex_context_t *ctx, const char *key, hex_item_t *result) +{ + + for (int i = 0; i < ctx->registry.size; i++) + { + if (strcmp(ctx->registry.entries[i].key, key) == 0) + { + *result = ctx->registry.entries[i].value; + return 1; + } + } + return 0; +}
A src/stack.c

@@ -0,0 +1,231 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Stack Implementation // +//////////////////////////////////////// + +// Free a token +void hex_free_token(hex_token_t *token) +{ + if (token) + { + free(token->value); + free(token); + } +} + +// Push functions +int hex_push(hex_context_t *ctx, hex_item_t item) +{ + + if (ctx->stack.top >= HEX_STACK_SIZE - 1) + { + hex_error(ctx, "Stack overflow"); + return 1; + } + hex_debug_item(ctx, "PUSH", item); + int result = 0; + if (item.type == HEX_TYPE_USER_SYMBOL) + { + hex_item_t value; + if (hex_get_symbol(ctx, item.token->value, &value)) + { + result = PUSH(ctx, value); + } + else + { + hex_error(ctx, "Undefined user symbol: %s", item.token->value); + FREE(ctx, value); + result = 1; + } + } + else if (item.type == HEX_TYPE_NATIVE_SYMBOL) + { + hex_debug_item(ctx, "CALL", item); + result = item.data.fn_value(ctx); + } + else + { + ctx->stack.entries[++ctx->stack.top] = item; + } + if (result == 0) + { + hex_debug_item(ctx, "DONE", item); + } + else + { + hex_debug_item(ctx, "FAIL", item); + } + return result; +} + +char *hex_process_string(hex_context_t *ctx, const char *value) +{ + int len = strlen(value); + char *processed_str = (char *)malloc(len + 1); + if (!processed_str) + { + hex_error(ctx, "Memory allocation failed"); + return NULL; + } + + char *dst = processed_str; + const char *src = value; + while (*src) + { + if (*src == '\\' && *(src + 1)) + { + src++; + switch (*src) + { + case 'n': + *dst++ = '\n'; + break; + case 't': + *dst++ = '\t'; + break; + case 'r': + *dst++ = '\r'; + break; + case 'b': + *dst++ = '\b'; + break; + case 'f': + *dst++ = '\f'; + break; + case 'v': + *dst++ = '\v'; + break; + // case '\\': + // *dst++ = '\\'; + // break; + case '\"': + *dst++ = '\"'; + break; + default: + *dst++ = '\\'; + *dst++ = *src; + break; + } + } + else + { + *dst++ = *src; + } + src++; + } + *dst = '\0'; + return processed_str; +} + +hex_item_t hex_string_item(hex_context_t *ctx, const char *value) +{ + hex_item_t item = {.type = HEX_TYPE_STRING, .data.str_value = hex_process_string(ctx, value)}; + return item; +} + +hex_item_t hex_integer_item(hex_context_t *ctx, int value) +{ + (void)(ctx); + hex_item_t item = {.type = HEX_TYPE_INTEGER, .data.int_value = value}; + return item; +} + +hex_item_t hex_quotation_item(hex_context_t *ctx, hex_item_t **quotation, int size) +{ + (void)(ctx); + hex_item_t item = {.type = HEX_TYPE_QUOTATION, .data.quotation_value = quotation, .quotation_size = size}; + return item; +} + +int hex_push_string(hex_context_t *ctx, const char *value) +{ + return PUSH(ctx, hex_string_item(ctx, value)); +} + +int hex_push_integer(hex_context_t *ctx, int value) +{ + return PUSH(ctx, hex_integer_item(ctx, value)); +} + +int hex_push_quotation(hex_context_t *ctx, hex_item_t **quotation, int size) +{ + return PUSH(ctx, hex_quotation_item(ctx, quotation, size)); +} + +int hex_push_symbol(hex_context_t *ctx, hex_token_t *token) +{ + add_to_stack_trace(ctx, token); + hex_item_t value; + if (hex_get_symbol(ctx, token->value, &value)) + { + value.token = token; + return PUSH(ctx, value); + } + else + { + hex_error(ctx, "Undefined symbol: %s", token->value); + return 1; + } +} + +// Pop function +hex_item_t hex_pop(hex_context_t *ctx) +{ + if (ctx->stack.top < 0) + { + hex_error(ctx, "Insufficient items on the stack"); + return (hex_item_t){.type = HEX_TYPE_INVALID}; + } + hex_debug_item(ctx, " POP", ctx->stack.entries[ctx->stack.top]); + return ctx->stack.entries[ctx->stack.top--]; +} + +// Free a stack item +void hex_free_item(hex_context_t *ctx, hex_item_t item) +{ + hex_debug_item(ctx, "FREE", item); + if (item.type == HEX_TYPE_STRING && item.data.str_value != NULL) + { + hex_debug(ctx, "FREE: ** string start"); + item.data.str_value = NULL; + free(item.data.str_value); + hex_debug(ctx, "FREE: ** string end"); + } + + else if (item.type == HEX_TYPE_QUOTATION && item.data.quotation_value != NULL) + { + hex_debug(ctx, "FREE: ** quotation start"); + hex_free_list(ctx, item.data.quotation_value, item.quotation_size); + item.data.quotation_value = NULL; + hex_debug(ctx, "FREE: ** quotation end"); + } + else if (item.type == HEX_TYPE_NATIVE_SYMBOL && item.token->value != NULL) + { + hex_debug(ctx, "FREE: ** native symbol start"); + item.token = NULL; + hex_free_token(item.token); + hex_debug(ctx, "FREE: ** native symbol end"); + } + else if (item.type == HEX_TYPE_USER_SYMBOL && item.token->value != NULL) + { + hex_debug(ctx, "FREE: ** user symbol start"); + item.token = NULL; + hex_free_token(item.token); + hex_debug(ctx, "FREE: ** user symbol end"); + } + else + { + hex_debug(ctx, "FREE: ** nothing to free"); + } +} + +void hex_free_list(hex_context_t *ctx, hex_item_t **quotation, int size) +{ + for (int i = 0; i < size; i++) + { + FREE(ctx, *quotation[i]); + } +}
A src/stacktrace.c

@@ -0,0 +1,43 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Stack trace implementation // +//////////////////////////////////////// + +// Add an entry to the circular stack trace +void add_to_stack_trace(hex_context_t *ctx, hex_token_t *token) +{ + int index = (ctx->stack_trace.start + ctx->stack_trace.size) % HEX_STACK_TRACE_SIZE; + + if (ctx->stack_trace.size < HEX_STACK_TRACE_SIZE) + { + // Buffer is not full; add item + ctx->stack_trace.entries[index] = *token; + ctx->stack_trace.size++; + } + else + { + // Buffer is full; overwrite the oldest item + ctx->stack_trace.entries[index] = *token; + ctx->stack_trace.start = (ctx->stack_trace.start + 1) % HEX_STACK_TRACE_SIZE; + } +} + +// Print the stack trace +void print_stack_trace(hex_context_t *ctx) +{ + if (!ctx->settings.stack_trace_enabled || !ctx->settings.errors_enabled || ctx->stack_trace.size <= 0) + { + return; + } + fprintf(stderr, "[stack trace] (most recent symbol first):\n"); + + for (int i = 0; i < ctx->stack_trace.size; i++) + { + int index = (ctx->stack_trace.start + ctx->stack_trace.size - 1 - i) % HEX_STACK_TRACE_SIZE; + hex_token_t token = ctx->stack_trace.entries[index]; + fprintf(stderr, " %s (%s:%d:%d)\n", token.value, token.position.filename, token.position.line, token.position.column); + } +}
A src/symbols.c

@@ -0,0 +1,2525 @@

+#ifndef HEX_H +#include "hex.h" +#endif + +//////////////////////////////////////// +// Native Symbol Implementations // +//////////////////////////////////////// + +// Definition symbols + +int hex_symbol_store(hex_context_t *ctx) +{ + + POP(ctx, name); + if (name.type == HEX_TYPE_INVALID) + { + FREE(ctx, name); + return 1; + } + POP(ctx, value); + if (value.type == HEX_TYPE_INVALID) + { + FREE(ctx, name); + FREE(ctx, value); + return 1; + } + if (name.type != HEX_TYPE_STRING) + { + hex_error(ctx, "Symbol name must be a string"); + FREE(ctx, name); + FREE(ctx, value); + return 1; + } + if (hex_set_symbol(ctx, name.data.str_value, value, 0) != 0) + { + hex_error(ctx, "Failed to store symbol '%s'", name.data.str_value); + FREE(ctx, name); + FREE(ctx, value); + return 1; + } + return 0; +} + +int hex_symbol_free(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type != HEX_TYPE_STRING) + { + FREE(ctx, item); + hex_error(ctx, "Symbol name must be a string"); + return 1; + } + if (hex_valid_native_symbol(ctx, item.data.str_value)) + { + hex_error(ctx, "Cannot free native symbol '%s'", item.data.str_value); + FREE(ctx, item); + return 1; + } + for (int i = 0; i < ctx->registry.size; i++) + { + if (strcmp(ctx->registry.entries[i].key, item.data.str_value) == 0) + { + free(ctx->registry.entries[i].key); + FREE(ctx, ctx->registry.entries[i].value); + for (int j = i; j < ctx->registry.size - 1; j++) + { + ctx->registry.entries[j] = ctx->registry.entries[j + 1]; + } + ctx->registry.size--; + FREE(ctx, item); + return 0; + } + } + FREE(ctx, item); + return 0; +} + +int hex_symbol_type(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + return hex_push_string(ctx, hex_type(item.type)); +} + +// Evaluation symbols + +int hex_symbol_i(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "Symbol '.' requires a quotation"); + FREE(ctx, item); + return 1; + } + for (int i = 0; i < item.quotation_size; i++) + { + if (hex_push(ctx, *item.data.quotation_value[i]) != 0) + { + FREE(ctx, item); + return 1; + } + } + return 0; +} + +// evaluate a string +int hex_symbol_eval(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type != HEX_TYPE_STRING) + { + hex_error(ctx, "Symbol '!' requires a string"); + FREE(ctx, item); + return 1; + } + return hex_interpret(ctx, item.data.str_value, "<!>", 1, 1); +} + +// IO Symbols + +int hex_symbol_puts(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + hex_raw_print_item(stdout, item); + printf("\n"); + return 0; +} + +int hex_symbol_warn(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + hex_raw_print_item(stderr, item); + printf("\n"); + return 0; +} + +int hex_symbol_print(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + hex_raw_print_item(stdout, item); + fflush(stdout); + return 0; +} + +int hex_symbol_gets(hex_context_t *ctx) +{ + + char input[HEX_STDIN_BUFFER_SIZE]; // Buffer to store the input (adjust size if needed) + char *p = input; +#if defined(__EMSCRIPTEN__) + p = em_fgets(input, 1024); +#else + p = fgets(input, sizeof(input), stdin); +#endif + + if (p != NULL) + { + // Strip the newline character at the end of the string + input[strcspn(input, "\n")] = '\0'; + + // Push the input string onto the stack + return hex_push_string(ctx, input); + } + else + { + hex_error(ctx, "Failed to read input"); + return 1; + } +} + +// Mathematical symbols +int hex_symbol_add(hex_context_t *ctx) +{ + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value + b.data.int_value); + } + hex_error(ctx, "Symbol '+' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_subtract(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value - b.data.int_value); + } + hex_error(ctx, "Symbol '-' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_multiply(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value * b.data.int_value); + } + hex_error(ctx, "'*' symbol requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_divide(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + if (b.data.int_value == 0) + { + hex_error(ctx, "Division by zero"); + return 1; + } + return hex_push_integer(ctx, a.data.int_value / b.data.int_value); + } + hex_error(ctx, "Symbol '/' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_modulo(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + if (b.data.int_value == 0) + { + hex_error(ctx, "Division by zero"); + return 1; + } + return hex_push_integer(ctx, a.data.int_value % b.data.int_value); + } + hex_error(ctx, "Symbol '%%' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +// Bit symbols + +int hex_symbol_bitand(hex_context_t *ctx) +{ + + POP(ctx, right); + if (right.type == HEX_TYPE_INVALID) + { + FREE(ctx, right); + return 1; + } + POP(ctx, left); + if (left.type == HEX_TYPE_INVALID) + { + FREE(ctx, left); + FREE(ctx, right); + return 1; + } + if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, left.data.int_value & right.data.int_value); + } + hex_error(ctx, "Symbol '&' requires two integers"); + FREE(ctx, left); + FREE(ctx, right); + return 1; +} + +int hex_symbol_bitor(hex_context_t *ctx) +{ + + POP(ctx, right); + if (right.type == HEX_TYPE_INVALID) + { + FREE(ctx, right); + return 1; + } + POP(ctx, left); + if (left.type == HEX_TYPE_INVALID) + { + FREE(ctx, left); + FREE(ctx, right); + return 1; + } + if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, left.data.int_value | right.data.int_value); + } + hex_error(ctx, "Symbol '|' requires two integers"); + FREE(ctx, left); + FREE(ctx, right); + return 1; +} + +int hex_symbol_bitxor(hex_context_t *ctx) +{ + + POP(ctx, right); + if (right.type == HEX_TYPE_INVALID) + { + FREE(ctx, right); + return 1; + } + POP(ctx, left); + if (left.type == HEX_TYPE_INVALID) + { + FREE(ctx, left); + FREE(ctx, right); + return 1; + } + if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, left.data.int_value ^ right.data.int_value); + } + hex_error(ctx, "Symbol '^' requires two integers"); + FREE(ctx, left); + FREE(ctx, right); + return 1; +} + +int hex_symbol_shiftleft(hex_context_t *ctx) +{ + + POP(ctx, right); + if (right.type == HEX_TYPE_INVALID) + { + FREE(ctx, right); + return 1; + } + POP(ctx, left); + if (left.type == HEX_TYPE_INVALID) + { + FREE(ctx, left); + FREE(ctx, right); + return 1; + } + if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, left.data.int_value << right.data.int_value); + } + hex_error(ctx, "Symbol '<<' requires two integers"); + FREE(ctx, left); + FREE(ctx, right); + return 1; +} + +int hex_symbol_shiftright(hex_context_t *ctx) +{ + + POP(ctx, right); + if (right.type == HEX_TYPE_INVALID) + { + FREE(ctx, right); + return 1; + } + POP(ctx, left); + if (left.type == HEX_TYPE_INVALID) + { + FREE(ctx, left); + FREE(ctx, right); + return 1; + } + if (left.type == HEX_TYPE_INTEGER && right.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, left.data.int_value >> right.data.int_value); + } + hex_error(ctx, "Symbol '>>' requires two integers"); + FREE(ctx, left); + FREE(ctx, right); + return 1; +} + +int hex_symbol_bitnot(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, ~item.data.int_value); + } + hex_error(ctx, "Symbol '~' requires an integer"); + FREE(ctx, item); + return 1; +} + +// Conversion symbols + +int hex_symbol_int(hex_context_t *ctx) +{ + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + return 1; + } + if (a.type == HEX_TYPE_STRING) + { + return hex_push_integer(ctx, strtol(a.data.str_value, NULL, 16)); + } + hex_error(ctx, "Symbol 'int' requires a string"); + FREE(ctx, a); + return 1; +} + +int hex_symbol_str(hex_context_t *ctx) +{ + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + return 1; + } + if (a.type == HEX_TYPE_INTEGER) + { + return hex_push_string(ctx, hex_itoa_hex(a.data.int_value)); + } + hex_error(ctx, "Symbol 'str' requires an integer"); + FREE(ctx, a); + return 1; +} + +int hex_symbol_dec(hex_context_t *ctx) +{ + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + return 1; + } + if (a.type == HEX_TYPE_INTEGER) + { + return hex_push_string(ctx, hex_itoa_dec(a.data.int_value)); + } + hex_error(ctx, "Symbol 'dec' requires an integer"); + FREE(ctx, a); + return 1; +} + +int hex_symbol_hex(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type == HEX_TYPE_STRING) + { + return hex_push_integer(ctx, strtol(item.data.str_value, NULL, 10)); + } + hex_error(ctx, "Symbol 'hex' requires a string representing a decimal integer"); + FREE(ctx, item); + return 1; +} + +int hex_symbol_ord(hex_context_t *ctx) +{ + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type == HEX_TYPE_STRING) + { + if (strlen(item.data.str_value) > 1) + { + return hex_push_integer(ctx, -1); + } + unsigned char *str = (unsigned char *)item.data.str_value; + if (str[0] < 128) + { + return hex_push_integer(ctx, str[0]); + } + else + { + return hex_push_integer(ctx, -1); + } + } + hex_error(ctx, "Symbol 'ord' requires a string"); + FREE(ctx, item); + return 1; +} + +int hex_symbol_chr(hex_context_t *ctx) +{ + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type == HEX_TYPE_INTEGER) + { + if (item.data.int_value >= 0 && item.data.int_value < 128) + { + char str[2] = {(char)item.data.int_value, '\0'}; + return hex_push_string(ctx, str); + } + else + { + return hex_push_string(ctx, ""); + } + } + hex_error(ctx, "Symbol 'chr' requires an integer"); + FREE(ctx, item); + return 1; +} + +// Comparison symbols + +static int hex_equal(hex_item_t a, hex_item_t b) +{ + if (a.type == HEX_TYPE_INVALID || b.type == HEX_TYPE_INVALID) + { + return 0; + } + if (a.type == HEX_TYPE_NATIVE_SYMBOL || a.type == HEX_TYPE_USER_SYMBOL) + { + return (strcmp(a.token->value, b.token->value) == 0); + } + if (a.type != b.type) + { + return 0; + } + if (a.type == HEX_TYPE_INTEGER) + { + return a.data.int_value == b.data.int_value; + } + if (a.type == HEX_TYPE_STRING) + { + return (strcmp(a.data.str_value, b.data.str_value) == 0); + } + if (a.type == HEX_TYPE_QUOTATION) + { + if (a.quotation_size != b.quotation_size) + { + return 0; + } + else + { + for (int i = 0; i < a.quotation_size; i++) + { + if (!hex_equal(*a.data.quotation_value[i], *b.data.quotation_value[i])) + { + return 0; + } + } + return 1; + } + } + return 0; +} + +static int hex_is_type_symbol(hex_item_t *item) +{ + if (item->type == HEX_TYPE_USER_SYMBOL || item->type == HEX_TYPE_NATIVE_SYMBOL) + { + return 1; + } + return 0; +} + +static int hex_greater(hex_context_t *ctx, hex_item_t *a, hex_item_t *b, char *symbol) +{ + if (a->type == HEX_TYPE_INTEGER && b->type == HEX_TYPE_INTEGER) + { + return a->data.int_value > b->data.int_value; + } + else if (a->type == HEX_TYPE_STRING && b->type == HEX_TYPE_STRING) + { + return strcmp(a->data.str_value, b->data.str_value) > 0; + } + else if (a->type == HEX_TYPE_QUOTATION && b->type == HEX_TYPE_QUOTATION) + { + // Compare quotations lexicographically + size_t min_size = a->quotation_size < b->quotation_size ? a->quotation_size : b->quotation_size; + int is_greater = 0; + + for (size_t i = 0; i < min_size; i++) + { + hex_item_t *it_a = a->data.quotation_value[i]; + hex_item_t *it_b = b->data.quotation_value[i]; + + // Perform element-wise comparison + if (it_a->type != it_b->type && !(hex_is_type_symbol(it_a) && hex_is_type_symbol(it_b))) + { + // Mismatched types, return false + return 0; + } + + if (it_a->type == HEX_TYPE_INTEGER) + { + if (it_a->data.int_value != it_b->data.int_value) + { + is_greater = it_a->data.int_value > it_b->data.int_value; + break; + } + } + else if (it_a->type == HEX_TYPE_STRING) + { + int cmp = strcmp(it_a->data.str_value, it_b->data.str_value); + if (cmp != 0) + { + is_greater = cmp > 0; + break; + } + } + else if (hex_is_type_symbol(it_a)) + { + int cmp = strcmp(it_a->token->value, it_b->token->value); + if (cmp != 0) + { + is_greater = cmp > 0; + break; + } + } + else + { + // Mismatched types, return false + return 0; + } + } + + if (!is_greater) + { + // If all compared elements are equal, compare sizes + return a->quotation_size > b->quotation_size; + } + return is_greater; + } + else + { + hex_error(ctx, "'%s' symbol requires two integers, two strings, or two quotations", symbol); + return -1; + } +} + +int hex_symbol_equal(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if ((a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) || (a.type == HEX_TYPE_STRING && b.type == HEX_TYPE_STRING) || (a.type == HEX_TYPE_QUOTATION && b.type == HEX_TYPE_QUOTATION)) + { + return hex_push_integer(ctx, hex_equal(a, b)); + } + // Different types => false + return hex_push_integer(ctx, 0); +} + +int hex_symbol_notequal(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if ((a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) || (a.type == HEX_TYPE_STRING && b.type == HEX_TYPE_STRING) || (a.type == HEX_TYPE_QUOTATION && b.type == HEX_TYPE_QUOTATION)) + { + return hex_push_integer(ctx, !hex_equal(a, b)); + } + // Different types => true + return hex_push_integer(ctx, 1); +} + +int hex_symbol_greater(hex_context_t *ctx) +{ + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + hex_item_t *pa = &a; + hex_item_t *pb = &b; + hex_push_integer(ctx, hex_greater(ctx, pa, pb, ">")); + return 0; +} + +int hex_symbol_less(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + hex_item_t *pa = &a; + hex_item_t *pb = &b; + hex_push_integer(ctx, hex_greater(ctx, pb, pa, "<")); + return 0; +} + +int hex_symbol_greaterequal(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + hex_item_t *pa = &a; + hex_item_t *pb = &b; + hex_push_integer(ctx, hex_greater(ctx, pa, pb, ">") || hex_equal(a, b)); + return 0; +} + +int hex_symbol_lessequal(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + hex_item_t *pa = &a; + hex_item_t *pb = &b; + hex_push_integer(ctx, !hex_greater(ctx, pb, pa, "<") || hex_equal(a, b)); + return 0; +} + +// Boolean symbols + +int hex_symbol_and(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value && b.data.int_value); + } + hex_error(ctx, "Symbol 'and' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_or(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value || b.data.int_value); + } + hex_error(ctx, "Symbol 'or' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +int hex_symbol_not(hex_context_t *ctx) +{ + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + return 1; + } + if (a.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, !a.data.int_value); + } + hex_error(ctx, "Symbol 'not' requires an integer"); + FREE(ctx, a); + return 1; +} + +int hex_symbol_xor(hex_context_t *ctx) +{ + + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + return 1; + } + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (a.type == HEX_TYPE_INTEGER && b.type == HEX_TYPE_INTEGER) + { + return hex_push_integer(ctx, a.data.int_value ^ b.data.int_value); + } + hex_error(ctx, "Symbol 'xor' requires two integers"); + FREE(ctx, a); + FREE(ctx, b); + return 1; +} + +// Quotation and String (List) Symbols + +int hex_symbol_cat(hex_context_t *ctx) +{ + + POP(ctx, value); + if (value.type == HEX_TYPE_INVALID) + { + FREE(ctx, value); + return 1; // Failed to pop value + } + + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, list); + FREE(ctx, value); + return 1; // Failed to pop list + } + + int result = 0; + + if (list.type == HEX_TYPE_QUOTATION && value.type == HEX_TYPE_QUOTATION) + { + // Concatenate two quotations + size_t newSize = list.quotation_size + value.quotation_size; + hex_item_t **newQuotation = (hex_item_t **)realloc( + list.data.quotation_value, newSize * sizeof(hex_item_t *)); + if (!newQuotation) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + // Append items from the second quotation + for (size_t i = 0; i < (size_t)value.quotation_size; i++) + { + newQuotation[list.quotation_size + i] = value.data.quotation_value[i]; + } + + list.data.quotation_value = newQuotation; + list.quotation_size = newSize; + result = hex_push_quotation(ctx, list.data.quotation_value, newSize); + } + } + else if (list.type == HEX_TYPE_STRING && value.type == HEX_TYPE_STRING) + { + // Concatenate two strings + size_t newLength = strlen(list.data.str_value) + strlen(value.data.str_value) + 1; + char *newStr = (char *)malloc(newLength); + if (!newStr) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + strcpy(newStr, list.data.str_value); + strcat(newStr, value.data.str_value); + result = hex_push_string(ctx, newStr); + } + } + else + { + hex_error(ctx, "Symbol 'cat' requires two quotations or two strings"); + result = 1; + } + + // Free resources if the operation fails + if (result != 0) + { + FREE(ctx, list); + FREE(ctx, value); + } + + return result; +} + +int hex_symbol_len(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + int result = 0; + if (item.type == HEX_TYPE_QUOTATION) + { + result = hex_push_integer(ctx, item.quotation_size); + } + else if (item.type == HEX_TYPE_STRING) + { + result = hex_push_integer(ctx, strlen(item.data.str_value)); + } + else + { + hex_error(ctx, "Symbol 'len' requires a quotation or a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, item); + } + return result; +} + +int hex_symbol_get(hex_context_t *ctx) +{ + + POP(ctx, index); + if (index.type == HEX_TYPE_INVALID) + { + FREE(ctx, index); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, list); + FREE(ctx, index); + return 1; + } + int result = 0; + if (list.type == HEX_TYPE_QUOTATION) + { + if (index.type != HEX_TYPE_INTEGER) + { + hex_error(ctx, "Index must be an integer"); + result = 1; + } + else if (index.data.int_value < 0 || index.data.int_value >= list.quotation_size) + { + hex_error(ctx, "Index out of range"); + result = 1; + } + else + { + result = hex_push(ctx, *list.data.quotation_value[index.data.int_value]); + } + } + else if (list.type == HEX_TYPE_STRING) + { + if (index.type != HEX_TYPE_INTEGER) + { + hex_error(ctx, "Index must be an integer"); + result = 1; + } + else if (index.data.int_value < 0 || index.data.int_value >= (int)strlen(list.data.str_value)) + { + hex_error(ctx, "Index out of range"); + result = 1; + } + else + { + char str[2] = {list.data.str_value[index.data.int_value], '\0'}; + result = hex_push_string(ctx, str); + } + } + else + { + hex_error(ctx, "Symbol 'get' requires a quotation or a string"); + result = 1; + } + if (result != 0) + { + + FREE(ctx, list); + FREE(ctx, index); + } + return result; +} + +int hex_symbol_index(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, list); + FREE(ctx, item); + return 1; + } + int result = -1; + if (list.type == HEX_TYPE_QUOTATION) + { + for (int i = 0; i < list.quotation_size; i++) + { + if (hex_equal(*list.data.quotation_value[i], item)) + { + result = i; + break; + } + } + } + else if (list.type == HEX_TYPE_STRING) + { + char *ptr = strstr(list.data.str_value, item.data.str_value); + if (ptr) + { + result = ptr - list.data.str_value; + } + } + else + { + hex_error(ctx, "Symbol 'index' requires a quotation or a string"); + FREE(ctx, list); + FREE(ctx, item); + return 1; + } + return hex_push_integer(ctx, result); +} + +// String symbols + +int hex_symbol_join(hex_context_t *ctx) +{ + + POP(ctx, separator); + if (separator.type == HEX_TYPE_INVALID) + { + FREE(ctx, separator); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, list); + FREE(ctx, separator); + return 1; + } + int result = 0; + if (list.type == HEX_TYPE_QUOTATION && separator.type == HEX_TYPE_STRING) + { + int length = 0; + for (int i = 0; i < list.quotation_size; i++) + { + if (list.data.quotation_value[i]->type == HEX_TYPE_STRING) + { + length += strlen(list.data.quotation_value[i]->data.str_value); + } + else + { + hex_error(ctx, "Quotation must contain only strings"); + FREE(ctx, list); + FREE(ctx, separator); + return 1; + } + } + if (result == 0) + { + length += (list.quotation_size - 1) * strlen(separator.data.str_value); + char *newStr = (char *)malloc(length + 1); + if (!newStr) + { + hex_error(ctx, "Memory allocation failed"); + FREE(ctx, list); + FREE(ctx, separator); + return 1; + } + newStr[0] = '\0'; + for (int i = 0; i < list.quotation_size; i++) + { + strcat(newStr, list.data.quotation_value[i]->data.str_value); + if (i < list.quotation_size - 1) + { + strcat(newStr, separator.data.str_value); + } + } + result = hex_push_string(ctx, newStr); + } + } + else + { + hex_error(ctx, "Symbol 'join' requires a quotation and a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, list); + FREE(ctx, separator); + } + return result; +} + +int hex_symbol_split(hex_context_t *ctx) +{ + POP(ctx, separator); + if (separator.type == HEX_TYPE_INVALID) + { + FREE(ctx, separator); + return 1; + } + POP(ctx, str); + if (str.type == HEX_TYPE_INVALID) + { + FREE(ctx, str); + FREE(ctx, separator); + return 1; + } + int result = 0; + if (str.type == HEX_TYPE_STRING && separator.type == HEX_TYPE_STRING) + { + if (strlen(separator.data.str_value) == 0) + { + // Separator is an empty string: split into individual characters + int size = strlen(str.data.str_value); + hex_item_t **quotation = (hex_item_t **)malloc(size * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + for (int i = 0; i < size; i++) + { + quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (!quotation[i]) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + break; + } + quotation[i]->type = HEX_TYPE_STRING; + quotation[i]->data.str_value = (char *)malloc(2); // Allocate 2 bytes: 1 for the character and 1 for null terminator + if (!quotation[i]->data.str_value) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + break; + } + quotation[i]->data.str_value[0] = str.data.str_value[i]; // Copy the single character + quotation[i]->data.str_value[1] = '\0'; // Null-terminate the string + } + if (result == 0) + { + result = hex_push_quotation(ctx, quotation, size); + } + } + } + else + { + // Separator is not empty: split as usual + char *token = strtok(str.data.str_value, separator.data.str_value); + int capacity = 2; + int size = 0; + hex_item_t **quotation = (hex_item_t **)malloc(capacity * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + while (token) + { + if (size >= capacity) + { + capacity *= 2; + quotation = (hex_item_t **)realloc(quotation, capacity * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + break; + } + } + quotation[size] = (hex_item_t *)malloc(sizeof(hex_item_t)); + quotation[size]->type = HEX_TYPE_STRING; + quotation[size]->data.str_value = strdup(token); + size++; + token = strtok(NULL, separator.data.str_value); + } + if (result == 0) + { + result = hex_push_quotation(ctx, quotation, size); + } + } + } + } + else + { + hex_error(ctx, "Symbol 'split' requires two strings"); + result = 1; + } + if (result != 0) + { + FREE(ctx, str); + FREE(ctx, separator); + } + return result; +} + +int hex_symbol_replace(hex_context_t *ctx) +{ + + POP(ctx, replacement); + if (replacement.type == HEX_TYPE_INVALID) + { + FREE(ctx, replacement); + return 1; + } + POP(ctx, search); + if (search.type == HEX_TYPE_INVALID) + { + FREE(ctx, search); + FREE(ctx, replacement); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, list); + FREE(ctx, search); + FREE(ctx, replacement); + return 1; + } + int result = 0; + if (list.type == HEX_TYPE_STRING && search.type == HEX_TYPE_STRING && replacement.type == HEX_TYPE_STRING) + { + char *str = list.data.str_value; + char *find = search.data.str_value; + char *replace = replacement.data.str_value; + char *ptr = strstr(str, find); + if (ptr) + { + int findLen = strlen(find); + int replaceLen = strlen(replace); + int newLen = strlen(str) - findLen + replaceLen + 1; + char *newStr = (char *)malloc(newLen); + if (!newStr) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + strncpy(newStr, str, ptr - str); + strcpy(newStr + (ptr - str), replace); + strcpy(newStr + (ptr - str) + replaceLen, ptr + findLen); + result = hex_push_string(ctx, newStr); + } + } + else + { + result = hex_push_string(ctx, str); + } + } + else + { + hex_error(ctx, "Symbol 'replace' requires three strings"); + result = 1; + } + if (result != 0) + { + FREE(ctx, list); + FREE(ctx, search); + FREE(ctx, replacement); + } + return result; +} + +// File symbols + +int hex_symbol_read(hex_context_t *ctx) +{ + + POP(ctx, filename); + if (filename.type == HEX_TYPE_INVALID) + { + FREE(ctx, filename); + return 1; + } + int result = 0; + if (filename.type == HEX_TYPE_STRING) + { + FILE *file = fopen(filename.data.str_value, "r"); + if (!file) + { + hex_error(ctx, "Could not open file for reading: %s", filename.data.str_value); + result = 1; + } + else + { + fseek(file, 0, SEEK_END); + long length = ftell(file); + fseek(file, 0, SEEK_SET); + + char *buffer = (char *)malloc(length + 1); + if (!buffer) + { + hex_error(ctx, "Memory allocation failed"); + result = 1; + } + else + { + size_t bytesRead = fread(buffer, 1, length, file); + buffer[bytesRead] = '\0'; + result = hex_push_string(ctx, buffer); + free(buffer); + } + fclose(file); + } + } + else + { + hex_error(ctx, "Symbol 'read' requires a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, filename); + } + return result; +} + +int hex_symbol_write(hex_context_t *ctx) +{ + + POP(ctx, filename); + if (filename.type == HEX_TYPE_INVALID) + { + FREE(ctx, filename); + return 1; + } + POP(ctx, data); + if (data.type == HEX_TYPE_INVALID) + { + FREE(ctx, data); + FREE(ctx, filename); + return 1; + } + int result = 0; + if (filename.type == HEX_TYPE_STRING) + { + if (data.type == HEX_TYPE_STRING) + { + FILE *file = fopen(filename.data.str_value, "w"); + if (file) + { + fputs(data.data.str_value, file); + fclose(file); + result = 0; + } + else + { + hex_error(ctx, "Could not open file for writing: %s", filename.data.str_value); + result = 1; + } + } + else + { + hex_error(ctx, "Symbol 'write' requires a string"); + result = 1; + } + } + else + { + hex_error(ctx, "Symbol 'write' requires a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, data); + FREE(ctx, filename); + } + return result; +} + +int hex_symbol_append(hex_context_t *ctx) +{ + + POP(ctx, filename); + if (filename.type == HEX_TYPE_INVALID) + { + FREE(ctx, filename); + return 1; + } + POP(ctx, data); + if (data.type == HEX_TYPE_INVALID) + { + FREE(ctx, data); + FREE(ctx, filename); + return 1; + } + int result = 0; + if (filename.type == HEX_TYPE_STRING) + { + if (data.type == HEX_TYPE_STRING) + { + FILE *file = fopen(filename.data.str_value, "a"); + if (file) + { + fputs(data.data.str_value, file); + fclose(file); + result = 0; + } + else + { + hex_error(ctx, "Could not open file for appending: %s", filename.data.str_value); + result = 1; + } + } + else + { + hex_error(ctx, "Symbol 'append' requires a string"); + result = 1; + } + } + else + { + hex_error(ctx, "Symbol 'append' requires a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, data); + FREE(ctx, filename); + } + return result; +} + +// Shell symbols + +int hex_symbol_args(hex_context_t *ctx) +{ + + hex_item_t **quotation = (hex_item_t **)malloc(ctx->argc * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + return 1; + } + else + { + for (int i = 0; i < ctx->argc; i++) + { + quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); + quotation[i]->type = HEX_TYPE_STRING; + quotation[i]->data.str_value = ctx->argv[i]; + } + if (hex_push_quotation(ctx, quotation, ctx->argc) != 0) + { + hex_free_list(ctx, quotation, ctx->argc); + return 1; + } + } + return 0; +} + +int hex_symbol_exit(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (item.type != HEX_TYPE_INTEGER) + { + hex_error(ctx, "Exit status must be an integer"); + FREE(ctx, item); + return 1; + } + int exit_status = item.data.int_value; + exit(exit_status); + return 0; // This line will never be reached, but it's here to satisfy the return type +} + +int hex_symbol_exec(hex_context_t *ctx) +{ + + POP(ctx, command); + if (command.type == HEX_TYPE_INVALID) + { + FREE(ctx, command); + return 1; + } + int result = 0; + if (command.type == HEX_TYPE_STRING) + { + int status = system(command.data.str_value); + result = hex_push_integer(ctx, status); + } + else + { + hex_error(ctx, "Symbol 'exec' requires a string"); + result = 1; + } + if (result != 0) + { + FREE(ctx, command); + } + return result; +} + +int hex_symbol_run(hex_context_t *ctx) +{ + + POP(ctx, command); + if (command.type == HEX_TYPE_INVALID) + { + FREE(ctx, command); + return 1; + } + if (command.type != HEX_TYPE_STRING) + { + hex_error(ctx, "Symbol 'run' requires a string"); + FREE(ctx, command); + return 1; + } + + char output[8192] = ""; + char error[8192] = ""; + int return_code = 0; + +#ifdef _WIN32 + // Windows implementation + HANDLE hOutputRead, hOutputWrite; + HANDLE hErrorRead, hErrorWrite; + SECURITY_ATTRIBUTES sa; + PROCESS_INFORMATION pi; + STARTUPINFO si; + DWORD exitCode; + + sa.nLength = sizeof(SECURITY_ATTRIBUTES); + sa.lpSecurityDescriptor = NULL; + sa.bInheritHandle = TRUE; + + // Create pipes for capturing stdout and stderr + if (!CreatePipe(&hOutputRead, &hOutputWrite, &sa, 0) || !CreatePipe(&hErrorRead, &hErrorWrite, &sa, 0)) + { + hex_error(ctx, "Failed to create pipes"); + FREE(ctx, command); + return 1; + } + + // Set up STARTUPINFO structure + ZeroMemory(&si, sizeof(STARTUPINFO)); + si.cb = sizeof(STARTUPINFO); + si.hStdOutput = hOutputWrite; + si.hStdError = hErrorWrite; + si.dwFlags |= STARTF_USESTDHANDLES; + + // Create the child process + if (!CreateProcess(NULL, command.data.str_value, NULL, NULL, TRUE, 0, NULL, NULL, &si, &pi)) + { + hex_error(ctx, "Failed to create process"); + FREE(ctx, command); + return 1; + } + + // Close write ends of the pipes + CloseHandle(hOutputWrite); + CloseHandle(hErrorWrite); + + // Read stdout + DWORD bytesRead; + char buffer[1024]; + while (ReadFile(hOutputRead, buffer, sizeof(buffer) - 1, &bytesRead, NULL) && bytesRead > 0) + { + buffer[bytesRead] = '\0'; + strcat(output, buffer); + } + + // Read stderr + while (ReadFile(hErrorRead, buffer, sizeof(buffer) - 1, &bytesRead, NULL) && bytesRead > 0) + { + buffer[bytesRead] = '\0'; + strcat(error, buffer); + } + + // Wait for the child process to finish and get the return code + WaitForSingleObject(pi.hProcess, INFINITE); + GetExitCodeProcess(pi.hProcess, &exitCode); + return_code = exitCode; + + // Close handles + CloseHandle(pi.hProcess); + CloseHandle(pi.hThread); + CloseHandle(hOutputRead); + CloseHandle(hErrorRead); +#else + // POSIX implementation (Linux/macOS) + int stdout_pipe[2]; + int stderr_pipe[2]; + if (pipe(stdout_pipe) != 0 || pipe(stderr_pipe) != 0) + { + hex_error(ctx, "Failed to create pipes"); + FREE(ctx, command); + return 1; + } + + pid_t pid = fork(); + if (pid == -1) + { + hex_error(ctx, "Failed to fork process"); + FREE(ctx, command); + return 1; + } + else if (pid == 0) + { + // Child process + close(stdout_pipe[0]); + close(stderr_pipe[0]); + dup2(stdout_pipe[1], STDOUT_FILENO); + dup2(stderr_pipe[1], STDERR_FILENO); + execl("/bin/sh", "sh", "-c", command.data.str_value, (char *)NULL); + exit(1); + } + else + { + // Parent process + close(stdout_pipe[1]); + close(stderr_pipe[1]); + + // Read stdout + FILE *stdout_fp = fdopen(stdout_pipe[0], "r"); + char path[1035]; + while (fgets(path, sizeof(path), stdout_fp) != NULL) + { + strcat(output, path); + } + fclose(stdout_fp); + + // Read stderr + FILE *stderr_fp = fdopen(stderr_pipe[0], "r"); + while (fgets(path, sizeof(path), stderr_fp) != NULL) + { + strcat(error, path); + } + fclose(stderr_fp); + + // Wait for child process to finish and get the return code + int status; + waitpid(pid, &status, 0); + return_code = WEXITSTATUS(status); + } +#endif + + // Push the return code, output, and error as a quotation + hex_item_t **quotation = (hex_item_t **)malloc(3 * sizeof(hex_item_t *)); + quotation[0] = (hex_item_t *)malloc(sizeof(hex_item_t)); + quotation[0]->type = HEX_TYPE_INTEGER; + quotation[0]->data.int_value = return_code; + + quotation[1] = (hex_item_t *)malloc(sizeof(hex_item_t)); + quotation[1]->type = HEX_TYPE_STRING; + quotation[1]->data.str_value = strdup(output); + + quotation[2] = (hex_item_t *)malloc(sizeof(hex_item_t)); + quotation[2]->type = HEX_TYPE_STRING; + quotation[2]->data.str_value = strdup(error); + + return hex_push_quotation(ctx, quotation, 3); +} + +// Control flow symbols + +int hex_symbol_if(hex_context_t *ctx) +{ + + POP(ctx, elseBlock); + if (elseBlock.type == HEX_TYPE_INVALID) + { + FREE(ctx, elseBlock); + return 1; + } + POP(ctx, thenBlock); + if (thenBlock.type == HEX_TYPE_INVALID) + { + FREE(ctx, thenBlock); + FREE(ctx, elseBlock); + return 1; + } + POP(ctx, condition); + if (condition.type == HEX_TYPE_INVALID) + { + FREE(ctx, condition); + FREE(ctx, thenBlock); + FREE(ctx, elseBlock); + return 1; + } + if (condition.type != HEX_TYPE_QUOTATION || thenBlock.type != HEX_TYPE_QUOTATION || elseBlock.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'if' symbol requires three quotations"); + return 1; + } + else + { + for (int i = 0; i < condition.quotation_size; i++) + { + if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) + { + FREE(ctx, condition); + FREE(ctx, thenBlock); + FREE(ctx, elseBlock); + return 1; + } + } + POP(ctx, evalResult); + if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) + { + for (int i = 0; i < thenBlock.quotation_size; i++) + { + if (hex_push(ctx, *thenBlock.data.quotation_value[i]) != 0) + { + FREE(ctx, condition); + FREE(ctx, thenBlock); + FREE(ctx, elseBlock); + return 1; + } + } + } + else + { + for (int i = 0; i < elseBlock.quotation_size; i++) + { + if (hex_push(ctx, *elseBlock.data.quotation_value[i]) != 0) + { + FREE(ctx, condition); + FREE(ctx, thenBlock); + FREE(ctx, elseBlock); + return 1; + } + } + } + } + return 0; +} + +int hex_symbol_when(hex_context_t *ctx) +{ + + POP(ctx, action); + if (action.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + return 1; + } + POP(ctx, condition); + if (condition.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + FREE(ctx, condition); + return 1; + } + int result = 0; + if (condition.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'when' symbol requires two quotations"); + result = 1; + } + else + { + for (int i = 0; i < condition.quotation_size; i++) + { + if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) + { + result = 1; + break; // Break if pushing the item failed + } + } + POP(ctx, evalResult); + if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) + { + for (int i = 0; i < action.quotation_size; i++) + { + if (hex_push(ctx, *action.data.quotation_value[i]) != 0) + { + result = 1; + break; + } + } + } + } + if (result != 0) + { + FREE(ctx, action); + FREE(ctx, condition); + } + return result; +} + +int hex_symbol_while(hex_context_t *ctx) +{ + + POP(ctx, action); + if (action.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + return 1; + } + POP(ctx, condition); + if (condition.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + FREE(ctx, condition); + return 1; + } + if (condition.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'while' symbol requires two quotations"); + FREE(ctx, action); + FREE(ctx, condition); + return 1; + } + else + { + while (1) + { + for (int i = 0; i < condition.quotation_size; i++) + { + if (hex_push(ctx, *condition.data.quotation_value[i]) != 0) + { + FREE(ctx, action); + FREE(ctx, condition); + return 1; + } + } + POP(ctx, evalResult); + if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value == 0) + { + break; + } + for (int i = 0; i < action.quotation_size; i++) + { + if (hex_push(ctx, *action.data.quotation_value[i]) != 0) + { + FREE(ctx, action); + FREE(ctx, condition); + return 1; + } + } + } + } + return 0; +} + +int hex_symbol_each(hex_context_t *ctx) +{ + + POP(ctx, action); + if (action.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'each' symbol requires two quotations"); + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + else + { + for (int i = 0; i < list.quotation_size; i++) + { + if (hex_push(ctx, *list.data.quotation_value[i]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + for (int j = 0; j < action.quotation_size; j++) + { + if (hex_push(ctx, *action.data.quotation_value[j]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + } + } + } + return 0; +} + +int hex_symbol_error(hex_context_t *ctx) +{ + + char *message = strdup(ctx->error); + ctx->error[0] = '\0'; + return hex_push_string(ctx, message); +} + +int hex_symbol_try(hex_context_t *ctx) +{ + + POP(ctx, catch_block); + if (catch_block.type == HEX_TYPE_INVALID) + { + FREE(ctx, catch_block); + return 1; + } + POP(ctx, try_block); + if (try_block.type == HEX_TYPE_INVALID) + { + FREE(ctx, catch_block); + FREE(ctx, try_block); + return 1; + } + if (try_block.type != HEX_TYPE_QUOTATION || catch_block.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'try' symbol requires two quotations"); + FREE(ctx, catch_block); + FREE(ctx, try_block); + return 1; + } + else + { + char prevError[256]; + strncpy(prevError, ctx->error, sizeof(ctx->error)); + ctx->error[0] = '\0'; + + ctx->settings.errors_enabled = 0; + for (int i = 0; i < try_block.quotation_size; i++) + { + if (hex_push(ctx, *try_block.data.quotation_value[i]) != 0) + { + ctx->settings.errors_enabled = 1; + } + } + ctx->settings.errors_enabled = 1; + + if (strcmp(ctx->error, "")) + { + for (int i = 0; i < catch_block.quotation_size; i++) + { + if (hex_push(ctx, *catch_block.data.quotation_value[i]) != 0) + { + FREE(ctx, catch_block); + FREE(ctx, try_block); + return 1; + } + } + } + + strncpy(ctx->error, prevError, sizeof(ctx->error)); + } + return 0; +} + +// Quotation symbols + +int hex_symbol_q(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + + hex_item_t *quotation = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + FREE(ctx, item); + return 1; + } + + *quotation = item; + + hex_item_t result; + result.type = HEX_TYPE_QUOTATION; + result.data.quotation_value = (hex_item_t **)malloc(sizeof(hex_item_t *)); + if (!result.data.quotation_value) + { + FREE(ctx, item); + free(quotation); + hex_error(ctx, "Memory allocation failed"); + return 1; + } + + result.data.quotation_value[0] = quotation; + result.quotation_size = 1; + + if (PUSH(ctx, result) != 0) + { + FREE(ctx, item); + free(quotation); + free(result.data.quotation_value); + return 1; + } + + return 0; +} + +int hex_symbol_map(hex_context_t *ctx) +{ + + POP(ctx, action); + if (action.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'map' symbol requires two quotations"); + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + else + { + hex_item_t **quotation = (hex_item_t **)malloc(list.quotation_size * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + for (int i = 0; i < list.quotation_size; i++) + { + if (hex_push(ctx, *list.data.quotation_value[i]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, i); + return 1; + } + for (int j = 0; j < action.quotation_size; j++) + { + if (hex_push(ctx, *action.data.quotation_value[j]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, i); + return 1; + } + } + quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); + *quotation[i] = hex_pop(ctx); + } + if (hex_push_quotation(ctx, quotation, list.quotation_size) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, list.quotation_size); + return 1; + } + } + return 0; +} + +int hex_symbol_filter(hex_context_t *ctx) +{ + + POP(ctx, action); + if (action.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + return 1; + } + POP(ctx, list); + if (list.type == HEX_TYPE_INVALID) + { + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + if (list.type != HEX_TYPE_QUOTATION || action.type != HEX_TYPE_QUOTATION) + { + hex_error(ctx, "'filter' symbol requires two quotations"); + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + else + { + hex_item_t **quotation = (hex_item_t **)malloc(list.quotation_size * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + FREE(ctx, action); + FREE(ctx, list); + return 1; + } + int count = 0; + for (int i = 0; i < list.quotation_size; i++) + { + if (hex_push(ctx, *list.data.quotation_value[i]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, count); + return 1; + } + for (int j = 0; j < action.quotation_size; j++) + { + if (hex_push(ctx, *action.data.quotation_value[j]) != 0) + { + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, count); + return 1; + } + } + POP(ctx, evalResult); + if (evalResult.type == HEX_TYPE_INTEGER && evalResult.data.int_value > 0) + { + quotation[count] = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (!quotation[count]) + { + hex_error(ctx, "Memory allocation failed"); + FREE(ctx, action); + FREE(ctx, list); + hex_free_list(ctx, quotation, count); + return 1; + } + *quotation[count] = *list.data.quotation_value[i]; + count++; + } + } + if (hex_push_quotation(ctx, quotation, count) != 0) + { + hex_error(ctx, "An error occurred while filtering the list"); + FREE(ctx, action); + FREE(ctx, list); + for (int i = 0; i < count; i++) + { + FREE(ctx, *quotation[i]); + } + return 1; + } + } + return 0; +} + +// Stack manipulation symbols + +int hex_symbol_swap(hex_context_t *ctx) +{ + + POP(ctx, a); + if (a.type == HEX_TYPE_INVALID) + { + FREE(ctx, a); + return 1; + } + POP(ctx, b); + if (b.type == HEX_TYPE_INVALID) + { + FREE(ctx, b); + FREE(ctx, a); + return 1; + } + if (PUSH(ctx, a) != 0) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + if (PUSH(ctx, b) != 0) + { + FREE(ctx, a); + FREE(ctx, b); + return 1; + } + return 0; +} + +int hex_symbol_dup(hex_context_t *ctx) +{ + + POP(ctx, item); + if (item.type == HEX_TYPE_INVALID) + { + FREE(ctx, item); + return 1; + } + if (PUSH(ctx, item) == 0 && PUSH(ctx, item) == 0) + { + return 0; + } + FREE(ctx, item); + return 1; +} + +int hex_symbol_stack(hex_context_t *ctx) +{ + + hex_item_t **quotation = (hex_item_t **)malloc((ctx->stack.top + 1) * sizeof(hex_item_t *)); + if (!quotation) + { + hex_error(ctx, "Memory allocation failed"); + return 1; + } + int count = 0; + for (int i = 0; i <= ctx->stack.top; i++) + { + quotation[i] = (hex_item_t *)malloc(sizeof(hex_item_t)); + if (!quotation[i]) + { + hex_error(ctx, "Memory allocation failed"); + hex_free_list(ctx, quotation, count); + return 1; + } + *quotation[i] = ctx->stack.entries[i]; + //*quotation[i] = HEX_STACK[i]; + count++; + } + + if (hex_push_quotation(ctx, quotation, ctx->stack.top + 1) != 0) + { + hex_error(ctx, "An error occurred while fetching stack contents"); + hex_free_list(ctx, quotation, count); + return 1; + } + return 0; +} + +int hex_symbol_clear(hex_context_t *ctx) +{ + + while (ctx->stack.top >= 0) + { + FREE(ctx, ctx->stack.entries[ctx->stack.top--]); + // FREE(ctx, HEX_STACK[HEX_TOP--]); + } + ctx->stack.top = -1; + return 0; +} + +int hex_symbol_pop(hex_context_t *ctx) +{ + + POP(ctx, item); + FREE(ctx, item); + return 0; +} + +//////////////////////////////////////// +// Native Symbol Registration // +//////////////////////////////////////// + +void hex_register_symbols(hex_context_t *ctx) +{ + hex_set_native_symbol(ctx, ":", hex_symbol_store); + hex_set_native_symbol(ctx, "#", hex_symbol_free); + hex_set_native_symbol(ctx, "type", hex_symbol_type); + hex_set_native_symbol(ctx, ".", hex_symbol_i); + hex_set_native_symbol(ctx, "!", hex_symbol_eval); + hex_set_native_symbol(ctx, "puts", hex_symbol_puts); + hex_set_native_symbol(ctx, "warn", hex_symbol_warn); + hex_set_native_symbol(ctx, "print", hex_symbol_print); + hex_set_native_symbol(ctx, "gets", hex_symbol_gets); + hex_set_native_symbol(ctx, "+", hex_symbol_add); + hex_set_native_symbol(ctx, "-", hex_symbol_subtract); + hex_set_native_symbol(ctx, "*", hex_symbol_multiply); + hex_set_native_symbol(ctx, "/", hex_symbol_divide); + hex_set_native_symbol(ctx, "%", hex_symbol_modulo); + hex_set_native_symbol(ctx, "&", hex_symbol_bitand); + hex_set_native_symbol(ctx, "|", hex_symbol_bitor); + hex_set_native_symbol(ctx, "^", hex_symbol_bitxor); + hex_set_native_symbol(ctx, "~", hex_symbol_bitnot); + hex_set_native_symbol(ctx, "<<", hex_symbol_shiftleft); + hex_set_native_symbol(ctx, ">>", hex_symbol_shiftright); + hex_set_native_symbol(ctx, "int", hex_symbol_int); + hex_set_native_symbol(ctx, "str", hex_symbol_str); + hex_set_native_symbol(ctx, "dec", hex_symbol_dec); + hex_set_native_symbol(ctx, "hex", hex_symbol_hex); + hex_set_native_symbol(ctx, "chr", hex_symbol_chr); + hex_set_native_symbol(ctx, "ord", hex_symbol_ord); + hex_set_native_symbol(ctx, "==", hex_symbol_equal); + hex_set_native_symbol(ctx, "!=", hex_symbol_notequal); + hex_set_native_symbol(ctx, ">", hex_symbol_greater); + hex_set_native_symbol(ctx, "<", hex_symbol_less); + hex_set_native_symbol(ctx, ">=", hex_symbol_greaterequal); + hex_set_native_symbol(ctx, "<=", hex_symbol_lessequal); + hex_set_native_symbol(ctx, "and", hex_symbol_and); + hex_set_native_symbol(ctx, "or", hex_symbol_or); + hex_set_native_symbol(ctx, "not", hex_symbol_not); + hex_set_native_symbol(ctx, "xor", hex_symbol_xor); + hex_set_native_symbol(ctx, "cat", hex_symbol_cat); + hex_set_native_symbol(ctx, "len", hex_symbol_len); + hex_set_native_symbol(ctx, "get", hex_symbol_get); + hex_set_native_symbol(ctx, "index", hex_symbol_index); + hex_set_native_symbol(ctx, "join", hex_symbol_join); + hex_set_native_symbol(ctx, "split", hex_symbol_split); + hex_set_native_symbol(ctx, "replace", hex_symbol_replace); + hex_set_native_symbol(ctx, "read", hex_symbol_read); + hex_set_native_symbol(ctx, "write", hex_symbol_write); + hex_set_native_symbol(ctx, "append", hex_symbol_append); + hex_set_native_symbol(ctx, "args", hex_symbol_args); + hex_set_native_symbol(ctx, "exit", hex_symbol_exit); + hex_set_native_symbol(ctx, "exec", hex_symbol_exec); + hex_set_native_symbol(ctx, "run", hex_symbol_run); + hex_set_native_symbol(ctx, "if", hex_symbol_if); + hex_set_native_symbol(ctx, "when", hex_symbol_when); + hex_set_native_symbol(ctx, "while", hex_symbol_while); + hex_set_native_symbol(ctx, "each", hex_symbol_each); + hex_set_native_symbol(ctx, "error", hex_symbol_error); + hex_set_native_symbol(ctx, "try", hex_symbol_try); + hex_set_native_symbol(ctx, "'", hex_symbol_q); + hex_set_native_symbol(ctx, "map", hex_symbol_map); + hex_set_native_symbol(ctx, "filter", hex_symbol_filter); + hex_set_native_symbol(ctx, "swap", hex_symbol_swap); + hex_set_native_symbol(ctx, "dup", hex_symbol_dup); + hex_set_native_symbol(ctx, "stack", hex_symbol_stack); + hex_set_native_symbol(ctx, "clear", hex_symbol_clear); + hex_set_native_symbol(ctx, "pop", hex_symbol_pop); +}