diff options
34 files changed, 6360 insertions, 405 deletions
@@ -25,6 +25,8 @@ SRCS = src/main.c \ src/lsp/lsp_main.c \ src/lsp/lsp_analysis.c \ src/lsp/lsp_index.c \ + src/lsp/lsp_project.c \ + src/lsp/cJSON.c \ src/zen/zen_facts.c \ src/repl/repl.c \ src/plugins/plugin_manager.c @@ -65,6 +65,7 @@ Join the discussion, share demos, ask questions, or report bugs in the official - [8. Memory Management](#8-memory-management) - [Defer](#defer) - [Autofree](#autofree) + - [Resource Semantics (Move by Default)](#resource-semantics-move-by-default) - [RAII / Drop Trait](#raii--drop-trait) - [9. Object Oriented Programming](#9-object-oriented-programming) - [Methods](#methods) @@ -84,12 +85,16 @@ Join the discussion, share demos, ask questions, or report bugs in the official - [Volatile](#volatile) - [Named Constraints](#named-constraints) - [15. Build Directives](#15-build-directives) +- [Tooling](#tooling) + - [Language Server (LSP)](#language-server-lsp) + - [REPL](#repl) - [Compiler Support & Compatibility](#compiler-support--compatibility) - [Test Suite Status](#test-suite-status) - [Building with Zig](#building-with-zig) - [C++ Interop](#c-interop) - [CUDA Interop](#cuda-interop) - [Contributing](#contributing) +- [Attributions](#attributions) --- @@ -189,6 +194,8 @@ struct Flags { } ``` +> **Note**: Structs use [Move Semantics](#move-semantics--copy-safety) by default. Fields can be accessed via `.` even on pointers (Auto-Dereference). + #### Enums Tagged unions (Sum types) capable of holding data. ```zc @@ -273,6 +280,21 @@ match shape { } ``` +#### Reference Binding +To inspect a value without taking ownership (moving it), use the `ref` keyword in the pattern. This is essential for types that implement Move Semantics (like `Option`, `Result`, non-Copy structs). + +```zc +var opt = Some(NonCopyVal{...}); +match opt { + Some(ref x) => { + // 'x' is a pointer to the value inside 'opt' + // 'opt' is NOT moved/consumed here + print(x.field); + }, + None => {} +} +``` + #### Loops ```zc // Range @@ -347,6 +369,9 @@ These operators are built-in language features and cannot be overloaded directly | `?.` | Safe Navigation | `ptr?.field` accesses field only if `ptr` is not NULL | | `?` | Try Operator | `res?` returns error if present (Result/Option types) | +**Auto-Dereference**: +Pointer field access (`ptr.field`) and method calls (`ptr.method()`) automatically dereference the pointer, equivalent to `(*ptr).field`. + ### 7. Printing and String Interpolation Zen C provides versatile options for printing to the console, including keywords and concise shorthands. @@ -409,6 +434,45 @@ Automatically free the variable when scope exits. autofree var types = malloc(1024); ``` +#### Resource Semantics (Move by Default) +Zen C treats types with destructors (like `File`, `Vec`, or malloc'd pointers) as **Resources**. To prevent double-free errors, resources cannot be implicitly duplicated. + +- **Move by Default**: Assigning a resource variable transfers ownership. The original variable becomes invalid (Moved). +- **Copy Types**: Types without destructors may opt-in to `Copy` behavior, making assignment a duplication. + +**Diagnostics & Philosophy**: +If you see an error "Use of moved value", the compiler is telling you: *"This type owns a resource (like memory or a handle) and blindly copying it is unsafe."* + +> **Contrast:** Unlike C/C++, Zen C does not implicitly duplicate resource-owning values. + +**Function Arguments**: +Passing a value to a function follows the same rules as assignment: resources are moved unless passed by reference. + +```zc +fn process(r: Resource) { ... } // 'r' is moved into function +fn peek(r: Resource*) { ... } // 'r' is borrowed (reference) +``` + +**Explicit Cloning**: +If you *do* want two copies of a resource, make it explicit: + +```zc +var b = a.clone(); // Calls the 'clone' method from the Clone trait +``` + +**Opt-in Copy (Value Types)**: +For small types without destructors: + +```zc +struct Point { x: int; y: int; } +impl Copy for Point {} // Opt-in to implicit duplication + +fn main() { + var p1 = Point { x: 1, y: 2 }; + var p2 = p1; // Copied. p1 stays valid. +} +``` + #### RAII / Drop Trait Implement `Drop` to run cleanup logic automatically. ```zc @@ -513,6 +577,45 @@ impl Drop for Resource { } ``` +> **Note:** If a variable is moved, `drop` is NOT called on the original variable. It adheres to [Resource Semantics](#resource-semantics-move-by-default). + +**Copy** + +Marker trait to opt-in to `Copy` behavior (implicit duplication) instead of Move semantics. Used via `@derive(Copy)`. + +> **Rule:** Types that implement `Copy` must not define a destructor (`Drop`). + +```zc +@derive(Copy) +struct Point { x: int; y: int; } + +fn main() { + var p1 = Point{x: 1, y: 2}; + var p2 = p1; // Copied! p1 remains valid. +} +``` + +**Clone** + +Implement `Clone` to allow explicit duplication of resource-owning types. + +```zc +import "std/mem.zc" + +struct MyBox { val: int; } + +impl Clone for MyBox { + fn clone(self) -> MyBox { + return MyBox{val: self.val}; + } +} + +fn main() { + var b1 = MyBox{val: 42}; + var b2 = b1.clone(); // Explicit copy +} +``` + #### Composition Use `use` to embed other structs. You can either mix them in (flatten fields) or name them (nest fields). @@ -626,7 +729,15 @@ Decorate functions and structs to modify compiler behavior. | `@weak` | Fn | Weak symbol linkage. | | `@section("name")` | Fn | Place code in specific section. | | `@noreturn` | Fn | Function does not return (e.g. exit). | -| `@derived(...)` | Struct | Auto-implement traits (e.g. `Debug`). | +| `@derive(...)` | Struct | Auto-implement traits. Supports `Debug`, `Eq` (Smart Derive), `Copy`, `Clone`. | + +### Smart Derives + +Zen C provides "Smart Derives" that respect Move Semantics: + +- **`@derive(Eq)`**: Generates an equality method that takes arguments by reference (`fn eq(self, other: T*)`). + - When comparing two non-Copy structs (`a == b`), the compiler automatically passes `b` by reference (`&b`) to avoid moving it. + - Recursive equality checks on fields also prefer pointer access to prevent ownership transfer. ### 14. Inline Assembly @@ -708,6 +819,72 @@ fn main() { ... } --- +## Tooling + +Zen C provides a built-in Language Server and REPL to enhance the development experience. + +### Language Server (LSP) + +The Zen C Language Server (LSP) supports standard LSP features for editor integration, providing: + +* **Go to Definition** +* **Find References** +* **Hover Information** +* **Completion** (Function/Struct names, Dot-completion for methods/fields) +* **Document Symbols** (Outline) +* **Signature Help** +* **Diagnostics** (Syntax/Semantic errors) + +To start the language server (typically configured in your editor's LSP settings): + +```bash +zc lsp +``` + +It communicates via standard I/O (JSON-RPC 2.0). + +### REPL + +The Read-Eval-Print Loop allows you to experiment with Zen C code interactively. + +```bash +zc repl +``` + +#### Features + +* **Interactive Coding**: Type expressions or statements for immediate evaluation. +* **Persistent History**: Commands are saved to `~/.zprep_history`. +* **Startup Script**: Auto-loads commands from `~/.zprep_init.zc`. + +#### Commands + +| Command | Description | +|:---|:---| +| `:help` | Show available commands. | +| `:reset` | Clear current session history (variables/functions). | +| `:vars` | Show active variables. | +| `:funcs` | Show user-defined functions. | +| `:structs` | Show user-defined structs. | +| `:imports` | Show active imports. | +| `:history` | Show session input history. | +| `:type <expr>` | Show the type of an expression. | +| `:c <stmt>` | Show the generated C code for a statement. | +| `:time <expr>` | Benchmark an expression (runs 1000 iterations). | +| `:edit [n]` | Edit command `n` (default: last) in `$EDITOR`. | +| `:save <file>` | Save the current session to a `.zc` file. | +| `:load <file>` | Load and execute a `.zc` file into the session. | +| `:watch <expr>` | Watch an expression (re-evaluated after every entry). | +| `:unwatch <n>` | Remove a watch. | +| `:undo` | Remove the last command from the session. | +| `:delete <n>` | Remove command at index `n`. | +| `:clear` | Clear the screen. | +| `:quit` | Exit the REPL. | +| `! <cmd>` | Run a shell command (e.g. `!ls`). | + +--- + + ## Compiler Support & Compatibility Zen C is designed to work with most C11 compilers. Some features rely on GNU C extensions, but these often work in other compilers. Use the `--cc` flag to switch backends. @@ -904,3 +1081,12 @@ make test * **Parser**: `src/parser/` - Recursive descent parser. * **Codegen**: `src/codegen/` - Transpiler logic (Zen C -> GNU C/C11). * **Standard Library**: `std/` - Written in Zen C itself. + +--- + +## Attributions + +This project uses the following third-party libraries: + +* **cJSON** (MIT License): Used for JSON parsing and generation in the Language Server. + * Copyright (c) 2009-2017 Dave Gamble and cJSON contributors diff --git a/src/analysis/typecheck.c b/src/analysis/typecheck.c index e3abf10..2b06bb4 100644 --- a/src/analysis/typecheck.c +++ b/src/analysis/typecheck.c @@ -69,10 +69,114 @@ static Symbol *tc_lookup(TypeChecker *tc, const char *name) return NULL; } +// ** Move Semantics Helpers ** + +static int is_safe_to_copy(TypeChecker *tc, Type *t) +{ + // Use parser's helper if available, or simple heuristic + return is_type_copy(tc->pctx, t); +} + +static void check_use_validity(TypeChecker *tc, ASTNode *var_node, Symbol *sym) +{ + if (!sym || !var_node) + { + return; + } + + if (sym->is_moved) + { + char msg[256]; + snprintf( + msg, 255, + "Use of moved value '%s'. This type owns resources and cannot be implicitly copied.", + sym->name); + tc_error(tc, var_node->token, msg); + } +} + +static void mark_symbol_moved(TypeChecker *tc, Symbol *sym, ASTNode *context_node) +{ + if (!sym) + { + return; + } + + // Only move if type is NOT Copy + Type *t = sym->type_info; + if (t && !is_safe_to_copy(tc, t)) + { + sym->is_moved = 1; + } +} + +static void mark_symbol_valid(TypeChecker *tc, Symbol *sym) +{ + if (sym) + { + sym->is_moved = 0; + } +} + // ** Node Checkers ** static void check_node(TypeChecker *tc, ASTNode *node); +static void check_expr_binary(TypeChecker *tc, ASTNode *node) +{ + check_node(tc, node->binary.left); + check_node(tc, node->binary.right); + + // Assignment Logic for Moves + if (strcmp(node->binary.op, "=") == 0) + { + // If RHS is a var, it might Move + if (node->binary.right->type == NODE_EXPR_VAR) + { + Symbol *rhs_sym = tc_lookup(tc, node->binary.right->var_ref.name); + if (rhs_sym) + { + mark_symbol_moved(tc, rhs_sym, node); + } + } + + // LHS is being (re-)initialized, so it becomes Valid. + if (node->binary.left->type == NODE_EXPR_VAR) + { + Symbol *lhs_sym = tc_lookup(tc, node->binary.left->var_ref.name); + if (lhs_sym) + { + mark_symbol_valid(tc, lhs_sym); + } + } + } +} + +static void check_expr_call(TypeChecker *tc, ASTNode *node) +{ + check_node(tc, node->call.callee); + + // Check arguments + ASTNode *arg = node->call.args; + while (arg) + { + check_node(tc, arg); + + // If argument is passed by VALUE, and it's a variable, it MOVES. + // If passed by ref (UNARY '&'), the child was checked but Is Not A Var Node itself. + if (arg->type == NODE_EXPR_VAR) + { + Symbol *sym = tc_lookup(tc, arg->var_ref.name); + if (sym) + { + mark_symbol_moved(tc, sym, node); + } + } + + arg = arg->next; + } +} + static void check_block(TypeChecker *tc, ASTNode *block) { tc_enter_scope(tc); @@ -140,6 +244,16 @@ static void check_var_decl(TypeChecker *tc, ASTNode *node) { check_type_compatibility(tc, decl_type, init_type, node->token); } + + // Move Analysis: If initializing from another variable, it moves. + if (node->var_decl.init_expr->type == NODE_EXPR_VAR) + { + Symbol *init_sym = tc_lookup(tc, node->var_decl.init_expr->var_ref.name); + if (init_sym) + { + mark_symbol_moved(tc, init_sym, node); + } + } } // If type is not explicit, we should ideally infer it from init_expr. @@ -187,6 +301,9 @@ static void check_expr_var(TypeChecker *tc, ASTNode *node) { node->type_info = sym->type_info; } + + // Check for Use-After-Move + check_use_validity(tc, node, sym); } static void check_node(TypeChecker *tc, ASTNode *node) @@ -240,15 +357,26 @@ static void check_node(TypeChecker *tc, ASTNode *node) tc_exit_scope(tc); break; case NODE_EXPR_BINARY: - check_node(tc, node->binary.left); - check_node(tc, node->binary.right); + check_expr_binary(tc, node); break; case NODE_EXPR_CALL: - check_node(tc, node->call.callee); - check_node(tc, node->call.args); + check_expr_call(tc, node); break; default: - // Generic recursion for lists. + // Generic recursion for lists and other nodes. + // Special case for Return to trigger move? + if (node->type == NODE_RETURN && node->ret.value) + { + // If returning a variable by value, it is moved. + if (node->ret.value->type == NODE_EXPR_VAR) + { + Symbol *sym = tc_lookup(tc, node->ret.value->var_ref.name); + if (sym) + { + mark_symbol_moved(tc, sym, node); + } + } + } break; } diff --git a/src/ast/ast.h b/src/ast/ast.h index 2233b09..508a247 100644 --- a/src/ast/ast.h +++ b/src/ast/ast.h @@ -303,6 +303,7 @@ struct ASTNode char *pattern; char *binding_name; int is_destructuring; + int is_ref; // New: Supports 'ref' binding (Some(ref x)) ASTNode *guard; ASTNode *body; int is_default; diff --git a/src/codegen/codegen.c b/src/codegen/codegen.c index 01c8204..b2f1ad1 100644 --- a/src/codegen/codegen.c +++ b/src/codegen/codegen.c @@ -246,38 +246,89 @@ static void codegen_match_internal(ParserContext *ctx, ASTNode *node, FILE *out, { if (strstr(g_config.cc, "tcc")) { - fprintf(out, "__typeof__(_m_%d.val) %s = _m_%d.val; ", id, - c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "__typeof__(&_m_%d.val) %s = &_m_%d.val; ", id, + c->match_case.binding_name, id); + } + else + { + fprintf(out, "__typeof__(_m_%d.val) %s = _m_%d.val; ", id, + c->match_case.binding_name, id); + } } else { - fprintf(out, "ZC_AUTO %s = _m_%d.val; ", c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "ZC_AUTO %s = &_m_%d.val; ", c->match_case.binding_name, id); + } + else + { + fprintf(out, "ZC_AUTO %s = _m_%d.val; ", c->match_case.binding_name, id); + } } } - if (is_result) + else if (is_result) // FIX: Changed 'if' to 'else if' to match original logic structure + // if needed, but original code had implicit fallthrough checks? No, + // checks match pattern. { if (strcmp(c->match_case.pattern, "Ok") == 0) { if (strstr(g_config.cc, "tcc")) { - fprintf(out, "__typeof__(_m_%d.val) %s = _m_%d.val; ", id, - c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "__typeof__(&_m_%d.val) %s = &_m_%d.val; ", id, + c->match_case.binding_name, id); + } + else + { + fprintf(out, "__typeof__(_m_%d.val) %s = _m_%d.val; ", id, + c->match_case.binding_name, id); + } } else { - fprintf(out, "ZC_AUTO %s = _m_%d.val; ", c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "ZC_AUTO %s = &_m_%d.val; ", c->match_case.binding_name, + id); + } + else + { + fprintf(out, "ZC_AUTO %s = _m_%d.val; ", c->match_case.binding_name, + id); + } } } else { if (strstr(g_config.cc, "tcc")) { - fprintf(out, "__typeof__(_m_%d.err) %s = _m_%d.err; ", id, - c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "__typeof__(&_m_%d.err) %s = &_m_%d.err; ", id, + c->match_case.binding_name, id); + } + else + { + fprintf(out, "__typeof__(_m_%d.err) %s = _m_%d.err; ", id, + c->match_case.binding_name, id); + } } else { - fprintf(out, "ZC_AUTO %s = _m_%d.err; ", c->match_case.binding_name, id); + if (c->match_case.is_ref) + { + fprintf(out, "ZC_AUTO %s = &_m_%d.err; ", c->match_case.binding_name, + id); + } + else + { + fprintf(out, "ZC_AUTO %s = _m_%d.err; ", c->match_case.binding_name, + id); + } } } } @@ -292,7 +343,18 @@ static void codegen_match_internal(ParserContext *ctx, ASTNode *node, FILE *out, { f = c->match_case.pattern; } - fprintf(out, "ZC_AUTO %s = _m_%d.data.%s; ", c->match_case.binding_name, id, f); + // Generic struct destructuring (e.g. MyStruct_Variant) + // Assuming data union or accessible field. + // Original: _m_%d.data.%s + if (c->match_case.is_ref) + { + fprintf(out, "ZC_AUTO %s = &_m_%d.data.%s; ", c->match_case.binding_name, id, + f); + } + else + { + fprintf(out, "ZC_AUTO %s = _m_%d.data.%s; ", c->match_case.binding_name, id, f); + } } } @@ -461,7 +523,7 @@ void codegen_expression(ParserContext *ctx, ASTNode *node, FILE *out) } fprintf(out, "%s__eq(&", base); codegen_expression(ctx, node->binary.left, out); - fprintf(out, ", "); + fprintf(out, ", &"); codegen_expression(ctx, node->binary.right, out); fprintf(out, ")"); if (strcmp(node->binary.op, "!=") == 0) @@ -641,21 +703,57 @@ void codegen_expression(ParserContext *ctx, ASTNode *node, FILE *out) char mixin_func_name[128]; sprintf(mixin_func_name, "%s__%s", base, method); + char *resolved_method_suffix = NULL; + if (!find_func(ctx, mixin_func_name)) { - // Method not found on primary struct, check mixins - ASTNode *def = find_struct_def(ctx, base); - if (def && def->type == NODE_STRUCT && def->strct.used_structs) + // Try resolving as a trait method: Struct__Trait_Method + StructRef *ref = ctx->parsed_impls_list; + while (ref) { - for (int k = 0; k < def->strct.used_struct_count; k++) + if (ref->node && ref->node->type == NODE_IMPL_TRAIT) { - char mixin_check[128]; - sprintf(mixin_check, "%s__%s", def->strct.used_structs[k], method); - if (find_func(ctx, mixin_check)) + if (strcmp(ref->node->impl_trait.target_type, base) == 0) { - call_base = def->strct.used_structs[k]; - need_cast = 1; - break; + char trait_mangled[256]; + sprintf(trait_mangled, "%s__%s_%s", base, + ref->node->impl_trait.trait_name, method); + if (find_func(ctx, trait_mangled)) + { + char *suffix = + xmalloc(strlen(ref->node->impl_trait.trait_name) + + strlen(method) + 2); + sprintf(suffix, "%s_%s", ref->node->impl_trait.trait_name, + method); + resolved_method_suffix = suffix; + break; + } + } + } + ref = ref->next; + } + + if (resolved_method_suffix) + { + method = resolved_method_suffix; + } + else + { + // Method not found on primary struct, check mixins + ASTNode *def = find_struct_def(ctx, base); + if (def && def->type == NODE_STRUCT && def->strct.used_structs) + { + for (int k = 0; k < def->strct.used_struct_count; k++) + { + char mixin_check[128]; + sprintf(mixin_check, "%s__%s", def->strct.used_structs[k], + method); + if (find_func(ctx, mixin_check)) + { + call_base = def->strct.used_structs[k]; + need_cast = 1; + break; + } } } } @@ -1685,7 +1783,6 @@ void codegen_node_single(ParserContext *ctx, ASTNode *node, FILE *out) } { char *tname = NULL; - // Use type_info with codegen_type_to_string if available if (node->type_info) { tname = codegen_type_to_string(node->type_info); @@ -1697,11 +1794,16 @@ void codegen_node_single(ParserContext *ctx, ASTNode *node, FILE *out) if (tname) { - // Cleanup attribute - ASTNode *def = find_struct_def(ctx, tname); + char *clean_type = tname; + if (strncmp(clean_type, "struct ", 7) == 0) + { + clean_type += 7; + } + + ASTNode *def = find_struct_def(ctx, clean_type); if (def && def->type_info && def->type_info->traits.has_drop) { - fprintf(out, "__attribute__((cleanup(%s__Drop_glue))) ", tname); + fprintf(out, "__attribute__((cleanup(%s__Drop_glue))) ", clean_type); } // Emit Variable with Type @@ -1731,6 +1833,18 @@ void codegen_node_single(ParserContext *ctx, ASTNode *node, FILE *out) if (inferred && strcmp(inferred, "__auto_type") != 0) { + char *clean_type = inferred; + if (strncmp(clean_type, "struct ", 7) == 0) + { + clean_type += 7; + } + + ASTNode *def = find_struct_def(ctx, clean_type); + if (def && def->type_info && def->type_info->traits.has_drop) + { + fprintf(out, "__attribute__((cleanup(%s__Drop_glue))) ", clean_type); + } + emit_var_decl_type(ctx, out, inferred, node->var_decl.name); add_symbol(ctx, node->var_decl.name, inferred, NULL); fprintf(out, " = "); diff --git a/src/codegen/codegen_decl.c b/src/codegen/codegen_decl.c index 8ada7d6..e7bd3f1 100644 --- a/src/codegen/codegen_decl.c +++ b/src/codegen/codegen_decl.c @@ -424,6 +424,28 @@ void emit_struct_defs(ParserContext *ctx, ASTNode *node, FILE *out) } } +// Helper to substitute 'Self' with replacement string +static char *substitute_proto_self(const char *type_str, const char *replacement) +{ + if (!type_str) + { + return NULL; + } + if (strcmp(type_str, "Self") == 0) + { + return xstrdup(replacement); + } + // Handle pointers (Self* -> replacement*) + if (strncmp(type_str, "Self", 4) == 0) + { + char *rest = (char *)type_str + 4; + char *buf = xmalloc(strlen(replacement) + strlen(rest) + 1); + sprintf(buf, "%s%s", replacement, rest); + return buf; + } + return xstrdup(type_str); +} + // Emit trait definitions. void emit_trait_defs(ASTNode *node, FILE *out) { @@ -440,8 +462,10 @@ void emit_trait_defs(ASTNode *node, FILE *out) ASTNode *m = node->trait.methods; while (m) { - fprintf(out, " %s (*%s)(", m->func.ret_type, - parse_original_method_name(m->func.name)); + char *ret_safe = substitute_proto_self(m->func.ret_type, "void*"); + fprintf(out, " %s (*%s)(", ret_safe, parse_original_method_name(m->func.name)); + free(ret_safe); + int has_self = (m->func.args && strstr(m->func.args, "self")); if (!has_self) { @@ -454,7 +478,32 @@ void emit_trait_defs(ASTNode *node, FILE *out) { fprintf(out, ", "); } - fprintf(out, "%s", m->func.args); + char *args_safe = xstrdup(m->func.args); + // TODO: better replace, but for now this works. + char *p = strstr(args_safe, "Self"); + while (p) + { + // Check word boundary + if ((p == args_safe || !isalnum(p[-1])) && !isalnum(p[4])) + { + int off = p - args_safe; + char *new_s = xmalloc(strlen(args_safe) + 10); + strncpy(new_s, args_safe, off); + new_s[off] = 0; + strcat(new_s, "void*"); + strcat(new_s, p + 4); + free(args_safe); + args_safe = new_s; + p = strstr(args_safe + off + 5, "Self"); + } + else + { + p = strstr(p + 1, "Self"); + } + } + + fprintf(out, "%s", args_safe); + free(args_safe); } fprintf(out, ");\n"); m = m->next; @@ -467,7 +516,9 @@ void emit_trait_defs(ASTNode *node, FILE *out) while (m) { const char *orig = parse_original_method_name(m->func.name); - fprintf(out, "%s %s__%s(%s* self", m->func.ret_type, node->trait.name, orig, + char *ret_sub = substitute_proto_self(m->func.ret_type, node->trait.name); + + fprintf(out, "%s %s__%s(%s* self", ret_sub, node->trait.name, orig, node->trait.name); int has_self = (m->func.args && strstr(m->func.args, "self")); @@ -478,17 +529,45 @@ void emit_trait_defs(ASTNode *node, FILE *out) char *comma = strchr(m->func.args, ','); if (comma) { - fprintf(out, ", %s", comma + 1); + // Substitute Self -> TraitName in wrapper args + char *args_sub = xstrdup(comma + 1); + char *p = strstr(args_sub, "Self"); + while (p) + { + int off = p - args_sub; + char *new_s = + xmalloc(strlen(args_sub) + strlen(node->trait.name) + 5); + strncpy(new_s, args_sub, off); + new_s[off] = 0; + strcat(new_s, node->trait.name); + strcat(new_s, p + 4); + free(args_sub); + args_sub = new_s; + p = strstr(args_sub + off + strlen(node->trait.name), "Self"); + } + + fprintf(out, ", %s", args_sub); + free(args_sub); } } else { - fprintf(out, ", %s", m->func.args); + fprintf(out, ", %s", m->func.args); // TODO: recursive subst } } fprintf(out, ") {\n"); - fprintf(out, " return self->vtable->%s(self->self", orig); + int ret_is_self = (strcmp(m->func.ret_type, "Self") == 0); + + if (ret_is_self) + { + // Special handling: return (Trait){.self = call(), .vtable = self->vtable} + fprintf(out, " void* ret = self->vtable->%s(self->self", orig); + } + else + { + fprintf(out, " return self->vtable->%s(self->self", orig); + } if (m->func.args) { @@ -510,7 +589,16 @@ void emit_trait_defs(ASTNode *node, FILE *out) } free(call_args); } - fprintf(out, ");\n}\n"); + fprintf(out, ");\n"); + + if (ret_is_self) + { + fprintf(out, " return (%s){.self = ret, .vtable = self->vtable};\n", + node->trait.name); + } + + fprintf(out, "}\n\n"); + free(ret_sub); m = m->next; } @@ -853,6 +941,13 @@ void emit_impl_vtables(ParserContext *ctx, FILE *out) emitted[count].strct = strct; count++; + if (0 == strcmp(trait, "Copy")) + { + // Marker trait, no runtime vtable needed + ref = ref->next; + continue; + } + fprintf(out, "%s_VTable %s_%s_VTable = {", trait, strct, trait); ASTNode *m = node->impl_trait.methods; diff --git a/src/codegen/codegen_main.c b/src/codegen/codegen_main.c index 7382827..97abfc7 100644 --- a/src/codegen/codegen_main.c +++ b/src/codegen/codegen_main.c @@ -14,33 +14,101 @@ static int struct_depends_on(ASTNode *s1, const char *target_name) return 0; } - // Only structs have dependencies that matter for ordering. - if (s1->type != NODE_STRUCT) + // Check structs + if (s1->type == NODE_STRUCT) { - return 0; - } + ASTNode *field = s1->strct.fields; + while (field) + { + if (field->type == NODE_FIELD && field->field.type) + { + char *type_str = field->field.type; - ASTNode *field = s1->strct.fields; - while (field) + // Skip pointers - they don't create ordering dependency. + if (strchr(type_str, '*')) + { + field = field->next; + continue; + } + + // Clean type string (remove struct/enum prefixes) + const char *clean = type_str; + if (strncmp(clean, "struct ", 7) == 0) + { + clean += 7; + } + else if (strncmp(clean, "enum ", 5) == 0) + { + clean += 5; + } + else if (strncmp(clean, "union ", 6) == 0) + { + clean += 6; + } + + // Check for match + size_t len = strlen(target_name); + if (strncmp(clean, target_name, len) == 0) + { + char next = clean[len]; + if (next == 0 || next == '[' || isspace(next)) + { + return 1; + } + } + } + field = field->next; + } + } + // Check enums (ADTs) + else if (s1->type == NODE_ENUM) { - if (field->type == NODE_FIELD && field->field.type) + ASTNode *variant = s1->enm.variants; + while (variant) { - char *type_str = field->field.type; - // Skip pointers - they don't create ordering dependency. - if (strchr(type_str, '*')) + if (variant->type == NODE_ENUM_VARIANT && variant->variant.payload) { - field = field->next; - continue; - } + char *type_str = type_to_string(variant->variant.payload); + if (type_str) + { + if (strchr(type_str, '*')) + { + free(type_str); + variant = variant->next; + continue; + } - // Check if this field's type matches target (struct or enum). - if (strcmp(type_str, target_name) == 0) - { - return 1; + const char *clean = type_str; + if (strncmp(clean, "struct ", 7) == 0) + { + clean += 7; + } + else if (strncmp(clean, "enum ", 5) == 0) + { + clean += 5; + } + else if (strncmp(clean, "union ", 6) == 0) + { + clean += 6; + } + + size_t len = strlen(target_name); + if (strncmp(clean, target_name, len) == 0) + { + char next = clean[len]; + if (next == 0 || next == '[' || isspace(next)) + { + free(type_str); + return 1; + } + } + free(type_str); + } } + variant = variant->next; } - field = field->next; } + return 0; } @@ -102,8 +170,8 @@ static ASTNode *topo_sort_structs(ASTNode *head) continue; } - // Enums and traits have no dependencies, emit first. - if (nodes[i]->type == NODE_ENUM || nodes[i]->type == NODE_TRAIT) + // Traits have no dependencies, emit first. + if (nodes[i]->type == NODE_TRAIT) { order[order_idx++] = i; emitted[i] = 1; @@ -111,7 +179,7 @@ static ASTNode *topo_sort_structs(ASTNode *head) continue; } - // For structs, check if all dependencies are emitted. + // For structs/enums, check if all dependencies are emitted. int can_emit = 1; for (int j = 0; j < count; j++) { diff --git a/src/lsp/cJSON.c b/src/lsp/cJSON.c new file mode 100644 index 0000000..f8ea73a --- /dev/null +++ b/src/lsp/cJSON.c @@ -0,0 +1,3259 @@ +/* + Copyright (c) 2009-2017 Dave Gamble and cJSON contributors + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +/* cJSON */ +/* JSON parser in C. */ + +/* disable warnings about old C89 functions in MSVC */ +#if !defined(_CRT_SECURE_NO_DEPRECATE) && defined(_MSC_VER) +#define _CRT_SECURE_NO_DEPRECATE +#endif + +#ifdef __GNUC__ +#pragma GCC visibility push(default) +#endif +#if defined(_MSC_VER) +#pragma warning(push) +/* disable warning about single line comments in system headers */ +#pragma warning(disable : 4001) +#endif + +#include <string.h> +#include <stdio.h> +#include <math.h> +#include <stdlib.h> +#include <limits.h> +#include <ctype.h> +#include <float.h> + +#ifdef ENABLE_LOCALES +#include <locale.h> +#endif + +#if defined(_MSC_VER) +#pragma warning(pop) +#endif +#ifdef __GNUC__ +#pragma GCC visibility pop +#endif + +#include "cJSON.h" + +/* define our own boolean type */ +#ifdef true +#undef true +#endif +#define true ((cJSON_bool)1) + +#ifdef false +#undef false +#endif +#define false ((cJSON_bool)0) + +/* define isnan and isinf for ANSI C, if in C99 or above, isnan and isinf has been defined in math.h + */ +#ifndef isinf +#define isinf(d) (isnan((d - d)) && !isnan(d)) +#endif +#ifndef isnan +#define isnan(d) (d != d) +#endif + +#ifndef NAN +#ifdef _WIN32 +#define NAN sqrt(-1.0) +#else +#define NAN 0.0 / 0.0 +#endif +#endif + +typedef struct +{ + const unsigned char *json; + size_t position; +} error; +static error global_error = {NULL, 0}; + +CJSON_PUBLIC(const char *) cJSON_GetErrorPtr(void) +{ + return (const char *)(global_error.json + global_error.position); +} + +CJSON_PUBLIC(char *) cJSON_GetStringValue(const cJSON *const item) +{ + if (!cJSON_IsString(item)) + { + return NULL; + } + + return item->valuestring; +} + +CJSON_PUBLIC(double) cJSON_GetNumberValue(const cJSON *const item) +{ + if (!cJSON_IsNumber(item)) + { + return (double)NAN; + } + + return item->valuedouble; +} + +/* This is a safeguard to prevent copy-pasters from using incompatible C and header files */ +#if (CJSON_VERSION_MAJOR != 1) || (CJSON_VERSION_MINOR != 7) || (CJSON_VERSION_PATCH != 19) +#error cJSON.h and cJSON.c have different versions. Make sure that both have the same. +#endif + +CJSON_PUBLIC(const char *) cJSON_Version(void) +{ + static char version[15]; + sprintf(version, "%i.%i.%i", CJSON_VERSION_MAJOR, CJSON_VERSION_MINOR, CJSON_VERSION_PATCH); + + return version; +} + +/* Case insensitive string comparison, doesn't consider two NULL pointers equal though */ +static int case_insensitive_strcmp(const unsigned char *string1, const unsigned char *string2) +{ + if ((string1 == NULL) || (string2 == NULL)) + { + return 1; + } + + if (string1 == string2) + { + return 0; + } + + for (; tolower(*string1) == tolower(*string2); (void)string1++, string2++) + { + if (*string1 == '\0') + { + return 0; + } + } + + return tolower(*string1) - tolower(*string2); +} + +typedef struct internal_hooks +{ + void *(CJSON_CDECL *allocate)(size_t size); + void(CJSON_CDECL *deallocate)(void *pointer); + void *(CJSON_CDECL *reallocate)(void *pointer, size_t size); +} internal_hooks; + +#if defined(_MSC_VER) +/* work around MSVC error C2322: '...' address of dllimport '...' is not static */ +static void *CJSON_CDECL internal_malloc(size_t size) +{ + return malloc(size); +} +static void CJSON_CDECL internal_free(void *pointer) +{ + free(pointer); +} +static void *CJSON_CDECL internal_realloc(void *pointer, size_t size) +{ + return realloc(pointer, size); +} +#else +#define internal_malloc malloc +#define internal_free free +#define internal_realloc realloc +#endif + +/* strlen of character literals resolved at compile time */ +#define static_strlen(string_literal) (sizeof(string_literal) - sizeof("")) + +static internal_hooks global_hooks = {internal_malloc, internal_free, internal_realloc}; + +static unsigned char *cJSON_strdup(const unsigned char *string, const internal_hooks *const hooks) +{ + size_t length = 0; + unsigned char *copy = NULL; + + if (string == NULL) + { + return NULL; + } + + length = strlen((const char *)string) + sizeof(""); + copy = (unsigned char *)hooks->allocate(length); + if (copy == NULL) + { + return NULL; + } + memcpy(copy, string, length); + + return copy; +} + +CJSON_PUBLIC(void) cJSON_InitHooks(cJSON_Hooks *hooks) +{ + if (hooks == NULL) + { + /* Reset hooks */ + global_hooks.allocate = malloc; + global_hooks.deallocate = free; + global_hooks.reallocate = realloc; + return; + } + + global_hooks.allocate = malloc; + if (hooks->malloc_fn != NULL) + { + global_hooks.allocate = hooks->malloc_fn; + } + + global_hooks.deallocate = free; + if (hooks->free_fn != NULL) + { + global_hooks.deallocate = hooks->free_fn; + } + + /* use realloc only if both free and malloc are used */ + global_hooks.reallocate = NULL; + if ((global_hooks.allocate == malloc) && (global_hooks.deallocate == free)) + { + global_hooks.reallocate = realloc; + } +} + +/* Internal constructor. */ +static cJSON *cJSON_New_Item(const internal_hooks *const hooks) +{ + cJSON *node = (cJSON *)hooks->allocate(sizeof(cJSON)); + if (node) + { + memset(node, '\0', sizeof(cJSON)); + } + + return node; +} + +/* Delete a cJSON structure. */ +CJSON_PUBLIC(void) cJSON_Delete(cJSON *item) +{ + cJSON *next = NULL; + while (item != NULL) + { + next = item->next; + if (!(item->type & cJSON_IsReference) && (item->child != NULL)) + { + cJSON_Delete(item->child); + } + if (!(item->type & cJSON_IsReference) && (item->valuestring != NULL)) + { + global_hooks.deallocate(item->valuestring); + item->valuestring = NULL; + } + if (!(item->type & cJSON_StringIsConst) && (item->string != NULL)) + { + global_hooks.deallocate(item->string); + item->string = NULL; + } + global_hooks.deallocate(item); + item = next; + } +} + +/* get the decimal point character of the current locale */ +static unsigned char get_decimal_point(void) +{ +#ifdef ENABLE_LOCALES + struct lconv *lconv = localeconv(); + return (unsigned char)lconv->decimal_point[0]; +#else + return '.'; +#endif +} + +typedef struct +{ + const unsigned char *content; + size_t length; + size_t offset; + size_t depth; /* How deeply nested (in arrays/objects) is the input at the current offset. */ + internal_hooks hooks; +} parse_buffer; + +/* check if the given size is left to read in a given parse buffer (starting with 1) */ +#define can_read(buffer, size) ((buffer != NULL) && (((buffer)->offset + size) <= (buffer)->length)) +/* check if the buffer can be accessed at the given index (starting with 0) */ +#define can_access_at_index(buffer, index) \ + ((buffer != NULL) && (((buffer)->offset + index) < (buffer)->length)) +#define cannot_access_at_index(buffer, index) (!can_access_at_index(buffer, index)) +/* get a pointer to the buffer at the position */ +#define buffer_at_offset(buffer) ((buffer)->content + (buffer)->offset) + +/* Parse the input text to generate a number, and populate the result into item. */ +static cJSON_bool parse_number(cJSON *const item, parse_buffer *const input_buffer) +{ + double number = 0; + unsigned char *after_end = NULL; + unsigned char *number_c_string; + unsigned char decimal_point = get_decimal_point(); + size_t i = 0; + size_t number_string_length = 0; + cJSON_bool has_decimal_point = false; + + if ((input_buffer == NULL) || (input_buffer->content == NULL)) + { + return false; + } + + /* copy the number into a temporary buffer and replace '.' with the decimal point + * of the current locale (for strtod) + * This also takes care of '\0' not necessarily being available for marking the end of the input + */ + for (i = 0; can_access_at_index(input_buffer, i); i++) + { + switch (buffer_at_offset(input_buffer)[i]) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + case '+': + case '-': + case 'e': + case 'E': + number_string_length++; + break; + + case '.': + number_string_length++; + has_decimal_point = true; + break; + + default: + goto loop_end; + } + } +loop_end: + /* malloc for temporary buffer, add 1 for '\0' */ + number_c_string = (unsigned char *)input_buffer->hooks.allocate(number_string_length + 1); + if (number_c_string == NULL) + { + return false; /* allocation failure */ + } + + memcpy(number_c_string, buffer_at_offset(input_buffer), number_string_length); + number_c_string[number_string_length] = '\0'; + + if (has_decimal_point) + { + for (i = 0; i < number_string_length; i++) + { + if (number_c_string[i] == '.') + { + /* replace '.' with the decimal point of the current locale (for strtod) */ + number_c_string[i] = decimal_point; + } + } + } + + number = strtod((const char *)number_c_string, (char **)&after_end); + if (number_c_string == after_end) + { + /* free the temporary buffer */ + input_buffer->hooks.deallocate(number_c_string); + return false; /* parse_error */ + } + + item->valuedouble = number; + + /* use saturation in case of overflow */ + if (number >= INT_MAX) + { + item->valueint = INT_MAX; + } + else if (number <= (double)INT_MIN) + { + item->valueint = INT_MIN; + } + else + { + item->valueint = (int)number; + } + + item->type = cJSON_Number; + + input_buffer->offset += (size_t)(after_end - number_c_string); + /* free the temporary buffer */ + input_buffer->hooks.deallocate(number_c_string); + return true; +} + +/* don't ask me, but the original cJSON_SetNumberValue returns an integer or double */ +CJSON_PUBLIC(double) cJSON_SetNumberHelper(cJSON *object, double number) +{ + if (number >= INT_MAX) + { + object->valueint = INT_MAX; + } + else if (number <= (double)INT_MIN) + { + object->valueint = INT_MIN; + } + else + { + object->valueint = (int)number; + } + + return object->valuedouble = number; +} + +/* Note: when passing a NULL valuestring, cJSON_SetValuestring treats this as an error and return + * NULL */ +CJSON_PUBLIC(char *) cJSON_SetValuestring(cJSON *object, const char *valuestring) +{ + char *copy = NULL; + size_t v1_len; + size_t v2_len; + /* if object's type is not cJSON_String or is cJSON_IsReference, it should not set valuestring + */ + if ((object == NULL) || !(object->type & cJSON_String) || (object->type & cJSON_IsReference)) + { + return NULL; + } + /* return NULL if the object is corrupted or valuestring is NULL */ + if (object->valuestring == NULL || valuestring == NULL) + { + return NULL; + } + + v1_len = strlen(valuestring); + v2_len = strlen(object->valuestring); + + if (v1_len <= v2_len) + { + /* strcpy does not handle overlapping string: [X1, X2] [Y1, Y2] => X2 < Y1 or Y2 < X1 */ + if (!(valuestring + v1_len < object->valuestring || + object->valuestring + v2_len < valuestring)) + { + return NULL; + } + strcpy(object->valuestring, valuestring); + return object->valuestring; + } + copy = (char *)cJSON_strdup((const unsigned char *)valuestring, &global_hooks); + if (copy == NULL) + { + return NULL; + } + if (object->valuestring != NULL) + { + cJSON_free(object->valuestring); + } + object->valuestring = copy; + + return copy; +} + +typedef struct +{ + unsigned char *buffer; + size_t length; + size_t offset; + size_t depth; /* current nesting depth (for formatted printing) */ + cJSON_bool noalloc; + cJSON_bool format; /* is this print a formatted print */ + internal_hooks hooks; +} printbuffer; + +/* realloc printbuffer if necessary to have at least "needed" bytes more */ +static unsigned char *ensure(printbuffer *const p, size_t needed) +{ + unsigned char *newbuffer = NULL; + size_t newsize = 0; + + if ((p == NULL) || (p->buffer == NULL)) + { + return NULL; + } + + if ((p->length > 0) && (p->offset >= p->length)) + { + /* make sure that offset is valid */ + return NULL; + } + + if (needed > INT_MAX) + { + /* sizes bigger than INT_MAX are currently not supported */ + return NULL; + } + + needed += p->offset + 1; + if (needed <= p->length) + { + return p->buffer + p->offset; + } + + if (p->noalloc) + { + return NULL; + } + + /* calculate new buffer size */ + if (needed > (INT_MAX / 2)) + { + /* overflow of int, use INT_MAX if possible */ + if (needed <= INT_MAX) + { + newsize = INT_MAX; + } + else + { + return NULL; + } + } + else + { + newsize = needed * 2; + } + + if (p->hooks.reallocate != NULL) + { + /* reallocate with realloc if available */ + newbuffer = (unsigned char *)p->hooks.reallocate(p->buffer, newsize); + if (newbuffer == NULL) + { + p->hooks.deallocate(p->buffer); + p->length = 0; + p->buffer = NULL; + + return NULL; + } + } + else + { + /* otherwise reallocate manually */ + newbuffer = (unsigned char *)p->hooks.allocate(newsize); + if (!newbuffer) + { + p->hooks.deallocate(p->buffer); + p->length = 0; + p->buffer = NULL; + + return NULL; + } + + memcpy(newbuffer, p->buffer, p->offset + 1); + p->hooks.deallocate(p->buffer); + } + p->length = newsize; + p->buffer = newbuffer; + + return newbuffer + p->offset; +} + +/* calculate the new length of the string in a printbuffer and update the offset */ +static void update_offset(printbuffer *const buffer) +{ + const unsigned char *buffer_pointer = NULL; + if ((buffer == NULL) || (buffer->buffer == NULL)) + { + return; + } + buffer_pointer = buffer->buffer + buffer->offset; + + buffer->offset += strlen((const char *)buffer_pointer); +} + +/* securely comparison of floating-point variables */ +static cJSON_bool compare_double(double a, double b) +{ + double maxVal = fabs(a) > fabs(b) ? fabs(a) : fabs(b); + return (fabs(a - b) <= maxVal * DBL_EPSILON); +} + +/* Render the number nicely from the given item into a string. */ +static cJSON_bool print_number(const cJSON *const item, printbuffer *const output_buffer) +{ + unsigned char *output_pointer = NULL; + double d = item->valuedouble; + int length = 0; + size_t i = 0; + unsigned char number_buffer[26] = {0}; /* temporary buffer to print the number into */ + unsigned char decimal_point = get_decimal_point(); + double test = 0.0; + + if (output_buffer == NULL) + { + return false; + } + + /* This checks for NaN and Infinity */ + if (isnan(d) || isinf(d)) + { + length = sprintf((char *)number_buffer, "null"); + } + else if (d == (double)item->valueint) + { + length = sprintf((char *)number_buffer, "%d", item->valueint); + } + else + { + /* Try 15 decimal places of precision to avoid nonsignificant nonzero digits */ + length = sprintf((char *)number_buffer, "%1.15g", d); + + /* Check whether the original double can be recovered */ + if ((sscanf((char *)number_buffer, "%lg", &test) != 1) || !compare_double((double)test, d)) + { + /* If not, print with 17 decimal places of precision */ + length = sprintf((char *)number_buffer, "%1.17g", d); + } + } + + /* sprintf failed or buffer overrun occurred */ + if ((length < 0) || (length > (int)(sizeof(number_buffer) - 1))) + { + return false; + } + + /* reserve appropriate space in the output */ + output_pointer = ensure(output_buffer, (size_t)length + sizeof("")); + if (output_pointer == NULL) + { + return false; + } + + /* copy the printed number to the output and replace locale + * dependent decimal point with '.' */ + for (i = 0; i < ((size_t)length); i++) + { + if (number_buffer[i] == decimal_point) + { + output_pointer[i] = '.'; + continue; + } + + output_pointer[i] = number_buffer[i]; + } + output_pointer[i] = '\0'; + + output_buffer->offset += (size_t)length; + + return true; +} + +/* parse 4 digit hexadecimal number */ +static unsigned parse_hex4(const unsigned char *const input) +{ + unsigned int h = 0; + size_t i = 0; + + for (i = 0; i < 4; i++) + { + /* parse digit */ + if ((input[i] >= '0') && (input[i] <= '9')) + { + h += (unsigned int)input[i] - '0'; + } + else if ((input[i] >= 'A') && (input[i] <= 'F')) + { + h += (unsigned int)10 + input[i] - 'A'; + } + else if ((input[i] >= 'a') && (input[i] <= 'f')) + { + h += (unsigned int)10 + input[i] - 'a'; + } + else /* invalid */ + { + return 0; + } + + if (i < 3) + { + /* shift left to make place for the next nibble */ + h = h << 4; + } + } + + return h; +} + +/* converts a UTF-16 literal to UTF-8 + * A literal can be one or two sequences of the form \uXXXX */ +static unsigned char utf16_literal_to_utf8(const unsigned char *const input_pointer, + const unsigned char *const input_end, + unsigned char **output_pointer) +{ + long unsigned int codepoint = 0; + unsigned int first_code = 0; + const unsigned char *first_sequence = input_pointer; + unsigned char utf8_length = 0; + unsigned char utf8_position = 0; + unsigned char sequence_length = 0; + unsigned char first_byte_mark = 0; + + if ((input_end - first_sequence) < 6) + { + /* input ends unexpectedly */ + goto fail; + } + + /* get the first utf16 sequence */ + first_code = parse_hex4(first_sequence + 2); + + /* check that the code is valid */ + if (((first_code >= 0xDC00) && (first_code <= 0xDFFF))) + { + goto fail; + } + + /* UTF16 surrogate pair */ + if ((first_code >= 0xD800) && (first_code <= 0xDBFF)) + { + const unsigned char *second_sequence = first_sequence + 6; + unsigned int second_code = 0; + sequence_length = 12; /* \uXXXX\uXXXX */ + + if ((input_end - second_sequence) < 6) + { + /* input ends unexpectedly */ + goto fail; + } + + if ((second_sequence[0] != '\\') || (second_sequence[1] != 'u')) + { + /* missing second half of the surrogate pair */ + goto fail; + } + + /* get the second utf16 sequence */ + second_code = parse_hex4(second_sequence + 2); + /* check that the code is valid */ + if ((second_code < 0xDC00) || (second_code > 0xDFFF)) + { + /* invalid second half of the surrogate pair */ + goto fail; + } + + /* calculate the unicode codepoint from the surrogate pair */ + codepoint = 0x10000 + (((first_code & 0x3FF) << 10) | (second_code & 0x3FF)); + } + else + { + sequence_length = 6; /* \uXXXX */ + codepoint = first_code; + } + + /* encode as UTF-8 + * takes at maximum 4 bytes to encode: + * 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx */ + if (codepoint < 0x80) + { + /* normal ascii, encoding 0xxxxxxx */ + utf8_length = 1; + } + else if (codepoint < 0x800) + { + /* two bytes, encoding 110xxxxx 10xxxxxx */ + utf8_length = 2; + first_byte_mark = 0xC0; /* 11000000 */ + } + else if (codepoint < 0x10000) + { + /* three bytes, encoding 1110xxxx 10xxxxxx 10xxxxxx */ + utf8_length = 3; + first_byte_mark = 0xE0; /* 11100000 */ + } + else if (codepoint <= 0x10FFFF) + { + /* four bytes, encoding 1110xxxx 10xxxxxx 10xxxxxx 10xxxxxx */ + utf8_length = 4; + first_byte_mark = 0xF0; /* 11110000 */ + } + else + { + /* invalid unicode codepoint */ + goto fail; + } + + /* encode as utf8 */ + for (utf8_position = (unsigned char)(utf8_length - 1); utf8_position > 0; utf8_position--) + { + /* 10xxxxxx */ + (*output_pointer)[utf8_position] = (unsigned char)((codepoint | 0x80) & 0xBF); + codepoint >>= 6; + } + /* encode first byte */ + if (utf8_length > 1) + { + (*output_pointer)[0] = (unsigned char)((codepoint | first_byte_mark) & 0xFF); + } + else + { + (*output_pointer)[0] = (unsigned char)(codepoint & 0x7F); + } + + *output_pointer += utf8_length; + + return sequence_length; + +fail: + return 0; +} + +/* Parse the input text into an unescaped cinput, and populate item. */ +static cJSON_bool parse_string(cJSON *const item, parse_buffer *const input_buffer) +{ + const unsigned char *input_pointer = buffer_at_offset(input_buffer) + 1; + const unsigned char *input_end = buffer_at_offset(input_buffer) + 1; + unsigned char *output_pointer = NULL; + unsigned char *output = NULL; + + /* not a string */ + if (buffer_at_offset(input_buffer)[0] != '\"') + { + goto fail; + } + + { + /* calculate approximate size of the output (overestimate) */ + size_t allocation_length = 0; + size_t skipped_bytes = 0; + while (((size_t)(input_end - input_buffer->content) < input_buffer->length) && + (*input_end != '\"')) + { + /* is escape sequence */ + if (input_end[0] == '\\') + { + if ((size_t)(input_end + 1 - input_buffer->content) >= input_buffer->length) + { + /* prevent buffer overflow when last input character is a backslash */ + goto fail; + } + skipped_bytes++; + input_end++; + } + input_end++; + } + if (((size_t)(input_end - input_buffer->content) >= input_buffer->length) || + (*input_end != '\"')) + { + goto fail; /* string ended unexpectedly */ + } + + /* This is at most how much we need for the output */ + allocation_length = (size_t)(input_end - buffer_at_offset(input_buffer)) - skipped_bytes; + output = (unsigned char *)input_buffer->hooks.allocate(allocation_length + sizeof("")); + if (output == NULL) + { + goto fail; /* allocation failure */ + } + } + + output_pointer = output; + /* loop through the string literal */ + while (input_pointer < input_end) + { + if (*input_pointer != '\\') + { + *output_pointer++ = *input_pointer++; + } + /* escape sequence */ + else + { + unsigned char sequence_length = 2; + if ((input_end - input_pointer) < 1) + { + goto fail; + } + + switch (input_pointer[1]) + { + case 'b': + *output_pointer++ = '\b'; + break; + case 'f': + *output_pointer++ = '\f'; + break; + case 'n': + *output_pointer++ = '\n'; + break; + case 'r': + *output_pointer++ = '\r'; + break; + case 't': + *output_pointer++ = '\t'; + break; + case '\"': + case '\\': + case '/': + *output_pointer++ = input_pointer[1]; + break; + + /* UTF-16 literal */ + case 'u': + sequence_length = utf16_literal_to_utf8(input_pointer, input_end, &output_pointer); + if (sequence_length == 0) + { + /* failed to convert UTF16-literal to UTF-8 */ + goto fail; + } + break; + + default: + goto fail; + } + input_pointer += sequence_length; + } + } + + /* zero terminate the output */ + *output_pointer = '\0'; + + item->type = cJSON_String; + item->valuestring = (char *)output; + + input_buffer->offset = (size_t)(input_end - input_buffer->content); + input_buffer->offset++; + + return true; + +fail: + if (output != NULL) + { + input_buffer->hooks.deallocate(output); + output = NULL; + } + + if (input_pointer != NULL) + { + input_buffer->offset = (size_t)(input_pointer - input_buffer->content); + } + + return false; +} + +/* Render the cstring provided to an escaped version that can be printed. */ +static cJSON_bool print_string_ptr(const unsigned char *const input, + printbuffer *const output_buffer) +{ + const unsigned char *input_pointer = NULL; + unsigned char *output = NULL; + unsigned char *output_pointer = NULL; + size_t output_length = 0; + /* numbers of additional characters needed for escaping */ + size_t escape_characters = 0; + + if (output_buffer == NULL) + { + return false; + } + + /* empty string */ + if (input == NULL) + { + output = ensure(output_buffer, sizeof("\"\"")); + if (output == NULL) + { + return false; + } + strcpy((char *)output, "\"\""); + + return true; + } + + /* set "flag" to 1 if something needs to be escaped */ + for (input_pointer = input; *input_pointer; input_pointer++) + { + switch (*input_pointer) + { + case '\"': + case '\\': + case '\b': + case '\f': + case '\n': + case '\r': + case '\t': + /* one character escape sequence */ + escape_characters++; + break; + default: + if (*input_pointer < 32) + { + /* UTF-16 escape sequence uXXXX */ + escape_characters += 5; + } + break; + } + } + output_length = (size_t)(input_pointer - input) + escape_characters; + + output = ensure(output_buffer, output_length + sizeof("\"\"")); + if (output == NULL) + { + return false; + } + + /* no characters have to be escaped */ + if (escape_characters == 0) + { + output[0] = '\"'; + memcpy(output + 1, input, output_length); + output[output_length + 1] = '\"'; + output[output_length + 2] = '\0'; + + return true; + } + + output[0] = '\"'; + output_pointer = output + 1; + /* copy the string */ + for (input_pointer = input; *input_pointer != '\0'; (void)input_pointer++, output_pointer++) + { + if ((*input_pointer > 31) && (*input_pointer != '\"') && (*input_pointer != '\\')) + { + /* normal character, copy */ + *output_pointer = *input_pointer; + } + else + { + /* character needs to be escaped */ + *output_pointer++ = '\\'; + switch (*input_pointer) + { + case '\\': + *output_pointer = '\\'; + break; + case '\"': + *output_pointer = '\"'; + break; + case '\b': + *output_pointer = 'b'; + break; + case '\f': + *output_pointer = 'f'; + break; + case '\n': + *output_pointer = 'n'; + break; + case '\r': + *output_pointer = 'r'; + break; + case '\t': + *output_pointer = 't'; + break; + default: + /* escape and print as unicode codepoint */ + sprintf((char *)output_pointer, "u%04x", *input_pointer); + output_pointer += 4; + break; + } + } + } + output[output_length + 1] = '\"'; + output[output_length + 2] = '\0'; + + return true; +} + +/* Invoke print_string_ptr (which is useful) on an item. */ +static cJSON_bool print_string(const cJSON *const item, printbuffer *const p) +{ + return print_string_ptr((unsigned char *)item->valuestring, p); +} + +/* Predeclare these prototypes. */ +static cJSON_bool parse_value(cJSON *const item, parse_buffer *const input_buffer); +static cJSON_bool print_value(const cJSON *const item, printbuffer *const output_buffer); +static cJSON_bool parse_array(cJSON *const item, parse_buffer *const input_buffer); +static cJSON_bool print_array(const cJSON *const item, printbuffer *const output_buffer); +static cJSON_bool parse_object(cJSON *const item, parse_buffer *const input_buffer); +static cJSON_bool print_object(const cJSON *const item, printbuffer *const output_buffer); + +/* Utility to jump whitespace and cr/lf */ +static parse_buffer *buffer_skip_whitespace(parse_buffer *const buffer) +{ + if ((buffer == NULL) || (buffer->content == NULL)) + { + return NULL; + } + + if (cannot_access_at_index(buffer, 0)) + { + return buffer; + } + + while (can_access_at_index(buffer, 0) && (buffer_at_offset(buffer)[0] <= 32)) + { + buffer->offset++; + } + + if (buffer->offset == buffer->length) + { + buffer->offset--; + } + + return buffer; +} + +/* skip the UTF-8 BOM (byte order mark) if it is at the beginning of a buffer */ +static parse_buffer *skip_utf8_bom(parse_buffer *const buffer) +{ + if ((buffer == NULL) || (buffer->content == NULL) || (buffer->offset != 0)) + { + return NULL; + } + + if (can_access_at_index(buffer, 4) && + (strncmp((const char *)buffer_at_offset(buffer), "\xEF\xBB\xBF", 3) == 0)) + { + buffer->offset += 3; + } + + return buffer; +} + +CJSON_PUBLIC(cJSON *) +cJSON_ParseWithOpts(const char *value, const char **return_parse_end, + cJSON_bool require_null_terminated) +{ + size_t buffer_length; + + if (NULL == value) + { + return NULL; + } + + /* Adding null character size due to require_null_terminated. */ + buffer_length = strlen(value) + sizeof(""); + + return cJSON_ParseWithLengthOpts(value, buffer_length, return_parse_end, + require_null_terminated); +} + +/* Parse an object - create a new root, and populate. */ +CJSON_PUBLIC(cJSON *) +cJSON_ParseWithLengthOpts(const char *value, size_t buffer_length, const char **return_parse_end, + cJSON_bool require_null_terminated) +{ + parse_buffer buffer = {0, 0, 0, 0, {0, 0, 0}}; + cJSON *item = NULL; + + /* reset error position */ + global_error.json = NULL; + global_error.position = 0; + + if (value == NULL || 0 == buffer_length) + { + goto fail; + } + + buffer.content = (const unsigned char *)value; + buffer.length = buffer_length; + buffer.offset = 0; + buffer.hooks = global_hooks; + + item = cJSON_New_Item(&global_hooks); + if (item == NULL) /* memory fail */ + { + goto fail; + } + + if (!parse_value(item, buffer_skip_whitespace(skip_utf8_bom(&buffer)))) + { + /* parse failure. ep is set. */ + goto fail; + } + + /* if we require null-terminated JSON without appended garbage, skip and then check for a null + * terminator */ + if (require_null_terminated) + { + buffer_skip_whitespace(&buffer); + if ((buffer.offset >= buffer.length) || buffer_at_offset(&buffer)[0] != '\0') + { + goto fail; + } + } + if (return_parse_end) + { + *return_parse_end = (const char *)buffer_at_offset(&buffer); + } + + return item; + +fail: + if (item != NULL) + { + cJSON_Delete(item); + } + + if (value != NULL) + { + error local_error; + local_error.json = (const unsigned char *)value; + local_error.position = 0; + + if (buffer.offset < buffer.length) + { + local_error.position = buffer.offset; + } + else if (buffer.length > 0) + { + local_error.position = buffer.length - 1; + } + + if (return_parse_end != NULL) + { + *return_parse_end = (const char *)local_error.json + local_error.position; + } + + global_error = local_error; + } + + return NULL; +} + +/* Default options for cJSON_Parse */ +CJSON_PUBLIC(cJSON *) cJSON_Parse(const char *value) +{ + return cJSON_ParseWithOpts(value, 0, 0); +} + +CJSON_PUBLIC(cJSON *) cJSON_ParseWithLength(const char *value, size_t buffer_length) +{ + return cJSON_ParseWithLengthOpts(value, buffer_length, 0, 0); +} + +#define cjson_min(a, b) (((a) < (b)) ? (a) : (b)) + +static unsigned char *print(const cJSON *const item, cJSON_bool format, + const internal_hooks *const hooks) +{ + static const size_t default_buffer_size = 256; + printbuffer buffer[1]; + unsigned char *printed = NULL; + + memset(buffer, 0, sizeof(buffer)); + + /* create buffer */ + buffer->buffer = (unsigned char *)hooks->allocate(default_buffer_size); + buffer->length = default_buffer_size; + buffer->format = format; + buffer->hooks = *hooks; + if (buffer->buffer == NULL) + { + goto fail; + } + + /* print the value */ + if (!print_value(item, buffer)) + { + goto fail; + } + update_offset(buffer); + + /* check if reallocate is available */ + if (hooks->reallocate != NULL) + { + printed = (unsigned char *)hooks->reallocate(buffer->buffer, buffer->offset + 1); + if (printed == NULL) + { + goto fail; + } + buffer->buffer = NULL; + } + else /* otherwise copy the JSON over to a new buffer */ + { + printed = (unsigned char *)hooks->allocate(buffer->offset + 1); + if (printed == NULL) + { + goto fail; + } + memcpy(printed, buffer->buffer, cjson_min(buffer->length, buffer->offset + 1)); + printed[buffer->offset] = '\0'; /* just to be sure */ + + /* free the buffer */ + hooks->deallocate(buffer->buffer); + buffer->buffer = NULL; + } + + return printed; + +fail: + if (buffer->buffer != NULL) + { + hooks->deallocate(buffer->buffer); + buffer->buffer = NULL; + } + + if (printed != NULL) + { + hooks->deallocate(printed); + printed = NULL; + } + + return NULL; +} + +/* Render a cJSON item/entity/structure to text. */ +CJSON_PUBLIC(char *) cJSON_Print(const cJSON *item) +{ + return (char *)print(item, true, &global_hooks); +} + +CJSON_PUBLIC(char *) cJSON_PrintUnformatted(const cJSON *item) +{ + return (char *)print(item, false, &global_hooks); +} + +CJSON_PUBLIC(char *) cJSON_PrintBuffered(const cJSON *item, int prebuffer, cJSON_bool fmt) +{ + printbuffer p = {0, 0, 0, 0, 0, 0, {0, 0, 0}}; + + if (prebuffer < 0) + { + return NULL; + } + + p.buffer = (unsigned char *)global_hooks.allocate((size_t)prebuffer); + if (!p.buffer) + { + return NULL; + } + + p.length = (size_t)prebuffer; + p.offset = 0; + p.noalloc = false; + p.format = fmt; + p.hooks = global_hooks; + + if (!print_value(item, &p)) + { + global_hooks.deallocate(p.buffer); + p.buffer = NULL; + return NULL; + } + + return (char *)p.buffer; +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_PrintPreallocated(cJSON *item, char *buffer, const int length, const cJSON_bool format) +{ + printbuffer p = {0, 0, 0, 0, 0, 0, {0, 0, 0}}; + + if ((length < 0) || (buffer == NULL)) + { + return false; + } + + p.buffer = (unsigned char *)buffer; + p.length = (size_t)length; + p.offset = 0; + p.noalloc = true; + p.format = format; + p.hooks = global_hooks; + + return print_value(item, &p); +} + +/* Parser core - when encountering text, process appropriately. */ +static cJSON_bool parse_value(cJSON *const item, parse_buffer *const input_buffer) +{ + if ((input_buffer == NULL) || (input_buffer->content == NULL)) + { + return false; /* no input */ + } + + /* parse the different types of values */ + /* null */ + if (can_read(input_buffer, 4) && + (strncmp((const char *)buffer_at_offset(input_buffer), "null", 4) == 0)) + { + item->type = cJSON_NULL; + input_buffer->offset += 4; + return true; + } + /* false */ + if (can_read(input_buffer, 5) && + (strncmp((const char *)buffer_at_offset(input_buffer), "false", 5) == 0)) + { + item->type = cJSON_False; + input_buffer->offset += 5; + return true; + } + /* true */ + if (can_read(input_buffer, 4) && + (strncmp((const char *)buffer_at_offset(input_buffer), "true", 4) == 0)) + { + item->type = cJSON_True; + item->valueint = 1; + input_buffer->offset += 4; + return true; + } + /* string */ + if (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == '\"')) + { + return parse_string(item, input_buffer); + } + /* number */ + if (can_access_at_index(input_buffer, 0) && ((buffer_at_offset(input_buffer)[0] == '-') || + ((buffer_at_offset(input_buffer)[0] >= '0') && + (buffer_at_offset(input_buffer)[0] <= '9')))) + { + return parse_number(item, input_buffer); + } + /* array */ + if (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == '[')) + { + return parse_array(item, input_buffer); + } + /* object */ + if (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == '{')) + { + return parse_object(item, input_buffer); + } + + return false; +} + +/* Render a value to text. */ +static cJSON_bool print_value(const cJSON *const item, printbuffer *const output_buffer) +{ + unsigned char *output = NULL; + + if ((item == NULL) || (output_buffer == NULL)) + { + return false; + } + + switch ((item->type) & 0xFF) + { + case cJSON_NULL: + output = ensure(output_buffer, 5); + if (output == NULL) + { + return false; + } + strcpy((char *)output, "null"); + return true; + + case cJSON_False: + output = ensure(output_buffer, 6); + if (output == NULL) + { + return false; + } + strcpy((char *)output, "false"); + return true; + + case cJSON_True: + output = ensure(output_buffer, 5); + if (output == NULL) + { + return false; + } + strcpy((char *)output, "true"); + return true; + + case cJSON_Number: + return print_number(item, output_buffer); + + case cJSON_Raw: + { + size_t raw_length = 0; + if (item->valuestring == NULL) + { + return false; + } + + raw_length = strlen(item->valuestring) + sizeof(""); + output = ensure(output_buffer, raw_length); + if (output == NULL) + { + return false; + } + memcpy(output, item->valuestring, raw_length); + return true; + } + + case cJSON_String: + return print_string(item, output_buffer); + + case cJSON_Array: + return print_array(item, output_buffer); + + case cJSON_Object: + return print_object(item, output_buffer); + + default: + return false; + } +} + +/* Build an array from input text. */ +static cJSON_bool parse_array(cJSON *const item, parse_buffer *const input_buffer) +{ + cJSON *head = NULL; /* head of the linked list */ + cJSON *current_item = NULL; + + if (input_buffer->depth >= CJSON_NESTING_LIMIT) + { + return false; /* to deeply nested */ + } + input_buffer->depth++; + + if (buffer_at_offset(input_buffer)[0] != '[') + { + /* not an array */ + goto fail; + } + + input_buffer->offset++; + buffer_skip_whitespace(input_buffer); + if (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == ']')) + { + /* empty array */ + goto success; + } + + /* check if we skipped to the end of the buffer */ + if (cannot_access_at_index(input_buffer, 0)) + { + input_buffer->offset--; + goto fail; + } + + /* step back to character in front of the first element */ + input_buffer->offset--; + /* loop through the comma separated array elements */ + do + { + /* allocate next item */ + cJSON *new_item = cJSON_New_Item(&(input_buffer->hooks)); + if (new_item == NULL) + { + goto fail; /* allocation failure */ + } + + /* attach next item to list */ + if (head == NULL) + { + /* start the linked list */ + current_item = head = new_item; + } + else + { + /* add to the end and advance */ + current_item->next = new_item; + new_item->prev = current_item; + current_item = new_item; + } + + /* parse next value */ + input_buffer->offset++; + buffer_skip_whitespace(input_buffer); + if (!parse_value(current_item, input_buffer)) + { + goto fail; /* failed to parse value */ + } + buffer_skip_whitespace(input_buffer); + } while (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == ',')); + + if (cannot_access_at_index(input_buffer, 0) || buffer_at_offset(input_buffer)[0] != ']') + { + goto fail; /* expected end of array */ + } + +success: + input_buffer->depth--; + + if (head != NULL) + { + head->prev = current_item; + } + + item->type = cJSON_Array; + item->child = head; + + input_buffer->offset++; + + return true; + +fail: + if (head != NULL) + { + cJSON_Delete(head); + } + + return false; +} + +/* Render an array to text */ +static cJSON_bool print_array(const cJSON *const item, printbuffer *const output_buffer) +{ + unsigned char *output_pointer = NULL; + size_t length = 0; + cJSON *current_element = item->child; + + if (output_buffer == NULL) + { + return false; + } + + /* Compose the output array. */ + /* opening square bracket */ + output_pointer = ensure(output_buffer, 1); + if (output_pointer == NULL) + { + return false; + } + + *output_pointer = '['; + output_buffer->offset++; + output_buffer->depth++; + + while (current_element != NULL) + { + if (!print_value(current_element, output_buffer)) + { + return false; + } + update_offset(output_buffer); + if (current_element->next) + { + length = (size_t)(output_buffer->format ? 2 : 1); + output_pointer = ensure(output_buffer, length + 1); + if (output_pointer == NULL) + { + return false; + } + *output_pointer++ = ','; + if (output_buffer->format) + { + *output_pointer++ = ' '; + } + *output_pointer = '\0'; + output_buffer->offset += length; + } + current_element = current_element->next; + } + + output_pointer = ensure(output_buffer, 2); + if (output_pointer == NULL) + { + return false; + } + *output_pointer++ = ']'; + *output_pointer = '\0'; + output_buffer->depth--; + + return true; +} + +/* Build an object from the text. */ +static cJSON_bool parse_object(cJSON *const item, parse_buffer *const input_buffer) +{ + cJSON *head = NULL; /* linked list head */ + cJSON *current_item = NULL; + + if (input_buffer->depth >= CJSON_NESTING_LIMIT) + { + return false; /* to deeply nested */ + } + input_buffer->depth++; + + if (cannot_access_at_index(input_buffer, 0) || (buffer_at_offset(input_buffer)[0] != '{')) + { + goto fail; /* not an object */ + } + + input_buffer->offset++; + buffer_skip_whitespace(input_buffer); + if (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == '}')) + { + goto success; /* empty object */ + } + + /* check if we skipped to the end of the buffer */ + if (cannot_access_at_index(input_buffer, 0)) + { + input_buffer->offset--; + goto fail; + } + + /* step back to character in front of the first element */ + input_buffer->offset--; + /* loop through the comma separated array elements */ + do + { + /* allocate next item */ + cJSON *new_item = cJSON_New_Item(&(input_buffer->hooks)); + if (new_item == NULL) + { + goto fail; /* allocation failure */ + } + + /* attach next item to list */ + if (head == NULL) + { + /* start the linked list */ + current_item = head = new_item; + } + else + { + /* add to the end and advance */ + current_item->next = new_item; + new_item->prev = current_item; + current_item = new_item; + } + + if (cannot_access_at_index(input_buffer, 1)) + { + goto fail; /* nothing comes after the comma */ + } + + /* parse the name of the child */ + input_buffer->offset++; + buffer_skip_whitespace(input_buffer); + if (!parse_string(current_item, input_buffer)) + { + goto fail; /* failed to parse name */ + } + buffer_skip_whitespace(input_buffer); + + /* swap valuestring and string, because we parsed the name */ + current_item->string = current_item->valuestring; + current_item->valuestring = NULL; + + if (cannot_access_at_index(input_buffer, 0) || (buffer_at_offset(input_buffer)[0] != ':')) + { + goto fail; /* invalid object */ + } + + /* parse the value */ + input_buffer->offset++; + buffer_skip_whitespace(input_buffer); + if (!parse_value(current_item, input_buffer)) + { + goto fail; /* failed to parse value */ + } + buffer_skip_whitespace(input_buffer); + } while (can_access_at_index(input_buffer, 0) && (buffer_at_offset(input_buffer)[0] == ',')); + + if (cannot_access_at_index(input_buffer, 0) || (buffer_at_offset(input_buffer)[0] != '}')) + { + goto fail; /* expected end of object */ + } + +success: + input_buffer->depth--; + + if (head != NULL) + { + head->prev = current_item; + } + + item->type = cJSON_Object; + item->child = head; + + input_buffer->offset++; + return true; + +fail: + if (head != NULL) + { + cJSON_Delete(head); + } + + return false; +} + +/* Render an object to text. */ +static cJSON_bool print_object(const cJSON *const item, printbuffer *const output_buffer) +{ + unsigned char *output_pointer = NULL; + size_t length = 0; + cJSON *current_item = item->child; + + if (output_buffer == NULL) + { + return false; + } + + /* Compose the output: */ + length = (size_t)(output_buffer->format ? 2 : 1); /* fmt: {\n */ + output_pointer = ensure(output_buffer, length + 1); + if (output_pointer == NULL) + { + return false; + } + + *output_pointer++ = '{'; + output_buffer->depth++; + if (output_buffer->format) + { + *output_pointer++ = '\n'; + } + output_buffer->offset += length; + + while (current_item) + { + if (output_buffer->format) + { + size_t i; + output_pointer = ensure(output_buffer, output_buffer->depth); + if (output_pointer == NULL) + { + return false; + } + for (i = 0; i < output_buffer->depth; i++) + { + *output_pointer++ = '\t'; + } + output_buffer->offset += output_buffer->depth; + } + + /* print key */ + if (!print_string_ptr((unsigned char *)current_item->string, output_buffer)) + { + return false; + } + update_offset(output_buffer); + + length = (size_t)(output_buffer->format ? 2 : 1); + output_pointer = ensure(output_buffer, length); + if (output_pointer == NULL) + { + return false; + } + *output_pointer++ = ':'; + if (output_buffer->format) + { + *output_pointer++ = '\t'; + } + output_buffer->offset += length; + + /* print value */ + if (!print_value(current_item, output_buffer)) + { + return false; + } + update_offset(output_buffer); + + /* print comma if not last */ + length = ((size_t)(output_buffer->format ? 1 : 0) + (size_t)(current_item->next ? 1 : 0)); + output_pointer = ensure(output_buffer, length + 1); + if (output_pointer == NULL) + { + return false; + } + if (current_item->next) + { + *output_pointer++ = ','; + } + + if (output_buffer->format) + { + *output_pointer++ = '\n'; + } + *output_pointer = '\0'; + output_buffer->offset += length; + + current_item = current_item->next; + } + + output_pointer = ensure(output_buffer, output_buffer->format ? (output_buffer->depth + 1) : 2); + if (output_pointer == NULL) + { + return false; + } + if (output_buffer->format) + { + size_t i; + for (i = 0; i < (output_buffer->depth - 1); i++) + { + *output_pointer++ = '\t'; + } + } + *output_pointer++ = '}'; + *output_pointer = '\0'; + output_buffer->depth--; + + return true; +} + +/* Get Array size/item / object item. */ +CJSON_PUBLIC(int) cJSON_GetArraySize(const cJSON *array) +{ + cJSON *child = NULL; + size_t size = 0; + + if (array == NULL) + { + return 0; + } + + child = array->child; + + while (child != NULL) + { + size++; + child = child->next; + } + + /* FIXME: Can overflow here. Cannot be fixed without breaking the API */ + + return (int)size; +} + +static cJSON *get_array_item(const cJSON *array, size_t index) +{ + cJSON *current_child = NULL; + + if (array == NULL) + { + return NULL; + } + + current_child = array->child; + while ((current_child != NULL) && (index > 0)) + { + index--; + current_child = current_child->next; + } + + return current_child; +} + +CJSON_PUBLIC(cJSON *) cJSON_GetArrayItem(const cJSON *array, int index) +{ + if (index < 0) + { + return NULL; + } + + return get_array_item(array, (size_t)index); +} + +static cJSON *get_object_item(const cJSON *const object, const char *const name, + const cJSON_bool case_sensitive) +{ + cJSON *current_element = NULL; + + if ((object == NULL) || (name == NULL)) + { + return NULL; + } + + current_element = object->child; + if (case_sensitive) + { + while ((current_element != NULL) && (current_element->string != NULL) && + (strcmp(name, current_element->string) != 0)) + { + current_element = current_element->next; + } + } + else + { + while ((current_element != NULL) && + (case_insensitive_strcmp((const unsigned char *)name, + (const unsigned char *)(current_element->string)) != 0)) + { + current_element = current_element->next; + } + } + + if ((current_element == NULL) || (current_element->string == NULL)) + { + return NULL; + } + + return current_element; +} + +CJSON_PUBLIC(cJSON *) cJSON_GetObjectItem(const cJSON *const object, const char *const string) +{ + return get_object_item(object, string, false); +} + +CJSON_PUBLIC(cJSON *) +cJSON_GetObjectItemCaseSensitive(const cJSON *const object, const char *const string) +{ + return get_object_item(object, string, true); +} + +CJSON_PUBLIC(cJSON_bool) cJSON_HasObjectItem(const cJSON *object, const char *string) +{ + return cJSON_GetObjectItem(object, string) ? 1 : 0; +} + +/* Utility for array list handling. */ +static void suffix_object(cJSON *prev, cJSON *item) +{ + prev->next = item; + item->prev = prev; +} + +/* Utility for handling references. */ +static cJSON *create_reference(const cJSON *item, const internal_hooks *const hooks) +{ + cJSON *reference = NULL; + if (item == NULL) + { + return NULL; + } + + reference = cJSON_New_Item(hooks); + if (reference == NULL) + { + return NULL; + } + + memcpy(reference, item, sizeof(cJSON)); + reference->string = NULL; + reference->type |= cJSON_IsReference; + reference->next = reference->prev = NULL; + return reference; +} + +static cJSON_bool add_item_to_array(cJSON *array, cJSON *item) +{ + cJSON *child = NULL; + + if ((item == NULL) || (array == NULL) || (array == item)) + { + return false; + } + + child = array->child; + /* + * To find the last item in array quickly, we use prev in array + */ + if (child == NULL) + { + /* list is empty, start new one */ + array->child = item; + item->prev = item; + item->next = NULL; + } + else + { + /* append to the end */ + if (child->prev) + { + suffix_object(child->prev, item); + array->child->prev = item; + } + } + + return true; +} + +/* Add item to array/object. */ +CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToArray(cJSON *array, cJSON *item) +{ + return add_item_to_array(array, item); +} + +#if defined(__clang__) || \ + (defined(__GNUC__) && ((__GNUC__ > 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ > 5)))) +#pragma GCC diagnostic push +#endif +#ifdef __GNUC__ +#pragma GCC diagnostic ignored "-Wcast-qual" +#endif +/* helper function to cast away const */ +static void *cast_away_const(const void *string) +{ + return (void *)string; +} +#if defined(__clang__) || \ + (defined(__GNUC__) && ((__GNUC__ > 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ > 5)))) +#pragma GCC diagnostic pop +#endif + +static cJSON_bool add_item_to_object(cJSON *const object, const char *const string, + cJSON *const item, const internal_hooks *const hooks, + const cJSON_bool constant_key) +{ + char *new_key = NULL; + int new_type = cJSON_Invalid; + + if ((object == NULL) || (string == NULL) || (item == NULL) || (object == item)) + { + return false; + } + + if (constant_key) + { + new_key = (char *)cast_away_const(string); + new_type = item->type | cJSON_StringIsConst; + } + else + { + new_key = (char *)cJSON_strdup((const unsigned char *)string, hooks); + if (new_key == NULL) + { + return false; + } + + new_type = item->type & ~cJSON_StringIsConst; + } + + if (!(item->type & cJSON_StringIsConst) && (item->string != NULL)) + { + hooks->deallocate(item->string); + } + + item->string = new_key; + item->type = new_type; + + return add_item_to_array(object, item); +} + +CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item) +{ + return add_item_to_object(object, string, item, &global_hooks, false); +} + +/* Add an item to an object with constant string as key */ +CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObjectCS(cJSON *object, const char *string, cJSON *item) +{ + return add_item_to_object(object, string, item, &global_hooks, true); +} + +CJSON_PUBLIC(cJSON_bool) cJSON_AddItemReferenceToArray(cJSON *array, cJSON *item) +{ + if (array == NULL) + { + return false; + } + + return add_item_to_array(array, create_reference(item, &global_hooks)); +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_AddItemReferenceToObject(cJSON *object, const char *string, cJSON *item) +{ + if ((object == NULL) || (string == NULL)) + { + return false; + } + + return add_item_to_object(object, string, create_reference(item, &global_hooks), &global_hooks, + false); +} + +CJSON_PUBLIC(cJSON *) cJSON_AddNullToObject(cJSON *const object, const char *const name) +{ + cJSON *null = cJSON_CreateNull(); + if (add_item_to_object(object, name, null, &global_hooks, false)) + { + return null; + } + + cJSON_Delete(null); + return NULL; +} + +CJSON_PUBLIC(cJSON *) cJSON_AddTrueToObject(cJSON *const object, const char *const name) +{ + cJSON *true_item = cJSON_CreateTrue(); + if (add_item_to_object(object, name, true_item, &global_hooks, false)) + { + return true_item; + } + + cJSON_Delete(true_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) cJSON_AddFalseToObject(cJSON *const object, const char *const name) +{ + cJSON *false_item = cJSON_CreateFalse(); + if (add_item_to_object(object, name, false_item, &global_hooks, false)) + { + return false_item; + } + + cJSON_Delete(false_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) +cJSON_AddBoolToObject(cJSON *const object, const char *const name, const cJSON_bool boolean) +{ + cJSON *bool_item = cJSON_CreateBool(boolean); + if (add_item_to_object(object, name, bool_item, &global_hooks, false)) + { + return bool_item; + } + + cJSON_Delete(bool_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) +cJSON_AddNumberToObject(cJSON *const object, const char *const name, const double number) +{ + cJSON *number_item = cJSON_CreateNumber(number); + if (add_item_to_object(object, name, number_item, &global_hooks, false)) + { + return number_item; + } + + cJSON_Delete(number_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) +cJSON_AddStringToObject(cJSON *const object, const char *const name, const char *const string) +{ + cJSON *string_item = cJSON_CreateString(string); + if (add_item_to_object(object, name, string_item, &global_hooks, false)) + { + return string_item; + } + + cJSON_Delete(string_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) +cJSON_AddRawToObject(cJSON *const object, const char *const name, const char *const raw) +{ + cJSON *raw_item = cJSON_CreateRaw(raw); + if (add_item_to_object(object, name, raw_item, &global_hooks, false)) + { + return raw_item; + } + + cJSON_Delete(raw_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) cJSON_AddObjectToObject(cJSON *const object, const char *const name) +{ + cJSON *object_item = cJSON_CreateObject(); + if (add_item_to_object(object, name, object_item, &global_hooks, false)) + { + return object_item; + } + + cJSON_Delete(object_item); + return NULL; +} + +CJSON_PUBLIC(cJSON *) cJSON_AddArrayToObject(cJSON *const object, const char *const name) +{ + cJSON *array = cJSON_CreateArray(); + if (add_item_to_object(object, name, array, &global_hooks, false)) + { + return array; + } + + cJSON_Delete(array); + return NULL; +} + +CJSON_PUBLIC(cJSON *) cJSON_DetachItemViaPointer(cJSON *parent, cJSON *const item) +{ + if ((parent == NULL) || (item == NULL) || (item != parent->child && item->prev == NULL)) + { + return NULL; + } + + if (item != parent->child) + { + /* not the first element */ + item->prev->next = item->next; + } + if (item->next != NULL) + { + /* not the last element */ + item->next->prev = item->prev; + } + + if (item == parent->child) + { + /* first element */ + parent->child = item->next; + } + else if (item->next == NULL) + { + /* last element */ + parent->child->prev = item->prev; + } + + /* make sure the detached item doesn't point anywhere anymore */ + item->prev = NULL; + item->next = NULL; + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromArray(cJSON *array, int which) +{ + if (which < 0) + { + return NULL; + } + + return cJSON_DetachItemViaPointer(array, get_array_item(array, (size_t)which)); +} + +CJSON_PUBLIC(void) cJSON_DeleteItemFromArray(cJSON *array, int which) +{ + cJSON_Delete(cJSON_DetachItemFromArray(array, which)); +} + +CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromObject(cJSON *object, const char *string) +{ + cJSON *to_detach = cJSON_GetObjectItem(object, string); + + return cJSON_DetachItemViaPointer(object, to_detach); +} + +CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromObjectCaseSensitive(cJSON *object, const char *string) +{ + cJSON *to_detach = cJSON_GetObjectItemCaseSensitive(object, string); + + return cJSON_DetachItemViaPointer(object, to_detach); +} + +CJSON_PUBLIC(void) cJSON_DeleteItemFromObject(cJSON *object, const char *string) +{ + cJSON_Delete(cJSON_DetachItemFromObject(object, string)); +} + +CJSON_PUBLIC(void) cJSON_DeleteItemFromObjectCaseSensitive(cJSON *object, const char *string) +{ + cJSON_Delete(cJSON_DetachItemFromObjectCaseSensitive(object, string)); +} + +/* Replace array/object items with new ones. */ +CJSON_PUBLIC(cJSON_bool) cJSON_InsertItemInArray(cJSON *array, int which, cJSON *newitem) +{ + cJSON *after_inserted = NULL; + + if (which < 0 || newitem == NULL) + { + return false; + } + + after_inserted = get_array_item(array, (size_t)which); + if (after_inserted == NULL) + { + return add_item_to_array(array, newitem); + } + + if (after_inserted != array->child && after_inserted->prev == NULL) + { + /* return false if after_inserted is a corrupted array item */ + return false; + } + + newitem->next = after_inserted; + newitem->prev = after_inserted->prev; + after_inserted->prev = newitem; + if (after_inserted == array->child) + { + array->child = newitem; + } + else + { + newitem->prev->next = newitem; + } + return true; +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_ReplaceItemViaPointer(cJSON *const parent, cJSON *const item, cJSON *replacement) +{ + if ((parent == NULL) || (parent->child == NULL) || (replacement == NULL) || (item == NULL)) + { + return false; + } + + if (replacement == item) + { + return true; + } + + replacement->next = item->next; + replacement->prev = item->prev; + + if (replacement->next != NULL) + { + replacement->next->prev = replacement; + } + if (parent->child == item) + { + if (parent->child->prev == parent->child) + { + replacement->prev = replacement; + } + parent->child = replacement; + } + else + { /* + * To find the last item in array quickly, we use prev in array. + * We can't modify the last item's next pointer where this item was the parent's child + */ + if (replacement->prev != NULL) + { + replacement->prev->next = replacement; + } + if (replacement->next == NULL) + { + parent->child->prev = replacement; + } + } + + item->next = NULL; + item->prev = NULL; + cJSON_Delete(item); + + return true; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemInArray(cJSON *array, int which, cJSON *newitem) +{ + if (which < 0) + { + return false; + } + + return cJSON_ReplaceItemViaPointer(array, get_array_item(array, (size_t)which), newitem); +} + +static cJSON_bool replace_item_in_object(cJSON *object, const char *string, cJSON *replacement, + cJSON_bool case_sensitive) +{ + if ((replacement == NULL) || (string == NULL)) + { + return false; + } + + /* replace the name in the replacement */ + if (!(replacement->type & cJSON_StringIsConst) && (replacement->string != NULL)) + { + cJSON_free(replacement->string); + } + replacement->string = (char *)cJSON_strdup((const unsigned char *)string, &global_hooks); + if (replacement->string == NULL) + { + return false; + } + + replacement->type &= ~cJSON_StringIsConst; + + return cJSON_ReplaceItemViaPointer(object, get_object_item(object, string, case_sensitive), + replacement); +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_ReplaceItemInObject(cJSON *object, const char *string, cJSON *newitem) +{ + return replace_item_in_object(object, string, newitem, false); +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_ReplaceItemInObjectCaseSensitive(cJSON *object, const char *string, cJSON *newitem) +{ + return replace_item_in_object(object, string, newitem, true); +} + +/* Create basic types: */ +CJSON_PUBLIC(cJSON *) cJSON_CreateNull(void) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_NULL; + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateTrue(void) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_True; + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateFalse(void) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_False; + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateBool(cJSON_bool boolean) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = boolean ? cJSON_True : cJSON_False; + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateNumber(double num) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_Number; + item->valuedouble = num; + + /* use saturation in case of overflow */ + if (num >= INT_MAX) + { + item->valueint = INT_MAX; + } + else if (num <= (double)INT_MIN) + { + item->valueint = INT_MIN; + } + else + { + item->valueint = (int)num; + } + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateString(const char *string) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_String; + item->valuestring = (char *)cJSON_strdup((const unsigned char *)string, &global_hooks); + if (!item->valuestring) + { + cJSON_Delete(item); + return NULL; + } + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateStringReference(const char *string) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item != NULL) + { + item->type = cJSON_String | cJSON_IsReference; + item->valuestring = (char *)cast_away_const(string); + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateObjectReference(const cJSON *child) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item != NULL) + { + item->type = cJSON_Object | cJSON_IsReference; + item->child = (cJSON *)cast_away_const(child); + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateArrayReference(const cJSON *child) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item != NULL) + { + item->type = cJSON_Array | cJSON_IsReference; + item->child = (cJSON *)cast_away_const(child); + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateRaw(const char *raw) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_Raw; + item->valuestring = (char *)cJSON_strdup((const unsigned char *)raw, &global_hooks); + if (!item->valuestring) + { + cJSON_Delete(item); + return NULL; + } + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateArray(void) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_Array; + } + + return item; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateObject(void) +{ + cJSON *item = cJSON_New_Item(&global_hooks); + if (item) + { + item->type = cJSON_Object; + } + + return item; +} + +/* Create Arrays: */ +CJSON_PUBLIC(cJSON *) cJSON_CreateIntArray(const int *numbers, int count) +{ + size_t i = 0; + cJSON *n = NULL; + cJSON *p = NULL; + cJSON *a = NULL; + + if ((count < 0) || (numbers == NULL)) + { + return NULL; + } + + a = cJSON_CreateArray(); + + for (i = 0; a && (i < (size_t)count); i++) + { + n = cJSON_CreateNumber(numbers[i]); + if (!n) + { + cJSON_Delete(a); + return NULL; + } + if (!i) + { + a->child = n; + } + else + { + suffix_object(p, n); + } + p = n; + } + + if (a && a->child) + { + a->child->prev = n; + } + + return a; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateFloatArray(const float *numbers, int count) +{ + size_t i = 0; + cJSON *n = NULL; + cJSON *p = NULL; + cJSON *a = NULL; + + if ((count < 0) || (numbers == NULL)) + { + return NULL; + } + + a = cJSON_CreateArray(); + + for (i = 0; a && (i < (size_t)count); i++) + { + n = cJSON_CreateNumber((double)numbers[i]); + if (!n) + { + cJSON_Delete(a); + return NULL; + } + if (!i) + { + a->child = n; + } + else + { + suffix_object(p, n); + } + p = n; + } + + if (a && a->child) + { + a->child->prev = n; + } + + return a; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateDoubleArray(const double *numbers, int count) +{ + size_t i = 0; + cJSON *n = NULL; + cJSON *p = NULL; + cJSON *a = NULL; + + if ((count < 0) || (numbers == NULL)) + { + return NULL; + } + + a = cJSON_CreateArray(); + + for (i = 0; a && (i < (size_t)count); i++) + { + n = cJSON_CreateNumber(numbers[i]); + if (!n) + { + cJSON_Delete(a); + return NULL; + } + if (!i) + { + a->child = n; + } + else + { + suffix_object(p, n); + } + p = n; + } + + if (a && a->child) + { + a->child->prev = n; + } + + return a; +} + +CJSON_PUBLIC(cJSON *) cJSON_CreateStringArray(const char *const *strings, int count) +{ + size_t i = 0; + cJSON *n = NULL; + cJSON *p = NULL; + cJSON *a = NULL; + + if ((count < 0) || (strings == NULL)) + { + return NULL; + } + + a = cJSON_CreateArray(); + + for (i = 0; a && (i < (size_t)count); i++) + { + n = cJSON_CreateString(strings[i]); + if (!n) + { + cJSON_Delete(a); + return NULL; + } + if (!i) + { + a->child = n; + } + else + { + suffix_object(p, n); + } + p = n; + } + + if (a && a->child) + { + a->child->prev = n; + } + + return a; +} + +/* Duplication */ +cJSON *cJSON_Duplicate_rec(const cJSON *item, size_t depth, cJSON_bool recurse); + +CJSON_PUBLIC(cJSON *) cJSON_Duplicate(const cJSON *item, cJSON_bool recurse) +{ + return cJSON_Duplicate_rec(item, 0, recurse); +} + +cJSON *cJSON_Duplicate_rec(const cJSON *item, size_t depth, cJSON_bool recurse) +{ + cJSON *newitem = NULL; + cJSON *child = NULL; + cJSON *next = NULL; + cJSON *newchild = NULL; + + /* Bail on bad ptr */ + if (!item) + { + goto fail; + } + /* Create new item */ + newitem = cJSON_New_Item(&global_hooks); + if (!newitem) + { + goto fail; + } + /* Copy over all vars */ + newitem->type = item->type & (~cJSON_IsReference); + newitem->valueint = item->valueint; + newitem->valuedouble = item->valuedouble; + if (item->valuestring) + { + newitem->valuestring = + (char *)cJSON_strdup((unsigned char *)item->valuestring, &global_hooks); + if (!newitem->valuestring) + { + goto fail; + } + } + if (item->string) + { + newitem->string = (item->type & cJSON_StringIsConst) + ? item->string + : (char *)cJSON_strdup((unsigned char *)item->string, &global_hooks); + if (!newitem->string) + { + goto fail; + } + } + /* If non-recursive, then we're done! */ + if (!recurse) + { + return newitem; + } + /* Walk the ->next chain for the child. */ + child = item->child; + while (child != NULL) + { + if (depth >= CJSON_CIRCULAR_LIMIT) + { + goto fail; + } + newchild = cJSON_Duplicate_rec( + child, depth + 1, true); /* Duplicate (with recurse) each item in the ->next chain */ + if (!newchild) + { + goto fail; + } + if (next != NULL) + { + /* If newitem->child already set, then crosswire ->prev and ->next and move on */ + next->next = newchild; + newchild->prev = next; + next = newchild; + } + else + { + /* Set newitem->child and move to it */ + newitem->child = newchild; + next = newchild; + } + child = child->next; + } + if (newitem && newitem->child) + { + newitem->child->prev = newchild; + } + + return newitem; + +fail: + if (newitem != NULL) + { + cJSON_Delete(newitem); + } + + return NULL; +} + +static void skip_oneline_comment(char **input) +{ + *input += static_strlen("//"); + + for (; (*input)[0] != '\0'; ++(*input)) + { + if ((*input)[0] == '\n') + { + *input += static_strlen("\n"); + return; + } + } +} + +static void skip_multiline_comment(char **input) +{ + *input += static_strlen("/*"); + + for (; (*input)[0] != '\0'; ++(*input)) + { + if (((*input)[0] == '*') && ((*input)[1] == '/')) + { + *input += static_strlen("*/"); + return; + } + } +} + +static void minify_string(char **input, char **output) +{ + (*output)[0] = (*input)[0]; + *input += static_strlen("\""); + *output += static_strlen("\""); + + for (; (*input)[0] != '\0'; (void)++(*input), ++(*output)) + { + (*output)[0] = (*input)[0]; + + if ((*input)[0] == '\"') + { + (*output)[0] = '\"'; + *input += static_strlen("\""); + *output += static_strlen("\""); + return; + } + else if (((*input)[0] == '\\') && ((*input)[1] == '\"')) + { + (*output)[1] = (*input)[1]; + *input += static_strlen("\""); + *output += static_strlen("\""); + } + } +} + +CJSON_PUBLIC(void) cJSON_Minify(char *json) +{ + char *into = json; + + if (json == NULL) + { + return; + } + + while (json[0] != '\0') + { + switch (json[0]) + { + case ' ': + case '\t': + case '\r': + case '\n': + json++; + break; + + case '/': + if (json[1] == '/') + { + skip_oneline_comment(&json); + } + else if (json[1] == '*') + { + skip_multiline_comment(&json); + } + else + { + json++; + } + break; + + case '\"': + minify_string(&json, (char **)&into); + break; + + default: + into[0] = json[0]; + json++; + into++; + } + } + + /* and null-terminate. */ + *into = '\0'; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsInvalid(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_Invalid; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsFalse(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_False; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsTrue(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xff) == cJSON_True; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsBool(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & (cJSON_True | cJSON_False)) != 0; +} +CJSON_PUBLIC(cJSON_bool) cJSON_IsNull(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_NULL; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsNumber(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_Number; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsString(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_String; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsArray(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_Array; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsObject(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_Object; +} + +CJSON_PUBLIC(cJSON_bool) cJSON_IsRaw(const cJSON *const item) +{ + if (item == NULL) + { + return false; + } + + return (item->type & 0xFF) == cJSON_Raw; +} + +CJSON_PUBLIC(cJSON_bool) +cJSON_Compare(const cJSON *const a, const cJSON *const b, const cJSON_bool case_sensitive) +{ + if ((a == NULL) || (b == NULL) || ((a->type & 0xFF) != (b->type & 0xFF))) + { + return false; + } + + /* check if type is valid */ + switch (a->type & 0xFF) + { + case cJSON_False: + case cJSON_True: + case cJSON_NULL: + case cJSON_Number: + case cJSON_String: + case cJSON_Raw: + case cJSON_Array: + case cJSON_Object: + break; + + default: + return false; + } + + /* identical objects are equal */ + if (a == b) + { + return true; + } + + switch (a->type & 0xFF) + { + /* in these cases and equal type is enough */ + case cJSON_False: + case cJSON_True: + case cJSON_NULL: + return true; + + case cJSON_Number: + if (compare_double(a->valuedouble, b->valuedouble)) + { + return true; + } + return false; + + case cJSON_String: + case cJSON_Raw: + if ((a->valuestring == NULL) || (b->valuestring == NULL)) + { + return false; + } + if (strcmp(a->valuestring, b->valuestring) == 0) + { + return true; + } + + return false; + + case cJSON_Array: + { + cJSON *a_element = a->child; + cJSON *b_element = b->child; + + for (; (a_element != NULL) && (b_element != NULL);) + { + if (!cJSON_Compare(a_element, b_element, case_sensitive)) + { + return false; + } + + a_element = a_element->next; + b_element = b_element->next; + } + + /* one of the arrays is longer than the other */ + if (a_element != b_element) + { + return false; + } + + return true; + } + + case cJSON_Object: + { + cJSON *a_element = NULL; + cJSON *b_element = NULL; + cJSON_ArrayForEach(a_element, a) + { + /* TODO This has O(n^2) runtime, which is horrible! */ + b_element = get_object_item(b, a_element->string, case_sensitive); + if (b_element == NULL) + { + return false; + } + + if (!cJSON_Compare(a_element, b_element, case_sensitive)) + { + return false; + } + } + + /* doing this twice, once on a and b to prevent true comparison if a subset of b + * TODO: Do this the proper way, this is just a fix for now */ + cJSON_ArrayForEach(b_element, b) + { + a_element = get_object_item(a, b_element->string, case_sensitive); + if (a_element == NULL) + { + return false; + } + + if (!cJSON_Compare(b_element, a_element, case_sensitive)) + { + return false; + } + } + + return true; + } + + default: + return false; + } +} + +CJSON_PUBLIC(void *) cJSON_malloc(size_t size) +{ + return global_hooks.allocate(size); +} + +CJSON_PUBLIC(void) cJSON_free(void *object) +{ + global_hooks.deallocate(object); + object = NULL; +} diff --git a/src/lsp/cJSON.h b/src/lsp/cJSON.h new file mode 100644 index 0000000..d59fedb --- /dev/null +++ b/src/lsp/cJSON.h @@ -0,0 +1,354 @@ +/* + Copyright (c) 2009-2017 Dave Gamble and cJSON contributors + + Permission is hereby granted, free of charge, to any person obtaining a copy + of this software and associated documentation files (the "Software"), to deal + in the Software without restriction, including without limitation the rights + to use, copy, modify, merge, publish, distribute, sublicense, and/or sell + copies of the Software, and to permit persons to whom the Software is + furnished to do so, subject to the following conditions: + + The above copyright notice and this permission notice shall be included in + all copies or substantial portions of the Software. + + THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE + AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, + OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN + THE SOFTWARE. +*/ + +#ifndef cJSON__h +#define cJSON__h + +#ifdef __cplusplus +extern "C" +{ +#endif + +#if !defined(__WINDOWS__) && \ + (defined(WIN32) || defined(WIN64) || defined(_MSC_VER) || defined(_WIN32)) +#define __WINDOWS__ +#endif + +#ifdef __WINDOWS__ + + /* When compiling for windows, we specify a specific calling convention to avoid issues where we + are being called from a project with a different default calling convention. For windows you + have 3 define options: + + CJSON_HIDE_SYMBOLS - Define this in the case where you don't want to ever dllexport symbols + CJSON_EXPORT_SYMBOLS - Define this on library build when you want to dllexport symbols (default) + CJSON_IMPORT_SYMBOLS - Define this if you want to dllimport symbol + + For *nix builds that support visibility attribute, you can define similar behavior by + + setting default visibility to hidden by adding + -fvisibility=hidden (for gcc) + or + -xldscope=hidden (for sun cc) + to CFLAGS + + then using the CJSON_API_VISIBILITY flag to "export" the same symbols the way + CJSON_EXPORT_SYMBOLS does + + */ + +#define CJSON_CDECL __cdecl +#define CJSON_STDCALL __stdcall + +/* export symbols by default, this is necessary for copy pasting the C and header file */ +#if !defined(CJSON_HIDE_SYMBOLS) && !defined(CJSON_IMPORT_SYMBOLS) && !defined(CJSON_EXPORT_SYMBOLS) +#define CJSON_EXPORT_SYMBOLS +#endif + +#if defined(CJSON_HIDE_SYMBOLS) +#define CJSON_PUBLIC(type) type CJSON_STDCALL +#elif defined(CJSON_EXPORT_SYMBOLS) +#define CJSON_PUBLIC(type) __declspec(dllexport) type CJSON_STDCALL +#elif defined(CJSON_IMPORT_SYMBOLS) +#define CJSON_PUBLIC(type) __declspec(dllimport) type CJSON_STDCALL +#endif +#else /* !__WINDOWS__ */ +#define CJSON_CDECL +#define CJSON_STDCALL + +#if (defined(__GNUC__) || defined(__SUNPRO_CC) || defined(__SUNPRO_C)) && \ + defined(CJSON_API_VISIBILITY) +#define CJSON_PUBLIC(type) __attribute__((visibility("default"))) type +#else +#define CJSON_PUBLIC(type) type +#endif +#endif + +/* project version */ +#define CJSON_VERSION_MAJOR 1 +#define CJSON_VERSION_MINOR 7 +#define CJSON_VERSION_PATCH 19 + +#include <stddef.h> + +/* cJSON Types: */ +#define cJSON_Invalid (0) +#define cJSON_False (1 << 0) +#define cJSON_True (1 << 1) +#define cJSON_NULL (1 << 2) +#define cJSON_Number (1 << 3) +#define cJSON_String (1 << 4) +#define cJSON_Array (1 << 5) +#define cJSON_Object (1 << 6) +#define cJSON_Raw (1 << 7) /* raw json */ + +#define cJSON_IsReference 256 +#define cJSON_StringIsConst 512 + + /* The cJSON structure: */ + typedef struct cJSON + { + /* next/prev allow you to walk array/object chains. Alternatively, use + * GetArraySize/GetArrayItem/GetObjectItem */ + struct cJSON *next; + struct cJSON *prev; + /* An array or object item will have a child pointer pointing to a chain of the items in the + * array/object. */ + struct cJSON *child; + + /* The type of the item, as above. */ + int type; + + /* The item's string, if type==cJSON_String and type == cJSON_Raw */ + char *valuestring; + /* writing to valueint is DEPRECATED, use cJSON_SetNumberValue instead */ + int valueint; + /* The item's number, if type==cJSON_Number */ + double valuedouble; + + /* The item's name string, if this item is the child of, or is in the list of subitems of an + * object. */ + char *string; + } cJSON; + + typedef struct cJSON_Hooks + { + /* malloc/free are CDECL on Windows regardless of the default calling convention of the + * compiler, so ensure the hooks allow passing those functions directly. */ + void *(CJSON_CDECL *malloc_fn)(size_t sz); + void(CJSON_CDECL *free_fn)(void *ptr); + } cJSON_Hooks; + + typedef int cJSON_bool; + +/* Limits how deeply nested arrays/objects can be before cJSON rejects to parse them. + * This is to prevent stack overflows. */ +#ifndef CJSON_NESTING_LIMIT +#define CJSON_NESTING_LIMIT 1000 +#endif + +/* Limits the length of circular references can be before cJSON rejects to parse them. + * This is to prevent stack overflows. */ +#ifndef CJSON_CIRCULAR_LIMIT +#define CJSON_CIRCULAR_LIMIT 10000 +#endif + + /* returns the version of cJSON as a string */ + CJSON_PUBLIC(const char *) cJSON_Version(void); + + /* Supply malloc, realloc and free functions to cJSON */ + CJSON_PUBLIC(void) cJSON_InitHooks(cJSON_Hooks *hooks); + + /* Memory Management: the caller is always responsible to free the results from all variants of + * cJSON_Parse (with cJSON_Delete) and cJSON_Print (with stdlib free, cJSON_Hooks.free_fn, or + * cJSON_free as appropriate). The exception is cJSON_PrintPreallocated, where the caller has + * full responsibility of the buffer. */ + /* Supply a block of JSON, and this returns a cJSON object you can interrogate. */ + CJSON_PUBLIC(cJSON *) cJSON_Parse(const char *value); + CJSON_PUBLIC(cJSON *) cJSON_ParseWithLength(const char *value, size_t buffer_length); + /* ParseWithOpts allows you to require (and check) that the JSON is null terminated, and to + * retrieve the pointer to the final byte parsed. */ + /* If you supply a ptr in return_parse_end and parsing fails, then return_parse_end will contain + * a pointer to the error so will match cJSON_GetErrorPtr(). */ + CJSON_PUBLIC(cJSON *) + cJSON_ParseWithOpts(const char *value, const char **return_parse_end, + cJSON_bool require_null_terminated); + CJSON_PUBLIC(cJSON *) + cJSON_ParseWithLengthOpts(const char *value, size_t buffer_length, + const char **return_parse_end, cJSON_bool require_null_terminated); + + /* Render a cJSON entity to text for transfer/storage. */ + CJSON_PUBLIC(char *) cJSON_Print(const cJSON *item); + /* Render a cJSON entity to text for transfer/storage without any formatting. */ + CJSON_PUBLIC(char *) cJSON_PrintUnformatted(const cJSON *item); + /* Render a cJSON entity to text using a buffered strategy. prebuffer is a guess at the final + * size. guessing well reduces reallocation. fmt=0 gives unformatted, =1 gives formatted */ + CJSON_PUBLIC(char *) cJSON_PrintBuffered(const cJSON *item, int prebuffer, cJSON_bool fmt); + /* Render a cJSON entity to text using a buffer already allocated in memory with given length. + * Returns 1 on success and 0 on failure. */ + /* NOTE: cJSON is not always 100% accurate in estimating how much memory it will use, so to be + * safe allocate 5 bytes more than you actually need */ + CJSON_PUBLIC(cJSON_bool) + cJSON_PrintPreallocated(cJSON *item, char *buffer, const int length, const cJSON_bool format); + /* Delete a cJSON entity and all subentities. */ + CJSON_PUBLIC(void) cJSON_Delete(cJSON *item); + + /* Returns the number of items in an array (or object). */ + CJSON_PUBLIC(int) cJSON_GetArraySize(const cJSON *array); + /* Retrieve item number "index" from array "array". Returns NULL if unsuccessful. */ + CJSON_PUBLIC(cJSON *) cJSON_GetArrayItem(const cJSON *array, int index); + /* Get item "string" from object. Case insensitive. */ + CJSON_PUBLIC(cJSON *) cJSON_GetObjectItem(const cJSON *const object, const char *const string); + CJSON_PUBLIC(cJSON *) + cJSON_GetObjectItemCaseSensitive(const cJSON *const object, const char *const string); + CJSON_PUBLIC(cJSON_bool) cJSON_HasObjectItem(const cJSON *object, const char *string); + /* For analysing failed parses. This returns a pointer to the parse error. You'll probably need + * to look a few chars back to make sense of it. Defined when cJSON_Parse() returns 0. 0 when + * cJSON_Parse() succeeds. */ + CJSON_PUBLIC(const char *) cJSON_GetErrorPtr(void); + + /* Check item type and return its value */ + CJSON_PUBLIC(char *) cJSON_GetStringValue(const cJSON *const item); + CJSON_PUBLIC(double) cJSON_GetNumberValue(const cJSON *const item); + + /* These functions check the type of an item */ + CJSON_PUBLIC(cJSON_bool) cJSON_IsInvalid(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsFalse(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsTrue(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsBool(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsNull(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsNumber(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsString(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsArray(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsObject(const cJSON *const item); + CJSON_PUBLIC(cJSON_bool) cJSON_IsRaw(const cJSON *const item); + + /* These calls create a cJSON item of the appropriate type. */ + CJSON_PUBLIC(cJSON *) cJSON_CreateNull(void); + CJSON_PUBLIC(cJSON *) cJSON_CreateTrue(void); + CJSON_PUBLIC(cJSON *) cJSON_CreateFalse(void); + CJSON_PUBLIC(cJSON *) cJSON_CreateBool(cJSON_bool boolean); + CJSON_PUBLIC(cJSON *) cJSON_CreateNumber(double num); + CJSON_PUBLIC(cJSON *) cJSON_CreateString(const char *string); + /* raw json */ + CJSON_PUBLIC(cJSON *) cJSON_CreateRaw(const char *raw); + CJSON_PUBLIC(cJSON *) cJSON_CreateArray(void); + CJSON_PUBLIC(cJSON *) cJSON_CreateObject(void); + + /* Create a string where valuestring references a string so + * it will not be freed by cJSON_Delete */ + CJSON_PUBLIC(cJSON *) cJSON_CreateStringReference(const char *string); + /* Create an object/array that only references it's elements so + * they will not be freed by cJSON_Delete */ + CJSON_PUBLIC(cJSON *) cJSON_CreateObjectReference(const cJSON *child); + CJSON_PUBLIC(cJSON *) cJSON_CreateArrayReference(const cJSON *child); + + /* These utilities create an Array of count items. + * The parameter count cannot be greater than the number of elements in the number array, + * otherwise array access will be out of bounds.*/ + CJSON_PUBLIC(cJSON *) cJSON_CreateIntArray(const int *numbers, int count); + CJSON_PUBLIC(cJSON *) cJSON_CreateFloatArray(const float *numbers, int count); + CJSON_PUBLIC(cJSON *) cJSON_CreateDoubleArray(const double *numbers, int count); + CJSON_PUBLIC(cJSON *) cJSON_CreateStringArray(const char *const *strings, int count); + + /* Append item to the specified array/object. */ + CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToArray(cJSON *array, cJSON *item); + CJSON_PUBLIC(cJSON_bool) cJSON_AddItemToObject(cJSON *object, const char *string, cJSON *item); + /* Use this when string is definitely const (i.e. a literal, or as good as), and will definitely + * survive the cJSON object. WARNING: When this function was used, make sure to always check + * that (item->type & cJSON_StringIsConst) is zero before writing to `item->string` */ + CJSON_PUBLIC(cJSON_bool) + cJSON_AddItemToObjectCS(cJSON *object, const char *string, cJSON *item); + /* Append reference to item to the specified array/object. Use this when you want to add an + * existing cJSON to a new cJSON, but don't want to corrupt your existing cJSON. */ + CJSON_PUBLIC(cJSON_bool) cJSON_AddItemReferenceToArray(cJSON *array, cJSON *item); + CJSON_PUBLIC(cJSON_bool) + cJSON_AddItemReferenceToObject(cJSON *object, const char *string, cJSON *item); + + /* Remove/Detach items from Arrays/Objects. */ + CJSON_PUBLIC(cJSON *) cJSON_DetachItemViaPointer(cJSON *parent, cJSON *const item); + CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromArray(cJSON *array, int which); + CJSON_PUBLIC(void) cJSON_DeleteItemFromArray(cJSON *array, int which); + CJSON_PUBLIC(cJSON *) cJSON_DetachItemFromObject(cJSON *object, const char *string); + CJSON_PUBLIC(cJSON *) + cJSON_DetachItemFromObjectCaseSensitive(cJSON *object, const char *string); + CJSON_PUBLIC(void) cJSON_DeleteItemFromObject(cJSON *object, const char *string); + CJSON_PUBLIC(void) cJSON_DeleteItemFromObjectCaseSensitive(cJSON *object, const char *string); + + /* Update array items. */ + CJSON_PUBLIC(cJSON_bool) + cJSON_InsertItemInArray(cJSON *array, int which, + cJSON *newitem); /* Shifts pre-existing items to the right. */ + CJSON_PUBLIC(cJSON_bool) + cJSON_ReplaceItemViaPointer(cJSON *const parent, cJSON *const item, cJSON *replacement); + CJSON_PUBLIC(cJSON_bool) cJSON_ReplaceItemInArray(cJSON *array, int which, cJSON *newitem); + CJSON_PUBLIC(cJSON_bool) + cJSON_ReplaceItemInObject(cJSON *object, const char *string, cJSON *newitem); + CJSON_PUBLIC(cJSON_bool) + cJSON_ReplaceItemInObjectCaseSensitive(cJSON *object, const char *string, cJSON *newitem); + + /* Duplicate a cJSON item */ + CJSON_PUBLIC(cJSON *) cJSON_Duplicate(const cJSON *item, cJSON_bool recurse); + /* Duplicate will create a new, identical cJSON item to the one you pass, in new memory that + * will need to be released. With recurse!=0, it will duplicate any children connected to the + * item. The item->next and ->prev pointers are always zero on return from Duplicate. */ + /* Recursively compare two cJSON items for equality. If either a or b is NULL or invalid, they + * will be considered unequal. case_sensitive determines if object keys are treated case + * sensitive (1) or case insensitive (0) */ + CJSON_PUBLIC(cJSON_bool) + cJSON_Compare(const cJSON *const a, const cJSON *const b, const cJSON_bool case_sensitive); + + /* Minify a strings, remove blank characters(such as ' ', '\t', '\r', '\n') from strings. + * The input pointer json cannot point to a read-only address area, such as a string constant, + * but should point to a readable and writable address area. */ + CJSON_PUBLIC(void) cJSON_Minify(char *json); + + /* Helper functions for creating and adding items to an object at the same time. + * They return the added item or NULL on failure. */ + CJSON_PUBLIC(cJSON *) cJSON_AddNullToObject(cJSON *const object, const char *const name); + CJSON_PUBLIC(cJSON *) cJSON_AddTrueToObject(cJSON *const object, const char *const name); + CJSON_PUBLIC(cJSON *) cJSON_AddFalseToObject(cJSON *const object, const char *const name); + CJSON_PUBLIC(cJSON *) + cJSON_AddBoolToObject(cJSON *const object, const char *const name, const cJSON_bool boolean); + CJSON_PUBLIC(cJSON *) + cJSON_AddNumberToObject(cJSON *const object, const char *const name, const double number); + CJSON_PUBLIC(cJSON *) + cJSON_AddStringToObject(cJSON *const object, const char *const name, const char *const string); + CJSON_PUBLIC(cJSON *) + cJSON_AddRawToObject(cJSON *const object, const char *const name, const char *const raw); + CJSON_PUBLIC(cJSON *) cJSON_AddObjectToObject(cJSON *const object, const char *const name); + CJSON_PUBLIC(cJSON *) cJSON_AddArrayToObject(cJSON *const object, const char *const name); + +/* When assigning an integer value, it needs to be propagated to valuedouble too. */ +#define cJSON_SetIntValue(object, number) \ + ((object) ? (object)->valueint = (object)->valuedouble = (number) : (number)) + /* helper for the cJSON_SetNumberValue macro */ + CJSON_PUBLIC(double) cJSON_SetNumberHelper(cJSON *object, double number); +#define cJSON_SetNumberValue(object, number) \ + ((object != NULL) ? cJSON_SetNumberHelper(object, (double)number) : (number)) + /* Change the valuestring of a cJSON_String object, only takes effect when type of object is + * cJSON_String */ + CJSON_PUBLIC(char *) cJSON_SetValuestring(cJSON *object, const char *valuestring); + +/* If the object is not a boolean type this does nothing and returns cJSON_Invalid else it returns + * the new type*/ +#define cJSON_SetBoolValue(object, boolValue) \ + ((object != NULL && ((object)->type & (cJSON_False | cJSON_True))) \ + ? (object)->type = ((object)->type & (~(cJSON_False | cJSON_True))) | \ + ((boolValue) ? cJSON_True : cJSON_False) \ + : cJSON_Invalid) + +/* Macro for iterating over an array or object */ +#define cJSON_ArrayForEach(element, array) \ + for (element = (array != NULL) ? (array)->child : NULL; element != NULL; \ + element = element->next) + + /* malloc/free objects using the malloc/free functions that have been set with cJSON_InitHooks + */ + CJSON_PUBLIC(void *) cJSON_malloc(size_t size); + CJSON_PUBLIC(void) cJSON_free(void *object); + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/src/lsp/json_rpc.c b/src/lsp/json_rpc.c index 903da71..9b281bc 100644 --- a/src/lsp/json_rpc.c +++ b/src/lsp/json_rpc.c @@ -1,194 +1,208 @@ - #include "json_rpc.h" +#include "cJSON.h" +#include "lsp_project.h" #include <stdio.h> #include <stdlib.h> #include <string.h> -// Basic JSON parsing helpers -char *get_json_string(const char *json, const char *key) +void lsp_check_file(const char *uri, const char *src); +void lsp_goto_definition(const char *uri, int line, int col); +void lsp_hover(const char *uri, int line, int col); +void lsp_completion(const char *uri, int line, int col); +void lsp_document_symbol(const char *uri); +void lsp_references(const char *uri, int line, int col); +void lsp_signature_help(const char *uri, int line, int col); + +// Helper to extract textDocument params +static void get_params(cJSON *root, char **uri, int *line, int *col) { - char search[256]; - sprintf(search, "\"%s\":\"", key); - char *p = strstr(json, search); - if (!p) + cJSON *params = cJSON_GetObjectItem(root, "params"); + if (!params) { - return NULL; + return; } - p += strlen(search); - char *end = strchr(p, '"'); - if (!end) + + cJSON *doc = cJSON_GetObjectItem(params, "textDocument"); + if (doc) { - return NULL; + cJSON *u = cJSON_GetObjectItem(doc, "uri"); + if (u && u->valuestring) + { + *uri = strdup(u->valuestring); + } } - int len = end - p; - char *res = malloc(len + 1); - strncpy(res, p, len); - res[len] = 0; - return res; -} -// Extract nested "text" from params/contentChanges/0/text or -// params/textDocument/text This is very hacky for MVP. proper JSON library -// needed. + cJSON *pos = cJSON_GetObjectItem(params, "position"); + if (pos) + { + cJSON *l = cJSON_GetObjectItem(pos, "line"); + cJSON *c = cJSON_GetObjectItem(pos, "character"); + if (l) + { + *line = l->valueint; + } + if (c) + { + *col = c->valueint; + } + } +} -char *get_text_content(const char *json) +void handle_request(const char *json_str) { - char *p = strstr(json, "\"text\":\""); - if (!p) + cJSON *json = cJSON_Parse(json_str); + if (!json) { - return NULL; + return; } - p += 8; - size_t cap = strlen(p); - char *res = malloc(cap + 1); - char *dst = res; + cJSON *method_item = cJSON_GetObjectItem(json, "method"); + if (!method_item || !method_item->valuestring) + { + cJSON_Delete(json); + return; + } + char *method = method_item->valuestring; - while (*p) + if (strcmp(method, "initialize") == 0) { - if (*p == '\\') + cJSON *params = cJSON_GetObjectItem(json, "params"); + char *root = NULL; + if (params) { - p++; - if (*p == 'n') - { - *dst++ = '\n'; - } - else if (*p == 'r') + cJSON *rp = cJSON_GetObjectItem(params, "rootPath"); + if (rp && rp->valuestring) { - *dst++ = '\r'; - } - else if (*p == 't') - { - *dst++ = '\t'; - } - else if (*p == '"') - { - *dst++ = '"'; - } - else if (*p == '\\') - { - *dst++ = '\\'; + root = strdup(rp->valuestring); } else { - *dst++ = *p; // preserve others + cJSON *ru = cJSON_GetObjectItem(params, "rootUri"); + if (ru && ru->valuestring) + { + root = strdup(ru->valuestring); + } } - p++; } - else if (*p == '"') + + if (root && strncmp(root, "file://", 7) == 0) { - break; // End of string. + char *clean = strdup(root + 7); + free(root); + root = clean; } - else + + lsp_project_init(root ? root : "."); + if (root) { - *dst++ = *p++; + free(root); } - } - *dst = 0; - return res; -} -// Helper to get line/char -void get_json_position(const char *json, int *line, int *col) -{ - char *pos = strstr(json, "\"position\":"); - if (!pos) - { - return; - } - char *l = strstr(pos, "\"line\":"); - if (l) - { - *line = atoi(l + 7); - } - char *c = strstr(pos, "\"character\":"); - if (c) - { - *col = atoi(c + 12); - } -} - -void lsp_check_file(const char *uri, const char *src); -void lsp_goto_definition(const char *uri, int line, int col); -void lsp_hover(const char *uri, int line, int col); -void lsp_completion(const char *uri, int line, int col); - -void handle_request(const char *json_str) -{ - if (strstr(json_str, "\"method\":\"initialize\"")) - { const char *response = "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":{" "\"capabilities\":{\"textDocumentSync\":1," "\"definitionProvider\":true,\"hoverProvider\":true," + "\"referencesProvider\":true,\"documentSymbolProvider\":true," + "\"signatureHelpProvider\":{\"triggerCharacters\":[\"(\"]}," "\"completionProvider\":{" "\"triggerCharacters\":[\".\"]}}}}"; fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(response), response); fflush(stdout); - return; } - - if (strstr(json_str, "\"method\":\"textDocument/didOpen\"") || - strstr(json_str, "\"method\":\"textDocument/didChange\"")) + else if (strcmp(method, "textDocument/didOpen") == 0 || + strcmp(method, "textDocument/didChange") == 0) { - - char *uri = get_json_string(json_str, "uri"); - char *text = get_text_content(json_str); - - if (uri && text) + cJSON *params = cJSON_GetObjectItem(json, "params"); + if (params) { - fprintf(stderr, "zls: Checking %s\n", uri); - lsp_check_file(uri, text); + cJSON *doc = cJSON_GetObjectItem(params, "textDocument"); + if (doc) + { + cJSON *uri = cJSON_GetObjectItem(doc, "uri"); + cJSON *text = cJSON_GetObjectItem(doc, "text"); + // For didChange, text is inside contentChanges + if (!text && strcmp(method, "textDocument/didChange") == 0) + { + cJSON *changes = cJSON_GetObjectItem(params, "contentChanges"); + if (changes && cJSON_GetArraySize(changes) > 0) + { + cJSON *change = cJSON_GetArrayItem(changes, 0); + text = cJSON_GetObjectItem(change, "text"); + } + } + + if (uri && uri->valuestring && text && text->valuestring) + { + lsp_check_file(uri->valuestring, text->valuestring); + } + } } - + } + else if (strcmp(method, "textDocument/definition") == 0) + { + char *uri = NULL; + int line = 0, col = 0; + get_params(json, &uri, &line, &col); if (uri) { + lsp_goto_definition(uri, line, col); free(uri); } - if (text) + } + else if (strcmp(method, "textDocument/hover") == 0) + { + char *uri = NULL; + int line = 0, col = 0; + get_params(json, &uri, &line, &col); + if (uri) { - free(text); + lsp_hover(uri, line, col); + free(uri); } } - - if (strstr(json_str, "\"method\":\"textDocument/definition\"")) + else if (strcmp(method, "textDocument/completion") == 0) { - char *uri = get_json_string(json_str, "uri"); + char *uri = NULL; int line = 0, col = 0; - get_json_position(json_str, &line, &col); - + get_params(json, &uri, &line, &col); if (uri) { - fprintf(stderr, "zls: Definition request at %d:%d\n", line, col); - lsp_goto_definition(uri, line, col); + lsp_completion(uri, line, col); free(uri); } } - - if (strstr(json_str, "\"method\":\"textDocument/hover\"")) + else if (strcmp(method, "textDocument/documentSymbol") == 0) { - char *uri = get_json_string(json_str, "uri"); + char *uri = NULL; + int line = 0, col = 0; // Unused for outline + get_params(json, &uri, &line, &col); + if (uri) + { + lsp_document_symbol(uri); + free(uri); + } + } + else if (strcmp(method, "textDocument/references") == 0) + { + char *uri = NULL; int line = 0, col = 0; - get_json_position(json_str, &line, &col); - + get_params(json, &uri, &line, &col); if (uri) { - fprintf(stderr, "zls: Hover request at %d:%d\n", line, col); - lsp_hover(uri, line, col); + lsp_references(uri, line, col); free(uri); } } - - if (strstr(json_str, "\"method\":\"textDocument/completion\"")) + else if (strcmp(method, "textDocument/signatureHelp") == 0) { - char *uri = get_json_string(json_str, "uri"); + char *uri = NULL; int line = 0, col = 0; - get_json_position(json_str, &line, &col); - + get_params(json, &uri, &line, &col); if (uri) { - fprintf(stderr, "zls: Completion request at %d:%d\n", line, col); - lsp_completion(uri, line, col); + lsp_signature_help(uri, line, col); free(uri); } } + + cJSON_Delete(json); } diff --git a/src/lsp/lsp_analysis.c b/src/lsp/lsp_analysis.c index d455894..e63ae4a 100644 --- a/src/lsp/lsp_analysis.c +++ b/src/lsp/lsp_analysis.c @@ -1,13 +1,11 @@ - #include "json_rpc.h" -#include "lsp_index.h" -#include "parser.h" +#include "cJSON.h" +#include "lsp_project.h" // Includes lsp_index.h, parser.h #include <ctype.h> #include <stdio.h> #include <stdlib.h> #include <string.h> - -static LSPIndex *g_index = NULL; +#include <unistd.h> typedef struct Diagnostic { @@ -23,17 +21,28 @@ typedef struct Diagnostic *tail; } DiagnosticList; -static ParserContext *g_ctx = NULL; -static char *g_last_src = NULL; +// Helper to send JSON response +static void send_json_response(cJSON *root) +{ + char *str = cJSON_PrintUnformatted(root); + if (str) + { + fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(str), str); + fflush(stdout); + free(str); + } + cJSON_Delete(root); +} // Callback for parser errors. void lsp_on_error(void *data, Token t, const char *msg) { DiagnosticList *list = (DiagnosticList *)data; - Diagnostic *d = xmalloc(sizeof(Diagnostic)); + // Simple allocation for MVP + Diagnostic *d = calloc(1, sizeof(Diagnostic)); d->line = t.line > 0 ? t.line - 1 : 0; d->col = t.col > 0 ? t.col - 1 : 0; - d->message = xstrdup(msg); + d->message = strdup(msg); d->next = NULL; if (!list->head) @@ -50,79 +59,79 @@ void lsp_on_error(void *data, Token t, const char *msg) void lsp_check_file(const char *uri, const char *json_src) { - // Prepare ParserContext (persistent). - if (g_ctx) + if (!g_project) { - - free(g_ctx); + // Fallback or lazy init? current dir + char cwd[1024]; + if (getcwd(cwd, sizeof(cwd))) + { + lsp_project_init(cwd); + } + else + { + lsp_project_init("."); + } } - g_ctx = calloc(1, sizeof(ParserContext)); - g_ctx->is_fault_tolerant = 1; + // Setup error capture on the global project context DiagnosticList diagnostics = {0}; - g_ctx->error_callback_data = &diagnostics; - g_ctx->on_error = lsp_on_error; - // Prepare Lexer. - // Cache source. - if (g_last_src) - { - free(g_last_src); - } - g_last_src = strdup(json_src); + // We attach the callback to 'g_project->ctx'. + // NOTE: If we use lsp_project_update_file, it uses g_project->ctx. + void *old_data = g_project->ctx->error_callback_data; + void (*old_cb)(void *, Token, const char *) = g_project->ctx->on_error; - Lexer l; - lexer_init(&l, json_src); + g_project->ctx->error_callback_data = &diagnostics; + g_project->ctx->on_error = lsp_on_error; - ASTNode *root = parse_program(g_ctx, &l); + // Update and Parse + lsp_project_update_file(uri, json_src); - if (g_index) - { - lsp_index_free(g_index); - } - g_index = lsp_index_new(); - if (root) - { - lsp_build_index(g_index, root); - } + // Restore + g_project->ctx->on_error = old_cb; + g_project->ctx->error_callback_data = old_data; - // Construct JSON Response (notification) + // Construct JSON Response (publishDiagnostics) + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddStringToObject(root, "method", "textDocument/publishDiagnostics"); - char *notification = malloc(128 * 1024); - char *p = notification; - p += sprintf(p, - "{\"jsonrpc\":\"2.0\",\"method\":\"textDocument/" - "publishDiagnostics\",\"params\":{\"uri\":\"%s\",\"diagnostics\":[", - uri); + cJSON *params = cJSON_CreateObject(); + cJSON_AddStringToObject(params, "uri", uri); + + cJSON *diag_array = cJSON_CreateArray(); Diagnostic *d = diagnostics.head; while (d) { + cJSON *diag = cJSON_CreateObject(); - p += sprintf(p, - "{\"range\":{\"start\":{\"line\":%d,\"character\":%d},\"end\":" - "{\"line\":%d," - "\"character\":%d}},\"severity\":1,\"message\":\"%s\"}", - d->line, d->col, d->line, d->col + 1, d->message); + cJSON *range = cJSON_CreateObject(); + cJSON *start = cJSON_CreateObject(); + cJSON_AddNumberToObject(start, "line", d->line); + cJSON_AddNumberToObject(start, "character", d->col); - if (d->next) - { - p += sprintf(p, ","); - } + cJSON *end = cJSON_CreateObject(); + cJSON_AddNumberToObject(end, "line", d->line); + cJSON_AddNumberToObject(end, "character", d->col + 1); + + cJSON_AddItemToObject(range, "start", start); + cJSON_AddItemToObject(range, "end", end); + + cJSON_AddItemToObject(diag, "range", range); + cJSON_AddNumberToObject(diag, "severity", 1); + cJSON_AddStringToObject(diag, "message", d->message); + + cJSON_AddItemToArray(diag_array, diag); d = d->next; } - p += sprintf(p, "]}}"); - - // Send notification. - long len = strlen(notification); - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", len, notification); - fflush(stdout); + cJSON_AddItemToObject(params, "diagnostics", diag_array); + cJSON_AddItemToObject(root, "params", params); - free(notification); + send_json_response(root); - // Cleanup. Diagnostic *cur = diagnostics.head; while (cur) { @@ -135,58 +144,126 @@ void lsp_check_file(const char *uri, const char *json_src) void lsp_goto_definition(const char *uri, int line, int col) { - if (!g_index) + ProjectFile *pf = lsp_project_get_file(uri); + LSPIndex *idx = pf ? pf->index : NULL; + + if (!idx) { return; } - LSPRange *r = lsp_find_at(g_index, line, col); - if (r && r->type == RANGE_REFERENCE) + LSPRange *r = lsp_find_at(idx, line, col); + const char *target_uri = uri; + int target_start_line = 0, target_start_col = 0; + int target_end_line = 0, target_end_col = 0; + int found = 0; + + // 1. Check Local Index + if (r) { - // Found reference, return definition - char resp[1024]; - sprintf(resp, - "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":{\"uri\":\"%s\"," - "\"range\":{\"start\":{" - "\"line\":%d,\"character\":%d},\"end\":{\"line\":%d,\"character\":%" - "d}}}}", - uri, r->def_line, r->def_col, r->def_line, r->def_col); - - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(resp), resp); - fflush(stdout); + if (r->type == RANGE_DEFINITION) + { + // Already at definition + target_start_line = r->start_line; + target_start_col = r->start_col; + target_end_line = r->end_line; + target_end_col = r->end_col; + found = 1; + } + else if (r->type == RANGE_REFERENCE && r->def_line >= 0) + { + LSPRange *def = lsp_find_at(idx, r->def_line, r->def_col); + int is_local = 0; + if (def && def->type == RANGE_DEFINITION) + { + // Check name congruence logic ... (simplified for now) + // Assume logic in previous version was correct about checking names + is_local = 1; + } + + if (is_local) + { + target_start_line = r->def_line; + target_start_col = r->def_col; + target_end_line = r->def_line; + target_end_col = r->def_col; // approx + found = 1; + } + } } - else if (r && r->type == RANGE_DEFINITION) + + // 2. Global Definition (if local failed) + if (!found && r && r->node) { - // Already at definition? Return itself. - char resp[1024]; - sprintf(resp, - "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":{\"uri\":\"%s\"," - "\"range\":{\"start\":{" - "\"line\":%d,\"character\":%d},\"end\":{\"line\":%d,\"character\":%" - "d}}}}", - uri, r->start_line, r->start_col, r->end_line, r->end_col); - - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(resp), resp); - fflush(stdout); + char *name = NULL; + if (r->node->type == NODE_EXPR_VAR) + { + name = r->node->var_ref.name; + } + else if (r->node->type == NODE_EXPR_CALL && r->node->call.callee->type == NODE_EXPR_VAR) + { + name = r->node->call.callee->var_ref.name; + } + + if (name) + { + DefinitionResult def = lsp_project_find_definition(name); + if (def.uri && def.range) + { + target_uri = def.uri; + target_start_line = def.range->start_line; + target_start_col = def.range->start_col; + target_end_line = def.range->end_line; + target_end_col = def.range->end_col; + found = 1; + } + } + } + + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); + + if (found) + { + cJSON *result = cJSON_CreateObject(); + cJSON_AddStringToObject(result, "uri", target_uri); + + cJSON *range = cJSON_CreateObject(); + cJSON *start = cJSON_CreateObject(); + cJSON_AddNumberToObject(start, "line", target_start_line); + cJSON_AddNumberToObject(start, "character", target_start_col); + + cJSON *end = cJSON_CreateObject(); + cJSON_AddNumberToObject(end, "line", target_end_line); + cJSON_AddNumberToObject(end, "character", target_end_col); + + cJSON_AddItemToObject(range, "start", start); + cJSON_AddItemToObject(range, "end", end); + cJSON_AddItemToObject(result, "range", range); + + cJSON_AddItemToObject(root, "result", result); } else { - // Null result - const char *null_resp = "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":null}"; - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(null_resp), null_resp); - fflush(stdout); + cJSON_AddNullToObject(root, "result"); } + + send_json_response(root); } void lsp_hover(const char *uri, int line, int col) { (void)uri; - if (!g_index) + ProjectFile *pf = lsp_project_get_file(uri); + LSPIndex *idx = pf ? pf->index : NULL; + + if (!idx) { return; } - LSPRange *r = lsp_find_at(g_index, line, col); + LSPRange *r = lsp_find_at(idx, line, col); char *text = NULL; if (r) @@ -197,7 +274,7 @@ void lsp_hover(const char *uri, int line, int col) } else if (r->type == RANGE_REFERENCE) { - LSPRange *def = lsp_find_at(g_index, r->def_line, r->def_col); + LSPRange *def = lsp_find_at(idx, r->def_line, r->def_col); if (def && def->type == RANGE_DEFINITION) { text = def->hover_text; @@ -205,44 +282,59 @@ void lsp_hover(const char *uri, int line, int col) } } + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); + if (text) { - char *json = malloc(16384); - // content: { kind: markdown, value: text } - sprintf(json, - "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":{\"contents\":{\"kind\":" - "\"markdown\"," - "\"value\":\"```c\\n%s\\n```\"}}}", - text); - - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(json), json); - fflush(stdout); - free(json); + cJSON *result = cJSON_CreateObject(); + cJSON *contents = cJSON_CreateObject(); + cJSON_AddStringToObject(contents, "kind", "markdown"); + + // Need to wrap in ```c code block + char *code_block = malloc(strlen(text) + 16); + sprintf(code_block, "```c\n%s\n```", text); + cJSON_AddStringToObject(contents, "value", code_block); + free(code_block); + + cJSON_AddItemToObject(result, "contents", contents); + cJSON_AddItemToObject(root, "result", result); } else { - const char *null_resp = "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":null}"; - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(null_resp), null_resp); - fflush(stdout); + cJSON_AddNullToObject(root, "result"); } + + send_json_response(root); } void lsp_completion(const char *uri, int line, int col) { - (void)uri; - if (!g_ctx) + ProjectFile *pf = lsp_project_get_file(uri); + // Need global project context + if (!g_project || !g_project->ctx || !pf) { return; } - // Context-aware completion (Dot access) - if (g_last_src) + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); + cJSON *items = cJSON_CreateArray(); + + // 1. Context-aware completion (Dot access) + // ... [Same logic as before, just constructing cJSON] ... + // Note: To save space/complexity in this rewrite, I'll streamline the dot completion logic + // or just implement the global fallback for now if the regex parsing is too complex to inline + // perfectly here. However, the original code had significant line-scanning logic. I will + // attempt to preserve the dot completion logic by copying it and using cJSON. + + int dot_completed = 0; + if (pf->source) { - // Simple line access int cur_line = 0; - - char *ptr = g_last_src; - // Fast forward to line + char *ptr = pf->source; while (*ptr && cur_line < line) { if (*ptr == '\n') @@ -254,14 +346,12 @@ void lsp_completion(const char *uri, int line, int col) if (col > 0 && ptr[col - 1] == '.') { - // Found dot! - // Scan backwards for identifier: [whitespace] [ident] . + // Found dot logic int i = col - 2; while (i >= 0 && (ptr[i] == ' ' || ptr[i] == '\t')) { i--; } - if (i >= 0) { int end_ident = i; @@ -270,7 +360,6 @@ void lsp_completion(const char *uri, int line, int col) i--; } int start_ident = i + 1; - if (start_ident <= end_ident) { int len = end_ident - start_ident + 1; @@ -278,9 +367,8 @@ void lsp_completion(const char *uri, int line, int col) strncpy(var_name, ptr + start_ident, len); var_name[len] = 0; + Symbol *sym = find_symbol_in_all(g_project->ctx, var_name); char *type_name = NULL; - Symbol *sym = find_symbol_in_all(g_ctx, var_name); - if (sym) { if (sym->type_info) @@ -297,7 +385,7 @@ void lsp_completion(const char *uri, int line, int col) { char clean_name[256]; char *src = type_name; - if (0 == strncmp(src, "struct ", 7)) + if (strncmp(src, "struct ", 7) == 0) { src += 7; } @@ -308,88 +396,356 @@ void lsp_completion(const char *uri, int line, int col) } *dst = 0; - // Lookup struct. - StructDef *sd = g_ctx->struct_defs; + StructDef *sd = g_project->ctx->struct_defs; while (sd) { - if (0 == strcmp(sd->name, clean_name)) + if (strcmp(sd->name, clean_name) == 0) { - char *json_fields = malloc(1024 * 1024); - char *pj = json_fields; - pj += sprintf(pj, "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":["); - - int ffirst = 1; if (sd->node && sd->node->strct.fields) { ASTNode *field = sd->node->strct.fields; while (field) { - if (!ffirst) - { - pj += sprintf(pj, ","); - } - pj += sprintf( - pj, - "{\"label\":\"%s\",\"kind\":5,\"detail\":\"field %s\"}", - field->field.name, field->field.type); // Kind 5 = Field - ffirst = 0; + cJSON *item = cJSON_CreateObject(); + cJSON_AddStringToObject(item, "label", field->field.name); + cJSON_AddNumberToObject(item, "kind", 5); // Field + char detail[256]; + sprintf(detail, "field %s", field->field.type); + cJSON_AddStringToObject(item, "detail", detail); + cJSON_AddItemToArray(items, item); field = field->next; } } - - pj += sprintf(pj, "]}"); - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", - strlen(json_fields), json_fields); - fflush(stdout); - free(json_fields); - free(json); - return; // Done, yippee. + dot_completed = 1; + break; } sd = sd->next; } + if (sym && sym->type_info) + { + free(type_name); + } } } } } } - char *json = xmalloc(1024 * 1024); // 1MB buffer. - char *p = json; - p += sprintf(p, "{\"jsonrpc\":\"2.0\",\"id\":1,\"result\":["); + if (!dot_completed) + { + // Global Completion + FuncSig *f = g_project->ctx->func_registry; + while (f) + { + cJSON *item = cJSON_CreateObject(); + cJSON_AddStringToObject(item, "label", f->name); + cJSON_AddNumberToObject(item, "kind", 3); // Function + char detail[256]; + sprintf(detail, "fn %s", f->name); + cJSON_AddStringToObject(item, "detail", detail); + cJSON_AddItemToArray(items, item); + f = f->next; + } + + StructDef *s = g_project->ctx->struct_defs; + while (s) + { + cJSON *item = cJSON_CreateObject(); + cJSON_AddStringToObject(item, "label", s->name); + cJSON_AddNumberToObject(item, "kind", 22); // Struct + char detail[256]; + sprintf(detail, "struct %s", s->name); + cJSON_AddStringToObject(item, "detail", detail); + cJSON_AddItemToArray(items, item); + s = s->next; + } + } + + cJSON_AddItemToObject(root, "result", items); + send_json_response(root); +} + +void lsp_document_symbol(const char *uri) +{ + ProjectFile *pf = lsp_project_get_file(uri); + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); + + if (!pf || !pf->index) + { + cJSON_AddNullToObject(root, "result"); + send_json_response(root); + return; + } + + cJSON *items = cJSON_CreateArray(); + LSPRange *r = pf->index->head; + while (r) + { + if (r->type == RANGE_DEFINITION && r->node) + { + char *name = NULL; + int kind = 0; + + if (r->node->type == NODE_FUNCTION) + { + name = r->node->func.name; + kind = 12; + } + else if (r->node->type == NODE_STRUCT) + { + name = r->node->strct.name; + kind = 23; + } + else if (r->node->type == NODE_VAR_DECL) + { + name = r->node->var_decl.name; + kind = 13; + } + else if (r->node->type == NODE_CONST) + { + name = r->node->var_decl.name; + kind = 14; + } + + if (name) + { + cJSON *item = cJSON_CreateObject(); + cJSON_AddStringToObject(item, "name", name); + cJSON_AddNumberToObject(item, "kind", kind); + + cJSON *loc = cJSON_CreateObject(); + cJSON_AddStringToObject(loc, "uri", uri); + + cJSON *range = cJSON_CreateObject(); + cJSON *start = cJSON_CreateObject(); + cJSON_AddNumberToObject(start, "line", r->start_line); + cJSON_AddNumberToObject(start, "character", r->start_col); + + cJSON *end = cJSON_CreateObject(); + cJSON_AddNumberToObject(end, "line", r->end_line); + cJSON_AddNumberToObject(end, "character", r->end_col); + + cJSON_AddItemToObject(range, "start", start); + cJSON_AddItemToObject(range, "end", end); + cJSON_AddItemToObject(loc, "range", range); + + cJSON_AddItemToObject(item, "location", loc); + cJSON_AddItemToArray(items, item); + } + } + r = r->next; + } + + cJSON_AddItemToObject(root, "result", items); + send_json_response(root); +} + +void lsp_references(const char *uri, int line, int col) +{ + ProjectFile *pf = lsp_project_get_file(uri); + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); + cJSON *items = cJSON_CreateArray(); + + if (pf && pf->index) + { + LSPRange *r = lsp_find_at(pf->index, line, col); + if (r && r->node) + { + char *name = NULL; + if (r->node->type == NODE_FUNCTION) + { + name = r->node->func.name; + } + else if (r->node->type == NODE_VAR_DECL) + { + name = r->node->var_decl.name; + } + else if (r->node->type == NODE_CONST) + { + name = r->node->var_decl.name; + } + else if (r->node->type == NODE_STRUCT) + { + name = r->node->strct.name; + } + else if (r->node->type == NODE_EXPR_VAR) + { + name = r->node->var_ref.name; + } + else if (r->node->type == NODE_EXPR_CALL && r->node->call.callee->type == NODE_EXPR_VAR) + { + name = r->node->call.callee->var_ref.name; + } + + if (name) + { + ReferenceResult *refs = lsp_project_find_references(name); + ReferenceResult *curr = refs; + while (curr) + { + cJSON *item = cJSON_CreateObject(); + cJSON_AddStringToObject(item, "uri", curr->uri); + cJSON *range = cJSON_CreateObject(); + cJSON *start = cJSON_CreateObject(); + cJSON_AddNumberToObject(start, "line", curr->range->start_line); + cJSON_AddNumberToObject(start, "character", curr->range->start_col); + + cJSON *end = cJSON_CreateObject(); + cJSON_AddNumberToObject(end, "line", curr->range->end_line); + cJSON_AddNumberToObject(end, "character", curr->range->end_col); + + cJSON_AddItemToObject(range, "start", start); + cJSON_AddItemToObject(range, "end", end); + cJSON_AddItemToObject(item, "range", range); + cJSON_AddItemToArray(items, item); + + ReferenceResult *next = curr->next; + free(curr); + curr = next; + } + } + } + } + + cJSON_AddItemToObject(root, "result", items); + send_json_response(root); +} + +void lsp_signature_help(const char *uri, int line, int col) +{ + ProjectFile *pf = lsp_project_get_file(uri); + cJSON *root = cJSON_CreateObject(); + cJSON_AddStringToObject(root, "jsonrpc", "2.0"); + cJSON_AddNumberToObject(root, "id", 1); - int first = 1; + if (!g_project || !g_project->ctx || !pf || !pf->source) + { + cJSON_AddNullToObject(root, "result"); + send_json_response(root); + return; + } - // Functions - FuncSig *f = g_ctx->func_registry; - while (f) + // ... [Scan backwards logic same as before] ... + char *ptr = pf->source; + int cur_line = 0; + while (*ptr && cur_line < line) { - if (!first) + if (*ptr == '\n') { - p += sprintf(p, ","); + cur_line++; } - p += sprintf(p, "{\"label\":\"%s\",\"kind\":3,\"detail\":\"fn %s(...)\"}", f->name, - f->name); // Kind 3 = Function - first = 0; - f = f->next; + ptr++; + } + if (ptr && col > 0) + { + ptr += col; } - // Structs - StructDef *s = g_ctx->struct_defs; - while (s) + if (ptr > pf->source + strlen(pf->source)) { - if (!first) + cJSON_AddNullToObject(root, "result"); + send_json_response(root); + return; + } + + int found = 0; + char *p = ptr - 1; + while (p >= pf->source) + { + if (*p == ')') { - p += sprintf(p, ","); + break; } - p += sprintf(p, "{\"label\":\"%s\",\"kind\":22,\"detail\":\"struct %s\"}", s->name, - s->name); // Kind 22 = Struct - first = 0; - s = s->next; + if (*p == '(') + { + // Found open paren + char *ident_end = p - 1; + while (ident_end >= pf->source && isspace(*ident_end)) + { + ident_end--; + } + if (ident_end < pf->source) + { + break; + } + char *ident_start = ident_end; + while (ident_start >= pf->source && (isalnum(*ident_start) || *ident_start == '_')) + { + ident_start--; + } + ident_start++; + + int len = ident_end - ident_start + 1; + if (len > 0 && len < 255) + { + char func_name[256]; + strncpy(func_name, ident_start, len); + func_name[len] = 0; + // Lookup + FuncSig *fn = g_project->ctx->func_registry; + while (fn) + { + if (strcmp(fn->name, func_name) == 0) + { + // Found it + cJSON *result = cJSON_CreateObject(); + cJSON *sigs = cJSON_CreateArray(); + cJSON *sig = cJSON_CreateObject(); + + char label[2048]; + char params[1024] = ""; + int first = 1; + for (int i = 0; i < fn->total_args; i++) + { + if (!first) + { + strcat(params, ", "); + } + char *tstr = type_to_string(fn->arg_types[i]); + if (tstr) + { + strcat(params, tstr); + free(tstr); + } + else + { + strcat(params, "unknown"); + } + first = 0; + } + char *ret_str = type_to_string(fn->ret_type); + sprintf(label, "fn %s(%s) -> %s", fn->name, params, + ret_str ? ret_str : "void"); + if (ret_str) + { + free(ret_str); + } + + cJSON_AddStringToObject(sig, "label", label); + cJSON_AddItemToObject(sig, "parameters", cJSON_CreateArray()); + cJSON_AddItemToArray(sigs, sig); + + cJSON_AddItemToObject(result, "signatures", sigs); + cJSON_AddItemToObject(root, "result", result); + found = 1; + break; + } + fn = fn->next; + } + } + break; + } + p--; } - p += sprintf(p, "]}"); + if (!found) + { + cJSON_AddNullToObject(root, "result"); + } - fprintf(stdout, "Content-Length: %ld\r\n\r\n%s", strlen(json), json); - fflush(stdout); - free(json); + send_json_response(root); } diff --git a/src/lsp/lsp_index.c b/src/lsp/lsp_index.c index d952b77..285dec3 100644 --- a/src/lsp/lsp_index.c +++ b/src/lsp/lsp_index.c @@ -148,6 +148,12 @@ void lsp_walk_node(LSPIndex *idx, ASTNode *node) lsp_walk_node(idx, node->var_decl.init_expr); } + else if (node->type == NODE_STRUCT) + { + char hover[256]; + sprintf(hover, "struct %s", node->strct.name); + lsp_index_add_def(idx, node->token, hover, node); + } // Reference logic. if (node->definition_token.line > 0 && node->definition_token.line != node->token.line) diff --git a/src/lsp/lsp_project.c b/src/lsp/lsp_project.c new file mode 100644 index 0000000..71e7d3c --- /dev/null +++ b/src/lsp/lsp_project.c @@ -0,0 +1,314 @@ +#include "lsp_project.h" +#include <dirent.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/stat.h> + +LSPProject *g_project = NULL; + +static void scan_dir(const char *dir_path); + +void lsp_project_init(const char *root_path) +{ + if (g_project) + { + return; + } + + fprintf(stderr, "zls: Initializing project at %s\n", root_path); + + g_project = xcalloc(1, sizeof(LSPProject)); + g_project->root_path = xstrdup(root_path); + + // Create a persistent global context + g_project->ctx = xcalloc(1, sizeof(ParserContext)); + g_project->ctx->is_fault_tolerant = 1; + + // Set a default error handler that just logs to stderr (or ignores) + // to prevent exit(1) during initial scan. + void lsp_default_on_error(void *data, Token t, const char *msg); + g_project->ctx->on_error = lsp_default_on_error; + + // Scan workspace + scan_dir(root_path); +} + +// Default error handler for indexing phase +void lsp_default_on_error(void *data, Token t, const char *msg) +{ + (void)data; + // We can log it if we want, but standard zpanic_at already printed it to stderr. + // The important thing is that we exist so zpanic_at returns. + // Maybe we suppress duplicates or just let it pass. + // Since zpanic_at printed "error: ...", we don't need to print again. +} + +static void scan_file(const char *path) +{ + // Skip if not .zc + const char *ext = strrchr(path, '.'); + if (!ext || strcmp(ext, ".zc") != 0) + { + return; + } + + fprintf(stderr, "zls: Indexing %s\n", path); + + char *src = load_file(path); + if (!src) + { + return; + } + + char uri[2048]; + sprintf(uri, "file://%s", path); + + lsp_project_update_file(uri, src); + free(src); +} + +static void scan_dir(const char *dir_path) +{ + DIR *d = opendir(dir_path); + if (!d) + { + return; + } + + struct dirent *dir; + while ((dir = readdir(d)) != NULL) + { + if (dir->d_name[0] == '.') + { + continue; + } + + char path[1024]; + snprintf(path, sizeof(path), "%s/%s", dir_path, dir->d_name); + + struct stat st; + if (stat(path, &st) == 0) + { + if (S_ISDIR(st.st_mode)) + { + scan_dir(path); + } + else if (S_ISREG(st.st_mode)) + { + scan_file(path); + } + } + } + closedir(d); +} + +ProjectFile *lsp_project_get_file(const char *uri) +{ + if (!g_project) + { + return NULL; + } + ProjectFile *curr = g_project->files; + while (curr) + { + if (strcmp(curr->uri, uri) == 0) + { + return curr; + } + curr = curr->next; + } + return NULL; +} + +static ProjectFile *add_project_file(const char *uri) +{ + ProjectFile *f = xcalloc(1, sizeof(ProjectFile)); + f->uri = xstrdup(uri); + // Simple path extraction from URI (file://...) + if (strncmp(uri, "file://", 7) == 0) + { + f->path = xstrdup(uri + 7); + } + else + { + f->path = xstrdup(uri); + } + + f->next = g_project->files; + g_project->files = f; + return f; +} + +void lsp_project_update_file(const char *uri, const char *src) +{ + if (!g_project) + { + return; + } + + ProjectFile *pf = lsp_project_get_file(uri); + if (!pf) + { + pf = add_project_file(uri); + } + + // Clear old index + if (pf->index) + { + lsp_index_free(pf->index); + pf->index = NULL; + } + + // Update source + if (pf->source) + { + free(pf->source); + } + pf->source = xstrdup(src); + + // Parse + Lexer l; + lexer_init(&l, src); + + ASTNode *root = parse_program(g_project->ctx, &l); + + // Build Index + pf->index = lsp_index_new(); + if (root) + { + lsp_build_index(pf->index, root); + } +} + +DefinitionResult lsp_project_find_definition(const char *name) +{ + // ... existing implementation ... + DefinitionResult res = {0}; + if (!g_project) + { + return res; + } + + ProjectFile *pf = g_project->files; + while (pf) + { + if (pf->index) + { + LSPRange *r = pf->index->head; + while (r) + { + if (r->type == RANGE_DEFINITION && r->node) + { + // Check name match + char *found_name = NULL; + if (r->node->type == NODE_FUNCTION) + { + found_name = r->node->func.name; + } + else if (r->node->type == NODE_VAR_DECL) + { + found_name = r->node->var_decl.name; + } + else if (r->node->type == NODE_CONST) + { + found_name = r->node->var_decl.name; + } + else if (r->node->type == NODE_STRUCT) + { + found_name = r->node->strct.name; + } + + if (found_name && strcmp(found_name, name) == 0) + { + res.uri = pf->uri; + res.range = r; + return res; + } + } + r = r->next; + } + } + pf = pf->next; + } + + return res; +} + +// Find all references to a symbol name project-wide + +ReferenceResult *lsp_project_find_references(const char *name) +{ + if (!g_project) + { + return NULL; + } + ReferenceResult *head = NULL; + ReferenceResult *tail = NULL; + + ProjectFile *pf = g_project->files; + while (pf) + { + if (pf->index) + { + LSPRange *r = pf->index->head; + while (r) + { + // We want REFERENCES that match the name + // Or DEFINITIONS that match the name (include decl) + char *scan_name = NULL; + + if (r->node) + { + if (r->node->type == NODE_FUNCTION) + { + scan_name = r->node->func.name; + } + else if (r->node->type == NODE_VAR_DECL) + { + scan_name = r->node->var_decl.name; + } + else if (r->node->type == NODE_CONST) + { + scan_name = r->node->var_decl.name; + } + else if (r->node->type == NODE_STRUCT) + { + scan_name = r->node->strct.name; + } + else if (r->node->type == NODE_EXPR_VAR) + { + scan_name = r->node->var_ref.name; + } + else if (r->node->type == NODE_EXPR_CALL && + r->node->call.callee->type == NODE_EXPR_VAR) + { + scan_name = r->node->call.callee->var_ref.name; + } + } + + if (scan_name && strcmp(scan_name, name) == 0) + { + ReferenceResult *new_res = calloc(1, sizeof(ReferenceResult)); + new_res->uri = pf->uri; + new_res->range = r; + + if (!head) + { + head = new_res; + tail = new_res; + } + else + { + tail->next = new_res; + tail = new_res; + } + } + + r = r->next; + } + } + pf = pf->next; + } + return head; +} diff --git a/src/lsp/lsp_project.h b/src/lsp/lsp_project.h new file mode 100644 index 0000000..9b4fe42 --- /dev/null +++ b/src/lsp/lsp_project.h @@ -0,0 +1,59 @@ +#ifndef LSP_PROJECT_H +#define LSP_PROJECT_H + +#include "parser.h" +#include "lsp_index.h" + +typedef struct ProjectFile +{ + char *path; // Absolute path + char *uri; // file:// URI + char *source; // Cached source content + LSPIndex *index; // File-specific index (local vars, refs) + struct ProjectFile *next; +} ProjectFile; + +typedef struct +{ + // Global symbol table (Structs, Functions, Globals) + // We reuse ParserContext for this, as it already supports registries. + ParserContext *ctx; + + // List of tracked files + ProjectFile *files; + + // Root directory + char *root_path; +} LSPProject; + +// Global project instance +extern LSPProject *g_project; + +// Initialize the project with a root directory +void lsp_project_init(const char *root_path); + +// Find a file in the project +ProjectFile *lsp_project_get_file(const char *uri); + +// Update a file (re-parse and re-index) +void lsp_project_update_file(const char *uri, const char *src); + +// Find definition globally +typedef struct +{ + char *uri; + LSPRange *range; +} DefinitionResult; + +DefinitionResult lsp_project_find_definition(const char *name); + +typedef struct ReferenceResult +{ + char *uri; + LSPRange *range; + struct ReferenceResult *next; +} ReferenceResult; + +ReferenceResult *lsp_project_find_references(const char *name); + +#endif diff --git a/src/parser/parser.h b/src/parser/parser.h index b3837eb..5757fbc 100644 --- a/src/parser/parser.h +++ b/src/parser/parser.h @@ -42,6 +42,7 @@ typedef struct Symbol Token decl_token; int is_const_value; int const_int_val; + int is_moved; struct Symbol *next; } Symbol; @@ -427,6 +428,10 @@ ASTNode *parse_trait(ParserContext *ctx, Lexer *l); ASTNode *parse_impl(ParserContext *ctx, Lexer *l); ASTNode *parse_impl_trait(ParserContext *ctx, Lexer *l); ASTNode *parse_test(ParserContext *ctx, Lexer *l); + +// Move semantics helpers +int is_type_copy(ParserContext *ctx, Type *t); +void check_move_usage(ParserContext *ctx, ASTNode *node, Token t); ASTNode *parse_include(ParserContext *ctx, Lexer *l); ASTNode *parse_import(ParserContext *ctx, Lexer *l); ASTNode *parse_comptime(ParserContext *ctx, Lexer *l); diff --git a/src/parser/parser_core.c b/src/parser/parser_core.c index c3c91fe..464c905 100644 --- a/src/parser/parser_core.c +++ b/src/parser/parser_core.c @@ -585,17 +585,19 @@ static ASTNode *generate_derive_impls(ParserContext *ctx, ASTNode *strct, char * ASTNode *fdef = find_struct_def(ctx, ft); if (fdef && fdef->type == NODE_ENUM) { - // Enum field: compare tags + // Enum field: compare tags (pointer access via auto-deref) sprintf(cmp, "self.%s.tag == other.%s.tag", fn, fn); } else if (fdef && fdef->type == NODE_STRUCT) { - // Struct field: use _eq function - sprintf(cmp, "%s__eq(&self.%s, other.%s)", ft, fn, fn); + // Struct field: use _eq function, pass addresses + // self.field is L-value, other.field is L-value (auto-deref from + // pointer) We need addresses of them: &self.field, &other.field + sprintf(cmp, "%s__eq(&self.%s, &other.%s)", ft, fn, fn); } else { - // Primitive or unknown: use == + // Primitive or unknown: use == (auto-deref) sprintf(cmp, "self.%s == other.%s", fn, fn); } strcat(body, cmp); @@ -610,7 +612,8 @@ static ASTNode *generate_derive_impls(ParserContext *ctx, ASTNode *strct, char * strcat(body, ";"); } code = xmalloc(4096 + 1024); - sprintf(code, "impl %s { fn eq(self, other: %s) -> bool { %s } }", name, name, body); + // Updated signature: other is a pointer T* + sprintf(code, "impl %s { fn eq(self, other: %s*) -> bool { %s } }", name, name, body); } else if (0 == strcmp(trait, "Debug")) { @@ -619,6 +622,12 @@ static ASTNode *generate_derive_impls(ParserContext *ctx, ASTNode *strct, char * sprintf(code, "impl %s { fn to_string(self) -> char* { return \"%s { ... }\"; } }", name, name); } + else if (0 == strcmp(trait, "Copy")) + { + // Marker trait for Copy/Move semantics + code = xmalloc(1024); + sprintf(code, "impl Copy for %s {}", name); + } if (code) { diff --git a/src/parser/parser_expr.c b/src/parser/parser_expr.c index f7590e3..092b86b 100644 --- a/src/parser/parser_expr.c +++ b/src/parser/parser_expr.c @@ -9,6 +9,69 @@ Type *get_field_type(ParserContext *ctx, Type *struct_type, const char *field_name); +int is_type_copy(ParserContext *ctx, Type *t) +{ + if (!t) + { + return 1; // Default to Copy for unknown types to avoid annoyance + } + + switch (t->kind) + { + case TYPE_INT: + case TYPE_I8: + case TYPE_I16: + case TYPE_I32: + case TYPE_I64: + case TYPE_U8: + case TYPE_U16: + case TYPE_U32: + case TYPE_U64: + case TYPE_F32: + case TYPE_F64: + case TYPE_BOOL: + case TYPE_CHAR: + case TYPE_VOID: + case TYPE_POINTER: // Pointers are Copy + case TYPE_FUNCTION: + case TYPE_ENUM: // Enums are integers + return 1; + + case TYPE_STRUCT: + // Structs are MOVE by default unless they implement Copy + if (check_impl(ctx, "Copy", t->name)) + { + return 1; + } + return 0; + + case TYPE_ARRAY: + // Arrays decay or are fixed size context dependent, but usually not simplistic copy + // For Zen-C safety, let's treat them as Copy if they are treated as pointers, + // but if it's a value assignment, C doesn't support it anyway unless wrapped in struct. + return 0; + + default: + return 1; + } +} + +void check_move_usage(ParserContext *ctx, ASTNode *node, Token t) +{ + if (!node) + { + return; + } + if (node->type == NODE_EXPR_VAR) + { + Symbol *sym = find_symbol_entry(ctx, node->var_ref.name); + if (sym && sym->is_moved) + { + zpanic_at(t, "Use of moved value '%s'", node->var_ref.name); + } + } +} + static int type_is_unsigned(Type *t) { if (!t) @@ -636,9 +699,6 @@ void analyze_lambda_captures(ParserContext *ctx, ASTNode *lambda) for (int i = 0; i < num_refs; i++) { free(all_refs[i]); - } - if (all_refs) - { free(all_refs); } } @@ -1758,6 +1818,31 @@ ASTNode *parse_primary(ParserContext *ctx, Lexer *l) } ASTNode *arg = parse_expression(ctx, l); + + // Move Semantics Logic (Added for known funcs) + check_move_usage(ctx, arg, arg ? arg->token : t1); + if (arg && arg->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, arg->var_ref.name); + if (!t) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + t = s->type_info; + } + } + + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + if (!head) { head = arg; @@ -1864,6 +1949,31 @@ ASTNode *parse_primary(ParserContext *ctx, Lexer *l) } ASTNode *arg = parse_expression(ctx, l); + + // Move Semantics Logic (Added) + check_move_usage(ctx, arg, arg ? arg->token : t1); + if (arg && arg->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, arg->var_ref.name); + if (!t) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + t = s->type_info; + } + } + + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + if (!head) { head = arg; @@ -2125,6 +2235,31 @@ ASTNode *parse_primary(ParserContext *ctx, Lexer *l) } ASTNode *arg = parse_expression(ctx, l); + + // Move Semantics Logic + check_move_usage(ctx, arg, arg ? arg->token : t1); + if (arg && arg->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, arg->var_ref.name); + if (!t) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + t = s->type_info; + } + } + + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + if (!head) { head = arg; @@ -3173,7 +3308,122 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) if (op.type == TOK_LPAREN) { ASTNode *call = ast_create(NODE_EXPR_CALL); - call->call.callee = lhs; + + // Method Resolution Logic (Struct Method -> Trait Method) + ASTNode *self_arg = NULL; + FuncSig *resolved_sig = NULL; + char *resolved_name = NULL; + + if (lhs->type == NODE_EXPR_MEMBER) + { + Type *lt = lhs->member.target->type_info; + int is_lhs_ptr = 0; + char *alloc_name = NULL; + char *struct_name = + resolve_struct_name_from_type(ctx, lt, &is_lhs_ptr, &alloc_name); + + if (struct_name) + { + char mangled[256]; + sprintf(mangled, "%s__%s", struct_name, lhs->member.field); + FuncSig *sig = find_func(ctx, mangled); + + if (!sig) + { + // Trait method lookup: Struct__Trait_Method + StructRef *ref = ctx->parsed_impls_list; + while (ref) + { + if (ref->node && ref->node->type == NODE_IMPL_TRAIT) + { + if (ref->node->impl_trait.target_type && + strcmp(ref->node->impl_trait.target_type, struct_name) == 0) + { + char trait_mangled[512]; + snprintf(trait_mangled, 512, "%s__%s_%s", struct_name, + ref->node->impl_trait.trait_name, lhs->member.field); + if (find_func(ctx, trait_mangled)) + { + sig = find_func(ctx, trait_mangled); + strcpy(mangled, trait_mangled); + break; + } + } + } + ref = ref->next; + } + } + + if (sig) + { + resolved_name = xstrdup(mangled); + resolved_sig = sig; + + // Create 'self' argument + ASTNode *obj = lhs->member.target; + + // Handle Reference/Pointer adjustment based on signature + if (sig->total_args > 0 && sig->arg_types[0] && + sig->arg_types[0]->kind == TYPE_POINTER) + { + if (!is_lhs_ptr) + { + // Function expects ptr, have value -> &obj + int is_rvalue = + (obj->type == NODE_EXPR_CALL || obj->type == NODE_EXPR_BINARY || + obj->type == NODE_EXPR_STRUCT_INIT || + obj->type == NODE_EXPR_CAST || obj->type == NODE_MATCH); + + ASTNode *addr = ast_create(NODE_EXPR_UNARY); + addr->unary.op = is_rvalue ? xstrdup("&_rval") : xstrdup("&"); + addr->unary.operand = obj; + addr->type_info = type_new_ptr(lt); + self_arg = addr; + } + else + { + self_arg = obj; + } + } + else + { + // Function expects value + if (is_lhs_ptr) + { + // Have ptr, need value -> *obj + ASTNode *deref = ast_create(NODE_EXPR_UNARY); + deref->unary.op = xstrdup("*"); + deref->unary.operand = obj; + if (lt && lt->kind == TYPE_POINTER && lt->inner) + { + deref->type_info = lt->inner; + } + self_arg = deref; + } + else + { + self_arg = obj; + } + } + } + } + if (alloc_name) + { + free(alloc_name); + } + } + + if (resolved_name) + { + ASTNode *callee = ast_create(NODE_EXPR_VAR); + callee->var_ref.name = resolved_name; + call->call.callee = callee; + } + else + { + call->call.callee = lhs; + } + ASTNode *head = NULL, *tail = NULL; char **arg_names = NULL; int arg_count = 0; @@ -3201,6 +3451,31 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) } ASTNode *arg = parse_expression(ctx, l); + + // Move Semantics Logic + check_move_usage(ctx, arg, arg ? arg->token : t1); + if (arg && arg->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, arg->var_ref.name); + if (!t) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + t = s->type_info; + } + } + + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, arg->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + if (!head) { head = arg; @@ -3230,12 +3505,44 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) { zpanic_at(lexer_peek(l), "Expected )"); } + + // Prepend 'self' argument if resolved + if (self_arg) + { + self_arg->next = head; + head = self_arg; + arg_count++; + + if (has_named) + { + // Prepend NULL to arg_names for self + char **new_names = xmalloc(sizeof(char *) * arg_count); + new_names[0] = NULL; + for (int i = 0; i < arg_count - 1; i++) + { + new_names[i + 1] = arg_names[i]; + } + free(arg_names); + arg_names = new_names; + } + } + call->call.args = head; call->call.arg_names = has_named ? arg_names : NULL; call->call.arg_count = arg_count; call->resolved_type = xstrdup("unknown"); - if (lhs->type_info && lhs->type_info->kind == TYPE_FUNCTION && lhs->type_info->inner) + + if (resolved_sig) + { + call->type_info = resolved_sig->ret_type; + if (call->type_info) + { + call->resolved_type = type_to_string(call->type_info); + } + } + else if (lhs->type_info && lhs->type_info->kind == TYPE_FUNCTION && + lhs->type_info->inner) { call->type_info = lhs->type_info->inner; } @@ -3535,6 +3842,33 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) sprintf(mangled, "%s__%s", struct_name, node->member.field); FuncSig *sig = find_func(ctx, mangled); + + if (!sig) + { + // Try resolving as a trait method: Struct__Trait__Method + StructRef *ref = ctx->parsed_impls_list; + while (ref) + { + if (ref->node && ref->node->type == NODE_IMPL_TRAIT) + { + const char *t_struct = ref->node->impl_trait.target_type; + if (t_struct && strcmp(t_struct, struct_name) == 0) + { + char trait_mangled[512]; + snprintf(trait_mangled, 512, "%s__%s_%s", struct_name, + ref->node->impl_trait.trait_name, node->member.field); + if (find_func(ctx, trait_mangled)) + { + strcpy(mangled, trait_mangled); // Update mangled name + sig = find_func(ctx, mangled); + break; + } + } + } + ref = ref->next; + } + } + if (sig) { // It is a method! Create a Function Type Info to carry the return @@ -3577,6 +3911,52 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) bin->binary.left = lhs; bin->binary.right = rhs; + // Move Semantics Logic + if (op.type == TOK_OP && is_token(op, "=")) // Assignment "=" + { + // 1. RHS is being read: Check validity + check_move_usage(ctx, rhs, op); + + // 2. Mark RHS as moved (Transfer ownership) if it's a Move type + if (rhs->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, rhs->var_ref.name); + // If type info not on var, try looking up symbol + if (!t) + { + Symbol *s = find_symbol_entry(ctx, rhs->var_ref.name); + if (s) + { + t = s->type_info; + } + } + + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, rhs->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + + // 3. LHS is being written: Resurrect (it is now valid) + if (lhs->type == NODE_EXPR_VAR) + { + Symbol *s = find_symbol_entry(ctx, lhs->var_ref.name); + if (s) + { + s->is_moved = 0; + } + } + } + else // All other binary ops (including +=, -=, etc. which read LHS first) + { + check_move_usage(ctx, lhs, op); + check_move_usage(ctx, rhs, op); + } + if (op.type == TOK_LANGLE) { bin->binary.op = xstrdup("<"); @@ -3864,6 +4244,32 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) FuncSig *sig = find_func(ctx, mangled); + if (!sig) + { + // Try resolving as a trait method: Struct__Trait__Method + StructRef *ref = ctx->parsed_impls_list; + while (ref) + { + if (ref->node && ref->node->type == NODE_IMPL_TRAIT) + { + const char *t_struct = ref->node->impl_trait.target_type; + if (t_struct && strcmp(t_struct, struct_name) == 0) + { + char trait_mangled[512]; + snprintf(trait_mangled, 512, "%s__%s_%s", struct_name, + ref->node->impl_trait.trait_name, method); + if (find_func(ctx, trait_mangled)) + { + strcpy(mangled, trait_mangled); // Update mangled name + sig = find_func(ctx, mangled); + break; + } + } + } + ref = ref->next; + } + } + if (sig) { ASTNode *call = ast_create(NODE_EXPR_CALL); @@ -3909,9 +4315,34 @@ ASTNode *parse_expr_prec(ParserContext *ctx, Lexer *l, Precedence min_prec) } } + // Handle RHS (Argument 2) Auto-Ref if needed + ASTNode *arg2 = rhs; + if (sig->total_args > 1 && sig->arg_types[1] && + sig->arg_types[1]->kind == TYPE_POINTER) + { + Type *rt = rhs->type_info; + int is_rhs_ptr = (rt && rt->kind == TYPE_POINTER); + if (!is_rhs_ptr) // Need pointer, have value + { + int is_rvalue = + (rhs->type == NODE_EXPR_CALL || rhs->type == NODE_EXPR_BINARY || + rhs->type == NODE_EXPR_STRUCT_INIT || + rhs->type == NODE_EXPR_CAST || rhs->type == NODE_MATCH); + + ASTNode *addr = ast_create(NODE_EXPR_UNARY); + addr->unary.op = is_rvalue ? xstrdup("&_rval") : xstrdup("&"); + addr->unary.operand = rhs; + if (rt) + { + addr->type_info = type_new_ptr(rt); + } + arg2 = addr; + } + } + call->call.args = arg1; - arg1->next = rhs; - rhs->next = NULL; + arg1->next = arg2; + arg2->next = NULL; call->type_info = sig->ret_type; call->resolved_type = type_to_string(sig->ret_type); diff --git a/src/parser/parser_stmt.c b/src/parser/parser_stmt.c index db47d16..7873d51 100644 --- a/src/parser/parser_stmt.c +++ b/src/parser/parser_stmt.c @@ -217,11 +217,11 @@ ASTNode *parse_match(ParserContext *ctx, Lexer *l) break; } - // --- 1. Parse Patterns (with OR and range support) --- + // Parse Patterns (with OR and range support) // Patterns can be: // - Single value: 1 - // - OR patterns: 1 || 2 or 1 or 2 - // - Range patterns: 1..5 or 1..=5 + // - OR patterns: 1 || 2 or 1 or 2 or 1, 2 + // - Range patterns: 1..5 or 1..=5 or 1..<5 char patterns_buf[1024]; patterns_buf[0] = 0; int pattern_count = 0; @@ -292,11 +292,22 @@ ASTNode *parse_match(ParserContext *ctx, Lexer *l) char *binding = NULL; int is_destructure = 0; - // --- 2. Handle Destructuring: Ok(v) --- + int is_ref = 0; + + // Handle Destructuring: Ok(v) // (Only allowed if we matched a single pattern, e.g. "Result::Ok(val)") if (!is_default && pattern_count == 1 && lexer_peek(l).type == TOK_LPAREN) { lexer_next(l); // eat ( + + // Check for 'ref' keyword + if (lexer_peek(l).type == TOK_IDENT && lexer_peek(l).len == 3 && + strncmp(lexer_peek(l).start, "ref", 3) == 0) + { + lexer_next(l); // eat 'ref' + is_ref = 1; + } + Token b = lexer_next(l); if (b.type != TOK_IDENT) { @@ -324,6 +335,14 @@ ASTNode *parse_match(ParserContext *ctx, Lexer *l) zpanic_at(lexer_peek(l), "Expected =>"); } + // Create scope for the case to hold the binding + enter_scope(ctx); + if (binding) + { + // If ref binding, mark as pointer type so auto-deref (->) works + add_symbol(ctx, binding, is_ref ? "void*" : "unknown", NULL); + } + ASTNode *body; Token pk = lexer_peek(l); if (pk.type == TOK_LBRACE) @@ -344,10 +363,13 @@ ASTNode *parse_match(ParserContext *ctx, Lexer *l) body = parse_expression(ctx, l); } + exit_scope(ctx); + ASTNode *c = ast_create(NODE_MATCH_CASE); c->match_case.pattern = pattern; c->match_case.binding_name = binding; c->match_case.is_destructuring = is_destructure; + c->match_case.is_ref = is_ref; // Store is_ref flag c->match_case.guard = guard; c->match_case.body = body; c->match_case.is_default = is_default; @@ -1235,6 +1257,29 @@ ASTNode *parse_var_decl(ParserContext *ctx, Lexer *l) n->var_decl.init_expr = init; + // Move Semantics Logic for Initialization + check_move_usage(ctx, init, init ? init->token : name_tok); + if (init && init->type == NODE_EXPR_VAR) + { + Type *t = find_symbol_type_info(ctx, init->var_ref.name); + if (!t) + { + Symbol *s = find_symbol_entry(ctx, init->var_ref.name); + if (s) + { + t = s->type_info; + } + } + if (!is_type_copy(ctx, t)) + { + Symbol *s = find_symbol_entry(ctx, init->var_ref.name); + if (s) + { + s->is_moved = 1; + } + } + } + // Global detection: Either no scope (yet) OR root scope (no parent) if (!ctx->current_scope || !ctx->current_scope->parent) { @@ -1483,6 +1528,11 @@ ASTNode *parse_return(ParserContext *ctx, Lexer *l) else { n->ret.value = parse_expression(ctx, l); + check_move_usage(ctx, n->ret.value, n->ret.value ? n->ret.value->token : lexer_peek(l)); + + // Note: Returning a non-Copy variable effectively moves it out. + // We could mark it as moved, but scope ends anyway. + // The critical part is checking we aren't returning an ALREADY moved value. } } @@ -1668,8 +1718,8 @@ ASTNode *parse_for(ParserContext *ctx, Lexer *l) char *iter_method = "iterator"; // Check for reference iteration: for x in &vec - if (obj_expr->type == NODE_EXPR_UNARY && - obj_expr->unary.op && strcmp(obj_expr->unary.op, "&") == 0) + if (obj_expr->type == NODE_EXPR_UNARY && obj_expr->unary.op && + strcmp(obj_expr->unary.op, "&") == 0) { obj_expr = obj_expr->unary.operand; iter_method = "iter_ref"; @@ -3348,6 +3398,12 @@ ASTNode *parse_impl(ParserContext *ctx, Lexer *l) char *na = patch_self_args(f->func.args, name2); free(f->func.args); f->func.args = na; + + // Register function for lookup + register_func(ctx, mangled, f->func.arg_count, f->func.defaults, f->func.arg_types, + f->func.ret_type_info, f->func.is_varargs, f->func.is_async, + f->token); + if (!h) { h = f; @@ -3374,6 +3430,12 @@ ASTNode *parse_impl(ParserContext *ctx, Lexer *l) char *na = patch_self_args(f->func.args, name2); free(f->func.args); f->func.args = na; + + // Register function for lookup + register_func(ctx, mangled, f->func.arg_count, f->func.defaults, + f->func.arg_types, f->func.ret_type_info, f->func.is_varargs, + f->func.is_async, f->token); + if (!h) { h = f; @@ -3403,14 +3465,16 @@ ASTNode *parse_impl(ParserContext *ctx, Lexer *l) // If target struct is generic, register this impl as a template ASTNode *def = find_struct_def(ctx, name2); - if (def && ((def->type == NODE_STRUCT && def->strct.is_template) || + if (def && ((def->type == NODE_STRUCT && def->strct.is_template) || (def->type == NODE_ENUM && def->enm.is_template))) { - const char *gp = "T"; - if (def->type == NODE_STRUCT && def->strct.generic_param_count > 0) - gp = def->strct.generic_params[0]; - // TODO: Enum generic params support if needed - register_impl_template(ctx, name2, gp, n); + const char *gp = "T"; + if (def->type == NODE_STRUCT && def->strct.generic_param_count > 0) + { + gp = def->strct.generic_params[0]; + } + // TODO: Enum generic params support if needed + register_impl_template(ctx, name2, gp, n); } return n; @@ -4217,30 +4281,37 @@ ASTNode *parse_import(ParserContext *ctx, Lexer *l) strncpy(fn, t.start + 1, ln); fn[ln] = 0; - // Resolve relative paths (if starts with ./ or ../ + // Resolve paths relative to current file char resolved_path[1024]; - if (fn[0] == '.' && (fn[1] == '/' || (fn[1] == '.' && fn[2] == '/'))) + int is_explicit_relative = (fn[0] == '.' && (fn[1] == '/' || (fn[1] == '.' && fn[2] == '/'))); + + // Try to resolve relative to current file if not absolute + if (fn[0] != '/') { - // Relative import - resolve relative to current file char *current_dir = xstrdup(g_current_filename); char *last_slash = strrchr(current_dir, '/'); if (last_slash) { *last_slash = 0; // Truncate to directory + + // Handle explicit relative differently? + // Existing logic enforced it. Let's try to verify existence first. + + // Construct candidate path const char *leaf = fn; - if (leaf[0] == '.' && leaf[1] == '/') + // Clean up ./ prefix for cleaner path construction if we want + // but keeping it is fine too, /path/to/./file works. + + snprintf(resolved_path, sizeof(resolved_path), "%s/%s", current_dir, leaf); + + // If it's an explicit relative path, OR if the file exists at this relative location + if (is_explicit_relative || access(resolved_path, R_OK) == 0) { - leaf += 2; + free(fn); + fn = xstrdup(resolved_path); } - snprintf(resolved_path, sizeof(resolved_path), "%s/%s", current_dir, leaf); - } - else - { - snprintf(resolved_path, sizeof(resolved_path), "%s", fn); } free(current_dir); - free(fn); - fn = xstrdup(resolved_path); } // Check if file exists, if not try system-wide paths diff --git a/src/parser/parser_utils.c b/src/parser/parser_utils.c index f96ed24..4973111 100644 --- a/src/parser/parser_utils.c +++ b/src/parser/parser_utils.c @@ -1520,7 +1520,8 @@ ASTNode *copy_ast_replacing(ASTNode *n, const char *p, const char *c, const char break; case NODE_IMPL_TRAIT: new_node->impl_trait.trait_name = xstrdup(n->impl_trait.trait_name); - new_node->impl_trait.target_type = replace_type_str(n->impl_trait.target_type, p, c, os, ns); + new_node->impl_trait.target_type = + replace_type_str(n->impl_trait.target_type, p, c, os, ns); new_node->impl_trait.methods = copy_ast_replacing(n->impl_trait.methods, p, c, os, ns); break; default: @@ -1884,13 +1885,15 @@ void instantiate_methods(ParserContext *ctx, GenericImplTemplate *it, it->struct_name, mangled_struct_name); it->impl_node->next = backup_next; // Restore - ASTNode *meth = NULL; - if (new_impl->type == NODE_IMPL) { + if (new_impl->type == NODE_IMPL) + { new_impl->impl.struct_name = xstrdup(mangled_struct_name); meth = new_impl->impl.methods; - } else if (new_impl->type == NODE_IMPL_TRAIT) { + } + else if (new_impl->type == NODE_IMPL_TRAIT) + { new_impl->impl_trait.target_type = xstrdup(mangled_struct_name); meth = new_impl->impl_trait.methods; } @@ -2009,7 +2012,7 @@ void instantiate_generic(ParserContext *ctx, const char *tpl, const char *arg, ASTNode *i = ast_create(NODE_STRUCT); i->strct.name = xstrdup(m); i->strct.is_template = 0; - + // Copy type attributes (e.g. has_drop) i->type_info = type_new(TYPE_STRUCT); i->type_info->name = xstrdup(m); @@ -2033,7 +2036,7 @@ void instantiate_generic(ParserContext *ctx, const char *tpl, const char *arg, ASTNode *i = ast_create(NODE_ENUM); i->enm.name = xstrdup(m); i->enm.is_template = 0; - + // Copy type attributes (e.g. has_drop) i->type_info = type_new(TYPE_ENUM); i->type_info->name = xstrdup(m); @@ -14,3 +14,5 @@ import "./std/map.zc" import "./std/json.zc" import "./std/path.zc" import "./std/mem.zc" +import "./std/stack.zc" +import "./std/queue.zc" @@ -15,6 +15,12 @@ trait Drop { fn drop(self); } +trait Copy {} + +trait Clone { + fn clone(self) -> Self; +} + struct Box<T> { ptr: T*; } diff --git a/std/queue.zc b/std/queue.zc new file mode 100644 index 0000000..2bbcfaa --- /dev/null +++ b/std/queue.zc @@ -0,0 +1,65 @@ +import "./option.zc" +import "./mem.zc" + +struct Queue<T> { + data: T*; + len: usize; + cap: usize; +} + +impl Queue<T> { + fn new() -> Queue<T> { + return Queue<T>{data: NULL, len: 0, cap: 0}; + } + + fn free(self) { + if (self.data) { + free(self.data); + self.data = NULL; + } + self.len = 0; + } + + fn clone(self) -> Queue<T> { + var new_queue = Queue<T>::new(); + new_queue.len = self.len; + new_queue.cap = self.cap; + new_queue.data = malloc(sizeof(T) * new_queue.cap); + memcpy(new_queue.data, self.data, sizeof(T) * new_queue.cap); + return new_queue; + } + + fn push(self, value: T) { + if (!self.data) { + self.cap = 8; + self.data = malloc(sizeof(T) * self.cap); + } + if (self.len == self.cap) { + self.cap = self.cap * 2; + self.data = realloc(self.data, sizeof(T) * self.cap); + } + + // Assigns it at the back of + self.data[self.len] = value; + self.len = self.len + 1; + } + + fn pop(self) -> Option<T> { + if (self.len > 0) { + var value = self.data[0]; + self.len = self.len - 1; + + // Shifts the data in the queue "forward" + memcpy(self.data, self.data + 1, sizeof(T) * self.len); + + return Option<T>::Some(value); + } + return Option<T>::None(); + } +} + +impl Drop for Queue<T> { + fn drop(self) { + self.free(); + } +} diff --git a/std/stack.zc b/std/stack.zc new file mode 100644 index 0000000..3d7c3c5 --- /dev/null +++ b/std/stack.zc @@ -0,0 +1,59 @@ +import "./option.zc" +import "./mem.zc" + +struct Stack<T> { + data: T*; + len: usize; + cap: usize; +} + +impl Stack<T> { + fn new() -> Stack<T> { + return Stack<T>{data: NULL, len: 0, cap: 0}; + } + + fn free(self) { + if (self.data) { + free(self.data); + self.data = NULL; + } + self.len = 0; + } + + fn clone(self) -> Stack<T> { + var new_stack = Stack<T>::new(); + new_stack.len = self.len; + new_stack.cap = self.cap; + new_stack.data = malloc(sizeof(T) * new_stack.cap); + memcpy(new_stack.data, self.data, sizeof(T) * new_stack.cap); + return new_stack; + } + + fn push(self, value: T) { + if (!self.data) { + self.cap = 8; + self.data = malloc(sizeof(T) * self.cap); + } + if (self.len == self.cap) { + self.cap = self.cap * 2; + self.data = realloc(self.data, sizeof(T) * self.cap); + } + self.data[self.len] = value; + self.len = self.len + 1; + } + + fn pop(self) -> Option<T> { + if (self.len > 0) { + var value = self.data[self.len - 1]; + self.len = self.len - 1; + return Option<T>::Some(value); + } + return Option<T>::None(); + } +} + +impl Drop for Stack<T> { + fn drop(self) { + self.free(); + } +} @@ -111,7 +111,6 @@ impl Vec<T> { self.grow(); } self.data[self.len] = item; - self.data[self.len] = item; self.len = self.len + 1; } diff --git a/tests/features/test_auto_deref.zc b/tests/features/test_auto_deref.zc new file mode 100644 index 0000000..b7b0c51 --- /dev/null +++ b/tests/features/test_auto_deref.zc @@ -0,0 +1,34 @@ + +struct Point { + x: int; + y: int; +} + +test "auto_deref" { + var p = Point { x: 10, y: 20 }; + var ptr = &p; + + // This uses the dot operator on a pointer + // In standard C, this requires ->. In Zen-C w/ Auto-Deref, . should work. + assert(ptr.x == 10, "read ptr.x failed"); + + ptr.y = 30; + assert(p.y == 30, "write ptr.y failed"); + + assert(ptr.y == 30, "read ptr.y after write failed"); + + var i = Inner { val: 42 }; + var o = Outer { inner: i }; + var p_out = &o; + + // Nested access: p.inner.val -> p->inner.val + assert(p_out.inner.val == 42, "Nested access failed"); +} + +struct Inner { + val: int; +} + +struct Outer { + inner: Inner; +} diff --git a/tests/features/test_copy_trait.zc b/tests/features/test_copy_trait.zc new file mode 100644 index 0000000..994ccee --- /dev/null +++ b/tests/features/test_copy_trait.zc @@ -0,0 +1,37 @@ + +trait Copy {} + +struct Point { + x: int; + y: int; +} + +impl Copy for Point {} + +struct Mover { + val: int; +} + +test "copy_trait" { + var p1 = Point { x: 10, y: 20 }; + var p2 = p1; // Copy, not move + + // Both should be valid + assert(p1.x == 10, "p1 invalid after copy"); + assert(p2.x == 10, "p2 invalid after copy"); + + // Modify p2, p1 should be unchanged + p2.x = 30; + assert(p1.x == 10, "p1 changed after p2 modification"); + assert(p2.x == 30, "p2 modification failed"); + + println "Copy Trait Works!"; +} + +test "move_default" { + var m1 = Mover { val: 1 }; + var m2 = m1; // Moved + + // Uncommenting this should cause compile error + // var m3 = m1; +} diff --git a/tests/features/test_match_ref.zc b/tests/features/test_match_ref.zc new file mode 100644 index 0000000..4e04b89 --- /dev/null +++ b/tests/features/test_match_ref.zc @@ -0,0 +1,42 @@ + +enum MyOption<T> { + Some(T), + None +} + +test "match_ref_int" { + var r = MyOption<int>::Some(42); + match r { + MyOption::Some(ref i) => { + // i is int* + assert(*i == 42, "int ref check failed"); + *i = 100; + }, + MyOption::None => assert(false, "fail") + } +} +// Mover struct to test move semantics vs ref +struct Mover { + val: int; +} + +test "match_ref_mover" { + var m = Mover { val: 100 }; + // MyOption<Mover> instantiation + var opt = MyOption<Mover>::Some(m); + + match opt { + MyOption::Some(ref x) => { + // x should be Mover* + assert(x.val == 100, "Mover ref access failed"); + x.val = 200; + }, + MyOption::None => assert(false, "Should be Some") + } + + // Check if modification persisted + match opt { + MyOption::Some(v) => assert(v.val == 200, "Mutation via ref failed"), + MyOption::None => {} + } +} diff --git a/tests/features/test_move_semantics.zc b/tests/features/test_move_semantics.zc new file mode 100644 index 0000000..bf0d717 --- /dev/null +++ b/tests/features/test_move_semantics.zc @@ -0,0 +1,55 @@ + +struct Point { + x: int; +} + +struct Mover { + val: int; +} + +test "basic_move" { + var p1 = Mover { val: 10 }; + var p2 = p1; // p1 moved to p2 + + // Valid usage of p2 + assert(p2.val == 10, "p2 should be valid"); + + // Invalid usage of p1 (Uncomment to test compiler error) + // var p3 = p1; +} + +test "primitive_copy" { + var i = 10; + var j = i; // Copy + var k = i; // Copy again - should be valid + assert(k == 10, "Primitive copy failed"); +} + +test "reassignment" { + var m1 = Mover { val: 1 }; + var m2 = m1; // m1 moved + + m1 = Mover { val: 2 }; // Resurrect m1 + var m3 = m1; // Valid now + assert(m3.val == 2, "Resurrection failed"); +} + +fn consume(m: Mover) { + assert(m.val == 10, "Func arg failed"); +} + +test "func_arg" { + var m = Mover { val: 10 }; + consume(m); // m moved + + // 2. Use after move (Call - Negative Test) + // consume(m); // Should fail: Use of moved value 'm' +} + +/* +// 3. Use after return (Negative Test) +fn fail_return(m: Mover) -> Mover { + var m2 = m; + return m; // Should fail: Use of moved value 'm' +} +*/ diff --git a/tests/features/test_resources.zc b/tests/features/test_resources.zc new file mode 100644 index 0000000..dc7b9f9 --- /dev/null +++ b/tests/features/test_resources.zc @@ -0,0 +1,59 @@ + +// Copy Trait +@derive(Copy) +struct Point { x: int; y: int; } + +// Clone Pattern +trait Clone { + fn clone(self) -> Self; +} + +struct Box { val: int; } + +impl Clone for Box { + fn clone(self) -> Box { + return Box{val: self.val}; + } +} + +// Re-initialization +struct Resource { ptr: void*; } + +// Function Param Helper +fn take_point(p: Point) { + if p.x != 10 { + // Error + } +} + +test "Resource Semantics" { + "Testing Resource Semantics..."; + + var p1 = Point{x: 10, y: 20}; + var p2 = p1; // Copied + + var b1 = Box{val: 99}; + var b2 = b1.clone(); + // var b3 = b1; // This would move if uncommented. + + if b2.val != 99 { + !"Clone failed"; + exit(1); + } + + // Re-initialization + // struct Resource (Global) + + var r1 = Resource{ptr: NULL}; + var r2 = r1; // Moved + + r1 = Resource{ptr: NULL}; // Re-init + var r3 = r1; // Valid again + + // Function Param Move (Simulated) + take_point(p1); + take_point(p1); // Still valid because Copy + + "Resource Semantics Passed."; +} + diff --git a/tests/features/test_smart_derive.zc b/tests/features/test_smart_derive.zc new file mode 100644 index 0000000..705a12f --- /dev/null +++ b/tests/features/test_smart_derive.zc @@ -0,0 +1,26 @@ + +@derive(Eq) +struct Container { + val: int; +} + +// Ensure derived eq uses pointers by trying to use 'c2' after comparison +test "eq_moves" { + var c1 = Container { val: 10 }; + var c2 = Container { val: 10 }; + + // This should call Container__eq(&c1, &c2) + // If it passed by value, c2 would be moved here + if c1 == c2 { + // c2 must still be valid + assert(c2.val == 10, "c2 moved during equality check!"); + } else { + assert(false, "c1 != c2"); + } + + // Explicitly verify c2 is still valid + c2.val = 20; + assert(c2.val == 20, "c2 invalid"); + + println "Smart Derive Eq Works!"; +} diff --git a/tests/memory/test_memory_safety.zc b/tests/memory/test_memory_safety.zc index 652fe21..020a7ba 100644 --- a/tests/memory/test_memory_safety.zc +++ b/tests/memory/test_memory_safety.zc @@ -183,7 +183,7 @@ test "test_immutable" { "Distance: {dist}"; var sum = accumulate(10, 5); "Accumulate: {sum}"; - var p3 = process_point(p1); + var p3 = process_point(Point { x: 0, y: 0 }); "Processed: ({p3.x}, {p3.y})"; counter_immut(); } diff --git a/tests/misc/test_edge_cases.zc b/tests/misc/test_edge_cases.zc index f1a4cf5..0d50744 100644 --- a/tests/misc/test_edge_cases.zc +++ b/tests/misc/test_edge_cases.zc @@ -9,7 +9,7 @@ fn test_empty_struct() { var e1 = Empty {}; var e2 = Empty {}; - if (!e1.eq(e2)) { + if (!e1.eq(&e2)) { println "FAIL: Empty struct eq failed"; exit(1); } @@ -35,12 +35,12 @@ fn test_many_fields() { var m2 = ManyFields { a: 1, b: 2, c: 3, d: 4, e: 5, f: 6, g: 7, h: 8 }; var m3 = ManyFields { a: 1, b: 2, c: 3, d: 4, e: 5, f: 6, g: 7, h: 9 }; // h differs - if (!m1.eq(m2)) { + if (!m1.eq(&m2)) { println "FAIL: equal structs not detected as equal"; exit(1); } - if (m1.eq(m3)) { + if (m1.eq(&m3)) { println "FAIL: different structs detected as equal"; exit(1); } diff --git a/tests/std/test_queue.zc b/tests/std/test_queue.zc new file mode 100644 index 0000000..4d40c42 --- /dev/null +++ b/tests/std/test_queue.zc @@ -0,0 +1,48 @@ +import "std/queue.zc" + +test "Queue Push/Pop" { + print "Testing Queue Push/Pop"; + var queue = Queue<i32>::new(); + defer queue.free(); + + print "Popping on an empty queue without pushing anything prior" + var v = queue.pop(); + assert(v.is_none(), "v should not have a valid value"); + + print "Pushing in three values..." + queue.push(123); + queue.push(456); + queue.push(789); + + v = queue.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + + v = queue.pop(); + assert(v.is_some() && v.unwrap() == 456, "v's value should be 456"); + + v = queue.pop(); + assert(v.is_some() && v.unwrap() == 789, "v's value should be 789"); + + print "Popping on an empty queue after pushing and popping three values" + v = queue.pop(); + assert(v.is_none(), "v should not have a valid value"); +} + +test "Queue Clone" { + print "Testing Queue Cloning"; + var queue = Queue<i32>::new(); + defer queue.free(); + queue.push(123); + var queue2 = queue.clone(); + defer queue2.free(); + + var v = queue2.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + v = queue2.pop(); + assert(v.is_none(), "v should not have a valid value"); + + v = queue.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + v = queue.pop(); + assert(v.is_none(), "v should not have a valid value"); +} diff --git a/tests/std/test_stack.zc b/tests/std/test_stack.zc new file mode 100644 index 0000000..ecc9d3c --- /dev/null +++ b/tests/std/test_stack.zc @@ -0,0 +1,48 @@ +import "std/stack.zc" + +test "Stack Push/Pop" { + print "Testing Stack Push/Pop"; + var stack = Stack<i32>::new(); + defer stack.free(); + + print "Popping on an empty stack without pushing anything prior" + var v = stack.pop(); + assert(v.is_none(), "v should not have a valid value"); + + print "Pushing in three values..." + stack.push(123); + stack.push(456); + stack.push(789); + + v = stack.pop(); + assert(v.is_some() && v.unwrap() == 789, "v's value should be 789"); + + v = stack.pop(); + assert(v.is_some() && v.unwrap() == 456, "v's value should be 456"); + + v = stack.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + + print "Popping on an empty stack after pushing and popping three values" + v = stack.pop(); + assert(v.is_none(), "v should not have a valid value"); +} + +test "Stack Clone" { + print "Testing Stack Cloning"; + var stack = Stack<i32>::new(); + defer stack.free(); + stack.push(123); + var stack2 = stack.clone(); + defer stack2.free(); + + var v = stack2.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + v = stack2.pop(); + assert(v.is_none(), "v should not have a valid value"); + + v = stack.pop(); + assert(v.is_some() && v.unwrap() == 123, "v's value should be 123"); + v = stack.pop(); + assert(v.is_none(), "v should not have a valid value"); +} |
