aboutsummaryrefslogtreecommitdiffhomepage
path: root/lib/Core
diff options
context:
space:
mode:
authorDan Liew <delcypher@gmail.com>2014-01-12 06:41:32 -0800
committerDan Liew <delcypher@gmail.com>2014-01-12 06:41:32 -0800
commit8c03dfa5ea9fe5176cbb82b70a36ffa93c70b91c (patch)
treea78ec50217f5732bd3df084c5bdf38821570d2cc /lib/Core
parent3eff4a22c77d70e1c5c3193eaa0b825e8e0b4a3f (diff)
parent5b2dcbbcf91062e463a040d58302706c612f03bd (diff)
downloadklee-8c03dfa5ea9fe5176cbb82b70a36ffa93c70b91c.tar.gz
Merge pull request #68 from MartinNowack/feature_kleeInternalFunctions
Feature klee internal functions
Diffstat (limited to 'lib/Core')
-rw-r--r--lib/Core/Executor.cpp50
-rw-r--r--lib/Core/Executor.h5
2 files changed, 52 insertions, 3 deletions
diff --git a/lib/Core/Executor.cpp b/lib/Core/Executor.cpp
index ef55f21f..bf672bb7 100644
--- a/lib/Core/Executor.cpp
+++ b/lib/Core/Executor.cpp
@@ -2729,20 +2729,63 @@ void Executor::terminateStateOnExit(ExecutionState &state) {
terminateState(state);
}
+const InstructionInfo & Executor::getLastNonKleeInternalInstruction(const ExecutionState &state,
+ Instruction ** lastInstruction) {
+ // unroll the stack of the applications state and find
+ // the last instruction which is not inside a KLEE internal function
+ ExecutionState::stack_ty::const_reverse_iterator it = state.stack.rbegin(),
+ itE = state.stack.rend();
+
+ // don't check beyond the outermost function (i.e. main())
+ itE--;
+
+ const InstructionInfo * ii = 0;
+ if (kmodule->internalFunctions.count(it->kf->function) == 0){
+ ii = state.prevPC->info;
+ *lastInstruction = state.prevPC->inst;
+ // Cannot return yet because even though
+ // it->function is not an internal function it might of
+ // been called from an internal function.
+ }
+
+ // Wind up the stack and check if we are in a KLEE internal function.
+ // We visit the entire stack because we want to return a CallInstruction
+ // that was not reached via any KLEE internal functions.
+ for (;it != itE; ++it) {
+ // check calling instruction and if it is contained in a KLEE internal function
+ const Function * f = (*it->caller).inst->getParent()->getParent();
+ if (kmodule->internalFunctions.count(f)){
+ ii = 0;
+ continue;
+ }
+ if (!ii){
+ ii = (*it->caller).info;
+ *lastInstruction = (*it->caller).inst;
+ }
+ }
+
+ if (!ii) {
+ // something went wrong, play safe and return the current instruction info
+ *lastInstruction = state.prevPC->inst;
+ return *state.prevPC->info;
+ }
+ return *ii;
+}
void Executor::terminateStateOnError(ExecutionState &state,
const llvm::Twine &messaget,
const char *suffix,
const llvm::Twine &info) {
std::string message = messaget.str();
static std::set< std::pair<Instruction*, std::string> > emittedErrors;
- const InstructionInfo &ii = *state.prevPC->info;
+ Instruction * lastInst;
+ const InstructionInfo &ii = getLastNonKleeInternalInstruction(state, &lastInst);
if (EmitAllErrors ||
- emittedErrors.insert(std::make_pair(state.prevPC->inst, message)).second) {
+ emittedErrors.insert(std::make_pair(lastInst, message)).second) {
if (ii.file != "") {
klee_message("ERROR: %s:%d: %s", ii.file.c_str(), ii.line, message.c_str());
} else {
- klee_message("ERROR: %s", message.c_str());
+ klee_message("ERROR: (location information missing) %s", message.c_str());
}
if (!EmitAllErrors)
klee_message("NOTE: now ignoring this error at this location");
@@ -2752,6 +2795,7 @@ void Executor::terminateStateOnError(ExecutionState &state,
if (ii.file != "") {
msg << "File: " << ii.file << "\n";
msg << "Line: " << ii.line << "\n";
+ msg << "assembly.ll line: " << ii.assemblyLine << "\n";
}
msg << "Stack: \n";
state.dumpStack(msg);
diff --git a/lib/Core/Executor.h b/lib/Core/Executor.h
index b7318a2c..7d82332c 100644
--- a/lib/Core/Executor.h
+++ b/lib/Core/Executor.h
@@ -340,6 +340,11 @@ private:
/// Get textual information regarding a memory address.
std::string getAddressInfo(ExecutionState &state, ref<Expr> address) const;
+ // Determines the \param lastInstruction of the \param state which is not KLEE
+ // internal and returns its InstructionInfo
+ const InstructionInfo & getLastNonKleeInternalInstruction(const ExecutionState &state,
+ llvm::Instruction** lastInstruction);
+
// remove state from queue and delete
void terminateState(ExecutionState &state);
// call exit handler and terminate state