/* * 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. */ // This tool demonstrates the use of PIN CONTEXT IARG and to examine the performance // change between the implementations. #include #include #include #include "pin.H" #include using std::string; using std::endl; using std::ofstream; ///////////////////// // GLOBAL VARIABLES ///////////////////// // A knob for defining the output file name KNOB KnobOutputFile(KNOB_MODE_WRITEONCE, "pintool", "o", "context.out", "specify output file name"); // A knob for selecting the needed CONTEXT KNOB KnobContextType(KNOB_MODE_WRITEONCE, "pintool", "c", "0" , "specify context to use (0-partial no regs in/out 1-const 2-canonical 3-partial all regs in/out 4-no context)"); KNOB KnobVerbose(KNOB_MODE_WRITEONCE, "pintool", "v", "0" , "verbose mode (specify 1 for verbose, default is 0)"); // ofstream object for handling the output ofstream OutFile; ///////////////////// // ANALYSIS FUNCTIONS ///////////////////// static void lookupRegister(CONTEXT * ctxt, ADDRINT iaddr, THREADID tid) { // For the integer registers, it is safe to use ADDRINT. But make sure to pass a pointer to it. ADDRINT val = PIN_GetContextReg(ctxt, REG_GDX); if (KnobVerbose.Value()) OutFile << hexstr(val) << " at addr " << hexstr(iaddr) << " " << decstr(tid) << endl; } static void lookupRegisterNoContext(ADDRINT iaddr, THREADID tid) { if (KnobVerbose.Value()) OutFile << "noContext " << " at addr " << hexstr(iaddr) << " " << decstr(tid) << endl; } ///////////////////// // INSTRUMENTATION FUNCTIONS ///////////////////// static VOID Trace(TRACE trace, VOID *v) { const static REGSET allRegs = PIN_GetFullContextRegsSet(); if (0 == KnobContextType.Value()) { REGSET regsin; REGSET regsout; TRACE_InsertCall(trace, IPOINT_BEFORE, (AFUNPTR)lookupRegister, IARG_PARTIAL_CONTEXT, ®sin, ®sout, IARG_INST_PTR, IARG_THREAD_ID, IARG_END); } else if (1 == KnobContextType.Value()) { TRACE_InsertCall(trace, IPOINT_BEFORE, (AFUNPTR)lookupRegister, IARG_CONST_CONTEXT, IARG_INST_PTR, IARG_THREAD_ID, IARG_END); } else if (3 == KnobContextType.Value()) { TRACE_InsertCall(trace, IPOINT_BEFORE, (AFUNPTR)lookupRegister, IARG_PARTIAL_CONTEXT, &allRegs, &allRegs, IARG_INST_PTR, IARG_THREAD_ID, IARG_END); } else if (4 == KnobContextType.Value()) { TRACE_InsertCall(trace, IPOINT_BEFORE, (AFUNPTR)lookupRegisterNoContext, IARG_INST_PTR, IARG_THREAD_ID, IARG_END); } else { TRACE_InsertCall(trace, IPOINT_BEFORE, (AFUNPTR)lookupRegister, IARG_CONTEXT, IARG_INST_PTR, IARG_THREAD_ID, IARG_END); } } ///////////////////// // MAIN FUNCTION ///////////////////// int main(int argc, char * argv[]) { // Initialize Pin PIN_Init(argc, argv); // Open the output file OutFile.open(KnobOutputFile.Value().c_str()); assert(OutFile.is_open()); // Add instrumentation TRACE_AddInstrumentFunction(Trace, 0); // Start running the application PIN_StartProgram(); // never returns return 1; }