source: icGREP/icgrep-devel/icgrep/pablo/codegenstate.h @ 4878

Last change on this file since 4878 was 4878, checked in by nmedfort, 4 years ago

More work on n-ary operations.

File size: 8.0 KB
Line 
1/*
2 *  Copyright (c) 2014 International Characters.
3 *  This software is licensed to the public under the Open Software License 3.0.
4 *  icgrep is a trademark of International Characters.
5 */
6
7#ifndef PS_PABLOS_H
8#define PS_PABLOS_H
9
10#include <pablo/pabloAST.h>
11#include <pablo/symbol_generator.h>
12#include <pablo/pe_advance.h>
13#include <pablo/pe_and.h>
14#include <pablo/pe_call.h>
15#include <pablo/pe_matchstar.h>
16#include <pablo/pe_next.h>
17#include <pablo/pe_not.h>
18#include <pablo/pe_ones.h>
19#include <pablo/pe_or.h>
20#include <pablo/pe_scanthru.h>
21#include <pablo/pe_sel.h>
22#include <pablo/pe_integer.h>
23#include <pablo/pe_string.h>
24#include <pablo/pe_var.h>
25#include <pablo/pe_xor.h>
26#include <pablo/pe_zeroes.h>
27#include <pablo/pe_count.h>
28#include <pablo/ps_assign.h>
29#include <pablo/ps_if.h>
30#include <pablo/ps_while.h>
31#include <pablo/function.h>
32#include <llvm/ADT/ArrayRef.h>
33#include <stdexcept>
34
35namespace pablo {
36
37class PabloBlock : public PabloAST, public StatementList {
38    friend class PabloAST;
39    friend class PabloBuilder;
40public:
41
42    static inline bool classof(const PabloBlock *) {
43        return true;
44    }
45    static inline bool classof(const Statement *) {
46        return false;
47    }
48    static inline bool classof(const PabloAST * e) {
49        return e->getClassTypeId() == ClassTypeId::Block;
50    }
51    static inline bool classof(const void *) {
52        return false;
53    }
54
55    inline static PabloBlock * Create(PabloFunction & function) noexcept {
56        return new PabloBlock(function.mSymbolTable);
57    }
58
59    inline static PabloBlock * Create(PabloBlock * const block) noexcept {
60        return new PabloBlock(block->mSymbolGenerator);
61    }
62
63    PabloAST * createAdvance(PabloAST * expr, const Integer::Type shiftAmount);
64
65    PabloAST * createAdvance(PabloAST * expr, PabloAST * shiftAmount);
66
67    PabloAST * createAdvance(PabloAST * expr, const Integer::Type shiftAmount, const std::string prefix);
68
69    PabloAST * createAdvance(PabloAST * expr, PabloAST * shiftAmount, const std::string prefix);
70
71    static inline Zeroes * createZeroes() {
72        return &mZeroes;
73    }
74
75    static inline Ones * createOnes() {
76        return &mOnes;
77    }
78
79    inline Call * createCall(Prototype * prototype, const std::vector<Var *> & args) {
80        return createCall(prototype, reinterpret_cast<const std::vector<PabloAST *> &>(args));
81    }
82
83    inline Call * createCall(Prototype * prototype, const std::vector<PabloAST *> & args) {
84        if (prototype == nullptr) {
85            throw std::runtime_error("Call object cannot be created with a Null prototype!");
86        }
87        if (args.size() != cast<Prototype>(prototype)->getNumOfParameters()) {
88            throw std::runtime_error("Invalid number of arguments passed into Call object!");
89        }
90        return createCall(static_cast<PabloAST *>(prototype), args);
91    }
92
93    Assign * createAssign(const std::string && prefix, PabloAST * expr);
94
95    inline Var * createVar(const std::string name) {
96        return createVar(getName(name, false));
97    }
98
99    inline Var * createVar(String * name) {
100        return createVar(cast<PabloAST>(name));
101    }
102
103    Next * createNext(Assign * assign, PabloAST * expr);
104
105    PabloAST * createAnd(PabloAST * expr1, PabloAST * expr2);
106
107    And * createAnd(const unsigned operands, PabloAST * value);
108
109    PabloAST * createAnd(PabloAST * expr1, PabloAST * expr2, const std::string prefix);
110
111    PabloAST * createNot(PabloAST * expr);
112
113    PabloAST * createNot(PabloAST * expr, const std::string prefix);
114
115    PabloAST * createOr(PabloAST * expr1, PabloAST * expr2);
116
117    Or * createOr(const unsigned operands, PabloAST * value);
118
119    PabloAST * createOr(PabloAST * expr1, PabloAST * expr2, const std::string prefix);
120
121    PabloAST * createXor(PabloAST * expr1, PabloAST * expr2);
122
123    PabloAST * createXor(PabloAST * expr1, PabloAST * expr2, const std::string prefix);
124
125    PabloAST * createMatchStar(PabloAST * marker, PabloAST * charclass);
126
127    PabloAST * createMatchStar(PabloAST * marker, PabloAST * charclass, const std::string prefix);
128
129    PabloAST * createScanThru(PabloAST * from, PabloAST * thru);
130
131    PabloAST * createScanThru(PabloAST * from, PabloAST * thru, const std::string prefix);
132
133    PabloAST * createSel(PabloAST * condition, PabloAST * trueExpr, PabloAST * falseExpr);
134
135    PabloAST * createSel(PabloAST * condition, PabloAST * trueExpr, PabloAST * falseExpr, const std::string prefix);
136
137    PabloAST * createCount(PabloAST * expr);
138   
139    PabloAST * createCount(PabloAST * expr, const std::string prefix);
140   
141    If * createIf(PabloAST * condition, const std::initializer_list<Assign *> definedVars, PabloBlock * body);
142
143    If * createIf(PabloAST * condition, const std::vector<Assign *> & definedVars, PabloBlock * body);
144
145    If * createIf(PabloAST * condition, std::vector<Assign *> && definedVars, PabloBlock * body);
146
147    While * createWhile(PabloAST * condition, const std::initializer_list<Next *> nextVars, PabloBlock * body);
148
149    While * createWhile(PabloAST * condition, const std::vector<Next *> & nextVars, PabloBlock * body);
150
151    While * createWhile(PabloAST * condition, std::vector<Next *> && nextVars, PabloBlock * body);
152
153    PabloAST * createMod64Advance(PabloAST * expr, const Integer::Type shiftAmount);
154
155    PabloAST * createMod64Advance(PabloAST * expr, PabloAST * shiftAmount);
156
157    PabloAST * createMod64Advance(PabloAST * expr, const Integer::Type shiftAmount, const std::string prefix);
158
159    PabloAST * createMod64Advance(PabloAST * expr, PabloAST * shiftAmount, const std::string prefix);
160
161    PabloAST * createMod64MatchStar(PabloAST * marker, PabloAST * charclass);
162
163    PabloAST * createMod64MatchStar(PabloAST * marker, PabloAST * charclass, const std::string prefix);
164
165    PabloAST * createMod64ScanThru(PabloAST * from, PabloAST * thru);
166
167    PabloAST * createMod64ScanThru(PabloAST * from, PabloAST * thru, const std::string prefix);
168
169
170    inline StatementList & statements() {
171        return *this;
172    }
173
174    inline const StatementList & statements() const {
175        return *this;
176    }
177
178    inline String * getName(const std::string name, const bool generated = true) const {
179        return mSymbolGenerator->get(name, generated);
180    }
181
182    inline String * makeName(const std::string prefix, const bool generated = true) const {
183        return mSymbolGenerator->make(prefix, generated);
184    }
185
186    inline Integer * getInteger(Integer::Type value) {
187        return mSymbolGenerator->getInteger(value);
188    }
189
190    inline PabloBlock * getParent() const {
191        return mParent;
192    }
193   
194    void setParent(PabloBlock * parent) {
195        mParent = parent;
196        // Add test to assert this block is in the same function.
197    }
198
199    void insert(Statement * const statement);
200
201    unsigned enumerateScopes(unsigned baseScopeIndex);
202   
203    inline unsigned getScopeIndex() const {
204        return mScopeIndex;
205    }
206   
207    void eraseFromParent(const bool recursively = false);
208
209    virtual ~PabloBlock();
210
211protected:
212
213    explicit PabloBlock(SymbolGenerator * symbolGenerator) noexcept;
214
215    PabloAST * renameNonNamedNode(PabloAST * expr, const std::string && prefix);
216
217    template<typename Type>
218    inline Type * insertAtInsertionPoint(Type * expr) {
219        if (isa<Statement>(expr)) {
220            if (LLVM_UNLIKELY(isa<If>(expr) || isa<While>(expr))) {
221                PabloBlock * const body = isa<If>(expr) ? cast<If>(expr)->getBody() : cast<While>(expr)->getBody();
222                body->setParent(this);
223                addUser(body);
224            }
225            insert(cast<Statement>(expr));
226        }
227        return expr;
228    }
229
230private:
231
232    Call * createCall(PabloAST * prototype, const std::vector<PabloAST *> &);
233
234    Var * createVar(PabloAST * name);
235
236private:       
237    static Zeroes                                       mZeroes;
238    static Ones                                         mOnes;
239    SymbolGenerator *                                   mSymbolGenerator; // TODO: need a better way of passing a symbol generator around
240    PabloBlock *                                        mParent;
241    unsigned                                            mScopeIndex;
242};
243
244}
245
246#endif // PS_PABLOS_H
Note: See TracBrowser for help on using the repository browser.