You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
237 lines
6.3 KiB
237 lines
6.3 KiB
/*
|
|
* Copyright 2002-2019 Intel Corporation.
|
|
*
|
|
* This software is provided to you as Sample Source Code as defined in the accompanying
|
|
* End User License Agreement for the Intel(R) Software Development Products ("Agreement")
|
|
* section 1.L.
|
|
*
|
|
* This software and the related documents are provided as is, with no express or implied
|
|
* warranties, other than those that are expressly stated in the License.
|
|
*/
|
|
|
|
#include <iostream>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include "pin.H"
|
|
using std::endl;
|
|
|
|
VOID * lastInstPtr;
|
|
BOOL isPredictedTaken;
|
|
VOID * predictedInstPtr;
|
|
UINT64 icount = 0;
|
|
UINT64 errors = 0;
|
|
BOOL isSkipped = TRUE; // always skip checking the first inst
|
|
|
|
// The tool assumes single-threaded application.
|
|
// This may not be the case on Windows 10.
|
|
// We arbitrary choose single thread to profile.
|
|
THREADID myThread = INVALID_THREADID;
|
|
|
|
ADDRINT IfMyThread(THREADID threadId)
|
|
{
|
|
// Profile only single thread at any time
|
|
return threadId == myThread;
|
|
}
|
|
|
|
VOID ThreadStart(THREADID tid, CONTEXT *ctxt, INT32 flags, VOID *v)
|
|
{
|
|
// Determine single thread to profile.
|
|
if (myThread == INVALID_THREADID) myThread = tid;
|
|
}
|
|
|
|
|
|
VOID CountError()
|
|
{
|
|
errors++;
|
|
if (errors > 100)
|
|
{
|
|
std::cerr << "Too many errors, giving up\n";
|
|
exit(errors);
|
|
}
|
|
}
|
|
|
|
VOID CheckFlow(VOID * instPtr, INT32 isTaken, VOID * fallthroughAddr, VOID * takenAddr, UINT32 stutters)
|
|
{
|
|
isPredictedTaken = isTaken;
|
|
|
|
icount++;
|
|
|
|
//fprintf(stderr,"Current: %p isTaken %d fallthroughAddr %p takenAddr %p\n", instPtr, isTaken, fallthroughAddr, takenAddr);
|
|
|
|
if (predictedInstPtr != instPtr && !isSkipped &&
|
|
!(stutters && (instPtr == lastInstPtr))) // An instruction which stutters can stay at the same IP
|
|
{
|
|
fprintf(stderr,"From: %p predicted InstPtr %p, actual InstPtr %p\n", lastInstPtr, predictedInstPtr, instPtr);
|
|
CountError();
|
|
}
|
|
|
|
isSkipped = FALSE;
|
|
|
|
if (isTaken)
|
|
{
|
|
predictedInstPtr = takenAddr;
|
|
}
|
|
else
|
|
{
|
|
predictedInstPtr = fallthroughAddr;
|
|
}
|
|
|
|
lastInstPtr = instPtr;
|
|
}
|
|
|
|
VOID Taken()
|
|
{
|
|
if (!isPredictedTaken)
|
|
{
|
|
fprintf(stderr,"%p taken but not predictedInstPtr\n", lastInstPtr);
|
|
CountError();
|
|
}
|
|
}
|
|
|
|
VOID Skip()
|
|
{
|
|
isSkipped = TRUE;
|
|
}
|
|
|
|
TLS_KEY ea_tls_key;
|
|
|
|
VOID SaveEa(THREADID tid, VOID * ea)
|
|
{
|
|
PIN_SetThreadData(ea_tls_key, ea, tid);
|
|
}
|
|
|
|
VOID CheckXlatAfter(THREADID tid, ADDRINT eax)
|
|
{
|
|
VOID * ea = PIN_GetThreadData(ea_tls_key, tid);
|
|
int actual = *(char*)&eax;
|
|
int expected = *(char*)ea;
|
|
if (expected != actual)
|
|
{
|
|
fprintf(stderr, "xlat actual %d expected %d\n",actual,expected);
|
|
errors++;
|
|
}
|
|
}
|
|
|
|
#if defined(TARGET_IA32)
|
|
VOID CheckXlat(INS ins)
|
|
{
|
|
if (INS_Opcode(ins) != XED_ICLASS_XLAT)
|
|
return;
|
|
INS_InsertCall(ins, IPOINT_BEFORE, AFUNPTR(SaveEa), IARG_THREAD_ID, IARG_MEMORYREAD_EA, IARG_END);
|
|
INS_InsertCall(ins, IPOINT_AFTER, AFUNPTR(CheckXlatAfter), IARG_THREAD_ID, IARG_REG_VALUE, REG_EAX, IARG_END);
|
|
}
|
|
#else
|
|
VOID CheckXlat(INS ins)
|
|
{}
|
|
#endif
|
|
|
|
|
|
VOID Instruction(INS ins, VOID *v)
|
|
{
|
|
CheckXlat(ins);
|
|
|
|
if (INS_HasFallThrough(ins) && INS_IsControlFlow(ins))
|
|
{
|
|
INS_InsertIfCall(ins, IPOINT_BEFORE, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_BEFORE, (AFUNPTR)CheckFlow,
|
|
IARG_INST_PTR,
|
|
IARG_BRANCH_TAKEN,
|
|
IARG_FALLTHROUGH_ADDR,
|
|
IARG_BRANCH_TARGET_ADDR,
|
|
IARG_UINT32, INS_Stutters(ins),
|
|
IARG_END);
|
|
}
|
|
if (!INS_HasFallThrough(ins) && INS_IsControlFlow(ins))
|
|
{
|
|
/*
|
|
* In this case there is no FallThrough address.
|
|
* It is simply handled by putting 0 as the fallthroughAddr (the actual value does not matter,
|
|
* because this control flow instruction is always taken).
|
|
*/
|
|
INS_InsertIfCall(ins, IPOINT_BEFORE, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_BEFORE, (AFUNPTR)CheckFlow,
|
|
IARG_INST_PTR,
|
|
IARG_BRANCH_TAKEN,
|
|
IARG_ADDRINT, 0,
|
|
IARG_BRANCH_TARGET_ADDR,
|
|
IARG_UINT32, INS_Stutters(ins),
|
|
IARG_END);
|
|
}
|
|
if (INS_HasFallThrough(ins) && !INS_IsControlFlow(ins))
|
|
{
|
|
INS_InsertIfCall(ins, IPOINT_BEFORE, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_BEFORE, (AFUNPTR)CheckFlow,
|
|
IARG_INST_PTR,
|
|
IARG_BRANCH_TAKEN,
|
|
IARG_FALLTHROUGH_ADDR,
|
|
IARG_ADDRINT, 0,
|
|
IARG_UINT32, INS_Stutters(ins),
|
|
IARG_END);
|
|
}
|
|
if (!INS_HasFallThrough(ins) && !INS_IsControlFlow(ins))
|
|
{
|
|
INS_InsertIfCall(ins, IPOINT_BEFORE, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_BEFORE, (AFUNPTR)CheckFlow,
|
|
IARG_INST_PTR,
|
|
IARG_BRANCH_TAKEN,
|
|
IARG_ADDRINT, 0,
|
|
IARG_ADDRINT, 0,
|
|
IARG_UINT32, INS_Stutters(ins),
|
|
IARG_END);
|
|
}
|
|
|
|
if (INS_IsValidForIpointTakenBranch(ins))
|
|
{
|
|
INS_InsertIfCall(ins, IPOINT_TAKEN_BRANCH, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_TAKEN_BRANCH, (AFUNPTR)Taken, IARG_END);
|
|
}
|
|
|
|
#if defined(TARGET_IA32) || defined(TARGET_IA32E)
|
|
if (INS_IsSysenter(ins))
|
|
{ // sysenter on x86 has some funny control flow that we can't correctly verify for now
|
|
INS_InsertIfCall(ins, IPOINT_BEFORE, AFUNPTR(IfMyThread), IARG_THREAD_ID, IARG_END);
|
|
INS_InsertThenCall(ins, IPOINT_BEFORE, (AFUNPTR)Skip, IARG_END);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
VOID SyscallEntry(THREADID threadIndex, CONTEXT *ctxt, SYSCALL_STANDARD std, VOID *v)
|
|
{
|
|
// Profile only single thread
|
|
if (myThread != threadIndex) return;
|
|
|
|
Skip();
|
|
}
|
|
|
|
VOID Fini(INT32 code, VOID *v)
|
|
{
|
|
if (code)
|
|
{
|
|
exit(code);
|
|
}
|
|
|
|
std::cerr << errors << " errors (" << icount << " instructions checked)" << endl;
|
|
exit(errors);
|
|
}
|
|
|
|
int main(INT32 argc, CHAR **argv)
|
|
{
|
|
PIN_Init(argc, argv);
|
|
// Use symbols to test handling of RTN of size 200000
|
|
PIN_InitSymbols();
|
|
|
|
ea_tls_key = PIN_CreateThreadDataKey(0);
|
|
|
|
INS_AddInstrumentFunction(Instruction, 0);
|
|
PIN_AddSyscallEntryFunction(SyscallEntry, 0);
|
|
|
|
// Add callbacks
|
|
PIN_AddThreadStartFunction(ThreadStart, 0);
|
|
PIN_AddFiniFunction(Fini, 0);
|
|
|
|
// Never returns
|
|
PIN_StartProgram();
|
|
|
|
return 0;
|
|
}
|