|
|
@ -124,33 +124,72 @@ string Process::doReadClassName (void * vptr)
|
|
|
|
return raw.substr(start,end-start);
|
|
|
|
return raw.substr(start,end-start);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//FIXME: cross-reference with ELF segment entries?
|
|
|
|
#include <mach/mach.h>
|
|
|
|
|
|
|
|
#include <mach/mach_vm.h>
|
|
|
|
|
|
|
|
#include <mach/vm_region.h>
|
|
|
|
|
|
|
|
#include <mach/vm_statistics.h>
|
|
|
|
|
|
|
|
#include <dlfcn.h>
|
|
|
|
|
|
|
|
|
|
|
|
void Process::getMemRanges( vector<t_memrange> & ranges )
|
|
|
|
void Process::getMemRanges( vector<t_memrange> & ranges )
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
|
|
|
|
char permissions[5]; // r/-, w/-, x/-, p/s, 0
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
FILE *mapFile = ::fopen("/proc/self/maps", "r");
|
|
|
|
kern_return_t kr;
|
|
|
|
size_t start, end, offset, device1, device2, node;
|
|
|
|
task_t the_task;
|
|
|
|
|
|
|
|
|
|
|
|
while (fgets(buffer, 1024, mapFile))
|
|
|
|
the_task = mach_task_self();
|
|
|
|
{
|
|
|
|
|
|
|
|
t_memrange temp;
|
|
|
|
vm_size_t vmsize;
|
|
|
|
temp.name[0] = 0;
|
|
|
|
vm_address_t address;
|
|
|
|
sscanf(buffer, "%zx-%zx %s %zx %2zx:%2zx %zu %[^\n]",
|
|
|
|
vm_region_basic_info_data_t info;
|
|
|
|
&start,
|
|
|
|
mach_msg_type_number_t info_count;
|
|
|
|
&end,
|
|
|
|
vm_region_flavor_t flavor;
|
|
|
|
(char*)&permissions,
|
|
|
|
memory_object_name_t object;
|
|
|
|
&offset, &device1, &device2, &node,
|
|
|
|
|
|
|
|
(char*)temp.name);
|
|
|
|
kr = KERN_SUCCESS;
|
|
|
|
temp.start = (void *) start;
|
|
|
|
address = 0;
|
|
|
|
temp.end = (void *) end;
|
|
|
|
|
|
|
|
temp.read = permissions[0] == 'r';
|
|
|
|
do {
|
|
|
|
temp.write = permissions[1] == 'w';
|
|
|
|
flavor = VM_REGION_BASIC_INFO;
|
|
|
|
temp.execute = permissions[2] == 'x';
|
|
|
|
info_count = VM_REGION_BASIC_INFO_COUNT;
|
|
|
|
temp.shared = permissions[3] == 's';
|
|
|
|
kr = vm_region(the_task, &address, &vmsize, flavor,
|
|
|
|
temp.valid = true;
|
|
|
|
(vm_region_info_t)&info, &info_count, &object);
|
|
|
|
ranges.push_back(temp);
|
|
|
|
if (kr == KERN_SUCCESS) {
|
|
|
|
|
|
|
|
if (info.reserved==1) {
|
|
|
|
|
|
|
|
address += vmsize;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
Dl_info dlinfo;
|
|
|
|
|
|
|
|
int dlcheck;
|
|
|
|
|
|
|
|
dlcheck = dladdr((const void*)address, &dlinfo);
|
|
|
|
|
|
|
|
if (dlcheck==0) {
|
|
|
|
|
|
|
|
dlinfo.dli_fname = "(none)";
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
t_memrange temp;
|
|
|
|
|
|
|
|
strncpy( temp.name, dlinfo.dli_fname, 1023 );
|
|
|
|
|
|
|
|
temp.name[1023] = 0;
|
|
|
|
|
|
|
|
temp.start = (void *) address;
|
|
|
|
|
|
|
|
temp.end = (void *) (address+vmsize);
|
|
|
|
|
|
|
|
temp.read = (info.protection & VM_PROT_READ);
|
|
|
|
|
|
|
|
temp.write = (info.protection & VM_PROT_WRITE);
|
|
|
|
|
|
|
|
temp.execute = (info.protection & VM_PROT_EXECUTE);
|
|
|
|
|
|
|
|
temp.shared = info.shared;
|
|
|
|
|
|
|
|
temp.valid = true;
|
|
|
|
|
|
|
|
ranges.push_back(temp);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
address += vmsize;
|
|
|
|
|
|
|
|
} else if (kr != KERN_INVALID_ADDRESS) {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (the_task != MACH_PORT_NULL) {
|
|
|
|
|
|
|
|
mach_port_deallocate(mach_task_self(), the_task);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} while (kr != KERN_INVALID_ADDRESS);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (the_task != MACH_PORT_NULL) {
|
|
|
|
|
|
|
|
mach_port_deallocate(mach_task_self(), the_task);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|