mirror of
https://github.com/krgamestudios/Toy.git
synced 2026-04-15 23:04:08 +10:00
Break and continue keywords are working
This commit is contained in:
@@ -8,12 +8,12 @@ DONE: chained if-then-else
|
|||||||
DONE: optional block around a path if it's only one statement
|
DONE: optional block around a path if it's only one statement
|
||||||
DONE: while-then
|
DONE: while-then
|
||||||
DONE: for-then
|
DONE: for-then
|
||||||
|
DONE: break and continue statements
|
||||||
|
|
||||||
TODO: string concat with the + operator
|
TODO: string concat with the + operator
|
||||||
TODO: increment & decrement operators
|
TODO: increment & decrement operators
|
||||||
TODO: a = b = c = 1;
|
TODO: a = b = c = 1;
|
||||||
TODO: are compounds shallow or deep copies?
|
TODO: are compounds shallow or deep copies?
|
||||||
TODO: break and continue statements
|
|
||||||
TODO: functions, and all of their features
|
TODO: functions, and all of their features
|
||||||
TODO: Assertion-based test scripts
|
TODO: Assertion-based test scripts
|
||||||
TODO: standard library
|
TODO: standard library
|
||||||
|
|||||||
@@ -10,6 +10,9 @@
|
|||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
|
||||||
|
//assigning to a byte from a short loses data
|
||||||
|
#define AS_USHORT(value) (*(unsigned short*)(&(value)))
|
||||||
|
|
||||||
void initCompiler(Compiler* compiler) {
|
void initCompiler(Compiler* compiler) {
|
||||||
initLiteralArray(&compiler->literalCache);
|
initLiteralArray(&compiler->literalCache);
|
||||||
compiler->bytecode = NULL;
|
compiler->bytecode = NULL;
|
||||||
@@ -144,7 +147,7 @@ static int writeLiteralTypeToCache(LiteralArray* literalCache, Literal literal)
|
|||||||
return pushLiteralArray(literalCache, lit);
|
return pushLiteralArray(literalCache, lit);
|
||||||
}
|
}
|
||||||
|
|
||||||
void writeCompiler(Compiler* compiler, Node* node) {
|
static void writeCompilerWithJumps(Compiler* compiler, Node* node, void* breakAddressesPtr, void* continueAddressesPtr) {
|
||||||
//grow if the bytecode space is too small
|
//grow if the bytecode space is too small
|
||||||
if (compiler->capacity < compiler->count + 1) {
|
if (compiler->capacity < compiler->count + 1) {
|
||||||
int oldCapacity = compiler->capacity;
|
int oldCapacity = compiler->capacity;
|
||||||
@@ -157,7 +160,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
switch(node->type) {
|
switch(node->type) {
|
||||||
//TODO: more types, like variables, etc.
|
//TODO: more types, like variables, etc.
|
||||||
case NODE_ERROR: {
|
case NODE_ERROR: {
|
||||||
fprintf(stderr, ERROR "[Internal] NODE_ERROR encountered in writeCompiler()\n" RESET);
|
fprintf(stderr, ERROR "[Internal] NODE_ERROR encountered in writeCompilerWithJumps()\n" RESET);
|
||||||
compiler->bytecode[compiler->count++] = OP_EOF; //1 byte
|
compiler->bytecode[compiler->count++] = OP_EOF; //1 byte
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -187,20 +190,20 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
|
|
||||||
case NODE_UNARY:
|
case NODE_UNARY:
|
||||||
//pass to the child node, then embed the unary command (print, negate, etc.)
|
//pass to the child node, then embed the unary command (print, negate, etc.)
|
||||||
writeCompiler(compiler, node->unary.child);
|
writeCompilerWithJumps(compiler, node->unary.child, breakAddressesPtr, continueAddressesPtr);
|
||||||
compiler->bytecode[compiler->count++] = (unsigned char)node->unary.opcode; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)node->unary.opcode; //1 byte
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_BINARY:
|
case NODE_BINARY:
|
||||||
//pass to the child nodes, then embed the binary command (math, etc.)
|
//pass to the child nodes, then embed the binary command (math, etc.)
|
||||||
writeCompiler(compiler, node->binary.left);
|
writeCompilerWithJumps(compiler, node->binary.left, breakAddressesPtr, continueAddressesPtr);
|
||||||
writeCompiler(compiler, node->binary.right);
|
writeCompilerWithJumps(compiler, node->binary.right, breakAddressesPtr, continueAddressesPtr);
|
||||||
compiler->bytecode[compiler->count++] = (unsigned char)node->binary.opcode; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)node->binary.opcode; //1 byte
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_GROUPING:
|
case NODE_GROUPING:
|
||||||
compiler->bytecode[compiler->count++] = (unsigned char)OP_GROUPING_BEGIN; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)OP_GROUPING_BEGIN; //1 byte
|
||||||
writeCompiler(compiler, node->grouping.child);
|
writeCompilerWithJumps(compiler, node->grouping.child, breakAddressesPtr, continueAddressesPtr);
|
||||||
compiler->bytecode[compiler->count++] = (unsigned char)OP_GROUPING_END; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)OP_GROUPING_END; //1 byte
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -208,7 +211,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
compiler->bytecode[compiler->count++] = (unsigned char)OP_SCOPE_BEGIN; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)OP_SCOPE_BEGIN; //1 byte
|
||||||
|
|
||||||
for (int i = 0; i < node->block.count; i++) {
|
for (int i = 0; i < node->block.count; i++) {
|
||||||
writeCompiler(compiler, &(node->block.nodes[i]));
|
writeCompilerWithJumps(compiler, &(node->block.nodes[i]), breakAddressesPtr, continueAddressesPtr);
|
||||||
}
|
}
|
||||||
|
|
||||||
compiler->bytecode[compiler->count++] = (unsigned char)OP_SCOPE_END; //1 byte
|
compiler->bytecode[compiler->count++] = (unsigned char)OP_SCOPE_END; //1 byte
|
||||||
@@ -234,7 +237,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_PAIR:
|
case NODE_PAIR:
|
||||||
fprintf(stderr, ERROR "[Internal] NODE_PAIR encountered in writeCompiler()\n" RESET);
|
fprintf(stderr, ERROR "[Internal] NODE_PAIR encountered in writeCompilerWithJumps()\n" RESET);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_VAR_TYPES: { //TODO: the "type" keyword
|
case NODE_VAR_TYPES: { //TODO: the "type" keyword
|
||||||
@@ -258,7 +261,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
|
|
||||||
case NODE_VAR_DECL: {
|
case NODE_VAR_DECL: {
|
||||||
//first, embed the expression (leaves it on the stack)
|
//first, embed the expression (leaves it on the stack)
|
||||||
writeCompiler(compiler, node->varDecl.expression);
|
writeCompilerWithJumps(compiler, node->varDecl.expression, breakAddressesPtr, continueAddressesPtr);
|
||||||
|
|
||||||
//write each piece of the declaration to the bytecode
|
//write each piece of the declaration to the bytecode
|
||||||
int identifierIndex = findLiteralIndex(&compiler->literalCache, node->varDecl.identifier);
|
int identifierIndex = findLiteralIndex(&compiler->literalCache, node->varDecl.identifier);
|
||||||
@@ -290,7 +293,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
|
|
||||||
case NODE_PATH_IF: {
|
case NODE_PATH_IF: {
|
||||||
//process the condition
|
//process the condition
|
||||||
writeCompiler(compiler, node->path.condition);
|
writeCompilerWithJumps(compiler, node->path.condition, breakAddressesPtr, continueAddressesPtr);
|
||||||
|
|
||||||
//cache the point to insert the jump distance at
|
//cache the point to insert the jump distance at
|
||||||
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
||||||
@@ -298,7 +301,7 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
compiler->count += sizeof(unsigned short); //2 bytes
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
|
|
||||||
//write the then path
|
//write the then path
|
||||||
writeCompiler(compiler, node->path.thenPath);
|
writeCompilerWithJumps(compiler, node->path.thenPath, breakAddressesPtr, continueAddressesPtr);
|
||||||
|
|
||||||
int jumpToEnd = 0;
|
int jumpToEnd = 0;
|
||||||
|
|
||||||
@@ -310,24 +313,31 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
//update the jumpToElse to point here
|
//update the jumpToElse to point here
|
||||||
compiler->bytecode[jumpToElse] = compiler->count;
|
AS_USHORT(compiler->bytecode[jumpToElse]) = compiler->count; //2 bytes
|
||||||
|
|
||||||
if (node->path.elsePath) {
|
if (node->path.elsePath) {
|
||||||
//if there's an else path, write it and
|
//if there's an else path, write it and
|
||||||
writeCompiler(compiler, node->path.elsePath);
|
writeCompilerWithJumps(compiler, node->path.elsePath, breakAddressesPtr, continueAddressesPtr);
|
||||||
|
|
||||||
//update the jumpToEnd to point here
|
//update the jumpToEnd to point here
|
||||||
compiler->bytecode[jumpToEnd] = compiler->count;
|
AS_USHORT(compiler->bytecode[jumpToEnd]) = compiler->count; //2 bytes
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_PATH_WHILE: {
|
case NODE_PATH_WHILE: {
|
||||||
|
//for breaks and continues
|
||||||
|
LiteralArray breakAddresses;
|
||||||
|
LiteralArray continueAddresses;
|
||||||
|
|
||||||
|
initLiteralArray(&breakAddresses);
|
||||||
|
initLiteralArray(&continueAddresses);
|
||||||
|
|
||||||
//cache the jump point
|
//cache the jump point
|
||||||
unsigned short jumpFromEnd = compiler->count;
|
unsigned short jumpToStart = compiler->count;
|
||||||
|
|
||||||
//process the condition
|
//process the condition
|
||||||
writeCompiler(compiler, node->path.condition);
|
writeCompilerWithJumps(compiler, node->path.condition, &breakAddresses, &continueAddresses);
|
||||||
|
|
||||||
//if false, jump to end
|
//if false, jump to end
|
||||||
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
||||||
@@ -335,27 +345,49 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
compiler->count += sizeof(unsigned short); //2 bytes
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
|
|
||||||
//write the body
|
//write the body
|
||||||
writeCompiler(compiler, node->path.thenPath);
|
writeCompilerWithJumps(compiler, node->path.thenPath, &breakAddresses, &continueAddresses);
|
||||||
|
|
||||||
//jump to condition
|
//jump to condition
|
||||||
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
||||||
compiler->bytecode[compiler->count] = jumpFromEnd;
|
AS_USHORT(compiler->bytecode[compiler->count]) = jumpToStart;
|
||||||
compiler->count += sizeof(unsigned short); //2 bytes
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
|
|
||||||
//jump from condition
|
//jump from condition
|
||||||
compiler->bytecode[jumpToEnd] = compiler->count;
|
AS_USHORT(compiler->bytecode[jumpToEnd]) = (unsigned short)compiler->count;
|
||||||
|
|
||||||
|
//set the breaks and continues
|
||||||
|
for (int i = 0; i < breakAddresses.count; i++) {
|
||||||
|
int point = AS_INTEGER(breakAddresses.literals[i]);
|
||||||
|
AS_USHORT(compiler->bytecode[point]) = (unsigned short)compiler->count;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < continueAddresses.count; i++) {
|
||||||
|
int point = AS_INTEGER(continueAddresses.literals[i]);
|
||||||
|
AS_USHORT(compiler->bytecode[point]) = jumpToStart;
|
||||||
|
}
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
freeLiteralArray(&breakAddresses);
|
||||||
|
freeLiteralArray(&continueAddresses);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case NODE_PATH_FOR: {
|
case NODE_PATH_FOR: {
|
||||||
|
//for breaks and continues
|
||||||
|
LiteralArray breakAddresses;
|
||||||
|
LiteralArray continueAddresses;
|
||||||
|
|
||||||
|
initLiteralArray(&breakAddresses);
|
||||||
|
initLiteralArray(&continueAddresses);
|
||||||
|
|
||||||
compiler->bytecode[compiler->count++] = OP_SCOPE_BEGIN; //1 byte
|
compiler->bytecode[compiler->count++] = OP_SCOPE_BEGIN; //1 byte
|
||||||
|
|
||||||
//initial setup
|
//initial setup
|
||||||
writeCompiler(compiler, node->path.preClause);
|
writeCompilerWithJumps(compiler, node->path.preClause, &breakAddresses, &continueAddresses);
|
||||||
|
|
||||||
//conditional
|
//conditional
|
||||||
unsigned short jumpFromEnd = compiler->count;
|
unsigned short jumpToStart = compiler->count;
|
||||||
writeCompiler(compiler, node->path.condition);
|
writeCompilerWithJumps(compiler, node->path.condition, &breakAddresses, &continueAddresses);
|
||||||
|
|
||||||
//if false jump to end
|
//if false jump to end
|
||||||
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
compiler->bytecode[compiler->count++] = OP_IF_FALSE_JUMP; //1 byte
|
||||||
@@ -364,24 +396,78 @@ void writeCompiler(Compiler* compiler, Node* node) {
|
|||||||
|
|
||||||
//write the body
|
//write the body
|
||||||
compiler->bytecode[compiler->count++] = OP_SCOPE_BEGIN; //1 byte
|
compiler->bytecode[compiler->count++] = OP_SCOPE_BEGIN; //1 byte
|
||||||
writeCompiler(compiler, node->path.thenPath);
|
writeCompilerWithJumps(compiler, node->path.thenPath, &breakAddresses, &continueAddresses);
|
||||||
compiler->bytecode[compiler->count++] = OP_SCOPE_END; //1 byte
|
compiler->bytecode[compiler->count++] = OP_SCOPE_END; //1 byte
|
||||||
|
|
||||||
|
//for-breaks actually jump to the bottom
|
||||||
|
int jumpToIncrement = compiler->count;
|
||||||
|
|
||||||
//evaluate third clause, restart
|
//evaluate third clause, restart
|
||||||
writeCompiler(compiler, node->path.postClause);
|
writeCompilerWithJumps(compiler, node->path.postClause, &breakAddresses, &continueAddresses);
|
||||||
|
|
||||||
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
||||||
compiler->bytecode[compiler->count] = jumpFromEnd;
|
AS_USHORT(compiler->bytecode[compiler->count]) = jumpToStart;
|
||||||
compiler->count += sizeof(unsigned short); //2 bytes
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
|
|
||||||
compiler->bytecode[jumpToEnd] = compiler->count;
|
AS_USHORT(compiler->bytecode[jumpToEnd]) = compiler->count;
|
||||||
|
|
||||||
compiler->bytecode[compiler->count++] = OP_SCOPE_END; //1 byte
|
compiler->bytecode[compiler->count++] = OP_SCOPE_END; //1 byte
|
||||||
|
|
||||||
|
//set the breaks and continues
|
||||||
|
for (int i = 0; i < breakAddresses.count; i++) {
|
||||||
|
int point = AS_INTEGER(breakAddresses.literals[i]);
|
||||||
|
AS_USHORT(compiler->bytecode[point]) = compiler->count;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < continueAddresses.count; i++) {
|
||||||
|
int point = AS_INTEGER(continueAddresses.literals[i]);
|
||||||
|
AS_USHORT(compiler->bytecode[point]) = jumpToIncrement;
|
||||||
|
}
|
||||||
|
|
||||||
|
//cleanup
|
||||||
|
freeLiteralArray(&breakAddresses);
|
||||||
|
freeLiteralArray(&continueAddresses);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NODE_PATH_BREAK: {
|
||||||
|
if (!breakAddressesPtr) {
|
||||||
|
fprintf(stderr, "Can't place a break statement here\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
//insert into bytecode
|
||||||
|
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
||||||
|
|
||||||
|
//push to the breakAddresses array
|
||||||
|
pushLiteralArray((LiteralArray*)breakAddressesPtr, TO_INTEGER_LITERAL(compiler->count));
|
||||||
|
|
||||||
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case NODE_PATH_CONTINUE: {
|
||||||
|
if (!continueAddressesPtr) {
|
||||||
|
fprintf(stderr, "Can't place a continue statement here\n");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
//insert into bytecode
|
||||||
|
compiler->bytecode[compiler->count++] = OP_JUMP; //1 byte
|
||||||
|
|
||||||
|
//push to the continueAddresses array
|
||||||
|
pushLiteralArray((LiteralArray*)continueAddressesPtr, TO_INTEGER_LITERAL(compiler->count));
|
||||||
|
|
||||||
|
compiler->count += sizeof(unsigned short); //2 bytes
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void writeCompiler(Compiler* compiler, Node* node) {
|
||||||
|
writeCompilerWithJumps(compiler, node, NULL, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
void freeCompiler(Compiler* compiler) {
|
void freeCompiler(Compiler* compiler) {
|
||||||
freeLiteralArray(&compiler->literalCache);
|
freeLiteralArray(&compiler->literalCache);
|
||||||
FREE(unsigned char, compiler->bytecode);
|
FREE(unsigned char, compiler->bytecode);
|
||||||
|
|||||||
@@ -588,7 +588,7 @@ static bool execFalseJump(Interpreter* interpreter) {
|
|||||||
int target = (int)readShort(interpreter->bytecode, &interpreter->count);
|
int target = (int)readShort(interpreter->bytecode, &interpreter->count);
|
||||||
|
|
||||||
if (target + interpreter->codeStart > interpreter->length) {
|
if (target + interpreter->codeStart > interpreter->length) {
|
||||||
printf("Jump out of range\n");
|
printf("Jump out of range (false jump)\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -65,6 +65,8 @@ void freeNode(Node* node) {
|
|||||||
case NODE_PATH_IF:
|
case NODE_PATH_IF:
|
||||||
case NODE_PATH_WHILE:
|
case NODE_PATH_WHILE:
|
||||||
case NODE_PATH_FOR:
|
case NODE_PATH_FOR:
|
||||||
|
case NODE_PATH_BREAK:
|
||||||
|
case NODE_PATH_CONTINUE:
|
||||||
freeNode(node->path.preClause);
|
freeNode(node->path.preClause);
|
||||||
freeNode(node->path.postClause);
|
freeNode(node->path.postClause);
|
||||||
freeNode(node->path.condition);
|
freeNode(node->path.condition);
|
||||||
|
|||||||
@@ -21,6 +21,8 @@ typedef enum NodeType {
|
|||||||
NODE_PATH_IF, //for control flow
|
NODE_PATH_IF, //for control flow
|
||||||
NODE_PATH_WHILE, //for control flow
|
NODE_PATH_WHILE, //for control flow
|
||||||
NODE_PATH_FOR, //for control flow
|
NODE_PATH_FOR, //for control flow
|
||||||
|
NODE_PATH_BREAK,
|
||||||
|
NODE_PATH_CONTINUE,
|
||||||
} NodeType;
|
} NodeType;
|
||||||
|
|
||||||
typedef struct NodeLiteral {
|
typedef struct NodeLiteral {
|
||||||
|
|||||||
@@ -985,6 +985,20 @@ static void forStmt(Parser* parser, Node** nodeHandle) {
|
|||||||
emitNodePath(nodeHandle, NODE_PATH_FOR, preClause, postClause, condition, thenPath, NULL);
|
emitNodePath(nodeHandle, NODE_PATH_FOR, preClause, postClause, condition, thenPath, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void breakStmt(Parser* parser, Node** nodeHandle) {
|
||||||
|
freeNode(*nodeHandle);
|
||||||
|
emitNodePath(nodeHandle, NODE_PATH_BREAK, NULL, NULL, NULL, NULL, NULL);
|
||||||
|
|
||||||
|
consume(parser, TOKEN_SEMICOLON, "Expected ';' at end of break statement");
|
||||||
|
}
|
||||||
|
|
||||||
|
static void continueStmt(Parser* parser, Node** nodeHandle) {
|
||||||
|
freeNode(*nodeHandle);
|
||||||
|
emitNodePath(nodeHandle, NODE_PATH_CONTINUE, NULL, NULL, NULL, NULL, NULL);
|
||||||
|
|
||||||
|
consume(parser, TOKEN_SEMICOLON, "Expected ';' at end of continue statement");
|
||||||
|
}
|
||||||
|
|
||||||
//precedence functions
|
//precedence functions
|
||||||
static void expressionStmt(Parser* parser, Node** nodeHandle) {
|
static void expressionStmt(Parser* parser, Node** nodeHandle) {
|
||||||
//BUGFIX: check for empty statements
|
//BUGFIX: check for empty statements
|
||||||
@@ -1043,6 +1057,18 @@ static void statement(Parser* parser, Node** nodeHandle) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//break
|
||||||
|
if (match(parser, TOKEN_BREAK)) {
|
||||||
|
breakStmt(parser, nodeHandle);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
//continue
|
||||||
|
if (match(parser, TOKEN_CONTINUE)) {
|
||||||
|
continueStmt(parser, nodeHandle);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
//default
|
//default
|
||||||
expressionStmt(parser, nodeHandle);
|
expressionStmt(parser, nodeHandle);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,18 @@
|
|||||||
|
|
||||||
|
|
||||||
//test true jump
|
//test true jump
|
||||||
if (true) {
|
if (true) {
|
||||||
assert true, "if-then failed";
|
assert true, "if-then failed (1)";
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
assert false, "if-then failed";
|
assert false, "if-then failed (2)";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
//test false jump
|
//test false jump
|
||||||
if (false) {
|
if (false) {
|
||||||
assert false, "if-then failed";
|
assert false, "if-then failed (3)";
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
assert true, "if-then failed";
|
assert true, "if-then failed (4)";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -36,4 +34,49 @@ for (var i = 0; i < 20; i = i + 1) {
|
|||||||
assert forCache == 19, "for-loop failed";
|
assert forCache == 19, "for-loop failed";
|
||||||
|
|
||||||
|
|
||||||
|
//test break - while
|
||||||
|
var breakWhileCache = 0;
|
||||||
|
while(true) {
|
||||||
|
breakWhileCache = breakWhileCache + 1;
|
||||||
|
|
||||||
|
if (breakWhileCache >= 7) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert breakWhileCache == 7, "break-while failed";
|
||||||
|
|
||||||
|
|
||||||
|
//test continue - while
|
||||||
|
var continueWhileCache = 0;
|
||||||
|
while (continueWhileCache < 10) {
|
||||||
|
continueWhileCache = continueWhileCache + 1;
|
||||||
|
|
||||||
|
if (continueWhileCache >= 7) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert continueWhileCache < 7, "continue-while failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test break - for
|
||||||
|
for (var i = 0; i < 10; i = i + 1) {
|
||||||
|
if (i >= 7) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert i < 7, "break-for failed";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//test break - continue
|
||||||
|
for (var i = 0; i < 10; i = i + 1) {
|
||||||
|
if (i >= 7) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
assert i < 7, "continue-for failed";
|
||||||
|
}
|
||||||
|
|
||||||
print "All good";
|
print "All good";
|
||||||
Reference in New Issue
Block a user