summaryrefslogtreecommitdiff
path: root/src/ast
diff options
context:
space:
mode:
Diffstat (limited to 'src/ast')
-rw-r--r--src/ast/Python3VisitorImpl.java274
-rw-r--r--src/ast/nodes/ArglistNode.java34
-rw-r--r--src/ast/nodes/AssignmentNode.java33
-rw-r--r--src/ast/nodes/AtomNode.java70
-rw-r--r--src/ast/nodes/AugassignNode.java5
-rw-r--r--src/ast/nodes/BlockNode.java32
-rw-r--r--src/ast/nodes/CompForNode.java58
-rw-r--r--src/ast/nodes/CompIterNode.java50
-rw-r--r--src/ast/nodes/CompNode.java5
-rw-r--r--src/ast/nodes/CompoundNode.java14
-rw-r--r--src/ast/nodes/DottedNameNode.java9
-rw-r--r--src/ast/nodes/ExprListNode.java68
-rw-r--r--src/ast/nodes/ExprNode.java97
-rw-r--r--src/ast/nodes/ForStmtNode.java40
-rw-r--r--src/ast/nodes/FuncdefNode.java33
-rw-r--r--src/ast/nodes/IfNode.java22
-rw-r--r--src/ast/nodes/ImportNode.java33
-rw-r--r--src/ast/nodes/Node.java85
-rw-r--r--src/ast/nodes/ParamdefNode.java20
-rw-r--r--src/ast/nodes/ParamlistNode.java12
-rw-r--r--src/ast/nodes/ReturnStmtNode.java8
-rw-r--r--src/ast/nodes/RootNode.java42
-rw-r--r--src/ast/nodes/SimpleStmtNode.java14
-rw-r--r--src/ast/nodes/SimpleStmtsNode.java8
-rw-r--r--src/ast/nodes/TestlistCompNode.java82
-rw-r--r--src/ast/nodes/TrailerNode.java29
-rw-r--r--src/ast/nodes/WhileStmtNode.java10
-rw-r--r--src/ast/types/AtomType.java3
-rw-r--r--src/ast/types/BoolType.java2
-rw-r--r--src/ast/types/ErrorType.java2
-rw-r--r--src/ast/types/FunctionType.java29
-rw-r--r--src/ast/types/ImportType.java12
-rw-r--r--src/ast/types/IntType.java2
-rw-r--r--src/ast/types/NoneType.java17
-rw-r--r--src/ast/types/ReservedWordsType.java12
-rw-r--r--src/ast/types/Type.java11
-rw-r--r--src/ast/types/VoidType.java7
37 files changed, 1017 insertions, 267 deletions
diff --git a/src/ast/Python3VisitorImpl.java b/src/ast/Python3VisitorImpl.java
index 604c8d2..1cf15b7 100644
--- a/src/ast/Python3VisitorImpl.java
+++ b/src/ast/Python3VisitorImpl.java
@@ -12,34 +12,33 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Overrides each `visitNODE` method from the base class.
*/
public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
+
/**
* Since a root can be a simple_stmts or a compound_stmt, this method
* returns a new `RootNode` with a list of them.
*
- * ```
- * root : NEWLINE* (simple_stmts | compound_stmt)* EOF;
- * ```
+ * ``` root : NEWLINE* (simple_stmts | compound_stmt)* EOF; ```
*/
public Node visitRoot(RootContext ctx) {
- ArrayList<Node> stmts = new ArrayList<Node>();
- ArrayList<Node> compStmts = new ArrayList<Node>();
+ ArrayList<Node> childs = new ArrayList<Node>();
- for (Simple_stmtsContext stm : ctx.simple_stmts()) {
- stmts.add(visit(stm));
- }
- for (Compound_stmtContext stm : ctx.compound_stmt()) {
- compStmts.add(visit(stm));
+ for (int i = 0; i < ctx.getChildCount(); i++) {
+ var child = ctx.getChild(i);
+
+ if (child instanceof Simple_stmtsContext) {
+ childs.add(visit((Simple_stmtsContext) child));
+ } else if (child instanceof Compound_stmtContext) {
+ childs.add(visit((Compound_stmtContext) child));
+ }
}
- return new RootNode(stmts, compStmts);
+ return new RootNode(childs);
}
/**
* Returns a `SimpleStmtsNode`, made by an array of SimpleStmtNode
*
- * ```
- * simple_stmts : simple_stmt (';' simple_stmt)* ';'? NEWLINE ;
- * ```
+ * ``` simple_stmts : simple_stmt (';' simple_stmt)* ';'? NEWLINE ; ```
*/
public Node visitSimple_stmts(Simple_stmtsContext ctx) {
ArrayList<Node> stmts = new ArrayList<Node>();
@@ -55,9 +54,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `CompoundNode`. It can be built by a different kind of
* statements, so only one of theme won't be null.
*
- * ```
- * compound_stmt : if_stmt | while_stmt | for_stmt | funcdef ;
- * ```
+ * ``` compound_stmt : if_stmt | while_stmt | for_stmt | funcdef ; ```
*/
public Node visitCompound_stmt(Compound_stmtContext ctx) {
Node ifStmt = null;
@@ -88,9 +85,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `SimpleStmtNode`. It can be built by a different kind of
* statements, so only one of theme won't be null.
*
- * ```
- * simple_stmt : assignment | expr | return_stmt | import_stm ;
- * ```
+ * ``` simple_stmt : assignment | expr | return_stmt | import_stm ; ```
*/
public Node visitSimple_stmt(Simple_stmtContext ctx) {
Node assignment = null;
@@ -120,10 +115,8 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns an `AssignmentNode`. It's made by left side, an assignment and a
* right side.
- *
- * ```
- * assignment : exprlist augassign exprlist ;
- * ```
+ *
+ * ``` assignment : exprlist augassign exprlist ; ```
*/
public Node visitAssignment(AssignmentContext ctx) {
Node lhr = visit(ctx.exprlist(0));
@@ -136,9 +129,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns a `ReturnStmtNode`. The returned exprlist can be null.
*
- * ```
- * return_stmt : 'return' exprlist? ;
- * ```
+ * ``` return_stmt : 'return' exprlist? ; ```
*/
public Node visitReturn_stmt(Return_stmtContext ctx) {
Node exprList = null;
@@ -153,10 +144,8 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `ImportNode`. An import can be made in different ways so we
* check the way in a module is imported (by from, by alias or by star).
*
- * ```
- * import_stm : 'import' dotted_name ('as' NAME)?
- * | 'from' dotted_name 'import' (NAME (',' NAME)* | '*') ;
- * ```
+ * ``` import_stm : 'import' dotted_name ('as' NAME)? | 'from' dotted_name
+ * 'import' (NAME (',' NAME)* | '*') ; ```
*/
public Node visitImport_stm(Import_stmContext ctx) {
boolean isFrom = ctx.FROM() != null;
@@ -177,9 +166,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns a `DottedNameNode` used in `import_stm`.
*
- * ```
- * dotted_name : NAME ('.' NAME)* ;
- * ```
+ * ``` dotted_name : NAME ('.' NAME)* ; ```
*/
public Node visitDotted_name(Dotted_nameContext ctx) {
ArrayList<TerminalNode> names = new ArrayList<TerminalNode>();
@@ -194,9 +181,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns a `FuncdefNode`. A paramlist can be null.
*
- * ```
- * funcdef : 'def' NAME '(' paramlist? ')' ':' block ;
- * ```
+ * ``` funcdef : 'def' NAME '(' paramlist? ')' ':' block ; ```
*/
public Node visitFuncdef(FuncdefContext ctx) {
Node paramlist = null;
@@ -214,9 +199,8 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `ParamlistNode`. We ignore the paramdef with default values
* (eg: is_used=False) because there is no test which uses this feature.
*
- * ```
- * paramlist : paramdef ('=' expr)? (',' paramdef ('=' expr)?)* ;
- *
+ * ``` paramlist : paramdef ('=' expr)? (',' paramdef ('=' expr)?)* ;
+ *
* ```
*/
public Node visitParamlist(ParamlistContext ctx) {
@@ -233,9 +217,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `ParamdefNode`. We ignore the paramdef with type annotation
* (eg: x : int) because there is no test which uses this feature.
*
- * ```
- * paramdef : NAME (':' expr)? ;
- * ```
+ * ``` paramdef : NAME (':' expr)? ; ```
*/
public Node visitParamdef(ParamdefContext ctx) {
return new ParamdefNode(ctx.NAME().toString());
@@ -245,11 +227,8 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns an `AugassignNode`. We don't provide all kinds of assignment
* below.
*
- * ```
- * augassign : '=' | '+=' | '-=' | '*=' | '@=' | '/=' | '%=' | '&=' | '|=' |
- * '^=' | '<<=' | '>>=' | '**=' | '//='
- * ;
- * ```
+ * ``` augassign : '=' | '+=' | '-=' | '*=' | '@=' | '/=' | '%=' | '&=' |
+ * '|=' | '^=' | '<<=' | '>>=' | '**=' | '//=' ; ```
*/
public Node visitAugassign(AugassignContext ctx) {
Node x = null;
@@ -262,20 +241,36 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
x = new AugassignNode(ctx.SUB_ASSIGN());
} else if (ctx.MULT_ASSIGN() != null) {
x = new AugassignNode(ctx.MULT_ASSIGN());
+ } else if (ctx.AT_ASSIGN() != null) {
+ x = new AugassignNode(ctx.AT_ASSIGN());
} else if (ctx.DIV_ASSIGN() != null) {
x = new AugassignNode(ctx.DIV_ASSIGN());
+ } else if (ctx.MOD_ASSIGN() != null) {
+ x = new AugassignNode(ctx.MOD_ASSIGN());
+ } else if (ctx.AND_ASSIGN() != null) {
+ x = new AugassignNode(ctx.AND_ASSIGN());
+ } else if (ctx.OR_ASSIGN() != null) {
+ x = new AugassignNode(ctx.OR_ASSIGN());
+ } else if (ctx.XOR_ASSIGN() != null) {
+ x = new AugassignNode(ctx.XOR_ASSIGN());
+ } else if (ctx.LEFT_SHIFT_ASSIGN() != null) {
+ x = new AugassignNode(ctx.LEFT_SHIFT_ASSIGN());
+ } else if (ctx.RIGHT_SHIFT_ASSIGN() != null) {
+ x = new AugassignNode(ctx.RIGHT_SHIFT_ASSIGN());
+ } else if (ctx.POWER_ASSIGN() != null) {
+ x = new AugassignNode(ctx.POWER_ASSIGN());
+ } else if (ctx.IDIV_ASSIGN() != null) {
+ x = new AugassignNode(ctx.IDIV_ASSIGN());
}
return x;
}
/**
- * Returns a `IfNode`.
- * FIXME: add support for elif statement.
+ * Returns a `IfNode`. FIXME: add support for elif statement.
*
- * ```
- * if_stmt : 'if' expr ':' block ('elif' expr ':' block)* ('else' ':' block)? ;
- * ```
+ * ``` if_stmt : 'if' expr ':' block ('elif' expr ':' block)* ('else' ':'
+ * block)? ; ```
*/
public Node visitIf_stmt(If_stmtContext ctx) {
var blocks = ctx.block();
@@ -292,9 +287,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns a `WhileStmtNode`. We do not provide 'else' branch.
*
- * ```
- * while_stmt : 'while' expr ':' block ('else' ':' block)? ;
- * ```
+ * ``` while_stmt : 'while' expr ':' block ('else' ':' block)? ; ```
*/
public Node visitWhile_stmt(While_stmtContext ctx) {
Node expr = visit(ctx.expr());
@@ -308,9 +301,7 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns a `ForSmtNode`. We do not provide 'else' branch.
*
- * ```
- * for_stmt : 'for' exprlist ':' block ('else' ':' block)? ;
- * ```
+ * ``` for_stmt : 'for' exprlist ':' block ('else' ':' block)? ; ```
*/
public Node visitFor_stmt(For_stmtContext ctx) {
Node exprList = visit(ctx.exprlist());
@@ -325,32 +316,30 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns a `BlockNode`. A block can be be a simple_stmts or a list of
* simple_stms and/or compound_stmt, so we just use a list for each kind.
*
- * ```
- * block : simple_stmts
- * | NEWLINE INDENT (simple_stmts | compound_stmt)+ DEDENT ;
- * ```
+ * ``` block : simple_stmts | NEWLINE INDENT (simple_stmts | compound_stmt)+
+ * DEDENT ; ```
*/
public Node visitBlock(BlockContext ctx) {
- ArrayList<Node> stmts = new ArrayList<Node>();
- ArrayList<Node> compStmts = new ArrayList<Node>();
+ ArrayList<Node> childs = new ArrayList<Node>();
- for (Simple_stmtsContext s : ctx.simple_stmts()) {
- stmts.add(visit(s));
- }
- for (Compound_stmtContext s : ctx.compound_stmt()) {
- compStmts.add(visit(s));
+ for (int i = 0; i < ctx.getChildCount(); i++) {
+ var child = ctx.getChild(i);
+
+ if (child instanceof Simple_stmtsContext) {
+ childs.add(visit((Simple_stmtsContext) child));
+ } else if (child instanceof Compound_stmtContext) {
+ childs.add(visit((Compound_stmtContext) child));
+ }
}
- return new BlockNode(stmts, compStmts);
+ return new BlockNode(childs);
}
/**
* Returns a `CompNode`. It should never be null.
*
- * ```
- * comp_op : '<' | '>' | '==' | '>=' | '<=' | '<>' | '!=' | 'in' | 'not' 'in' |
- * 'is' | 'is' 'not' ;
- * ```
+ * ``` comp_op : '<' | '>' | '==' | '>=' | '<=' | '<>' | '!=' | 'in' | 'not'
+ * 'in' | 'is' | 'is' 'not' ; ```
*/
public Node visitComp_op(Comp_opContext ctx) {
Node comp = null;
@@ -382,13 +371,11 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
* Returns an `ExprNode`. An expession can be a different kind of
* sub-expression. We do not provide all kinds of expr(s).
*
- * ```
- * expr : atom trailer* | expr '**' expr | ('+' | '-' | '~')+ expr | expr ('*' |
- * '@' | '/' | '%' | '//') expr | expr ('+' | '-') expr | expr ('<<' | '>>')
- * expr | expr '&' expr | expr '^' expr | expr '|' expr | 'not' expr | expr
- * comp_op expr | expr 'and' expr | expr 'or' expr | expr 'if' expr 'else' expr
- * ;
- * ```
+ * ``` expr : atom trailer* | expr '**' expr | ('+' | '-' | '~')+ expr |
+ * expr ('*' | '@' | '/' | '%' | '//') expr | expr ('+' | '-') expr | expr
+ * ('<<' | '>>') expr | expr '&' expr | expr '^' expr | expr '|' expr |
+ * 'not' expr | expr comp_op expr | expr 'and' expr | expr 'or' expr | expr
+ * 'if' expr 'else' expr ; ```
*/
public Node visitExpr(ExprContext ctx) {
Node atom = null;
@@ -438,41 +425,57 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
/**
* Returns an `AtomNode`.
- * FIXME: add support for testlist_comp
*
- * ```
- * atom : '(' testlist_comp? ')' | '[' testlist_comp? ']' | '{' testlist_comp?
- * '}' | NAME | NUMBER | STRING+ | '...' | 'None' | 'True' | 'False' ;
- * ```
+ * ``` atom : '(' testlist_comp? ')' | '[' testlist_comp? ']' | '{'
+ * testlist_comp? '}' | NAME | NUMBER | STRING+ | '...' | 'None' | 'True' |
+ * 'False' ; ```
*/
public Node visitAtom(AtomContext ctx) {
+ Testlist_compContext tlc = ctx.testlist_comp();
if (ctx.NUMBER() != null) {
- return new AtomNode(ctx.NUMBER().toString());
+ return new AtomNode(ctx.NUMBER().toString(), null);
+ } else if (ctx.NONE() != null) {
+ return new AtomNode(ctx.NONE().toString(), null);
} else if (ctx.TRUE() != null) {
- return new AtomNode(ctx.TRUE().toString());
+ return new AtomNode(ctx.TRUE().toString(), null);
} else if (ctx.FALSE() != null) {
- return new AtomNode(ctx.FALSE().toString());
+ return new AtomNode(ctx.FALSE().toString(), null);
} else if (ctx.NAME() != null) {
- return new AtomNode(ctx.NAME().toString());
- } else if (ctx.STRING() != null) {
-
+ return new AtomNode(ctx.NAME().toString(), null);
+ } else if (!ctx.STRING().isEmpty()) {
var varName = "";
for (var x : ctx.STRING()) {
varName += x;
}
+ return new AtomNode(varName, null);
+ } else if (ctx.OPEN_BRACE() != null && ctx.CLOSE_BRACE() != null) {
+ return manageCompListContext(tlc);
+ } else if (ctx.OPEN_BRACK() != null && ctx.CLOSE_BRACK() != null) {
+ return manageCompListContext(tlc);
+ } else if (ctx.OPEN_PAREN() != null && ctx.CLOSE_PAREN() != null) {
+ return manageCompListContext(tlc);
+ }
+ return new AtomNode(null, null);
+ }
- return new AtomNode(varName);
+ /**
+ * Supporting function for `visitAtom`. Returns an `AtomNode` with
+ * `testlist_comp` set if the context is not null. Otherwise, returns an
+ * `AtomNode` with nulls.
+ */
+ public AtomNode manageCompListContext(Testlist_compContext tlc) {
+ if (tlc != null) {
+ Node testlist_comp = visit(tlc);
+ return new AtomNode(null, testlist_comp);
}
- return new AtomNode(ctx.NONE().toString());
+ return new AtomNode(null, null);
}
/**
* Returns an `TrailerNode`.
*
- * ```
- * trailer : '(' arglist? ')' | '[' expr (',' expr)* ','? ']' | '.' NAME | '['
- * expr? ':' expr? (':' expr? )? ']' ;
- * ```
+ * ``` trailer : '(' arglist? ')' | '[' expr (',' expr)* ','? ']' | '.' NAME
+ * | '[' expr? ':' expr? (':' expr? )? ']' ; ```
*/
public Node visitTrailer(TrailerContext ctx) {
Node arglist = null;
@@ -491,29 +494,28 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
methodCall = ctx.NAME();
}
- return new TrailerNode(arglist, exprs, methodCall);
+ return new TrailerNode(arglist, exprs, methodCall, ctx.OPEN_PAREN() != null);
}
/**
- * Returns a `Node`.
- * FIXME: what to do in case of list??
+ * Returns a `Node`. FIXME: what to do in case of list??
*
- * ```
- * exprlist : expr (',' expr )* ','? ;
- * ```
+ * ``` exprlist : expr (',' expr )* ','? ; ```
*/
public Node visitExprlist(ExprlistContext ctx) {
- Node exp = visit(ctx.expr(0));
+ ArrayList<Node> exprlist = new ArrayList<Node>();
+
+ for (ExprContext c : ctx.expr()) {
+ exprlist.add(visit(c));
+ }
- return exp;
+ return new ExprListNode(exprlist);
}
/**
* Returns a `ArglistNode`.
*
- * ```
- * arglist : argument (',' argument)* ','? ;
- * ```
+ * ``` arglist : argument (',' argument)* ','? ; ```
*/
public Node visitArglist(ArglistContext ctx) {
ArrayList<Node> arguments = new ArrayList<Node>();
@@ -524,4 +526,56 @@ public class Python3VisitorImpl extends Python3ParserBaseVisitor<Node> {
return new ArglistNode(arguments);
}
+
+ /**
+ * Returns a `TestlistCompNode`.
+ *
+ * ``` testlist_comp : expr (comp_for | (',' expr)* ','?) ; ```
+ */
+ public Node visitTestlist_comp(Testlist_compContext ctx) {
+ ArrayList<Node> exprlist = new ArrayList<Node>();
+
+ for (ExprContext c : ctx.expr()) {
+ exprlist.add(visit(c));
+ }
+ Comp_forContext cfc = ctx.comp_for();
+ Node comp = null;
+ if (cfc != null) {
+ comp = visit(ctx.comp_for());
+ }
+ return new TestlistCompNode(exprlist, comp);
+ }
+
+ /**
+ * Returns a `CompForNode`.
+ *
+ * ``` comp_for : 'for' exprlist 'in' expr comp_iter? ;```
+ */
+ public Node visitComp_for(Comp_forContext ctx) {
+ Node exprlist = visit(ctx.exprlist());
+ Node expr = visit(ctx.expr());
+ Comp_iterContext cic = ctx.comp_iter();
+
+ if (cic != null) {
+ Node comp = visit(ctx.comp_iter());
+ return new CompForNode(exprlist, expr, comp);
+ }
+ return new CompForNode(exprlist, expr, null);
+ }
+
+ /**
+ * Returns a `CompIterNode`.
+ *
+ * ``` comp_iter : comp_for | comp_if ; ;```
+ */
+ public Node visitComp_iter(Comp_iterContext ctx) {
+ // TODO: Implement comp_if
+ // Node iter = visit(ctx.comp_if());
+ Comp_forContext cfc = ctx.comp_for();
+ Node forNode = null;
+ if (cfc != null) {
+ forNode = visit(ctx.comp_for());
+ }
+ return new CompIterNode(forNode);
+ }
}
diff --git a/src/ast/nodes/ArglistNode.java b/src/ast/nodes/ArglistNode.java
index f0c33e1..425e4c6 100644
--- a/src/ast/nodes/ArglistNode.java
+++ b/src/ast/nodes/ArglistNode.java
@@ -1,15 +1,16 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
+import java.util.Arrays;
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `arglist` statement of the grammar.
*/
public class ArglistNode implements Node {
+
protected ArrayList<Node> arguments;
public ArglistNode(ArrayList<Node> arguments) {
@@ -18,15 +19,40 @@ public class ArglistNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
for (var arg : arguments) {
- errors.addAll(arg.checkSemantics(ST, _nesting));
+ if (arg instanceof ExprNode) {
+ ExprNode argExpr = (ExprNode) arg;
+ String argName = argExpr.getId();
+
+ // TODO: check fucking IntType for params
+ // TODO: remove fucking comments
+ if (argName != null) {
+ if (Arrays.asList(bif).contains(argName)) {
+ continue;
+ }
+
+ if (ST.lookup(argName) != null && ST.lookup(argName).getType() instanceof ImportType) {
+ continue;
+ }
+
+ if (ST.nslookup(argName) < 0 && argExpr.typeCheck() instanceof AtomType) {
+ errors.add(new SemanticError("name '" + argName + "' is not defined."));
+ }
+ } else {
+ errors.addAll(arg.checkSemantics(ST, _nesting));
+ }
+ }
}
return errors;
}
+ public int getArgumentNumber() {
+ return arguments.size();
+ }
+
@Override
public Type typeCheck() {
return new VoidType();
diff --git a/src/ast/nodes/AssignmentNode.java b/src/ast/nodes/AssignmentNode.java
index 627e842..558e392 100644
--- a/src/ast/nodes/AssignmentNode.java
+++ b/src/ast/nodes/AssignmentNode.java
@@ -1,33 +1,48 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `assignment` statement of the grammar.
*/
public class AssignmentNode implements Node {
- private Node lhr;
- private Node assign;
- private Node rhr;
+
+ private final ExprListNode lhr;
+ private final Node assign;
+ private final ExprListNode rhr;
public AssignmentNode(Node lhr, Node assign, Node rhr) {
- this.lhr = lhr;
+ this.lhr = (ExprListNode) lhr;
this.assign = assign;
- this.rhr = rhr;
+ this.rhr = (ExprListNode) rhr;
}
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
- errors.addAll(lhr.checkSemantics(ST, _nesting));
+ // errors.addAll(lhr.checkSemantics(ST, _nesting));
errors.addAll(assign.checkSemantics(ST, _nesting));
errors.addAll(rhr.checkSemantics(ST, _nesting));
+ int lsize = lhr.getSize();
+
+ // FIXME: unused variable
+ // int rsize = rhr.getSize();
+ // if (lsize == rsize) {
+ for (int i = 0; i < lsize; i++) {
+ ExprNode latom = (ExprNode) lhr.getElem(i);
+ ST.insert(latom.getId(), new AtomType(), _nesting, "");
+ // ExprNode ratom = (ExprNode) rhr.getElem(i);
+ }
+ // } else {
+ // FIX: sgravata da piĆ¹ problemi che altro
+ // errors.add(new SemanticError("ValueError: different size of left or right side assignment"));
+ // }
+
return errors;
}
diff --git a/src/ast/nodes/AtomNode.java b/src/ast/nodes/AtomNode.java
index 4dcb926..47a15d7 100644
--- a/src/ast/nodes/AtomNode.java
+++ b/src/ast/nodes/AtomNode.java
@@ -1,30 +1,87 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
+import java.util.regex.Matcher;
+import java.util.regex.Pattern;
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `atom` statement of the grammar.
*/
public class AtomNode implements Node {
+
protected String val;
+ protected TestlistCompNode exprlist;
- public AtomNode(String val) {
+ public AtomNode(String val, Node exprlist) {
this.val = val;
+ this.exprlist = (TestlistCompNode) exprlist;
+ }
+
+ /**
+ * Returns the identifier of the `AtomNode` if it's not `null`, otherwise
+ * returns `null`.
+ */
+ public String getId() {
+ return this.val;
}
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- return new ArrayList<SemanticError>();
+ var errors = new ArrayList();
+
+ if (val != null) {
+ if ((this.typeCheck() instanceof AtomType) && ST.nslookup(this.getId()) < 0) {
+ errors.add(new SemanticError("name '" + this.getId() + "' is not defined."));
+ } else {
+ // System.out.println("exist " + this.typeCheck());
+ }
+ }
+
+ if (exprlist != null) {
+ errors.addAll(exprlist.checkSemantics(ST, _nesting));
+ }
+
+ return errors;
}
- // FIXME: this type for atom
+ // ENHANCE: return more specific types
@Override
public Type typeCheck() {
- return new AtomType();
+ if (this.val == null) {
+ return new VoidType();
+ }
+
+ Pattern noneVariable = Pattern.compile("^(None)$");
+ Pattern booleanVariable = Pattern.compile("^(True|False)$");
+ Pattern reservedWords = Pattern.compile("^(continue|break|int|float)$");
+ // this regex should match every possible atom name written in this format: CHAR
+ // (CHAR | DIGIT)*
+ Pattern simpleVariable = Pattern.compile("^[a-zA-Z][a-zA-Z0-9]*$", Pattern.CASE_INSENSITIVE);
+
+ Matcher noneVariableMatcher = noneVariable.matcher(this.val);
+ Matcher booleanVariableMatcher = booleanVariable.matcher(this.val);
+ Matcher reservedWordsMatcher = reservedWords.matcher(this.val);
+ Matcher simpleVariableMatcher = simpleVariable.matcher(this.val);
+
+ boolean matchFoundNone = noneVariableMatcher.find();
+ boolean matchFoundBoolean = booleanVariableMatcher.find();
+ boolean matchFoundContinueBreak = reservedWordsMatcher.find();
+ boolean matchFoundSimpleVariable = simpleVariableMatcher.find();
+
+ if (matchFoundBoolean) {
+ return new BoolType();
+ } else if (matchFoundContinueBreak) {
+ return new ReservedWordsType();
+ } else if (matchFoundNone) {
+ return new NoneType();
+ } else if (matchFoundSimpleVariable) {
+ return new AtomType(); // could be a variable or a fuction
+ } else {
+ return new VoidType(); // could be any type of data
+ }
}
// TODO: add code generation for atom node
@@ -35,6 +92,7 @@ public class AtomNode implements Node {
@Override
public String toPrint(String prefix) {
+ // FIXME: can be a testlist_comp with two expr and two atoms
if (val != null) {
return prefix + "Atom(" + val + ")\n";
}
diff --git a/src/ast/nodes/AugassignNode.java b/src/ast/nodes/AugassignNode.java
index 629fbcd..7ced63e 100644
--- a/src/ast/nodes/AugassignNode.java
+++ b/src/ast/nodes/AugassignNode.java
@@ -12,7 +12,8 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Node for the `augassign` statement of the grammar.
*/
public class AugassignNode implements Node {
- private TerminalNode val;
+
+ private final TerminalNode val;
public AugassignNode(TerminalNode val) {
this.val = val;
@@ -20,7 +21,7 @@ public class AugassignNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- return new ArrayList<SemanticError>();
+ return new ArrayList();
}
// FIXME: use the right type
diff --git a/src/ast/nodes/BlockNode.java b/src/ast/nodes/BlockNode.java
index 6b07f49..d9a151d 100644
--- a/src/ast/nodes/BlockNode.java
+++ b/src/ast/nodes/BlockNode.java
@@ -1,16 +1,29 @@
package ast.nodes;
-import java.util.ArrayList;
-
import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
/**
- * Node for `block` statement of the grammar.
- * It extends the `RootNode`.
+ * Node for `block` statement of the grammar. It extends the `RootNode`.
*/
public class BlockNode extends RootNode {
- public BlockNode(ArrayList<Node> stmts, ArrayList<Node> compoundStmts) {
- super(stmts, compoundStmts);
+
+ public BlockNode(ArrayList<Node> childs) {
+ super(childs);
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ // Check semantics for each child
+ for (Node child : childs) {
+ errors.addAll(child.checkSemantics(ST, _nesting));
+ }
+
+ return errors;
}
@Override
@@ -23,11 +36,8 @@ public class BlockNode extends RootNode {
String str = prefix + "Block\n";
prefix += " ";
- for (Node stmt : stmts) {
- str += stmt.toPrint(prefix);
- }
- for (Node stmt : compoundStmts) {
- str += stmt.toPrint(prefix);
+ for (Node child : childs) {
+ str += child.toPrint(prefix);
}
return str;
diff --git a/src/ast/nodes/CompForNode.java b/src/ast/nodes/CompForNode.java
new file mode 100644
index 0000000..b7c6924
--- /dev/null
+++ b/src/ast/nodes/CompForNode.java
@@ -0,0 +1,58 @@
+package ast.nodes;
+
+import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
+
+/**
+ * Node for the `comp_for` statement of the grammar. 'for' exprlist 'in' expr
+ * comp_iter?
+ */
+public class CompForNode implements Node {
+
+ protected ExprListNode exprlist;
+ protected ExprNode single_expr;
+ protected CompIterNode comp_iter;
+
+ public CompForNode(Node exprlist, Node single_expr, Node comp_iter) {
+ this.exprlist = (ExprListNode) exprlist;
+ this.single_expr = (ExprNode) single_expr;
+ this.comp_iter = (CompIterNode) comp_iter;
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ errors.addAll(exprlist.checkSemantics(ST, _nesting));
+ errors.addAll(single_expr.checkSemantics(ST, _nesting));
+ if (comp_iter != null) {
+ errors.addAll(comp_iter.checkSemantics(ST, _nesting));
+ }
+ return errors;
+ }
+
+ @Override
+ public Type typeCheck() {
+ return new VoidType();
+ }
+
+ // TODO: add code generation for arglist node
+ @Override
+ public String codeGeneration() {
+ return "";
+ }
+
+ @Override
+ public String toPrint(String prefix) {
+ String str = prefix + "CompForNode\n";
+
+ prefix += " ";
+ str += exprlist.toPrint(prefix);
+ str += single_expr.toPrint(prefix);
+ str += comp_iter.toPrint(prefix);
+ return str;
+ }
+
+}
diff --git a/src/ast/nodes/CompIterNode.java b/src/ast/nodes/CompIterNode.java
new file mode 100644
index 0000000..8de9f2f
--- /dev/null
+++ b/src/ast/nodes/CompIterNode.java
@@ -0,0 +1,50 @@
+package ast.nodes;
+
+import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
+
+/**
+ * Node for the `comp_iter` statement of the grammar.
+ */
+public class CompIterNode implements Node {
+
+ protected CompForNode comp_for;
+ // protected CompIfNode compIfNode;
+
+ public CompIterNode(Node comp_for) {
+ this.comp_for = (CompForNode) comp_for;
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ if (comp_for != null) {
+ errors.addAll(comp_for.checkSemantics(ST, _nesting));
+ }
+ return errors;
+ }
+
+ @Override
+ public Type typeCheck() {
+ return new VoidType();
+ }
+
+ // TODO: add code generation for arglist node
+ @Override
+ public String codeGeneration() {
+ return "";
+ }
+
+ @Override
+ public String toPrint(String prefix) {
+ String str = prefix + "CompIterNode\n";
+
+ prefix += " ";
+ str += comp_for.toPrint(prefix);
+ return str;
+ }
+
+}
diff --git a/src/ast/nodes/CompNode.java b/src/ast/nodes/CompNode.java
index 33976bf..22906df 100644
--- a/src/ast/nodes/CompNode.java
+++ b/src/ast/nodes/CompNode.java
@@ -11,7 +11,8 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Node for the `comp_op` statement of the grammar.
*/
public class CompNode implements Node {
- private TerminalNode op;
+
+ private final TerminalNode op;
public CompNode(TerminalNode op) {
this.op = op;
@@ -19,7 +20,7 @@ public class CompNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- return new ArrayList<SemanticError>();
+ return new ArrayList();
}
// TODO: it should be boolean, right?
diff --git a/src/ast/nodes/CompoundNode.java b/src/ast/nodes/CompoundNode.java
index e64be9e..845f05e 100644
--- a/src/ast/nodes/CompoundNode.java
+++ b/src/ast/nodes/CompoundNode.java
@@ -1,19 +1,19 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `compound_node` statement of the grammar.
*/
public class CompoundNode implements Node {
- private Node ifNode;
- private Node funcDef;
- private Node forStmt;
- private Node whileStmt;
+
+ private final Node ifNode;
+ private final Node funcDef;
+ private final Node forStmt;
+ private final Node whileStmt;
public CompoundNode(Node ifNode, Node funcDef, Node forStmt, Node whileStmt) {
this.ifNode = ifNode;
@@ -24,7 +24,7 @@ public class CompoundNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
if (ifNode != null) {
errors.addAll(ifNode.checkSemantics(ST, _nesting));
diff --git a/src/ast/nodes/DottedNameNode.java b/src/ast/nodes/DottedNameNode.java
index 46d1b61..23f14c1 100644
--- a/src/ast/nodes/DottedNameNode.java
+++ b/src/ast/nodes/DottedNameNode.java
@@ -11,6 +11,7 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Node for the `dooted_name` statement of the grammar.
*/
public class DottedNameNode implements Node {
+
protected ArrayList<TerminalNode> names;
public DottedNameNode(ArrayList<TerminalNode> names) {
@@ -19,14 +20,18 @@ public class DottedNameNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ for (int i = 0; i < names.size(); ++i) {
+ ST.insert(names.get(i).toString(), this.typeCheck(), _nesting, null);
+ }
return errors;
}
@Override
public Type typeCheck() {
- return new VoidType();
+ return new ImportType();
}
// NOTE: we do not provide code generation for this node in the same way
diff --git a/src/ast/nodes/ExprListNode.java b/src/ast/nodes/ExprListNode.java
new file mode 100644
index 0000000..4760db8
--- /dev/null
+++ b/src/ast/nodes/ExprListNode.java
@@ -0,0 +1,68 @@
+package ast.nodes;
+
+import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
+
+/**
+ * Node for the `expr_list` statement of the grammar.
+ */
+public class ExprListNode implements Node {
+
+ private final ArrayList<Node> exprs;
+
+ public ExprListNode(ArrayList<Node> exprs) {
+ this.exprs = exprs;
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList<>();
+
+ for (var expr : exprs) {
+ errors.addAll(expr.checkSemantics(ST, _nesting));
+ }
+
+ return errors;
+ }
+
+ public int getSize() {
+ return exprs.size();
+ }
+
+ /**
+ * Returns the i-th expressions of `exprs` field. If the index is greater or
+ * equals than the size return `null`.
+ */
+ public Node getElem(int i) {
+ if (i >= this.exprs.size()) {
+ return null;
+ }
+ return exprs.get(i);
+ }
+
+ @Override
+ public Type typeCheck() {
+ return new VoidType();
+ }
+
+ // TODO: code generation for expr list
+ @Override
+ public String codeGeneration() {
+ return "";
+ }
+
+ @Override
+ public String toPrint(String prefix) {
+ String str = prefix + "ExprList\n";
+
+ prefix += " ";
+ for (var param : exprs) {
+ str += param.toPrint(prefix);
+ }
+
+ return str;
+ }
+
+}
diff --git a/src/ast/nodes/ExprNode.java b/src/ast/nodes/ExprNode.java
index 25c2016..873d537 100644
--- a/src/ast/nodes/ExprNode.java
+++ b/src/ast/nodes/ExprNode.java
@@ -1,34 +1,97 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
+import java.util.Arrays;
+import semanticanalysis.STentry;
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `expr` statement of the grammar.
*/
public class ExprNode implements Node {
- private Node atom;
- private Node compOp;
- private String op;
- private ArrayList<Node> exprs;
- private ArrayList<Node> trailers;
+
+ private final AtomNode atom;
+ private final Node compOp;
+ private final String op;
+ private final ArrayList<Node> exprs;
+ private final ArrayList<Node> trailers;
public ExprNode(Node atom, Node compOp, ArrayList<Node> exprs, String op, ArrayList<Node> trailers) {
- this.atom = atom;
+ this.atom = (AtomNode) atom;
this.compOp = compOp;
this.exprs = exprs;
this.op = op;
this.trailers = trailers;
}
- @Override
- public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ /**
+ * Returns the i-th expressions of `exprs` field. If the index is greater or
+ * equals than the size return `null`.
+ */
+ public Node getExpr(int i) {
+ if (i >= this.exprs.size()) {
+ return null;
+ }
+ return this.exprs.get(i);
+ }
+
+ /**
+ * Returns the identifier of the `AtomNode` if it's not `null`, otherwise
+ * returns `null`.
+ */
+ public String getId() {
if (atom != null) {
+ return ((AtomNode) this.atom).getId();
+ } else {
+ return null;
+ }
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ // check if the atom is a function
+ if (atom != null && !trailers.isEmpty()) {
+
+ // check if the atom is not a built-in function
+ if (!Arrays.asList(bif).contains(atom.getId())) {
+
+ errors.addAll(atom.checkSemantics(ST, _nesting));
+
+ TrailerNode trailer = (TrailerNode) trailers.get(0);
+ String funName = atom.getId();
+
+ // TODO: it isnt a function, it could be a variable
+ STentry fun = ST.lookup(funName);
+
+ if (fun != null && !(fun.getType() instanceof ImportType)) {
+ if (!(fun.getType() instanceof FunctionType)) {
+ for (var t : trailers) {
+ errors.addAll(t.checkSemantics(ST, _nesting));
+ }
+
+ } else {
+ FunctionType ft = (FunctionType) fun.getType();
+ int paramNumber = ft.getParamNumber();
+ int argNumber = trailer.getArgumentNumber();
+
+ if (paramNumber != argNumber) {
+ errors.add(new SemanticError(funName + "() takes " + String.valueOf(paramNumber)
+ + " positional arguments but " + String.valueOf(argNumber) + " were given."));
+ }
+ }
+ }
+ } else {
+ for (var trailer : trailers) {
+ errors.addAll(trailer.checkSemantics(ST, _nesting));
+ }
+
+ }
+ } else if (atom != null) {
errors.addAll(atom.checkSemantics(ST, _nesting));
}
@@ -40,16 +103,16 @@ public class ExprNode implements Node {
errors.addAll(expr.checkSemantics(ST, _nesting));
}
- for (var trailer : trailers) {
- errors.addAll(trailer.checkSemantics(ST, _nesting));
- }
-
return errors;
}
// FIXME: type for the expr
@Override
public Type typeCheck() {
+ if (this.atom != null) {
+ return this.atom.typeCheck();
+ }
+
return new VoidType();
}
@@ -73,11 +136,11 @@ public class ExprNode implements Node {
}
for (var expr : exprs) {
- str += expr.toPrint(prefix);
+ str = expr.toPrint(prefix);
}
for (var trailer : trailers) {
- str += trailer.toPrint(prefix);
+ str = trailer.toPrint(prefix);
}
if (op != null) {
diff --git a/src/ast/nodes/ForStmtNode.java b/src/ast/nodes/ForStmtNode.java
index 6d94bb2..74c6ffc 100644
--- a/src/ast/nodes/ForStmtNode.java
+++ b/src/ast/nodes/ForStmtNode.java
@@ -1,26 +1,56 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `for_stmt` statement of the grammar.
*/
public class ForStmtNode implements Node {
- private Node exprList;
- private Node block;
+
+ private final Node exprList;
+ private final Node block;
public ForStmtNode(Node exprList, Node block) {
this.exprList = exprList;
this.block = block;
}
+ /**
+ * This methods check the semantics of the operation `for i in list: block`.
+ *
+ * After the `for` keyword, it's parsed as a list of expression. We verified
+ * that the last expression is always gonna be `expr comp_op expr`. We
+ * comp_op must be the `in` keyword. `i` could be of any lenght (e.g. `a, b,
+ * c`).
+ *
+ * In this function we define the following approch: we save in the
+ * SymbolTable every atom until the last expression in the expression's
+ * list. For the last element, we know that is in the `expr comp_op expr`
+ * format, so we take the left element and save it in the SymbolicTable as
+ * an atom.
+ */
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ // Save every atom in the expression's list, except the last one
+ var l = (ExprListNode) exprList;
+ for (int i = 0; i < l.getSize() - 1; ++i) {
+ var e = (ExprNode) l.getElem(i);
+ ST.insert(e.getId(), e.typeCheck(), _nesting, "");
+ }
+
+ // Manage the last expression of expression's list
+ // Get the ExprNode
+ var left = (ExprNode) l.getElem(l.getSize() - 1);
+ // Get the left side expression of the operation, that
+ // corresponds to the first element of the expression list.
+ var atomLeft = (ExprNode) left.getExpr(0);
+ // ENHANCE: check that the comp_op is the `in` keyword
+ ST.insert(atomLeft.getId(), atomLeft.typeCheck(), _nesting, "");
errors.addAll(exprList.checkSemantics(ST, _nesting));
errors.addAll(block.checkSemantics(ST, _nesting));
diff --git a/src/ast/nodes/FuncdefNode.java b/src/ast/nodes/FuncdefNode.java
index 67bc772..c4f5846 100644
--- a/src/ast/nodes/FuncdefNode.java
+++ b/src/ast/nodes/FuncdefNode.java
@@ -1,7 +1,9 @@
package ast.nodes;
import java.util.ArrayList;
+import java.util.HashMap;
+import semanticanalysis.STentry;
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
import ast.types.*;
@@ -11,9 +13,10 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Node for the `funcdef` statement of the grammar.
*/
public class FuncdefNode implements Node {
- private TerminalNode name;
- private Node paramlist;
- private Node block;
+
+ private final TerminalNode name;
+ private final Node paramlist;
+ private final Node block;
public FuncdefNode(TerminalNode name, Node paramlist, Node block) {
this.name = name;
@@ -23,13 +26,30 @@ public class FuncdefNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
+ int paramNumber = ((ParamlistNode) paramlist).getParamNumber();
+ Type returnType = this.block.typeCheck();
+ FunctionType ft = new FunctionType(paramNumber, returnType);
+
+ ST.insert(this.name.toString(), ft, _nesting, "");
+
+ HashMap<String, STentry> HM = new HashMap();
+
+ ST.add(HM);
+
+ ST.insert(this.name.toString(), ft, _nesting + 1, "");
if (paramlist != null) {
- errors.addAll(paramlist.checkSemantics(ST, _nesting));
+ errors.addAll(paramlist.checkSemantics(ST, _nesting + 1));
}
- errors.addAll(block.checkSemantics(ST, _nesting));
+ // TODO: think to the fucking offset
+ // Offset is increased for the possible return value
+ ST.increaseoffset();
+
+ errors.addAll(block.checkSemantics(ST, _nesting + 1));
+
+ ST.remove();
return errors;
}
@@ -46,6 +66,7 @@ public class FuncdefNode implements Node {
return "";
}
+ @Override
public String toPrint(String prefix) {
String str = prefix + "Funcdef(" + name + ")\n";
diff --git a/src/ast/nodes/IfNode.java b/src/ast/nodes/IfNode.java
index 50dde4a..b0e1ddb 100644
--- a/src/ast/nodes/IfNode.java
+++ b/src/ast/nodes/IfNode.java
@@ -1,18 +1,18 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `if` statement of the grammar.
*/
public class IfNode implements Node {
- private Node guard;
- private Node thenbranch;
- private Node elsebranch;
+
+ private final Node guard;
+ private final Node thenbranch;
+ private final Node elsebranch;
public IfNode(Node guard, Node thenbranch, Node elsebranch) {
this.guard = guard;
@@ -22,7 +22,7 @@ public class IfNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
errors.addAll(guard.checkSemantics(ST, _nesting));
errors.addAll(thenbranch.checkSemantics(ST, _nesting));
@@ -39,14 +39,14 @@ public class IfNode implements Node {
if (guard.typeCheck() instanceof BoolType) {
Type thenexp = thenbranch.typeCheck();
Type elseexp = elsebranch.typeCheck();
- if (thenexp.getClass().equals(elseexp.getClass()))
- return thenexp;
- else {
- System.out.println("Type Error: incompatible types in then and else branches");
+ if (thenexp.getClass().equals(elseexp.getClass())) {
+ return thenexp;
+ }else {
+ System.out.println("Type Error: incompatible types in then and else branches.");
return new ErrorType();
}
} else {
- System.out.println("Type Error: non boolean condition in if");
+ System.out.println("Type Error: non boolean condition in if.");
return new ErrorType();
}
}
diff --git a/src/ast/nodes/ImportNode.java b/src/ast/nodes/ImportNode.java
index e264c99..7e95ee3 100644
--- a/src/ast/nodes/ImportNode.java
+++ b/src/ast/nodes/ImportNode.java
@@ -1,20 +1,20 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `import_stmt` statement of the grammar.
*/
public class ImportNode implements Node {
- private Node dottedName;
- private boolean isFrom;
- private boolean importAs;
- private boolean importAll;
- private ArrayList<String> names;
+
+ private final Node dottedName;
+ private final boolean isFrom;
+ private final boolean importAs;
+ private final boolean importAll;
+ private final ArrayList<String> names;
public ImportNode(Node dottedName, boolean isFrom, boolean importAs, boolean importAll,
ArrayList<String> names) {
@@ -27,14 +27,26 @@ public class ImportNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
+
+ if (isFrom) {
+ for (int i = 0; i < names.size(); ++i) {
+ ST.insert(names.get(i), this.typeCheck(), _nesting, null);
+ }
+ } else {
+ errors.addAll(dottedName.checkSemantics(ST, _nesting));
+ }
+
+ if (importAs) {
+ ST.insert(names.get(names.size() - 1), this.typeCheck(), _nesting, null);
+ }
return errors;
}
@Override
public Type typeCheck() {
- return new VoidType();
+ return new ImportType();
}
// NOTE: we do not want to provide a code generation for this statement
@@ -63,8 +75,9 @@ public class ImportNode implements Node {
}
for (int i = 0; i < names.size(); ++i) {
- if (i == 0 && importAs)
+ if (i == 0 && importAs) {
continue;
+ }
str += prefix + names.get(i) + "\n";
}
diff --git a/src/ast/nodes/Node.java b/src/ast/nodes/Node.java
index 949531e..9fb58b8 100644
--- a/src/ast/nodes/Node.java
+++ b/src/ast/nodes/Node.java
@@ -1,19 +1,92 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Base interface for a Node.
*/
public interface Node {
+
+ static final String[] bif = {
+ "abs",
+ "aiter",
+ "all",
+ "anext",
+ "any",
+ "ascii",
+ "bin",
+ "bool",
+ "breakpoint",
+ "bytearray",
+ "bytes",
+ "callable",
+ "chr",
+ "classmethod",
+ "compile",
+ "complex",
+ "delattr",
+ "dict",
+ "dir",
+ "divmod",
+ "enumerate",
+ "eval",
+ "exec",
+ "exit",
+ "filter",
+ "float",
+ "format",
+ "frozenset",
+ "getattr",
+ "globals",
+ "hasattr",
+ "hash",
+ "help",
+ "hex",
+ "id",
+ "input",
+ "int",
+ "isinstance",
+ "issubclass",
+ "iter",
+ "len",
+ "list",
+ "locals",
+ "map",
+ "max",
+ "memoryview",
+ "min",
+ "next",
+ "object",
+ "oct",
+ "open",
+ "ord",
+ "pow",
+ "print",
+ "property",
+ "range",
+ "repr",
+ "reversed",
+ "round",
+ "set",
+ "setattr",
+ "slice",
+ "sorted",
+ "staticmethod",
+ "str",
+ "sum",
+ "super",
+ "tuple",
+ "type",
+ "vars",
+ "zip",
+ "__import__"};
+
/**
* Checks semantics for a given node for a SymbolTable ST and a level of
- * nesting.
- * Returns a list of `SemanticError`.
+ * nesting. Returns a list of `SemanticError`.
*/
ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting);
@@ -29,8 +102,8 @@ public interface Node {
String codeGeneration();
/**
- * Returns a string for a given node with a prefix.
- * It used when an AST wants to be visualized on screen.
+ * Returns a string for a given node with a prefix. It used when an AST
+ * wants to be visualized on screen.
*/
String toPrint(String prefix);
}
diff --git a/src/ast/nodes/ParamdefNode.java b/src/ast/nodes/ParamdefNode.java
index 481282e..6cdb2d0 100644
--- a/src/ast/nodes/ParamdefNode.java
+++ b/src/ast/nodes/ParamdefNode.java
@@ -1,14 +1,32 @@
package ast.nodes;
import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
/**
* Node for the `paramdef` statement of the grammar. Extends the `AtomNode`
* class.
*/
public class ParamdefNode extends AtomNode {
+
public ParamdefNode(String val) {
- super(val);
+ super(val, null);
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ var errors = new ArrayList();
+ String paramName = this.getId();
+
+ if (!ST.top_lookup(paramName)) {
+ ST.insert(paramName, this.typeCheck(), _nesting, "");
+ } else {
+ errors.add(new SemanticError("Duplicate argument '" + paramName + "' in function definition."));
+ }
+
+ return errors;
}
// FIXME: it should returns the param' type
diff --git a/src/ast/nodes/ParamlistNode.java b/src/ast/nodes/ParamlistNode.java
index 144eb13..cf75d08 100644
--- a/src/ast/nodes/ParamlistNode.java
+++ b/src/ast/nodes/ParamlistNode.java
@@ -1,16 +1,16 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `param_list` statement of the grammar.
*/
public class ParamlistNode implements Node {
- private ArrayList<Node> params;
+
+ private final ArrayList<Node> params;
public ParamlistNode(ArrayList<Node> _params) {
params = _params;
@@ -18,7 +18,7 @@ public class ParamlistNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
for (var param : params) {
errors.addAll(param.checkSemantics(ST, _nesting));
@@ -27,6 +27,10 @@ public class ParamlistNode implements Node {
return errors;
}
+ public int getParamNumber() {
+ return params.size();
+ }
+
@Override
public Type typeCheck() {
return new VoidType();
diff --git a/src/ast/nodes/ReturnStmtNode.java b/src/ast/nodes/ReturnStmtNode.java
index 0f2dd74..8e34813 100644
--- a/src/ast/nodes/ReturnStmtNode.java
+++ b/src/ast/nodes/ReturnStmtNode.java
@@ -1,16 +1,16 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `return_stmt` statement of the grammar.
*/
public class ReturnStmtNode implements Node {
- private Node exprList;
+
+ private final Node exprList;
public ReturnStmtNode(Node exprList) {
this.exprList = exprList;
@@ -18,7 +18,7 @@ public class ReturnStmtNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
if (this.exprList != null) {
errors.addAll(this.exprList.checkSemantics(ST, _nesting));
diff --git a/src/ast/nodes/RootNode.java b/src/ast/nodes/RootNode.java
index e0989e8..5bbce8c 100644
--- a/src/ast/nodes/RootNode.java
+++ b/src/ast/nodes/RootNode.java
@@ -1,36 +1,41 @@
package ast.nodes;
-import java.util.ArrayList;
-
-import semanticanalysis.SemanticError;
-import semanticanalysis.SymbolTable;
import ast.types.*;
+import java.util.ArrayList;
+import java.util.HashMap;
+import semanticanalysis.*;
/**
* Node for the `root` statement of the grammar.
*/
public class RootNode implements Node {
+
// stms and compundStmts are protected because they are reused for a
// BlockNode
- protected ArrayList<Node> stmts;
- protected ArrayList<Node> compoundStmts;
+ protected ArrayList<Node> childs;
- public RootNode(ArrayList<Node> stmts, ArrayList<Node> compoundStmts) {
- this.stmts = stmts;
- this.compoundStmts = compoundStmts;
+ public RootNode(ArrayList<Node> childs) {
+ this.childs = childs;
}
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
- for (Node stmt : stmts) {
- errors.addAll(stmt.checkSemantics(ST, _nesting));
- }
- for (Node stmt : compoundStmts) {
- errors.addAll(stmt.checkSemantics(ST, _nesting));
+ // Create a new HashMap for the current scope
+ HashMap<String, STentry> HM = new HashMap();
+
+ // Add the HashMap to the SymbolTable
+ ST.add(HM);
+
+ // Check semantics for each child
+ for (Node child : childs) {
+ errors.addAll(child.checkSemantics(ST, _nesting));
}
+ // Remove the HashMap from the SymbolTable
+ ST.remove();
+
return errors;
}
@@ -51,11 +56,8 @@ public class RootNode implements Node {
prefix += " ";
- for (Node stmt : stmts) {
- str += stmt.toPrint(prefix);
- }
- for (Node stmt : compoundStmts) {
- str += stmt.toPrint(prefix);
+ for (Node child : childs) {
+ str += child.toPrint(prefix);
}
return str;
diff --git a/src/ast/nodes/SimpleStmtNode.java b/src/ast/nodes/SimpleStmtNode.java
index 5f844cb..b2bc880 100644
--- a/src/ast/nodes/SimpleStmtNode.java
+++ b/src/ast/nodes/SimpleStmtNode.java
@@ -1,19 +1,19 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `simple_stmt` statement of the grammar.
*/
public class SimpleStmtNode implements Node {
- private Node assignment;
- private Node expr;
- private Node returnStmt;
- private Node importStmt;
+
+ private final Node assignment;
+ private final Node expr;
+ private final Node returnStmt;
+ private final Node importStmt;
public SimpleStmtNode(Node assignment, Node expr, Node returnStmt, Node importStmt) {
this.assignment = assignment;
@@ -24,7 +24,7 @@ public class SimpleStmtNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
if (assignment != null) {
errors.addAll(assignment.checkSemantics(ST, _nesting));
diff --git a/src/ast/nodes/SimpleStmtsNode.java b/src/ast/nodes/SimpleStmtsNode.java
index 66c8e2c..ae1044b 100644
--- a/src/ast/nodes/SimpleStmtsNode.java
+++ b/src/ast/nodes/SimpleStmtsNode.java
@@ -1,16 +1,16 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `simple_stmts` statement of the grammar.
*/
public class SimpleStmtsNode implements Node {
- private ArrayList<Node> stmts;
+
+ private final ArrayList<Node> stmts;
public SimpleStmtsNode(ArrayList<Node> stmts) {
this.stmts = stmts;
@@ -18,7 +18,7 @@ public class SimpleStmtsNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
for (Node stmt : stmts) {
errors.addAll(stmt.checkSemantics(ST, _nesting));
diff --git a/src/ast/nodes/TestlistCompNode.java b/src/ast/nodes/TestlistCompNode.java
new file mode 100644
index 0000000..32049f5
--- /dev/null
+++ b/src/ast/nodes/TestlistCompNode.java
@@ -0,0 +1,82 @@
+package ast.nodes;
+
+import ast.types.*;
+import java.util.ArrayList;
+import semanticanalysis.SemanticError;
+import semanticanalysis.SymbolTable;
+
+/**
+ * Node for the `testlist_comp` statement of the grammar.
+ */
+public class TestlistCompNode implements Node {
+
+ private final ArrayList<Node> exprs;
+ private final CompForNode comp;
+
+ public TestlistCompNode(ArrayList<Node> exprs, Node comp) {
+ this.exprs = exprs;
+ this.comp = (CompForNode) comp;
+ }
+
+ @Override
+ public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
+ ArrayList<SemanticError> errors = new ArrayList<>();
+
+ if (comp != null) {
+ // if comp is set, then we save the atom in the ST (we assume the first expr is
+ // an atom)
+ String id = ((ExprNode) exprs.get(0)).getId();
+ Type t = ((ExprNode) exprs.get(0)).typeCheck();
+ ST.insert(id, t, _nesting, "");
+ // errors.addAll(comp.checkSemantics(ST, _nesting));
+ } else {
+ // if comp is not set, then exprs is a list of 1 or more element
+ for (var param : exprs) {
+ var exp = (ExprNode) param;
+ ST.insert(exp.getId(), exp.typeCheck(), _nesting, "");
+ errors.addAll(param.checkSemantics(ST, _nesting));
+ }
+ }
+
+ return errors;
+ }
+
+ public int getSize() {
+ return exprs.size();
+ }
+
+ /**
+ * Returns the i-th expressions of `exprs` field. If the index is greater or
+ * equals than the size return `null`.
+ */
+ public Node getElem(int i) {
+ if (i >= this.exprs.size()) {
+ return null;
+ }
+ return exprs.get(i);
+ }
+
+ @Override
+ public Type typeCheck() {
+ return new VoidType();
+ }
+
+ // TODO: code generation for expr list
+ @Override
+ public String codeGeneration() {
+ return "";
+ }
+
+ @Override
+ public String toPrint(String prefix) {
+ String str = prefix + "Testlist_comp\n";
+
+ prefix += " ";
+ for (var param : exprs) {
+ str += param.toPrint(prefix);
+ }
+
+ return str;
+ }
+
+}
diff --git a/src/ast/nodes/TrailerNode.java b/src/ast/nodes/TrailerNode.java
index b0a0ee7..aaa72f3 100644
--- a/src/ast/nodes/TrailerNode.java
+++ b/src/ast/nodes/TrailerNode.java
@@ -11,22 +11,25 @@ import org.antlr.v4.runtime.tree.TerminalNode;
* Node for the `trailer` statement of the grammar.
*/
public class TrailerNode implements Node {
- private Node arglist;
- private ArrayList<Node> exprs;
- private TerminalNode methodCall;
- private boolean isEmpty;
- public TrailerNode(Node arglist, ArrayList<Node> exprs, TerminalNode methodCall) {
+ private final Node arglist;
+ private final ArrayList<Node> exprs;
+ private final TerminalNode methodCall;
+ private final boolean isParenthesis;
+ private final boolean isEmpty;
+
+ public TrailerNode(Node arglist, ArrayList<Node> exprs, TerminalNode methodCall, boolean isParenthesis) {
this.arglist = arglist;
this.exprs = exprs;
this.methodCall = methodCall;
+ this.isParenthesis = isParenthesis;
- this.isEmpty = (this.arglist == null && this.exprs.size() == 0 && this.methodCall == null);
+ this.isEmpty = (this.arglist == null && this.exprs.isEmpty() && this.methodCall == null);
}
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
if (arglist != null) {
errors.addAll(arglist.checkSemantics(ST, _nesting));
@@ -39,6 +42,18 @@ public class TrailerNode implements Node {
return errors;
}
+ public int getArgumentNumber() {
+ if (arglist == null) {
+ return 0;
+ }
+
+ return ((ArglistNode) arglist).getArgumentNumber();
+ }
+
+ public boolean isParenthesis() {
+ return this.isParenthesis;
+ }
+
@Override
public Type typeCheck() {
return new VoidType();
diff --git a/src/ast/nodes/WhileStmtNode.java b/src/ast/nodes/WhileStmtNode.java
index 6353ec1..1db01ea 100644
--- a/src/ast/nodes/WhileStmtNode.java
+++ b/src/ast/nodes/WhileStmtNode.java
@@ -1,17 +1,17 @@
package ast.nodes;
+import ast.types.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.types.*;
/**
* Node for the `while_stmt` statement of the grammar.
*/
public class WhileStmtNode implements Node {
- private Node expr;
- private Node block;
+
+ private final Node expr;
+ private final Node block;
public WhileStmtNode(Node expr, Node block) {
this.expr = expr;
@@ -20,7 +20,7 @@ public class WhileStmtNode implements Node {
@Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
- ArrayList<SemanticError> errors = new ArrayList<SemanticError>();
+ ArrayList<SemanticError> errors = new ArrayList();
errors.addAll(expr.checkSemantics(ST, _nesting));
errors.addAll(block.checkSemantics(ST, _nesting));
diff --git a/src/ast/types/AtomType.java b/src/ast/types/AtomType.java
index 0387ec1..fc1c69e 100644
--- a/src/ast/types/AtomType.java
+++ b/src/ast/types/AtomType.java
@@ -2,9 +2,10 @@ package ast.types;
/**
* An tom type.
- * TODO: do I need to use this one?
*/
public class AtomType extends Type {
+
+ @Override
public String toPrint(String prefix) {
return prefix + "Atom\n";
}
diff --git a/src/ast/types/BoolType.java b/src/ast/types/BoolType.java
index 20c2750..01e2cd5 100644
--- a/src/ast/types/BoolType.java
+++ b/src/ast/types/BoolType.java
@@ -4,6 +4,8 @@ package ast.types;
* A boolean type. A bool is True or False.
*/
public class BoolType extends Type {
+
+ @Override
public String toPrint(String prefix) {
return prefix + "Bool\n";
}
diff --git a/src/ast/types/ErrorType.java b/src/ast/types/ErrorType.java
index 4a7a0cf..71176d1 100644
--- a/src/ast/types/ErrorType.java
+++ b/src/ast/types/ErrorType.java
@@ -4,6 +4,8 @@ package ast.types;
* Error type.
*/
public class ErrorType extends Type {
+
+ @Override
public String toPrint(String prefix) {
return prefix + "Error\n";
}
diff --git a/src/ast/types/FunctionType.java b/src/ast/types/FunctionType.java
new file mode 100644
index 0000000..5e6adaa
--- /dev/null
+++ b/src/ast/types/FunctionType.java
@@ -0,0 +1,29 @@
+package ast.types;
+
+/**
+ * A Function type.
+ */
+public class FunctionType extends Type {
+
+ private final int paramNumber;
+ private final Type returnType;
+
+ public FunctionType(int paramNumber, Type returnType) {
+ this.paramNumber = paramNumber;
+ this.returnType = returnType;
+ }
+
+ // Return the length of the parameters
+ public int getParamNumber() {
+ return paramNumber;
+ }
+
+ public Type getReturnType() {
+ return returnType;
+ }
+
+ @Override
+ public String toPrint(String prefix) {
+ return prefix + "Function\n";
+ }
+}
diff --git a/src/ast/types/ImportType.java b/src/ast/types/ImportType.java
new file mode 100644
index 0000000..892de10
--- /dev/null
+++ b/src/ast/types/ImportType.java
@@ -0,0 +1,12 @@
+package ast.types;
+
+/**
+ * A type for the imported names.
+ */
+public class ImportType extends Type {
+
+ @Override
+ public String toPrint(String prefix) {
+ return prefix + "Import\n";
+ }
+}
diff --git a/src/ast/types/IntType.java b/src/ast/types/IntType.java
index a29518b..6483d93 100644
--- a/src/ast/types/IntType.java
+++ b/src/ast/types/IntType.java
@@ -4,6 +4,8 @@ package ast.types;
* An integer type.
*/
public class IntType extends Type {
+
+ @Override
public String toPrint(String prefix) {
return prefix + "Int\n";
}
diff --git a/src/ast/types/NoneType.java b/src/ast/types/NoneType.java
new file mode 100644
index 0000000..730add9
--- /dev/null
+++ b/src/ast/types/NoneType.java
@@ -0,0 +1,17 @@
+package ast.types;
+
+/**
+ * A none type. None return unit.
+ */
+public class NoneType extends Type {
+
+ @Override
+ public String toPrint(String prefix) {
+ return prefix + "None\n";
+ }
+
+ @Override
+ public String toString() {
+ return "None";
+ }
+}
diff --git a/src/ast/types/ReservedWordsType.java b/src/ast/types/ReservedWordsType.java
new file mode 100644
index 0000000..da72890
--- /dev/null
+++ b/src/ast/types/ReservedWordsType.java
@@ -0,0 +1,12 @@
+package ast.types;
+
+/**
+ * A type for the continue and break statements.
+ */
+public class ReservedWordsType extends Type {
+
+ @Override
+ public String toPrint(String prefix) {
+ return prefix + "ReservedWords\n";
+ }
+}
diff --git a/src/ast/types/Type.java b/src/ast/types/Type.java
index 6bff8b9..6190106 100644
--- a/src/ast/types/Type.java
+++ b/src/ast/types/Type.java
@@ -1,26 +1,25 @@
package ast.types;
+import ast.nodes.*;
import java.util.ArrayList;
-
import semanticanalysis.SemanticError;
import semanticanalysis.SymbolTable;
-import ast.nodes.*;
/**
* A node which represents a type class.
*/
public class Type implements Node {
+
public boolean isEqual(Type A, Type B) {
- if (A.getClass().equals(B.getClass()))
- return true;
- else
- return false;
+ return A.getClass().equals(B.getClass());
}
+ @Override
public String toPrint(String s) {
return s;
}
+ @Override
public ArrayList<SemanticError> checkSemantics(SymbolTable ST, int _nesting) {
// It is never invoked
return null;
diff --git a/src/ast/types/VoidType.java b/src/ast/types/VoidType.java
index d15933f..2d21da6 100644
--- a/src/ast/types/VoidType.java
+++ b/src/ast/types/VoidType.java
@@ -4,7 +4,14 @@ package ast.types;
* A void type. Voids return nothing.
*/
public class VoidType extends Type {
+
+ @Override
public String toPrint(String prefix) {
return prefix + "Void\n";
}
+
+ @Override
+ public String toString() {
+ return "Void";
+ }
}