8 #include <unordered_map>
9 #include <unordered_set>
131 return "push " + std::to_string(
word);
143 using Prog = std::vector<Instr>;
148 using Mem = std::unordered_map<Word32, Word32>;
154 std::unordered_map<int, float>
169 std::unordered_map<int, float> result =
val;
170 for (
auto &r : result) {
171 if (other.
val.find(r.first) != other.
val.end()) {
172 result.at(r.first) += other.
val.at(r.first);
175 for (
auto &r : other.
val) {
176 if (result.find(r.first) == result.end()) {
177 result.emplace(std::make_pair(r.first, r.second));
190 std::unordered_map<int, float> result =
val;
191 for (
auto &r : result) {
192 result.at(r.first) += w;
203 std::unordered_map<int, float> result =
val;
204 for (
auto &r : result) {
205 if (other.
val.find(r.first) != other.
val.end()) {
206 result.at(r.first) -= other.
val.at(r.first);
209 for (
auto &r : other.
val) {
210 if (result.find(r.first) == result.end()) {
211 result.emplace(std::make_pair(r.first, -1.0f * r.second));
224 std::unordered_map<int, float> result =
val;
225 for (
auto &r : result) {
226 result.at(r.first) *= w;
236 std::unordered_map<int, float> result =
val;
237 for (
auto &r : result) {
238 result.emplace(std::make_pair(r.first, std::abs(r.second)));
271 std::unordered_map<int, float>
358 result.erase(a.first);
394 Sym *tmp_left, *tmp_right;
398 if (cvals.find(
var_idx) != cvals.end()) {
499 float eval(
const std::unordered_map<int, float> &cvals,
500 const float eps)
const {
535 std::unordered_map<int, float> nvals(cvals);
536 for (
const auto &a :
assign) {
537 nvals.emplace(a.first, a.second);
539 if (
left->
eval(nvals, eps) <= 0.0f) {
548 if (cvals.find(
var_idx) != cvals.end()) {
552 "\x1b[33m Warning!! var_%d should be specified to "
553 "correctly evaluate this "
554 "symbolic expression\x1b[39m\n",
560 return std::abs(
left->
eval(cvals, eps) -
564 return left->
eval(cvals, eps) * (-1.0f) + eps;
567 return std::max(
left->
eval(cvals, eps),
571 return std::min(
left->
eval(cvals, eps),
593 Grad grad(
const std::unordered_map<int, float> &cvals,
594 const float eps)
const {
630 std::unordered_map<int, float> nvals(cvals);
631 for (
const auto &a :
assign) {
632 nvals.emplace(a.first, a.second);
634 if (
left->
eval(nvals, eps) <= 0.0f) {
643 std::unordered_map<int, float> tmp = {{
var_idx, 1.0f}};
653 }
else if (lv > rv) {
660 return left->
grad(cvals, eps) * (-1.0f);
695 std::string result =
"";
715 if (convert_to_num) {
718 result += std::to_string((
int)tmp_word);
720 result += std::to_string(tmp_word);
723 result += std::to_string(
word);
731 for (
const auto &a :
assign) {
732 result += std::to_string(a.first) +
"->";
733 float tmp = a.second;
735 result += std::to_string((
int)tmp) +
",";
737 result += std::to_string(tmp) +
",";
746 result +=
"var_" + std::to_string(
var_idx);
814 float p = 1.0f / (1.0f + (float)
high - (
float)
low);
815 for (
int i =
low; i <=
high; i++) {
816 vals.emplace_back(i);
817 probs.emplace_back(p);
846 std::vector<std::vector<long long>> v(
n + 1,
847 std::vector<long long>(
n + 1, 0));
848 for (
int i = 0; i < v.size(); i++) {
852 for (
int j = 1; j < v.size(); j++) {
853 for (
int k = 1; k < j; k++) {
854 v[j][k] = (v[j - 1][k - 1] + v[j - 1][k]);
858 for (
int i = 0; i <=
n; i++) {
859 vals.emplace_back(i);
860 probs.emplace_back((
float)v[
n][i] * std::pow(
p, (
float)i) *
861 std::pow(1.0f -
p, (
float)
n - (
float)i));
870 using SMem = std::unordered_map<Word32, Sym>;
962 std::unordered_map<int, DiscreteDist> &var2dist,
963 std::vector<std::vector<int>> &D) {
964 int total_num_pvar_combinations = D.size();
968 for (
int i = 0; i < total_num_pvar_combinations; i++) {
971 std::unordered_map<int, float> tmp_assign;
972 for (
auto &vd : var2dist) {
973 tmp_assign.emplace(vd.first, D[i][j]);
974 tmp_p *= vd.second.probs[j];
982 q_denominator =
new Sym(
988 return std::make_pair(q_numerator, q_denominator);
1001 float eval(std::unordered_map<int, float> ¶ms,
float eps,
1002 std::unordered_map<int, DiscreteDist> &var2dist,
1003 std::vector<std::vector<int>> &D) {
1004 std::pair<Sym *, Sym *> mq =
marginalize(var2dist, D);
1005 Sym *q_n = mq.first;
1006 Sym *q_d = mq.second;
1008 float v_n = q_n->
eval(params, eps);
1009 float v_d = q_d->
eval(params, eps);
1029 std::unordered_map<int, DiscreteDist> &var2dist,
1030 std::vector<std::vector<int>> &D) {
1031 std::pair<Sym *, Sym *> mq =
marginalize(var2dist, D);
1032 Sym *q_left = mq.first;
1033 Sym *q_right = mq.second;
1035 return Sym(symtype, q_left, q_right);
1119 std::string
toString(
bool include_memory =
true)
const {
1120 std::string
expr =
"";
1122 if (include_memory) {
1123 expr +=
"Concrete Memory: {";
1125 for (
auto &t :
mem) {
1128 expr += (
"var_" + std::to_string(((
int)t.first)) +
": " +
1129 std::to_string((
int)tmp_word)) +
1132 expr += (
"var_" + std::to_string(((
int)t.first)) +
": " +
1133 std::to_string(tmp_word)) +
1139 expr +=
"Symbolic Memory: {";
1140 for (
auto &t :
smem) {
1141 expr +=
"var_" + std::to_string((
int)t.first) +
": " +
1142 t.second.toString(
true) +
", ";
1147 expr +=
"Path Constraints: ";
1165 while (tmp != NULL) {
1166 printf(
"%s, ", tmp->
data.toString(
false).c_str());
1188 std::unordered_map<std::string,
1189 std::pair<bool, std::unordered_map<int, float>>>;
1205 std::unordered_map<int, std::vector<std::tuple<Sym, Mem, SymProb>>>;
1227 printf(
"PC: %d\n",
data.
pc);
Class representing an instruction.
Definition: type.h:53
Instr(InstrType instr)
Constructor for an instruction without additional data.
Definition: type.h:62
Word32 word
Definition: type.h:56
InstrType instr
Definition: type.h:55
Instr(InstrType instr, Word32 word)
Constructor for an instruction with additional data.
Definition: type.h:69
void print() const
Prints a human-readable representation of the instruction.
Definition: type.h:74
std::string toString() const
Converts the instruction to a string representation.
Definition: type.h:80
Node for a Doubly Linked List.
Definition: utils.h:131
LLNode * next
Pointer to the next node in the linked list.
Definition: utils.h:134
T data
The data stored in the node.
Definition: utils.h:133
Doubly Linked List Implementation.
Definition: utils.h:156
Definition: compiler.h:11
float wordToFloat(uint32_t word)
Converts a 32-bit word representation to a float value.
Definition: utils.h:51
std::unordered_map< int, std::vector< std::tuple< Sym, Mem, SymProb > >> ProbPathConstraintsTable
Alias for a table of probabilistic path constraints.
Definition: type.h:1205
InstrType
Enum representing different instruction types.
Definition: type.h:24
bool is_integer(float x)
Checks if a float is an integer.
Definition: utils.h:25
SymType
Enum representing different symbolic expression types.
Definition: type.h:247
uint32_t Word32
Alias for 32-bit unsigned integer.
Definition: type.h:19
uint32_t FloatToWord(float val)
Converts a float value to a 32-bit word representation.
Definition: utils.h:36
std::unordered_map< Word32, Word32 > Mem
Alias for memory, represented as an unordered map of 32-bit words.
Definition: type.h:148
std::vector< Instr > Prog
Alias for a program, represented as a vector of instructions.
Definition: type.h:143
Node * expr(Token *&token, char *user_input)
Parses an expression from a C-like language program.
Definition: parser.h:210
std::unordered_map< Word32, Sym > SMem
Alias for symbolic memory, represented as an unordered map of symbolic expressions.
Definition: type.h:870
std::unordered_map< std::string, std::pair< bool, std::unordered_map< int, float > >> PathConstraintsTable
Alias for a table of path constraints.
Definition: type.h:1189
Represents a bernoulli distribution derived from DiscreteDist.
Definition: type.h:827
float p
Definition: type.h:828
BernoulliDist(float p)
Definition: type.h:830
Represents a binomial distribution derived from DiscreteDist.
Definition: type.h:841
int n
Definition: type.h:842
float p
Definition: type.h:843
BinomialDist(int n, float p)
Definition: type.h:845
Represents a discrete probability distribution.
Definition: type.h:787
std::vector< int > vals
Definition: type.h:788
DiscreteDist()
Default constructor for DiscreteDist.
Definition: type.h:794
std::vector< float > probs
Definition: type.h:789
Struct representing the gradient of a symbolic expression.
Definition: type.h:153
Grad abs()
Compute the absolute values of the gradient.
Definition: type.h:235
Grad operator-(const Grad &other)
Overloaded subtraction operator for subtracting two gradients.
Definition: type.h:202
Grad operator*(float w)
Overloaded multiplication operator for multiplying the gradient by a constant.
Definition: type.h:223
std::unordered_map< int, float > val
Definition: type.h:155
Grad(std::unordered_map< int, float > val)
Constructor for the gradient.
Definition: type.h:161
Grad operator+(float w)
Overloaded addition operator for adding a constant to the gradient.
Definition: type.h:189
Grad operator+(const Grad &other)
Overloaded addition operator for adding two gradients.
Definition: type.h:168
Represents a symbolic probability with a numerator and denominator.
Definition: type.h:878
SymProb operator*(SymProb other)
Multiplies two SymProb instances.
Definition: type.h:921
std::pair< Sym *, Sym * > marginalize(std::unordered_map< int, DiscreteDist > &var2dist, std::vector< std::vector< int >> &D)
Marginalizes the SymProb over given variable assignments.
Definition: type.h:961
std::string toString()
Converts SymProb to a string.
Definition: type.h:910
SymProb()
Default constructor for SymProb.
Definition: type.h:890
SymProb * pmul(SymProb *other)
Multiplies two SymProb instances.
Definition: type.h:940
Sym * denominator
Definition: type.h:881
Sym * numerator
Definition: type.h:879
Sym query(SymType symtype, Sym &other, std::unordered_map< int, DiscreteDist > &var2dist, std::vector< std::vector< int >> &D)
Queries the SymProb with a given symbolic type, another symbolic expression, and variable assignments...
Definition: type.h:1028
SymProb(Sym *numerator, Sym *denominator)
Parameterized constructor for SymProb.
Definition: type.h:902
float eval(std::unordered_map< int, float > ¶ms, float eps, std::unordered_map< int, DiscreteDist > &var2dist, std::vector< std::vector< int >> &D)
Evaluates the SymProb for given parameters, epsilon, variable assignments, and distributions.
Definition: type.h:1001
Struct representing the symbolic state of the symbolic execution.
Definition: type.h:1042
SymProb * p
Definition: type.h:1050
SymProb * cond_p
Definition: type.h:1051
bool has_observed_p_cond
Definition: type.h:1055
SymState(int pc, int var_cnt, Mem &mem, SMem &smem, Linkedlist< Sym > &symbolic_stack, std::vector< Sym > &path_constraints, SymProb *p, SymProb *cond_p, bool has_observed_p_cond)
Constructor for symbolic state with specified values.
Definition: type.h:1081
std::vector< Sym > path_constraints
Definition: type.h:1049
int var_cnt
Definition: type.h:1044
SymState * copy()
Create a copy object.
Definition: type.h:1098
void print() const
Prints a human-readable representation of the symbolic state.
Definition: type.h:1162
Linkedlist< Sym > symbolic_stack
Definition: type.h:1047
int pc
Definition: type.h:1043
SMem smem
Definition: type.h:1046
SymState()
Default constructor for symbolic state.
Definition: type.h:1060
std::string toString(bool include_memory=true) const
Returns the human-redable string representation of concrete memory, symbolic memory and path constrai...
Definition: type.h:1119
Mem mem
Definition: type.h:1045
void set_concrete_val(int var_id, float val)
Sets a concrete value for a variable in the symbolic state.
Definition: type.h:1108
Struct representing a symbolic expression.
Definition: type.h:265
std::unordered_map< int, float > assign
Definition: type.h:272
Sym(SymType symtype, Sym *left)
Constructor for Sym with a specified type and left child.
Definition: type.h:284
Grad grad(const std::unordered_map< int, float > &cvals, const float eps) const
Computes the gradient of the symbolic expression given concrete variable values.
Definition: type.h:593
Sym * right
Definition: type.h:268
Sym * psimplify(const Mem &cvals)
Simplifies the symbolic expression by evaluating constant subexpressions.
Definition: type.h:393
Sym * copy()
Definition: type.h:319
Sym(SymType symtype, Sym *left, Sym *right, Word32 word, int var_idx)
Definition: type.h:312
void gather_var_ids(std::unordered_set< int > &result) const
Gathers variable indices from the symbolic expression.
Definition: type.h:325
Sym(SymType symtype, Sym *left, std::unordered_map< int, float > &assign)
Definition: type.h:286
Sym(SymType symtype, Word32 val)
Constructor for Sym with a specified type and Word32 value.
Definition: type.h:304
SymType symtype
Definition: type.h:266
Sym * left
Definition: type.h:267
std::string toString(bool convert_to_num) const
Converts the symbolic expression to a string representation.
Definition: type.h:694
float eval(const std::unordered_map< int, float > &cvals, const float eps) const
Evaluates the symbolic expression given concrete variable values.
Definition: type.h:499
Sym()
Default constructor for Sym.
Definition: type.h:277
Sym(SymType symtype, Sym *left, Sym *right)
Constructor for Sym with a specified type, left child, and right child.
Definition: type.h:296
int var_idx
Definition: type.h:270
Word32 word
Definition: type.h:269
Struct representing a trace in symbolic execution.
Definition: type.h:1210
Trace(SymState data, std::vector< Trace > children)
Constructor for a trace.
Definition: type.h:1219
void print(int indent_cnt=0) const
Prints a human-readable representation of the trace.
Definition: type.h:1226
std::vector< Trace > children
Definition: type.h:1212
SymState data
Definition: type.h:1211
Utility funcitons and classes.