Static Value-Flow Analysis
Loading...
Searching...
No Matches
BreakConstantExpr.cpp
Go to the documentation of this file.
1/* SVF - Static Value-Flow Analysis Framework
2Copyright (C) 2015 Yulei Sui
3Copyright (C) 2015 Jingling Xue
4
5This library is free software; you can redistribute it and/or
6modify it under the terms of the GNU Lesser General Public
7License as published by the Free Software Foundation; either
8version 2.1 of the License, or (at your option) any later version.
9
10This library is distributed in the hope that it will be useful,
11but WITHOUT ANY WARRANTY; without even the implied warranty of
12MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13Lesser General Public License for more details.
14
15You should have received a copy of the GNU Lesser General Public
16License along with this library; if not, write to the Free Software
17Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18*/
19
20/*
21 * BreakConstantExpr.cpp
22 *
23 * Created on: Oct 8, 2013
24 * Author: Yulei Sui
25 */
26
27
28
29//===- BreakConstantGEPs.cpp - Change constant GEPs into GEP instructions - --//
30//
31// The SAFECode Compiler
32//
33// This file was developed by the LLVM research group and is distributed under
34// the University of Illinois Open Source License. See LICENSE.TXT for details.
35//
36//===----------------------------------------------------------------------===//
37//
38// This pass changes all GEP constant expressions into GEP instructions. This
39// permits the rest of SAFECode to put run-time checks on them if necessary.
40//
41//===----------------------------------------------------------------------===//
42
43
44#include "llvm/ADT/Statistic.h"
45#include "llvm/IR/Constants.h"
46#include "llvm/IR/InstrTypes.h"
47#include "llvm/IR/Instruction.h"
48#include "llvm/IR/Instructions.h"
49#include "llvm/IR/LLVMContext.h"
50#include "llvm/IR/InstIterator.h"
51
52#include "SVF-LLVM/BasicTypes.h"
54#include "Util/GeneralType.h"
55#include "Util/SVFUtil.h"
56
57#include <iostream>
58#include <map>
59#include <utility>
60
61using namespace SVF;
62
63// Identifier variable for the pass
66
67#define DEBUG_TYPE "break-constgeps"
68
69// Statistics
70STATISTIC (GEPChanges, "Number of Converted GEP Constant Expressions");
71STATISTIC (TotalChanges, "Number of Converted Constant Expressions");
72
73//
74// Function: hasConstantGEP()
75//
76// Description:
77// This function determines whether the given value is a constant expression
78// that has a constant GEP expression embedded within it.
79//
80// Inputs:
81// V - The value to check.
82//
83// Return value:
84// nullptr - This value is not a constant expression with a constant expression
85// GEP within it.
86// ~nullptr - A pointer to the value casted into a ConstantExpr is returned.
87//
88static ConstantExpr *
90{
91 if (ConstantExpr * CE = SVFUtil::dyn_cast<ConstantExpr>(V))
92 {
93 if (CE->getOpcode() == Instruction::GetElementPtr)
94 {
95 return CE;
96 }
97 else
98 {
99 for (u32_t index = 0; index < CE->getNumOperands(); ++index)
100 {
101 if (hasConstantGEP (CE->getOperand(index)))
102 return CE;
103 }
104 }
105 }
106
107 return nullptr;
108}
109
110// Description:
111// This function determines whether the given value is a constant expression
112// that has a constant binary or unary operator expression embedded within it.
113static ConstantExpr *
115{
116 if (ConstantExpr * CE = SVFUtil::dyn_cast<ConstantExpr>(V))
117 {
118 if (Instruction::isBinaryOp(CE->getOpcode()) || Instruction::isUnaryOp(CE->getOpcode()))
119 {
120 return CE;
121 }
122 else
123 {
124 for (u32_t index = 0; index < CE->getNumOperands(); ++index)
125 {
126 if (hasConstantBinaryOrUnaryOp (CE->getOperand(index)))
127 return CE;
128 }
129 }
130 }
131
132 return nullptr;
133}
134
135// Description:
136// Return true if this is a constant Gep or binaryOp or UnaryOp expression
137static ConstantExpr *
139{
141 {
142 return gep;
143 }
145 {
146 return buop;
147 }
148 else
149 {
150 return nullptr;
151 }
152}
153
154
155//
156// Function: convertExpression()
157//
158// Description:
159// Convert a constant expression into an instruction. This routine does *not*
160// perform any recursion, so the resulting instruction may have constant
161// expression operands.
162//
163static Instruction*
165{
166 //
167 // Convert this constant expression into a regular instruction.
168 //
169 if (CE->getOpcode() == Instruction::GetElementPtr)
170 ++GEPChanges;
171 ++TotalChanges;
172 Instruction* Result = CE->getAsInstruction();
173 Result->insertBefore(InsertPt);
174 return Result;
175}
176
177//
178// Method: runOnFunction()
179//
180// Description:
181// Entry point for this LLVM pass.
182//
183// Return value:
184// true - The function was modified.
185// false - The function was not modified.
186//
187bool
189{
190 bool modified = false;
191 for (Module::iterator F = module.begin(), E = module.end(); F != E; ++F)
192 {
193 // Worklist of values to check for constant GEP expressions
194 std::vector<Instruction* > Worklist;
195
196 //
197 // Initialize the worklist by finding all instructions that have one or more
198 // operands containing a constant GEP expression.
199 //
200 for (Function::iterator BB = (*F).begin(); BB != (*F).end(); ++BB)
201 {
202 for (BasicBlock::iterator i = BB->begin(); i != BB->end(); ++i)
203 {
204 //
205 // Scan through the operands of this instruction. If it is a constant
206 // expression GEP, insert an instruction GEP before the instruction.
207 //
208 Instruction* I = &(*i);
209 for (u32_t index = 0; index < I->getNumOperands(); ++index)
210 {
211 if (hasConstantExpr(I->getOperand(index)))
212 {
213 Worklist.push_back (I);
214 }
215 }
216 }
217 }
218
219 //
220 // Determine whether we will modify anything.
221 //
222 if (Worklist.size()) modified = true;
223
224 //
225 // While the worklist is not empty, take an item from it, convert the
226 // operands into instructions if necessary, and determine if the newly
227 // added instructions need to be processed as well.
228 //
229 while (Worklist.size())
230 {
231 Instruction* I = Worklist.back();
232 Worklist.pop_back();
233
234 //
235 // Scan through the operands of this instruction and convert each into an
236 // instruction. Note that this works a little differently for phi
237 // instructions because the new instruction must be added to the
238 // appropriate predecessor block.
239 //
240 if (PHINode * PHI = SVFUtil::dyn_cast<PHINode>(I))
241 {
242 for (u32_t index = 0; index < PHI->getNumIncomingValues(); ++index)
243 {
244 //
245 // For PHI Nodes, if an operand is a constant expression with a GEP, we
246 // want to insert the new instructions in the predecessor basic block.
247 //
248 // Note: It seems that it's possible for a phi to have the same
249 // incoming basic block listed multiple times; this seems okay as long
250 // the same value is listed for the incoming block.
251 //
252 Instruction* InsertPt = PHI->getIncomingBlock(index)->getTerminator();
253 if (ConstantExpr * CE = hasConstantExpr(PHI->getIncomingValue(index)))
254 {
256 for (u32_t i2 = index; i2 < PHI->getNumIncomingValues(); ++i2)
257 {
258 if ((PHI->getIncomingBlock (i2)) == PHI->getIncomingBlock (index))
259 PHI->setIncomingValue (i2, NewInst);
260 }
261 Worklist.push_back (NewInst);
262 }
263 }
264 }
265 else
266 {
267 for (u32_t index = 0; index < I->getNumOperands(); ++index)
268 {
269 //
270 // For other instructions, we want to insert instructions replacing
271 // constant expressions immediately before the instruction using the
272 // constant expression.
273 //
274 if (ConstantExpr * CE = hasConstantExpr(I->getOperand(index)))
275 {
277 I->replaceUsesOfWith (CE, NewInst);
278 Worklist.push_back (NewInst);
279 }
280 }
281 }
282 }
283
284 }
285 return modified;
286}
287
288
289
290
static Instruction * convertExpression(ConstantExpr *CE, Instruction *InsertPt)
static ConstantExpr * hasConstantExpr(Value *V)
STATISTIC(GEPChanges, "Number of Converted GEP Constant Expressions")
static ConstantExpr * hasConstantBinaryOrUnaryOp(Value *V)
static ConstantExpr * hasConstantGEP(Value *V)
unsigned u32_t
Definition CommandLine.h:18
int index
Definition cJSON.h:170
virtual bool runOnModule(Module &M)
for isBitcode
Definition BasicTypes.h:68
llvm::Instruction Instruction
Definition BasicTypes.h:87
llvm::Value Value
LLVM Basic classes.
Definition BasicTypes.h:82
llvm::ConstantExpr ConstantExpr
Definition BasicTypes.h:120
llvm::IRBuilder IRBuilder
Definition BasicTypes.h:74
llvm::Module Module
Definition BasicTypes.h:84
llvm::PHINode PHINode
Definition BasicTypes.h:165
unsigned u32_t
Definition GeneralType.h:47