123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179 |
- #import "../kern_utils.h"
- #import "kmem.h"
- #define MAX_CHUNK_SIZE 0xFFF
- void remote_read_overwrite(mach_port_t task_port,
- uint64_t remote_address,
- uint64_t local_address,
- uint64_t length) {
- kern_return_t err;
-
- mach_vm_size_t outsize = 0;
- err = mach_vm_read_overwrite(task_port, (mach_vm_address_t)remote_address, (mach_vm_size_t)length, (mach_vm_address_t)local_address, &outsize);
- if (err != KERN_SUCCESS){
- return;
- }
-
- if (outsize != length){
- return;
- }
- }
- void remote_write(mach_port_t remote_task_port,
- uint64_t remote_address,
- uint64_t local_address,
- uint64_t length) {
- kern_return_t err = mach_vm_write(remote_task_port,
- (mach_vm_address_t)remote_address,
- (vm_offset_t)local_address,
- (mach_msg_type_number_t)length);
- if (err != KERN_SUCCESS) {
- return;
- }
- }
- uint64_t binary_load_address() {
- kern_return_t err;
- mach_msg_type_number_t region_count = VM_REGION_BASIC_INFO_COUNT_64;
- memory_object_name_t object_name = MACH_PORT_NULL;
- mach_vm_size_t target_first_size = 0x1000;
- mach_vm_address_t target_first_addr = 0x0;
- struct vm_region_basic_info_64 region = {0};
- err = mach_vm_region(mach_task_self(), &target_first_addr, &target_first_size, VM_REGION_BASIC_INFO_64, (vm_region_info_t)®ion, ®ion_count, &object_name);
-
- if (err != KERN_SUCCESS) {
- return -1;
- }
-
- return target_first_addr;
- }
- size_t kread(uint64_t where, void *p, size_t size) {
- int rv;
- size_t offset = 0;
- while (offset < size) {
- mach_vm_size_t sz, chunk = MAX_CHUNK_SIZE;
- if (chunk > size - offset) {
- chunk = size - offset;
- }
- rv = mach_vm_read_overwrite(tfp0, where + offset, chunk, (mach_vm_address_t)p + offset, &sz);
- if (rv || sz == 0) {
- fprintf(stderr, "[e] error reading kernel @%p\n", (void *)(offset + where));
- break;
- }
- offset += sz;
- }
- return offset;
- }
- size_t kwrite(uint64_t where, const void *p, size_t size) {
- int rv;
- size_t offset = 0;
- while (offset < size) {
- size_t chunk = MAX_CHUNK_SIZE;
- if (chunk > size - offset) {
- chunk = size - offset;
- }
- rv = mach_vm_write(tfp0, where + offset, (mach_vm_offset_t)p + offset, chunk);
- if (rv) {
- fprintf(stderr, "[e] error writing kernel @%p\n", (void *)(offset + where));
- break;
- }
- offset += chunk;
- }
- return offset;
- }
- uint64_t kalloc(vm_size_t size){
- mach_vm_address_t address = 0;
- mach_vm_allocate(tfp0, (mach_vm_address_t *)&address, size, VM_FLAGS_ANYWHERE);
- return address;
- }
- void kfree(mach_vm_address_t address, vm_size_t size){
- mach_vm_deallocate(tfp0, address, size);
- }
- uint16_t rk16(uint64_t kaddr) {
- uint16_t val = 0;
- kread(kaddr, &val, sizeof(val));
- return val;
- }
- uint32_t rk32(uint64_t kaddr) {
- uint32_t val = 0;
- kread(kaddr, &val, sizeof(val));
- return val;
- }
- uint64_t rk64(uint64_t kaddr) {
- uint64_t val = 0;
- kread(kaddr, &val, sizeof(val));
- return val;
- }
- void wk16(uint64_t kaddr, uint16_t val) {
- kwrite(kaddr, &val, sizeof(val));
- }
- void wk32(uint64_t kaddr, uint32_t val) {
- kwrite(kaddr, &val, sizeof(val));
- }
- void wk64(uint64_t kaddr, uint64_t val) {
- kwrite(kaddr, &val, sizeof(val));
- }
- // thx Siguza
- typedef struct {
- uint64_t prev;
- uint64_t next;
- uint64_t start;
- uint64_t end;
- } kmap_hdr_t;
- uint64_t zm_fix_addr(uint64_t addr) {
- static kmap_hdr_t zm_hdr = {0, 0, 0, 0};
- if (zm_hdr.start == 0) {
- // xxx rk64(0) ?!
- uint64_t zone_map = rk64(offset_zonemap);
- fprintf(stderr, "zone_map: %llx \n", zone_map);
- // hdr is at offset 0x10, mutexes at start
- size_t r = kread(zone_map + 0x10, &zm_hdr, sizeof(zm_hdr));
- fprintf(stderr, "zm_range: 0x%llx - 0x%llx (read 0x%zx, exp 0x%zx)\n", zm_hdr.start, zm_hdr.end, r, sizeof(zm_hdr));
- if (r != sizeof(zm_hdr) || zm_hdr.start == 0 || zm_hdr.end == 0) {
- fprintf(stderr, "kread of zone_map failed!\n");
- exit(1);
- }
- if (zm_hdr.end - zm_hdr.start > 0x100000000) {
- fprintf(stderr, "zone_map is too big, sorry.\n");
- exit(1);
- }
- }
- uint64_t zm_tmp = (zm_hdr.start & 0xffffffff00000000) | ((addr) & 0xffffffff);
- return zm_tmp < zm_hdr.start ? zm_tmp + 0x100000000 : zm_tmp;
- }
- int kstrcmp(uint64_t kstr, const char* str) {
- // XXX be safer, dont just assume you wont cause any
- // page faults by this
- size_t len = strlen(str) + 1;
- char *local = (char *)malloc(len + 1);
- local[len] = '\0';
- int ret = 1;
-
- if (kread(kstr, local, len) == len) {
- ret = strcmp(local, str);
- }
-
- free(local);
- return ret;
- }
|