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.
170 lines
4.7 KiB
170 lines
4.7 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 <stdlib.h>
|
|
#include <dlfcn.h>
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/syscall.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#define MAX_COMMAND_LINE_SIZE 15 // the size for the array of arguments to execv (this value is arbitrary)
|
|
#define EXPORT_SYM extern "C"
|
|
|
|
EXPORT_SYM bool AfterAttach1();
|
|
|
|
EXPORT_SYM bool AfterAttach2();
|
|
|
|
static int MAX_SIZE = 128; /*maximum line size*/
|
|
|
|
enum ExitType {
|
|
RES_SUCCESS = 0, // 0
|
|
RES_FORK_FAILED, // 1
|
|
RES_EXEC_FAILED, // 2
|
|
RES_LOAD_FAILED, // 3
|
|
RES_RES_INVALID_ARGS // 4
|
|
};
|
|
|
|
|
|
bool AfterAttach1()
|
|
{
|
|
// Pin sets an anslysis function here to notify the application when Pin attaches to it.
|
|
return false;
|
|
}
|
|
|
|
bool AfterAttach2()
|
|
{
|
|
// Pin sets an anslysis function here to notify the application when Pin attaches to it.
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
* block only the signals in the list: "signalsListToBlock"
|
|
*/
|
|
void BlockSignals(int signalsListToBlock[] , int len, sigset_t * sigmask)
|
|
{
|
|
sigemptyset(sigmask);
|
|
int i;
|
|
for(i=0; i< len; ++i)
|
|
sigaddset(sigmask, signalsListToBlock[i]);
|
|
pthread_sigmask(SIG_SETMASK, sigmask, NULL);
|
|
}
|
|
|
|
|
|
/*
|
|
Expected argv arguments:
|
|
[1] first image to load
|
|
[2] second image to load
|
|
[3] pin executable
|
|
[4] Pin flags (e.g. -slow_asserts)
|
|
>> zero or more flags possible
|
|
[5] "-probe"
|
|
[6] "-t"
|
|
[7] tool
|
|
[8] output file
|
|
[9] represent if SIGTRAP should be blocked by the application
|
|
argv[9]=0 - SIGTRAP shouldn't be blocked
|
|
argv[9]=1 - SIGTRAP should be blocked
|
|
*/
|
|
|
|
int main(int argc, char** argv)
|
|
{
|
|
if(argc < 9)
|
|
{
|
|
fprintf(stderr, "No enough arguments\n" );
|
|
fflush(stderr);
|
|
exit(RES_RES_INVALID_ARGS);
|
|
}
|
|
if (argc > MAX_COMMAND_LINE_SIZE){ // added: -pid attachPid -o NULL, omitted: argv[0..2], argv[9]
|
|
fprintf(stderr, "Too many arguments\n" );
|
|
fflush(stderr);
|
|
exit(RES_RES_INVALID_ARGS);
|
|
}
|
|
if (strcmp(argv[argc-1], "1") == 0) // Need to block the SIGTRAP signal
|
|
{
|
|
int sigList[1] = {SIGTRAP};
|
|
sigset_t sigmask;
|
|
BlockSignals(sigList, 1, &sigmask);
|
|
}
|
|
|
|
pid_t parentPid = getpid();
|
|
pid_t child = fork();
|
|
if (child < 0)
|
|
{
|
|
perror("fork failed while creating application process");
|
|
exit(RES_FORK_FAILED);
|
|
}
|
|
|
|
if (child)
|
|
{
|
|
// inside parent
|
|
while(!AfterAttach1())
|
|
{
|
|
sleep(1);
|
|
}
|
|
|
|
void *handle = dlopen(argv[1], RTLD_LAZY);
|
|
if (!handle)
|
|
{
|
|
fprintf(stderr, " Failed to load: %s because: %s\n", argv[1], dlerror());
|
|
fflush(stderr);
|
|
exit(RES_LOAD_FAILED);
|
|
}
|
|
|
|
while(!AfterAttach2())
|
|
{
|
|
sleep(1);
|
|
}
|
|
|
|
handle = dlopen(argv[2], RTLD_LAZY);
|
|
if (!handle)
|
|
{
|
|
fprintf(stderr, " Failed to load: %s because: %s\n", argv[2], dlerror());
|
|
fflush(stderr);
|
|
exit(RES_LOAD_FAILED);
|
|
}
|
|
|
|
while(1)
|
|
{
|
|
// expected to be stopped by tool.
|
|
sleep(1);
|
|
}
|
|
}
|
|
|
|
if ( child == 0 )
|
|
{
|
|
// inside child
|
|
char attachPid[MAX_SIZE];
|
|
snprintf(attachPid ,MAX_SIZE , "%d", parentPid);
|
|
char* args[MAX_COMMAND_LINE_SIZE] = {NULL}; // arguments for execv command
|
|
int args_count = 0;
|
|
int argv_count = 3; // to start from argv[3]...
|
|
args[args_count++] = argv[argv_count++]; // by convention, first arg is the filename of the executed file (pin)
|
|
args[args_count++] = (char*)"-pid";
|
|
args[args_count++] = attachPid;
|
|
while (strcmp(argv[argv_count], "-t") != 0){ // additional Pin flags (optional)
|
|
args[args_count++] = argv[argv_count++]; // including "-probe" (mandatory for test)
|
|
}
|
|
args[args_count++] = argv[argv_count++]; // "-t"
|
|
args[args_count++] = argv[argv_count++]; // tool
|
|
args[args_count++] = (char*)"-o";
|
|
args[args_count++] = argv[argv_count++]; // output file
|
|
args[args_count++] = NULL; // end
|
|
execv(argv[3], (char * const *)args); // never returns
|
|
perror("execv failed while trying to attach Pin to the application\n");
|
|
exit(RES_EXEC_FAILED);
|
|
}
|
|
return RES_SUCCESS;
|
|
}
|
|
|