🍊 Orange Compilerβ„’

The Orange Compilerβ„’ is a simple compiler used alongside the Orange Languageβ„’ to teach the inner workings of a compiler in a simple and enjoyable manner. It has a small hint of data analysis as an added bonus.

πŸ“ Checkpoints

Checkpoints are given by our professor, although they can change during the semester and they are more like 'symbolic' checkpoints to help us complete the project. These checkpoints represent the core components of the compiler. It is strongly recommended to follow the order of the checkpoints since they follow the class syllabus and the compiler is built in a secuential manner (cannot skip to a further component without finishing previous ones).

βœ… Checkpoint #0

  • Project proposal
  • Token list
  • Syntax diagrams

βœ… Checkpoint #1: Lexical and Syntactical Analysis

βœ… Checkpoint #2: Basic variable semantics

  • Procedure directory
  • Variable table

βœ… Checkpoint #3: Expression semantics

  • Semantic cube
  • Code generation for arithmetic expressions and secuential statutes
  • Assignment
  • Read
  • Write
  • Etc.

βœ… Checkpoint #4: Code generation for conditional statutes

  • Decisions
  • Cycles
    • WHILE
    • DO WHILE
    • FOR

βœ… Checkpoint #5: Code generation for functions

  • Function definition
    • VOID
    • TYPED
  • Function call

βœ… Checkpoint #6: Memory map for virtual machine

  • Memory management
  • Execution of arithmetic expressions
  • Secuential statutes

βœ… Checkpoint #7: Code generation for arrays and structured types

  • Execution of conditional statutes

⚠️ Checkpoint #8: First version of documentation

  • For a specific section
    • Code generation
    • Virtual machine

πŸ—’οΈ Log

πŸ“† August 23 πŸ•“ 23:13

Created repository. Started planning and design. Started checkpoint #0.


πŸ“† August 26 πŸ•“ 17:22

Made list with tokens.


πŸ“† August 29 πŸ•“ 19:03

Finished checkpoint #0 (proposal, token list and syntax diagrams). Things can still be changed or added, depending on teacher's feedback (more special functions, more structured types, etc).


πŸ“† August 30 πŸ•“ 20:22

Designing and documenting test for the Orange Compiler.


πŸ“† September 01 πŸ•“ 15:29

Minor changes to syntax diagrams and proposal document. Added first input files to testing folder.


πŸ“† September 02 πŸ•“ 23:16

Finished first version of parser. Instead of staying with PLY, I gave SLY a go. I felt that it has a little bit better syntax, and in the worst case scenario, I don't think its that complicated to change it to PLY again. I finished the parser based on the LittleDuck homework, and my proposal's syntax diagrams. As I was making the parser, I noticed several areas where problems my arise (reduction conflicts, redundance, etc.), but this version only serves as a backup while I keep working on fixing said issues. Also, I still don't know why Git/Github is not signing my commits.


πŸ“† September 05 πŸ•“ 17:10

Solved a reduce/reduce conflict with the block contents. I have a slight suspicion that the same issue might happen again with other rules, but the fix is very simple. For the block contents I had the rules:

blockcontents -> statute | statute blockcontents | empty

And all I did was remove the 'statute' rule. It fixes the conflict because now if there is only one statute in the block, it can exit through the empty rule, instead of having to decide.

I think this will happen again because I have a vague memory of doing something similar in other rules, but I believe those problems will become easily visible if I keep making more tests to poke at the grammar rules.


πŸ“† September 05 πŸ•“ 19:35

Watched recorded class (September 20) where the teacher talks about semantics. This recording was needed to begin writing basic variable semantics and the development of a procedure directory and a variable table.


πŸ“† September 06 πŸ•“ 16:48

Watched recorded class (September 23) where the teacher talks about semantics and focuses more on development and refinement of the function directory/variable table. Started development of function directory and variable table components.


πŸ“† September 09 πŸ•“ 23:38

Main program now accepts arguments to run/test the scanner, the parser, or both. Finished an implementation for function directory and variable table. When the parser enters a new function, I added a rule BEFORE the current rule ends where the current context is switched. This means that for a rule like < MAIN LPAREN RPAREN block > I added the rule < MAIN changecontext LPAREN RPAREN block > so that the context changes for everything inside the main block and not everything after it. The implementation works by creating an object for the function directory and an object for the variable table. When the parser passes through a new context, the variable table is cleared. All the variables declared in said context are added to the variable table, and then the whole variable table is added to the current function/context in the function directory. I also added some tags to identify important parts in the project. I still to make sure everything is working correctly with the function directory and the variable tables, but for now its good progress. In later commits, I want to add more testing input files. I also want to implement a testing framework (pytest or unittest) or maybe develop my own testing, but that will depend on which I find more convenient. I should also refine my testing document (add more testing points, design tests, document more, etc.).


πŸ“† September 13 πŸ•“ 20:34

Deleted token COLON because I noticed I didn't actually use it in any rule. I added it because I've used it in other programming languages, so I assumed I would use in my own, but after careful inspection, I noticed that by the way some statutes are organized, the COLON is never used. Struggled with Python's local module importing, so I decided to create a GENERAL testing file with multiple classes that represent each input file and have it at the root folder so that I don't have problems with imports. At this point I will add more tests before I continue with more complex topics like scope and memory allocation. I also renamed and restructured the files and folders. Also, the introduction of INPUT 0, this input file will be constantly changed, so it will not have tests written for it. Input 0 is for development and to see real time changes.


πŸ“† September 14 πŸ•“ 20:23

Before I jump into creating the semantic cube and other semantic functions, I want to test that the parser is working (or at least most of it) so I don't run into bigger problems later. I noticed that in a < RETURN_BLOCK >, if the user writes something like:

return a + b;

It loops forever, but if the user writes:

return (a + b);

It works fine. I fixed it by returning an < exp > instead of a < factor >. This could later bring problems, but I really doubt it. Just in case im stating it in this commit.


πŸ“† September 15 πŸ•“ 20:10

Changed the way in which errors are detected. Instead of a having a property in the scanner and parser, I created a new object that tracks the global status (by this I mean that for each input the status resets). This works well for testing and for running it in main, since every time a new input is provided, a new status checker is given as well. Also, this new object allows me to track errors in a more specific way, since now semantic errors can be provided specifically from different points of the compiler, and this will be very useful when making new tests because I can test if the semantic error came from a variable declaration, type mismatch, etc.


πŸ“† September 16 πŸ•“ 00:34

VarTable now checks for global variables before declaration. This was made possible by passing the current FuncDir to the addVar() method in the VarTable class, which also passes it to the checkVar() method in the same class. This ensures that before adding a new variable, it looks for the variable name in the current context, but also in the global scope.


πŸ“† September 20 πŸ•“ 17:23

Refactored testing

The old system was comparing emojis that report status. There is a STATUS object that keeps track of the program's status. When an error is expected in any part of the code (maybe in the parser, lexer, etc.), the property had to be manually changed to indicate later that an error occured (the property changed indicated the type of error that happened).

# OrangeStatus() Object

self.lexStatus      = 'βœ…'
self.syntaxStatus   = 'βœ…'
self.semanticStatus = 'βœ…'

And the tests looked like:

# test_GENERAL.py

def test_LEX(self):
    assert self.status.lexStatus == 'βœ…'

def test_SYNTAX(self):
    assert self.status.syntaxStatus == 'βœ…'

def test_SEMANTICS(self):
    assert self.status.semanticStatus == '❌'

The new system is creating custom error clases based on Exception:

# status.py

class lexicalError(Exception):
    pass

class syntacticalError(Exception):
    pass

class semanticError(Exception):
    pass

Python allows to make these custom errors empty and pass a message when the error is raised, giving more flexibility when reporting why the error happened

# scanner.py

# Error handling rule
def error(self, t):
    self.index += 1
    raise lexicalError("❌ Illegal character '%s'" % t.value[0])

And now in the testing file, we only have to expect an exception type:

# test_GENERAL.py

class TestInput02:
    def test_exception_raised(self):
        with pytest.raises(lexicalError):
            # Initialize a different compiler with the needed file
            status, lexer, parser = initializeCompiler('input_02.txt')

Parser Bugs

  • Reorganized grammars to make the file feel cleaner.
  • Refactored variable declarations. Instead of four rules considering if there are global variables/functions, now there is only one which includes global variable declaration and functions regardless, but now global variable declaration and function declaration include an empty rule in case they are not called.
  • Every block now has a variable declaration at the beginning. This allows for easier structure and variable scopes.
  • Now statutes don't have variable declarations. This prevents local variable nesting (like in for i -> for j -> for k)
  • First error rule defined. A specific rule was created to detect when a variable wants to identify as a reserved word. It goes fo < var >, and the token pattern it looks for is an unpacked tuple with all the reserved words in scanner.py. Other specific errors will be made in this way.

Scanner

  • Added a tuple with reserved words

Variable Table

  • I rolled back to previous variable checking, since I realized I don't need to check for global variables when DECLARING. So I left a bit of the code as a comment because I will need it when the variable called in the local scope doesn't exist, but it does in the global scope.

Orange Testing.md

  • Tests added

Inputs

  • Added input_08.txt
  • Added input_09.txt
  • Added input_10.txt
  • Added input_11.txt

πŸ“† September 21 πŸ•“ 10:14

  • Watched the recording of class 4 (Sep27). In this class the teacher talked more about semantics, precedence in arithmetic operations, type matching and the "semantic cube".
  • Watched the recording of class 5 (Sep30). In this class the teacher talked about syntax flow, polish vectors, quadruples and intermediate code generation.
  • Created first version of semantic cube

πŸ“† September 21 πŸ•“ 22:29

  • Watched the recording of class 6 (Oct04). In this class the teacher talked more about quadruple generation.
  • Refactored variable declaration rules in the parser and fixed some bugs.
  • Started quadruple generator class
  • Started some functions in the function directory that will aid the quadruple generatorw

πŸ“† September 24 πŸ•“ 22:19

  • Variables are now looked up in the current scope and the global scope when trying to be used. There are still some details to fix/add, like instead of leaving it as a function checkVar() for the OrangeFunctionDirectory, the checking function will be added to the quadruple generator. This would mean that when generating a quadruple for the variable found, the QuadMachine will look for said variable in the FunctionDirectory.
  • Input 12 added
  • Input 13 added
  • Input 14 added
  • Input 15 added
  • Input 16 added
  • Some tests (4 & 6) needed small corrections. The way they were written only tested for syntax and lex, so now that we actually care about undeclared variables, they didn't work. They still test the same thing they were intended to test, but I had to add a variable declaration to make the Orange Code work.
  • If main, global or any function doesn't have variable declaration, they will still be added to the function directory with an empty table (for consistency and ease of search).

πŸ“† September 25 πŸ•“ 21:37

  • Quadruple Machine now generates quadruples for arithmetic operations (+ * - /). It first uses the Orange Function Directory's checkVar function when adding operands. Then when an operator is added to the operator stack, it pops two operands and an operator and validates with the semantic cube if the operation is valid, and if not, a semantic error is raised. If its valid, a quadruple is generated and added to the quadruple array.
  • Temporary variables are stored with strings (T1, T2, ...). This should change when I learn about implementing memory.
  • No tests added in this run, since some core things to quadruple generation are yet to be added. This means that if I added the tests, I would have to refactor them later.
  • Constant variable handling is still pending. I added starter code, but I don't know if thats the way to do it. Maybe in a later class recording the teacher will explain how to do it.

πŸ“† September 27 πŸ•“ 17:01

  • Refactored parser's arithmetic expressions. I noticed that the way the grammar was written was wrong, since even though it works for precedence, it doesn't for order of operations (it was designed for right association). Fixed it for relational and logical as well.
  • Fixed parenthesis usage. Before, the parenthesis were basically ignored. Now a 'fake floor' is created every time a parenthesis is observed, and its removed whenever the parenthesis ends.
  • Added input_17
  • Added a basic test for quadruple generation, to see if correct order, precedence, associativity and generation are achieved.

πŸ“† September 27 πŸ•“ 17:19

  • Added input_18
  • Added another test similar to test_17 just to keep check of associativity, order of operations, parenthesis and quadruple generation, but with a different input

πŸ“† September 27 πŸ•“ 20:03

  • Modified some tests. Some tests broke because with quadruple generation, some things that still don't exist are verified, hence errors are raised
  • Added boolean type
  • Added quadruple generation for assignment
  • Added error raising for 'type mismatch' in the quadruple generation function

πŸ“† September 28 πŸ•“ 14:52

  • Quadruple generation for print statements (write).
  • Changed regular expression in the scanner for constant strings. This was because it used to match anything, any amount of times inside two quotation marks, causing an error to identify ALL of the parameters in a print as one.

πŸ“† September 28 πŸ•“ 16:18

  • Quadruple generation for inputs (read).
  • Quadruple generation for constant variables (int, float, bool, string)

πŸ“† September 30 πŸ•“ 16:03

  • Quadruple generation for conditional statements. This includes individual IF statements, IF statements with an ELSE statement, and nested conditional statements.
  • Testing for conditional quadruple generation. This includes three tests (lonely IF, IF with an ELSE, nested IFs).
  • Refactored . This was because ANY block could have variable declaration, and this included conditionals, loops, etc. That caused trouble because every block tried to create a new function with the current context, causing a semantic error for 'Function already declared'. All I did was make another type of block that included variable declaration, and changed the rules that actually needed it (main block, global, functions). And everything else remained with a normal block (without being able to declare variables).
  • Quadruple machine now tracks quadruple number for later reference. Also, a new stack was added to keep track of unfilled jump positions.
  • Added input_20
  • Added input_21
  • Added input_22

πŸ“† October 02 πŸ•“ 17:41

  • Watched recorded class for October 11. The teacher talked about code generation for WHILE and DO WHILE loops.
  • Quadruple generation for a WHILE loop. A this point the jumps work, but I feel the code could be better (in a more readable/understandable way). All tests pass, but I want to refactor a bit the 'fillJumps()' function and create more tests (nested while loops, while loops with if statements, loops & conditionals, etc.)
  • Added input_23

πŸ“† October 02 πŸ•“ 18:18

  • Refactored fillJumps() function and jump filling functionality.

πŸ“† October 02 πŸ•“ 23:26

  • Added DO WHILE loops. I wasn't planning on adding a DO WHILE, but while watching the recording of class 7, the teacher added an example for a DO WHILE that included nested WHILE loops and conditionals, and that seemed to me like the perfect example to test everything, so I just decided to give it a shot now that I'm at it.
  • Watched recorded class for October 14. The teacher talked about code generation for FOR loops
  • Added input_24

πŸ“† October 04 πŸ•“ 18:24

  • Added FOR loops.
  • Added '++' to semantic cube. It only works for integers, and its only used in FOR LOOPS. This could be refactored, but for now it made it easy for me incrementing the control variable in a FOR loop and saving its value to itself instead of a temporary variable.
  • Added some custom mismatch errors in quadruple machine
  • Minor bug fix in openjumpslot that confused the p[-7] position looking for a token. Since a DO WHILE loop and a FOR LOOP both use the 'do' keyword, and oddly enough in the same position.
  • Added input_25
  • Added input_26
  • Added input_27

πŸ“† October 06 πŸ•“ 00:29

  • Watched class recording of October 21. The teacher talked about memory management, virtual memory allocation for functions and quadruple generation for function definition.
  • Watched class recording of November 01. The teacher talked about memory management, virtual memory allocation for functions and quadruple generation for function definition (with examples).
  • Function directory's directory now stores a functions "address" (at what quadruple the function starts), another table for parameters, a string that represents it's signature, and a structure to keep track of size needed (it tracks amount of variable scopes and types)
  • Variable checking in the function directory now takes into account checking parameters too
  • Adding parameters now works and also updates the functions signature (only when declared)
  • Refactored scanner by "eliminating" reserved words, and merging them into the ID token. Now everything is read as a token, but if said ID matches a reserved word, its type is changed to that reserved word. This prevents the regular expression from matching prefixes (ex. can't have a variable named 'doppler' because it starts with 'do').
  • Added a varType to VarTable that keeps track of a variable type for a variable declaration line.
  • Refactored parser rules for function and variable definition
  • Added many mid point instructions in parser (for quadruple generation, type tracking, etc.)
  • Restructured several rules (broke recursion into a single variable rule for easier data extraction)
  • Added quadruples for function calls and definition (ERA, GOSUB, PARAM, ENDFUNC)
  • Fixed/updated a couple of tests, but I still have to update almost half (because previously tests didn't have the 'params' table, the signature, quadruple number, etc.)

πŸ“† October 07 πŸ•“ 14:44

  • Added some actions in between grammar rules for function calling (things like checking the function signature is the same as the function calling it,- checking that the function being called exists, etc.)
  • Fixed tests
  • Added input_28

πŸ“† October 07 πŸ•“ 15:17

  • Watched class recording for November 04. In this class the teacher explained more in depth memory management and gave some tips about how we could design a structure for memory and a virtual machine. The teacher also explained how to store in memory structured data like arrays and matrices.

πŸ“† October 18 πŸ•“ 23:08

  • Refactored everything to work with a memory address instead of a variable name
  • Memory management
  • Added constant table to store constants
  • Refactored tests
  • Executed tests by hand to confirm everything works (some bugs were found and fixed). Did not execute test 28 yet.

πŸ“† November 01 πŸ•“ 16:09

  • Refactored tests
  • Separated tests into their own file to avoid problems with a virtual machine execution
  • Memory addresses reset after context switch
  • OBJ file is generated after the parser finishes and is read and executed by the virtual machine
  • RETURN quadruple added
  • VirtualMachine developed.
    • Instructions are added as individual functions that are called through a switch (to speed things up)
    • An instruction pointer counter is added and updated as the instructions are read and executed
    • A "breadcrumb" list is added to keep track of instruction numbers to go back when calling a function
    • The function directory and the constant table are brought back to reference them throughout execution

πŸ“† November 08 πŸ•“ 20:52

  • Fixed recursion
  • Fixed multiple function calls overwriting the return value
  • Added an extra quadruple to store a function return value
  • Fixed tests to accomodate new quadruple for return values

Reference

Official Documentation

SLY Documentation

Pytest Documentation

VS Code Settings Sync

Tools

Regex 101

Blogs

Beginner's Book - Python Constructors: Default & Parameterized

Refactoring Guru - Singleton Design Pattern

Finxter Blog - How to call a parents class method

Datagy.io - Check if dictionary is empty

GeeksForGeeks - Ternary Operator in Python

W3Schools - Python Raise an Exception

Note.nkmk.me - Python unpack values

Splunk Tool - Python evenly space output data with varying string lengths

El Libro de Python - Switch en Python

Towards Data Science - Do not use python pickle unless you know all these facts

Codigo Piton - Como hacer switch case en python

StackOverflow

Method arguments in Python

How do I convert a tuple of tuples to a one-dimensional list using list comprehension?

How do I flatten deeply nested tuples?

Import a file from a subdirectory

How to compare type of an object

Matching special characters and letters in regex

Tags

TODO: Tasks to be done

HACK: Neat ideas to solve problems or cool ideas

FIXME: It works, but not always (could break or not tested)

BUG: It doesn't work

WARNING: Could potentially have problems

DOC: Documentation pending for this section