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.
189 lines
4.9 KiB
189 lines
4.9 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.
|
|
*/
|
|
|
|
/*
|
|
* Record the memory read and written by variable size instructions
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include "pin.H"
|
|
using std::string;
|
|
|
|
/* Introduce ambiguity into the definition of UINT32.
|
|
* Now we have two: LEVEL_BASE::UINT32 and breakUINT32::UINT32.
|
|
* Since there are no references to UINT32 in this code, it should be fine
|
|
* to do this, but this provides a check for Mantis 2344, which was
|
|
* that IARG_END used a naked UINT32, rather than LEVEL_BASE::UINT32.
|
|
*/
|
|
|
|
namespace breakUINT32 {
|
|
typedef void UINT32;
|
|
}
|
|
|
|
using namespace breakUINT32;
|
|
|
|
KNOB<string> KnobOutput(KNOB_MODE_WRITEONCE, "pintool", "o", "reptool.out", "output file");
|
|
|
|
FILE *out = 0;
|
|
UINT8 oldVal = 0;
|
|
UINT8 * ea;
|
|
|
|
// Change the first byte of string
|
|
VOID Change(UINT8 * addr)
|
|
{
|
|
ea = addr;
|
|
|
|
oldVal = *ea;
|
|
*ea = '1';
|
|
}
|
|
|
|
// Restore first byte of string
|
|
VOID Restore()
|
|
{
|
|
*ea = oldVal;
|
|
}
|
|
|
|
VOID SaveEa(UINT8 * addr)
|
|
{
|
|
ea = addr;
|
|
}
|
|
|
|
VOID RecordMemory(VOID * ip, ADDRINT testNum, char * type, UINT8 * addr, INT32 size)
|
|
{
|
|
// We pass addr == 0 at IPOINT_AFTER, and pick up the address we saved at IPOINT_BEFORE.
|
|
if (addr == 0)
|
|
{
|
|
addr = ea;
|
|
}
|
|
else
|
|
{
|
|
ea = addr;
|
|
}
|
|
|
|
fprintf(out, "Test %lu ", (unsigned long)testNum);
|
|
fprintf(out, "%s size %d: ",type,size);
|
|
for (INT32 i = 0; i < size; i++)
|
|
{
|
|
fprintf(out, "%c", addr[i]);
|
|
}
|
|
|
|
fprintf(out, "\n");
|
|
}
|
|
|
|
// Is called for every instruction and instruments reads and writes
|
|
VOID Trace(TRACE trace, VOID *v)
|
|
{
|
|
// Ignore everything that is not in the main function
|
|
RTN rtn = TRACE_Rtn(trace);
|
|
if (!RTN_Valid(rtn))
|
|
return;
|
|
|
|
if (RTN_Name(rtn) != "main")
|
|
return;
|
|
|
|
for (BBL bbl = TRACE_BblHead(trace); BBL_Valid(bbl); bbl = BBL_Next(bbl))
|
|
{
|
|
for (INS ins = BBL_InsHead(bbl); INS_Valid(ins); ins = INS_Next(ins))
|
|
{
|
|
if (!INS_HasRealRep(ins))
|
|
continue;
|
|
|
|
if (INS_IsMemoryRead(ins))
|
|
{
|
|
INS_InsertCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)Change,
|
|
IARG_MEMORYREAD_EA,
|
|
IARG_END);
|
|
|
|
INS_InsertPredicatedCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)RecordMemory,
|
|
IARG_INST_PTR,
|
|
IARG_REG_VALUE, REG_EDX,
|
|
IARG_PTR, "read1",
|
|
IARG_MEMORYREAD_EA,
|
|
IARG_MEMORYREAD_SIZE,
|
|
IARG_END);
|
|
|
|
INS_InsertCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)Restore,
|
|
IARG_END);
|
|
|
|
INS_InsertPredicatedCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)RecordMemory,
|
|
IARG_INST_PTR,
|
|
IARG_REG_VALUE, REG_EDX,
|
|
IARG_PTR, "read1",
|
|
IARG_MEMORYREAD_EA,
|
|
IARG_MEMORYREAD_SIZE,
|
|
IARG_END);
|
|
}
|
|
|
|
if (INS_HasMemoryRead2(ins))
|
|
{
|
|
INS_InsertPredicatedCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)RecordMemory,
|
|
IARG_INST_PTR,
|
|
IARG_REG_VALUE, REG_EDX,
|
|
IARG_PTR, "read2",
|
|
IARG_MEMORYREAD2_EA,
|
|
IARG_MEMORYREAD_SIZE,
|
|
// IARG_CONTEXT requires a bridge and tests liveness of
|
|
// helper virtual registers
|
|
IARG_CONTEXT,
|
|
IARG_END);
|
|
}
|
|
|
|
if (INS_IsMemoryWrite(ins))
|
|
{
|
|
INS_InsertCall(
|
|
ins, IPOINT_BEFORE, (AFUNPTR)SaveEa,
|
|
IARG_MEMORYWRITE_EA,
|
|
IARG_END);
|
|
|
|
INS_InsertPredicatedCall(
|
|
ins, IPOINT_AFTER, (AFUNPTR)RecordMemory,
|
|
IARG_INST_PTR,
|
|
IARG_REG_VALUE, REG_EDX,
|
|
IARG_PTR, "write",
|
|
IARG_ADDRINT, ADDRINT(0),
|
|
IARG_MEMORYWRITE_SIZE,
|
|
IARG_END);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
VOID Fini(INT32 code, VOID *v)
|
|
{
|
|
if (out != stdout)
|
|
fclose(out);
|
|
}
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
PIN_Init(argc, argv);
|
|
PIN_InitSymbols();
|
|
|
|
out = fopen(KnobOutput.Value().c_str(), "w");
|
|
if (!out) {
|
|
fprintf(stderr, "Can't open output file... using stdout\n");
|
|
out = stdout;
|
|
}
|
|
|
|
TRACE_AddInstrumentFunction(Trace, 0);
|
|
PIN_AddFiniFunction(Fini, 0);
|
|
|
|
// Never returns
|
|
PIN_StartProgram();
|
|
|
|
return 0;
|
|
}
|