-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcodegen.hpp
139 lines (130 loc) · 4.29 KB
/
codegen.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
#include "Parser.hpp"
#include "Lexer.hpp"
#include <llvm/ADT/APFloat.h>
#include <llvm/ADT/STLExtras.h>
#include <llvm/IR/BasicBlock.h>
#include <llvm/IR/Constants.h>
#include <llvm/IR/DerivedTypes.h>
#include <llvm/IR/Function.h>
#include <llvm/IR/IRBuilder.h>
#include <llvm/IR/LLVMContext.h>
#include <llvm/IR/Module.h>
#include <llvm/IR/NoFolder.h>
#include <llvm/IR/Type.h>
#include <llvm/IR/Verifier.h>
#include <memory>
#include <string>
#include <variant>
#include <vector>
#include <unordered_map>
/**
* DeclaringFunctionParameterData is a data container which holds the metadata needed
* for each declared parameter of a function.
*/
struct DeclaringFunctionParameterData
{
// The name of the parameter
std::string name;
// The type of the parameter (currently only pointer (array) and int32)
llvm::Type *type;
};
/**
* In each if/else statement, we have three basic blocks which are listed here.
*/
struct IfElseBasicBlocks
{
// Executes when the condition is true.
llvm::BasicBlock *then_block;
// Executes when the condition is false. Might be an empty basic block which simply jumps to aftermath.
llvm::BasicBlock *else_block;
// After "then" and "else" blocks, control flow merges with this block.
llvm::BasicBlock *aftermath_block;
};
/**
* In each for loop, we have three basic blocks which are listed here.
*/
struct ForLoopBasicBlocks
{
// Check for the condition of the for loop and either terminates it, or jumps to the loop
llvm::BasicBlock *condition_block;
// After each iteration, this code gets executed which usually just increments a counter
llvm::BasicBlock *step_block;
// After "then" and "else" blocks, control flow merges with this block.
llvm::BasicBlock *body_block;
// After the for loop ends, control flow merges with this block.
llvm::BasicBlock *aftermath_block;
};
class CodeGenerator
{
public:
// Different types of arithmetic operators of the language
enum class Operator {
PLUS = PLUS,
MINUS = MINUS,
MULT = TIMES,
LESS_THAN = LSS,
EQUALS = DOUBLE_EQ,
};
private:
// LLVM codegen variables
std::unique_ptr<llvm::LLVMContext> the_context;
std::unique_ptr<llvm::IRBuilder<llvm::NoFolder>> builder;
std::unique_ptr<llvm::Module> the_module;
// List of declared variables in current scope
std::unordered_map<std::string, llvm::Value *> local_variables;
// A list of operators which should be applied to values in semantic stack later
std::vector<CodeGenerator::Operator> operator_stack;
// Each variable or function is either int or void
enum class VariableType {
VOID,
INT,
};
// Types which can be pushed in the semantic stack
typedef std::variant<llvm::Value *, llvm::Function *, IfElseBasicBlocks, ForLoopBasicBlocks, CodeGenerator::VariableType, int> SemanticStaticObject;
// The good old semantic stack
std::vector<SemanticStaticObject> semantic_stack;
// Name of the variable/function we are declaring
std::string declaring_pid_name, declaring_function_name;
// The list of parameters of this function which we are currently declaring.
std::vector<DeclaringFunctionParameterData> declaring_function_params;
// True if we are in global scope, otherwise false.
bool in_global_scope;
// Function calls
void generate_prelude();
llvm::Value *dereference_ptr_if_needed(llvm::Value *);
public:
// Function calls
explicit CodeGenerator();
void print_code();
void print_code(const char *);
void void_type();
void int_type();
void declaring_pid(const char *);
void variable_declared();
void array_declared();
void function_start();
void array_param();
void int_param();
void function_params_end();
void function_end();
void pid(const char *);
void immediate(int);
void pop_expression();
void assign();
void array();
void immediate_val(int);
void save_operator(CodeGenerator::Operator);
void negate();
void calculate();
void call();
void insert_return(bool is_void);
void if_condition();
void if_no_else_end();
void if_else_begin();
void if_else_end();
void for_condition_begin();
void for_condition_end();
void for_step_end();
void for_end();
void insert_break();
};