diff options
author | Stephan Unverwerth <s.unverwerth@gmx.de> | 2020-03-11 19:27:43 +0100 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2020-03-12 09:25:49 +0100 |
commit | f3a9eba987a8346dc2c13cedac3865eee38a10a4 (patch) | |
tree | ff51343e5d7ed95715aff7aed8131280f060c14b /Userland | |
parent | 17705d23fbe700d03bcf32a79bb63a2b074e44b9 (diff) | |
download | serenity-f3a9eba987a8346dc2c13cedac3865eee38a10a4.zip |
LibJS: Add Javascript lexer and parser
This adds a basic Javascript lexer and parser. It can parse the
currently existing demo programs. More work needs to be done to
turn it into a complete parser than can parse arbitrary JS Code.
The lexer outputs tokens with preceeding whitespace and comments
in the trivia member. This should allow us to generate the exact
source code by concatenating the generated tokens.
The parser is written in a way that it always returns a complete
syntax tree. Error conditions are represented as nodes in the
tree. This simplifies the code and allows it to be used as an
early stage parser, e.g for parsing JS documents in an IDE while
editing the source code.:
Diffstat (limited to 'Userland')
-rw-r--r-- | Userland/js.cpp | 73 |
1 files changed, 50 insertions, 23 deletions
diff --git a/Userland/js.cpp b/Userland/js.cpp index cc6e09a2ad..6c919fdb6b 100644 --- a/Userland/js.cpp +++ b/Userland/js.cpp @@ -28,20 +28,20 @@ #include <LibJS/AST.h> #include <LibJS/Interpreter.h> #include <LibJS/Object.h> +#include <LibJS/Parser.h> #include <LibJS/PrimitiveString.h> #include <LibJS/Value.h> #include <stdio.h> -#define PROGRAM 4 +#define PROGRAM 6 -static void build_program(JS::Program&, JS::Heap&); +static NonnullOwnPtr<JS::Program> build_program(JS::Heap&); int main() { JS::Interpreter interpreter; - auto program = make<JS::Program>(); - build_program(*program, interpreter.heap()); + auto program = build_program(interpreter.heap()); program->dump(0); auto result = interpreter.run(*program); @@ -55,7 +55,7 @@ int main() } #if PROGRAM == 1 -void build_program(JS::Program& program, JS::Heap&) +NonnullOwnPtr<JS::Program> build_program(JS::Heap&) { // function foo() { return (1 + 2) + 3; } // foo(); @@ -70,11 +70,13 @@ void build_program(JS::Program& program, JS::Heap&) make<JS::Literal>(JS::Value(2))), make<JS::Literal>(JS::Value(3)))); - program.append<JS::FunctionDeclaration>("foo", move(block)); - program.append<JS::CallExpression>("foo"); + auto program = make<JS::Program>(); + program->append<JS::FunctionDeclaration>("foo", move(block)); + program->append<JS::ExpressionStatement>(make<JS::CallExpression>("foo")); + return program; } #elif PROGRAM == 2 -void build_program(JS::Program& program, JS::Heap&) +NonnullOwnPtr<JS::Program> build_program(JS::Heap&) { // c = 1; // function foo() { @@ -84,10 +86,11 @@ void build_program(JS::Program& program, JS::Heap&) // } // foo(); - program.append<JS::AssignmentExpression>( + auto program = make<JS::Program>(); + program->append<JS::ExpressionStatement>(make<JS::AssignmentExpression>( JS::AssignmentOp::Assign, make<JS::Identifier>("c"), - make<JS::Literal>(JS::Value(1))); + make<JS::Literal>(JS::Value(1)))); auto block = make<JS::BlockStatement>(); block->append<JS::VariableDeclaration>( @@ -107,11 +110,14 @@ void build_program(JS::Program& program, JS::Heap&) make<JS::Identifier>("a"), make<JS::Identifier>("b")), make<JS::Identifier>("c"))); - program.append<JS::FunctionDeclaration>("foo", move(block)); - program.append<JS::CallExpression>("foo"); + + program->append<JS::FunctionDeclaration>("foo", move(block)); + program->append<JS::ExpressionStatement>(make<JS::CallExpression>("foo")); + + return program; } #elif PROGRAM == 3 -void build_program(JS::Program& program, JS::Heap&) +NonnullOwnPtr<JS::Program> build_program(JS::Heap&) { // function foo() { // var x = {}; @@ -124,13 +130,15 @@ void build_program(JS::Program& program, JS::Heap&) make<JS::Identifier>("x"), make<JS::ObjectExpression>(), JS::DeclarationType::Var); - block->append<JS::CallExpression>("$gc"); + block->append<JS::ExpressionStatement>(make<JS::CallExpression>("$gc")); - program.append<JS::FunctionDeclaration>("foo", move(block)); - program.append<JS::CallExpression>("foo"); + auto program = make<JS::Program>(); + program->append<JS::FunctionDeclaration>("foo", move(block)); + program->append<JS::ExpressionStatement>(make<JS::CallExpression>("foo")); + return program; } #elif PROGRAM == 4 -void build_program(JS::Program& program, JS::Heap&) +NonnullOwnPtr<JS::Program> build_program(JS::Heap&) { // function foo() { // function bar() { @@ -147,19 +155,38 @@ void build_program(JS::Program& program, JS::Heap&) auto block_foo = make<JS::BlockStatement>(); block_foo->append<JS::FunctionDeclaration>("bar", move(block_bar)); - block_foo->append<JS::CallExpression>("bar"); + block_foo->append<JS::ExpressionStatement>(make<JS::CallExpression>("bar")); block_foo->append<JS::ReturnStatement>(make<JS::Identifier>("y")); - program.append<JS::FunctionDeclaration>("foo", move(block_foo)); - program.append<JS::CallExpression>("foo"); + auto program = make<JS::Program>(); + program->append<JS::FunctionDeclaration>("foo", move(block_foo)); + program->append<JS::ExpressionStatement>(make<JS::CallExpression>("foo")); + return program; } #elif PROGRAM == 5 -void build_program(JS::Program& program, JS::Heap& heap) +NonnullOwnPtr<JS::Program> build_program(JS::Heap& heap) { // "hello friends".length - program.append<JS::MemberExpression>( + auto program = make<JS::Program>(); + program->append<JS::ExpressionStatement>(make<JS::MemberExpression>( make<JS::Literal>(JS::Value(js_string(heap, "hello friends"))), - make<JS::Identifier>("length")); + make<JS::Identifier>("length"))); + + return program; +} +#elif PROGRAM == 6 +NonnullOwnPtr<JS::Program> 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 |