/*
* Copyright (c) 2015 PLUMgrid, Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma once
#include <vector>
#include <bitset>
#include <string>
#include <memory>
#include <algorithm>
#include <stdint.h>
#include "common.h"
#include "bcc_exception.h"
#include "scope.h"
#define REVISION_MASK 0xfff
#define MAJOR_VER_POS 22
#define MAJOR_VER_MASK ~((1 << MAJOR_VER_POS) - 1)
#define MINOR_VER_POS 12
#define MINOR_VER_MASK (~((1 << MINOR_VER_POS) - 1) & (~(MAJOR_VER_MASK)))
#define GET_MAJOR_VER(version) ((version & MAJOR_VER_MASK) >> MAJOR_VER_POS)
#define GET_MINOR_VER(version) ((version & MINOR_VER_MASK) >> MINOR_VER_POS)
#define GET_REVISION(version) (version & REVISION_MASK)
#define MAKE_VERSION(major, minor, rev) \
((major << MAJOR_VER_POS) | \
(minor << MINOR_VER_POS) | \
(rev & REVISION_MASK))
#define STATUS_RETURN __attribute((warn_unused_result)) StatusTuple
namespace ebpf {
namespace cc {
using std::unique_ptr;
using std::move;
using std::string;
using std::vector;
using std::bitset;
using std::find;
typedef unique_ptr<string> String;
#define NODE_EXPRESSIONS(EXPAND) \
EXPAND(IdentExprNode, ident_expr_node) \
EXPAND(AssignExprNode, assign_expr_node) \
EXPAND(PacketExprNode, packet_expr_node) \
EXPAND(IntegerExprNode, integer_expr_node) \
EXPAND(StringExprNode, string_expr_node) \
EXPAND(BinopExprNode, binop_expr_node) \
EXPAND(UnopExprNode, unop_expr_node) \
EXPAND(BitopExprNode, bitop_expr_node) \
EXPAND(GotoExprNode, goto_expr_node) \
EXPAND(ReturnExprNode, return_expr_node) \
EXPAND(MethodCallExprNode, method_call_expr_node) \
EXPAND(TableIndexExprNode, table_index_expr_node)
#define NODE_STATEMENTS(EXPAND) \
EXPAND(ExprStmtNode, expr_stmt_node) \
EXPAND(BlockStmtNode, block_stmt_node) \
EXPAND(IfStmtNode, if_stmt_node) \
EXPAND(OnValidStmtNode, onvalid_stmt_node) \
EXPAND(SwitchStmtNode, switch_stmt_node) \
EXPAND(CaseStmtNode, case_stmt_node) \
EXPAND(StructVariableDeclStmtNode, struct_variable_decl_stmt_node) \
EXPAND(IntegerVariableDeclStmtNode, integer_variable_decl_stmt_node) \
EXPAND(StructDeclStmtNode, struct_decl_stmt_node) \
EXPAND(StateDeclStmtNode, state_decl_stmt_node) \
EXPAND(ParserStateStmtNode, parser_state_stmt_node) \
EXPAND(MatchDeclStmtNode, match_decl_stmt_node) \
EXPAND(MissDeclStmtNode, miss_decl_stmt_node) \
EXPAND(FailureDeclStmtNode, failure_decl_stmt_node) \
EXPAND(TableDeclStmtNode, table_decl_stmt_node) \
EXPAND(FuncDeclStmtNode, func_decl_stmt_node)
#define EXPAND_NODES(EXPAND) \
NODE_EXPRESSIONS(EXPAND) \
NODE_STATEMENTS(EXPAND)
class Visitor;
// forward declare all classes
#define FORWARD(type, func) class type;
EXPAND_NODES(FORWARD)
#undef FORWARD
#define DECLARE(type) \
typedef unique_ptr<type> Ptr; \
virtual StatusTuple accept(Visitor* v);
class Node {
public:
typedef unique_ptr<Node> Ptr;
Node() : line_(-1), column_(-1) {}
virtual ~Node() {}
virtual StatusTuple accept(Visitor* v) = 0;
int line_;
int column_;
string text_;
};
template <typename... Args>
StatusTuple mkstatus_(Node *n, const char *fmt, Args... args) {
StatusTuple status = StatusTuple(n->line_ ? n->line_ : -1, fmt, args...);
if (n->line_ > 0)
status.append_msg("\n" + n->text_);
return status;
}
static inline StatusTuple mkstatus_(Node *n, const char *msg) {
StatusTuple status = StatusTuple(n->line_ ? n->line_ : -1, msg);
if (n->line_ > 0)
status.append_msg("\n" + n->text_);
return status;
}
class StmtNode : public Node {
public:
typedef unique_ptr<StmtNode> Ptr;
virtual StatusTuple accept(Visitor* v) = 0;
};
typedef vector<StmtNode::Ptr> StmtNodeList;
class ExprNode : public Node {
public:
typedef unique_ptr<ExprNode> Ptr;
virtual StatusTuple accept(Visitor* v) = 0;
enum expr_type { STRUCT, INTEGER, STRING, VOID, UNKNOWN };
enum prop_flag { READ = 0, WRITE, PROTO, IS_LHS, IS_REF, IS_PKT, LAST };
expr_type typeof_;
StructDeclStmtNode *struct_type_;
size_t bit_width_;
bitset<LAST> flags_;
unique_ptr<BitopExprNode> bitop_;
ExprNode() : typeof_(UNKNOWN), struct_type_(NULL), flags_(1 << READ) {}
void copy_type(const ExprNode& other) {
typeof_ = other.typeof_;
struct_type_ = other.struct_type_;
bit_width_ = other.bit_width_;
flags_ = other.flags_;
}
bool is_lhs() const { return flags_[IS_LHS]; }
bool is_ref() const { return flags_[IS_REF]; }
bool is_pkt() const { return flags_[IS_PKT]; }
};
typedef vector<ExprNode::Ptr> ExprNodeList;
class IdentExprNode : public ExprNode {
public:
DECLARE(IdentExprNode)
string name_;
string sub_name_;
string scope_name_;
VariableDeclStmtNode *decl_;
VariableDeclStmtNode *sub_decl_;
IdentExprNode(const IdentExprNode& other) {
name_ = other.name_;
sub_name_ = other.sub_name_;
scope_name_ = other.scope_name_;
decl_ = other.decl_;
sub_decl_ = other.sub_decl_;
}
IdentExprNode::Ptr copy() const {
return IdentExprNode::Ptr(new IdentExprNode(*this));
}
explicit IdentExprNode(const string& id) : name_(id) {}
explicit IdentExprNode(const char* id) : name_(id) {}
void prepend_scope(const string& id) {
scope_name_ = id;
}
void append_scope(const string& id) {
scope_name_ = move(name_);
name_ = id;
}
void prepend_dot(const string& id) {
sub_name_ = move(name_);
name_ = id;
}
void append_dot(const string& id) {
// we don't support nested struct so keep all subs as single variable
if (!sub_name_.empty()) {
sub_name_ += "." + id;
} else {
sub_name_ = id;
}
}
const string& full_name() {
if (full_name_.size()) {
return full_name_; // lazy init
}
if (scope_name_.size()) {
full_name_ += scope_name_ + "::";
}
full_name_ += name_;
if (sub_name_.size()) {
full_name_ += "." + sub_name_;
}
return full_name_;
}
const char* c_str() const { return name_.c_str(); }
private:
string full_name_;
};
class BitopExprNode : public ExprNode {
public:
DECLARE(BitopExprNode)
ExprNode::Ptr expr_;
size_t bit_offset_;
size_t bit_width_;
BitopExprNode(const string& bofs, const string& bsz)
: bit_offset_(strtoul(bofs.c_str(), NULL, 0)), bit_width_(strtoul(bsz.c_str(), NULL, 0)) {}
};
typedef vector<IdentExprNode::Ptr> IdentExprNodeList;
class AssignExprNode : public ExprNode {
public:
DECLARE(AssignExprNode)
//IdentExprNode *id_;
ExprNode::Ptr lhs_;
ExprNode::Ptr rhs_;
AssignExprNode(IdentExprNode::Ptr id, ExprNode::Ptr rhs)
: lhs_(move(id)), rhs_(move(rhs)) {
//id_ = (IdentExprNode *)lhs_.get();
lhs_->flags_[ExprNode::IS_LHS] = true;
}
AssignExprNode(ExprNode::Ptr lhs, ExprNode::Ptr rhs)
: lhs_(move(lhs)), rhs_(move(rhs)) {
//id_ = nullptr;
lhs_->flags_[ExprNode::IS_LHS] = true;
}
};
class PacketExprNode : public ExprNode {
public:
DECLARE(PacketExprNode)
IdentExprNode::Ptr id_;
explicit PacketExprNode(IdentExprNode::Ptr id) : id_(move(id)) {}
};
class StringExprNode : public ExprNode {
public:
DECLARE(StringExprNode)
string val_;
explicit StringExprNode(string *val) : val_(move(*val)) {
delete val;
}
explicit StringExprNode(const string &val) : val_(val) {}
};
class IntegerExprNode : public ExprNode {
public:
DECLARE(IntegerExprNode)
size_t bits_;
string val_;
IntegerExprNode(string* val, string* bits)
: bits_(strtoul(bits->c_str(), NULL, 0)), val_(move(*val)) {
delete val;
delete bits;
}
explicit IntegerExprNode(string* val)
: bits_(0), val_(move(*val)) {
delete val;
}
explicit IntegerExprNode(const string& val) : bits_(0), val_(val) {}
explicit IntegerExprNode(const string& val, size_t bits) : bits_(bits), val_(val) {}
};
class BinopExprNode : public ExprNode {
public:
DECLARE(BinopExprNode)
ExprNode::Ptr lhs_;
int op_;
ExprNode::Ptr rhs_;
BinopExprNode(ExprNode::Ptr lhs, int op, ExprNode::Ptr rhs)
: lhs_(move(lhs)), op_(op), rhs_(move(rhs))
{}
};
class UnopExprNode : public ExprNode {
public:
DECLARE(UnopExprNode)
ExprNode::Ptr expr_;
int op_;
UnopExprNode(int op, ExprNode::Ptr expr) : expr_(move(expr)), op_(op) {}
};
class GotoExprNode : public ExprNode {
public:
DECLARE(GotoExprNode)
bool is_continue_;
IdentExprNode::Ptr id_;
GotoExprNode(IdentExprNode::Ptr id, bool is_continue = false)
: is_continue_(is_continue), id_(move(id)) {}
};
class ReturnExprNode : public ExprNode {
public:
DECLARE(ReturnExprNode)
ExprNode::Ptr expr_;
ReturnExprNode(ExprNode::Ptr expr)
: expr_(move(expr)) {}
};
class BlockStmtNode : public StmtNode {
public:
DECLARE(BlockStmtNode)
explicit BlockStmtNode(StmtNodeList stmts = StmtNodeList())
: stmts_(move(stmts)), scope_(NULL) {}
~BlockStmtNode() { delete scope_; }
StmtNodeList stmts_;
Scopes::VarScope* scope_;
};
class MethodCallExprNode : public ExprNode {
public:
DECLARE(MethodCallExprNode)
IdentExprNode::Ptr id_;
ExprNodeList args_;
BlockStmtNode::Ptr block_;
MethodCallExprNode(IdentExprNode::Ptr id, ExprNodeList&& args, int lineno)
: id_(move(id)), args_(move(args)), block_(make_unique<BlockStmtNode>()) {
line_ = lineno;
}
};
class TableIndexExprNode : public ExprNode {
public:
DECLARE(TableIndexExprNode)
IdentExprNode::Ptr id_;
IdentExprNode::Ptr sub_;
ExprNode::Ptr index_;
TableDeclStmtNode *table_;
VariableDeclStmtNode *sub_decl_;
TableIndexExprNode(IdentExprNode::Ptr id, ExprNode::Ptr index)
: id_(move(id)), index_(move(index)), table_(nullptr), sub_decl_(nullptr)
{}
};
class ExprStmtNode : public StmtNode {
public:
DECLARE(ExprStmtNode)
ExprNode::Ptr expr_;
explicit ExprStmtNode(ExprNode::Ptr expr) : expr_(move(expr)) {}
};
class IfStmtNode : public StmtNode {
public:
DECLARE(IfStmtNode)
ExprNode::Ptr cond_;
StmtNode::Ptr true_block_;
StmtNode::Ptr false_block_;
// create an if () {} expression
IfStmtNode(ExprNode::Ptr cond, StmtNode::Ptr true_block)
: cond_(move(cond)), true_block_(move(true_block)) {}
// create an if () {} else {} expression
IfStmtNode(ExprNode::Ptr cond, StmtNode::Ptr true_block, StmtNode::Ptr false_block)
: cond_(move(cond)), true_block_(move(true_block)),
false_block_(move(false_block)) {}
};
class OnValidStmtNode : public StmtNode {
public:
DECLARE(OnValidStmtNode)
IdentExprNode::Ptr cond_;
StmtNode::Ptr block_;
StmtNode::Ptr else_block_;
// create an onvalid () {} expression
OnValidStmtNode(IdentExprNode::Ptr cond, StmtNode::Ptr block)
: cond_(move(cond)), block_(move(block)) {}
// create an onvalid () {} else {} expression
OnValidStmtNode(IdentExprNode::Ptr cond, StmtNode::Ptr block, StmtNode::Ptr else_block)
: cond_(move(cond)), block_(move(block)),
else_block_(move(else_block)) {}
};
class SwitchStmtNode : public StmtNode {
public:
DECLARE(SwitchStmtNode)
ExprNode::Ptr cond_;
BlockStmtNode::Ptr block_;
SwitchStmtNode(ExprNode::Ptr cond, BlockStmtNode::Ptr block)
: cond_(move(cond)), block_(move(block)) {}
};
class CaseStmtNode : public StmtNode {
public:
DECLARE(CaseStmtNode)
IntegerExprNode::Ptr value_;
BlockStmtNode::Ptr block_;
CaseStmtNode(IntegerExprNode::Ptr value, BlockStmtNode::Ptr block)
: value_(move(value)), block_(move(block)) {}
explicit CaseStmtNode(BlockStmtNode::Ptr block) : block_(move(block)) {}
};
class VariableDeclStmtNode : public StmtNode {
public:
typedef unique_ptr<VariableDeclStmtNode> Ptr;
virtual StatusTuple accept(Visitor* v) = 0;
enum storage_type { INTEGER, STRUCT, STRUCT_REFERENCE };
IdentExprNode::Ptr id_;
ExprNodeList init_;
enum storage_type storage_type_;
size_t bit_width_;
size_t bit_offset_;
int slot_;
string scope_id_;
explicit VariableDeclStmtNode(IdentExprNode::Ptr id, storage_type t, size_t bit_width = 0, size_t bit_offset = 0)
: id_(move(id)), storage_type_(t), bit_width_(bit_width), bit_offset_(bit_offset), slot_(0) {}
const char* scope_id() const { return scope_id_.c_str(); }
bool is_struct() { return (storage_type_ == STRUCT || storage_type_ == STRUCT_REFERENCE); }
bool is_pointer() { return (storage_type_ == STRUCT_REFERENCE); }
};
typedef vector<VariableDeclStmtNode::Ptr> FormalList;
class StructVariableDeclStmtNode : public VariableDeclStmtNode {
public:
DECLARE(StructVariableDeclStmtNode)
IdentExprNode::Ptr struct_id_;
StructVariableDeclStmtNode(IdentExprNode::Ptr struct_id, IdentExprNode::Ptr id,
VariableDeclStmtNode::storage_type t = VariableDeclStmtNode::STRUCT)
: VariableDeclStmtNode(move(id), t), struct_id_(move(struct_id)) {}
};
class IntegerVariableDeclStmtNode : public VariableDeclStmtNode {
public:
DECLARE(IntegerVariableDeclStmtNode)
IntegerVariableDeclStmtNode(IdentExprNode::Ptr id, const string& bits)
: VariableDeclStmtNode(move(id), VariableDeclStmtNode::INTEGER, strtoul(bits.c_str(), NULL, 0)) {}
};
class StructDeclStmtNode : public StmtNode {
public:
DECLARE(StructDeclStmtNode)
IdentExprNode::Ptr id_;
FormalList stmts_;
size_t bit_width_;
bool packed_;
StructDeclStmtNode(IdentExprNode::Ptr id, FormalList&& stmts = FormalList())
: id_(move(id)), stmts_(move(stmts)), bit_width_(0), packed_(false) {}
VariableDeclStmtNode* field(const string& name) const;
int indexof(const string& name) const;
bool is_packed() const { return packed_; }
};
class ParserStateStmtNode : public StmtNode {
public:
DECLARE(ParserStateStmtNode)
IdentExprNode::Ptr id_;
StmtNode* next_state_;
string scope_id_;
explicit ParserStateStmtNode(IdentExprNode::Ptr id)
: id_(move(id)) {}
static Ptr make(const IdentExprNode::Ptr& id) {
return Ptr(new ParserStateStmtNode(id->copy()));
}
string scoped_name() const { return scope_id_ + id_->name_; }
};
class StateDeclStmtNode : public StmtNode {
public:
DECLARE(StateDeclStmtNode)
struct Sub {
IdentExprNode::Ptr id_;
BlockStmtNode::Ptr block_;
ParserStateStmtNode::Ptr parser_;
Scopes::StateScope* scope_;
Sub(decltype(id_) id, decltype(block_) block, decltype(parser_) parser, decltype(scope_) scope)
: id_(move(id)), block_(move(block)), parser_(move(parser)), scope_(scope) {}
~Sub() { delete scope_; }
Sub(Sub&& other) : scope_(NULL) {
*this = move(other);
}
Sub& operator=(Sub&& other) {
if (this == &other) {
return *this;
}
id_ = move(other.id_);
block_ = move(other.block_);
parser_ = move(other.parser_);
std::swap(scope_, other.scope_);
return *this;
}
};
IdentExprNode::Ptr id_;
StmtNodeList init_;
string scope_id_;
ParserStateStmtNode::Ptr parser_;
vector<Sub> subs_;
StateDeclStmtNode() {}
StateDeclStmtNode(IdentExprNode::Ptr id, BlockStmtNode::Ptr block) : id_(move(id)) {
subs_.push_back(Sub(make_unique<IdentExprNode>(""), move(block), ParserStateStmtNode::Ptr(), NULL));
}
StateDeclStmtNode(IdentExprNode::Ptr id1, IdentExprNode::Ptr id2, BlockStmtNode::Ptr block)
: id_(move(id1)) {
subs_.push_back(Sub(move(id2), move(block), ParserStateStmtNode::Ptr(), NULL));
}
string scoped_name() const { return scope_id_ + id_->name_; }
vector<Sub>::iterator find_sub(const string& id) {
return find_if(subs_.begin(), subs_.end(), [&id] (const Sub& sub) {
if (sub.id_->name_ == id)
return true;
return false;
});
}
};
class MatchDeclStmtNode : public StmtNode {
public:
DECLARE(MatchDeclStmtNode)
IdentExprNode::Ptr id_;
FormalList formals_;
BlockStmtNode::Ptr block_;
MatchDeclStmtNode(IdentExprNode::Ptr id, FormalList&& formals, BlockStmtNode::Ptr block)
: id_(move(id)), formals_(move(formals)), block_(move(block)) {}
};
class MissDeclStmtNode : public StmtNode {
public:
DECLARE(MissDeclStmtNode)
IdentExprNode::Ptr id_;
FormalList formals_;
BlockStmtNode::Ptr block_;
MissDeclStmtNode(IdentExprNode::Ptr id, FormalList&& formals, BlockStmtNode::Ptr block)
: id_(move(id)), formals_(move(formals)), block_(move(block)) {}
};
class FailureDeclStmtNode : public StmtNode {
public:
DECLARE(FailureDeclStmtNode)
IdentExprNode::Ptr id_;
FormalList formals_;
BlockStmtNode::Ptr block_;
FailureDeclStmtNode(IdentExprNode::Ptr id, FormalList&& formals, BlockStmtNode::Ptr block)
: id_(move(id)), formals_(move(formals)), block_(move(block)) {}
};
class TableDeclStmtNode : public StmtNode {
public:
DECLARE(TableDeclStmtNode)
IdentExprNode::Ptr table_type_;
IdentExprNodeList templates_;
IdentExprNode::Ptr id_;
StructDeclStmtNode *key_type_;
StructDeclStmtNode *leaf_type_;
IdentExprNode * key_id() { return templates_.at(0).get(); }
IdentExprNode * leaf_id() { return templates_.at(1).get(); }
IdentExprNode * type_id() { return templates_.at(2).get(); }
IdentExprNode * policy_id() { return templates_.at(3).get(); }
size_t size_;
TableDeclStmtNode(IdentExprNode::Ptr table_type, IdentExprNodeList&& templates,
IdentExprNode::Ptr id, string* size)
: table_type_(move(table_type)), templates_(move(templates)), id_(move(id)),
key_type_(nullptr), leaf_type_(nullptr), size_(strtoul(size->c_str(), NULL, 0)) {
delete size;
}
};
class FuncDeclStmtNode : public StmtNode {
public:
DECLARE(FuncDeclStmtNode)
IdentExprNode::Ptr id_;
FormalList formals_;
BlockStmtNode::Ptr block_;
Scopes::StateScope* scope_;
FuncDeclStmtNode(IdentExprNode::Ptr id, FormalList&& formals, BlockStmtNode::Ptr block)
: id_(move(id)), formals_(move(formals)), block_(move(block)), scope_(NULL) {}
};
class Visitor {
public:
typedef StatusTuple Ret;
virtual ~Visitor() {}
#define VISIT(type, func) virtual STATUS_RETURN visit_##func(type* n) = 0;
EXPAND_NODES(VISIT)
#undef VISIT
};
#undef DECLARE
} // namespace cc
} // namespace ebpf