/* * Copyright (c) 2020, Andreas Kling * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * 1. Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #define PROGRAM 6 static NonnullOwnPtr build_program(JS::Heap&); int main() { JS::Interpreter interpreter; auto program = build_program(interpreter.heap()); program->dump(0); auto result = interpreter.run(*program); dbg() << "Interpreter returned " << result; printf("%s\n", result.to_string().characters()); dbg() << "Collecting garbage on exit..."; interpreter.heap().collect_garbage(); return 0; } #if PROGRAM == 1 NonnullOwnPtr build_program(JS::Heap&) { // function foo() { return (1 + 2) + 3; } // foo(); auto block = make(); block->append( make( JS::BinaryOp::Plus, make( JS::BinaryOp::Plus, make(JS::Value(1)), make(JS::Value(2))), make(JS::Value(3)))); auto program = make(); program->append("foo", move(block)); program->append(make("foo")); return program; } #elif PROGRAM == 2 NonnullOwnPtr build_program(JS::Heap&) { // c = 1; // function foo() { // var a = 5; // var b = 7; // return a + b + c; // } // foo(); auto program = make(); program->append(make( JS::AssignmentOp::Assign, make("c"), make(JS::Value(1)))); auto block = make(); block->append( make("a"), make(JS::Value(5)), JS::DeclarationType::Var); block->append( make("b"), make(JS::Value(7)), JS::DeclarationType::Var); block->append( make( JS::BinaryOp::Plus, make( JS::BinaryOp::Plus, make("a"), make("b")), make("c"))); program->append("foo", move(block)); program->append(make("foo")); return program; } #elif PROGRAM == 3 NonnullOwnPtr build_program(JS::Heap&) { // function foo() { // var x = {}; // $gc(); // } // foo(); auto block = make(); block->append( make("x"), make(), JS::DeclarationType::Var); block->append(make("$gc")); auto program = make(); program->append("foo", move(block)); program->append(make("foo")); return program; } #elif PROGRAM == 4 NonnullOwnPtr build_program(JS::Heap&) { // function foo() { // function bar() { // var y = 6; // } // // bar() // return y; // } // foo(); //I should return `undefined` because y is bound to the inner-most enclosing function, i.e the nested one (bar()), therefore, it's undefined in the scope of foo() auto block_bar = make(); block_bar->append(make("y"), make(JS::Value(6)), JS::DeclarationType::Var); auto block_foo = make(); block_foo->append("bar", move(block_bar)); block_foo->append(make("bar")); block_foo->append(make("y")); auto program = make(); program->append("foo", move(block_foo)); program->append(make("foo")); return program; } #elif PROGRAM == 5 NonnullOwnPtr build_program(JS::Heap& heap) { // "hello friends".length auto program = make(); program->append(make( make(JS::Value(js_string(heap, "hello friends"))), make("length"))); return program; } #elif PROGRAM == 6 NonnullOwnPtr build_program(JS::Heap&) { const char* source = "var foo = 1;\n" "function bar() {\n" " return 38;\n" "}\n" "foo = {};\n" "foo = bar() + 4;\n" "foo;\n"; auto parser = JS::Parser(JS::Lexer(source)); return parser.parse_program(); } #endif