forked from cseagle/blc
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcover.hh
92 lines (87 loc) · 5.17 KB
/
cover.hh
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
/* ###
* IP: GHIDRA
*
* 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.
*/
/// \file cover.hh
/// \brief Classes describing the topological scope of variables within a function
#ifndef __CPUI_COVER__
#define __CPUI_COVER__
#include "type.hh"
class PcodeOp;
class FlowBlock;
class Varnode;
/// \brief The topological scope of a variable within a basic block
///
/// Within a basic block, the topological scope of a variable can be considered
/// a contiguous range of p-code operations. This range can be described with
/// a \e start and \e stop PcodeOp object, indicating all p-code operations between
/// the two inclusive. The \e start and \e stop may hold special encodings meaning:
/// - From the beginning of the block
/// - To the end of the block
class CoverBlock {
const PcodeOp *start; ///< Beginning of the range
const PcodeOp *stop; ///< End of the range
public:
CoverBlock(void) { start = (const PcodeOp *)0; stop = (const PcodeOp *)0; } ///< Construct empty/uncovered block
static uintm getUIndex(const PcodeOp *op); ///< Get the comparison index for a PcodeOp
const PcodeOp *getStart(void) const { return start; } ///< Get the start of the range
const PcodeOp *getStop(void) const { return stop; } ///< Get the end of the range
void clear(void) { start = (const PcodeOp *)0; stop = (const PcodeOp *)0; } ///< Clear \b this block to empty/uncovered
void setAll(void) {
start = (const PcodeOp *)0; stop = (const PcodeOp *)1; } ///< Mark whole block as covered
void setBegin(const PcodeOp *begin) {
start = begin; if (stop==(const PcodeOp *)0) stop = (const PcodeOp *)1; } ///< Reset start of range
void setEnd(const PcodeOp *end) { stop = end; } ///< Reset end of range
int4 intersect(const CoverBlock &op2) const; ///< Compute intersection with another CoverBlock
bool empty(void) const {
return ((start==(const PcodeOp *)0)&&(stop==(const PcodeOp *)0)); } ///< Return \b true if \b this is empty/uncovered
bool contain(const PcodeOp *point) const; ///< Check containment of given point
int4 boundary(const PcodeOp *point) const; ///< Characterize given point as boundary
void merge(const CoverBlock &op2); ///< Merge another CoverBlock into \b this
void print(ostream &s) const; ///< Dump a description to stream
};
/// \brief A description of the topological scope of a single variable object
///
/// The \b topological \b scope of a variable within a function is the set of
/// locations within the code of the function where that variable holds a variable.
/// For the decompiler, a high-level variable in this sense, HighVariable, is a collection
/// of Varnode objects. In order to merge Varnodes into a HighVariable, the topological
/// scope of each Varnode must not intersect because that would mean the high-level variable
/// holds different values at the same point in the function.
///
/// Internally this is implemented as a map from basic block to their non-empty CoverBlock
class Cover {
map<int4,CoverBlock> cover; ///< block index -> CoverBlock
static const CoverBlock emptyBlock; ///< Global empty CoverBlock for blocks not covered by \b this
void addRefRecurse(const FlowBlock *bl); ///< Fill-in \b this recursively from the given block
public:
void clear(void) { cover.clear(); } ///< Clear \b this to an empty Cover
int4 compareTo(const Cover &op2) const; ///< Give ordering of \b this and another Cover
const CoverBlock &getCoverBlock(int4 i) const; ///< Get the CoverBlock corresponding to the i-th block
int4 intersect(const Cover &op2) const; ///< Characterize the intersection between \b this and another Cover.
int4 intersectByBlock(int4 blk,const Cover &op2) const; ///< Characterize the intersection on a specific block
void intersectList(vector<int4> &listout,const Cover &op2,int4 level) const;
bool contain(const PcodeOp *op,int4 max) const;
int4 containVarnodeDef(const Varnode *vn) const;
void merge(const Cover &op2); ///< Merge \b this with another Cover block by block
void rebuild(const Varnode *vn); ///< Reset \b this based on def-use of a single Varnode
void addDefPoint(const Varnode *vn); ///< Reset to the single point where the given Varnode is defined
void addRefPoint(const PcodeOp *ref,const Varnode *vn); ///< Add a variable read to \b this Cover
// void remove_refpoint(const PcodeOp *ref,const Varnode *vn) {
// rebuild(vn); } // Cheap but inefficient
void print(ostream &s) const; ///< Dump a description of \b this cover to stream
map<int4,CoverBlock>::const_iterator begin(void) const { return cover.begin(); } ///< Get beginning of CoverBlocks
map<int4,CoverBlock>::const_iterator end(void) const { return cover.end(); } ///< Get end of CoverBlocks
};
#endif