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
|
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SKSL_SYMBOL
#define SKSL_SYMBOL
#include "include/private/SkSLIRNode.h"
#include "include/private/SkSLProgramElement.h"
namespace SkSL {
/**
* Represents a symboltable entry.
*/
class Symbol : public IRNode {
public:
enum class Kind {
kExternal = (int) ProgramElement::Kind::kLast + 1,
kField,
kFunctionDeclaration,
kType,
kUnresolvedFunction,
kVariable,
kFirst = kExternal,
kLast = kVariable
};
Symbol(int offset, Kind kind, skstd::string_view name, const Type* type = nullptr)
: INHERITED(offset, (int) kind)
, fName(name)
, fType(type) {
SkASSERT(kind >= Kind::kFirst && kind <= Kind::kLast);
}
~Symbol() override {}
const Type& type() const {
SkASSERT(fType);
return *fType;
}
Kind kind() const {
return (Kind) fKind;
}
skstd::string_view name() const {
return fName;
}
/**
* Use is<T> to check the type of a symbol.
* e.g. replace `sym.kind() == Symbol::Kind::kVariable` with `sym.is<Variable>()`.
*/
template <typename T>
bool is() const {
return this->kind() == T::kSymbolKind;
}
/**
* Use as<T> to downcast symbols. e.g. replace `(Variable&) sym` with `sym.as<Variable>()`.
*/
template <typename T>
const T& as() const {
SkASSERT(this->is<T>());
return static_cast<const T&>(*this);
}
template <typename T>
T& as() {
SkASSERT(this->is<T>());
return static_cast<T&>(*this);
}
private:
skstd::string_view fName;
const Type* fType;
using INHERITED = IRNode;
friend class Type;
};
} // namespace SkSL
#endif
|