Add Linux Support
Added a separate memtest python file that uses linux syscalls to read and write to memory. Due to permissions on linux dolphin-memory-lib needs to be ran as sudo as of now.
This commit is contained in:
parent
b6ca272c60
commit
35a716630f
2 changed files with 191 additions and 0 deletions
187
memtest_lin.py
Normal file
187
memtest_lin.py
Normal file
|
@ -0,0 +1,187 @@
|
||||||
|
import ctypes
|
||||||
|
import struct
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
from subprocess import check_output
|
||||||
|
from ctypes import sizeof, addressof, POINTER, pointer
|
||||||
|
|
||||||
|
# Various Linux structs needed for operation
|
||||||
|
|
||||||
|
class iovec(ctypes.Structure):
|
||||||
|
_fields_ = [("iov_base",ctypes.c_void_p),("iov_len",ctypes.c_size_t)]
|
||||||
|
|
||||||
|
libc = ctypes.cdll.LoadLibrary("libc.so.6")
|
||||||
|
vm = libc.process_vm_readv
|
||||||
|
vm.argtypes = [ctypes.c_int, POINTER(iovec), ctypes.c_ulong, POINTER(iovec), ctypes.c_ulong, ctypes.c_ulong]
|
||||||
|
vmwrite = libc.process_vm_writev
|
||||||
|
vmwrite.argtypes = [ctypes.c_int, POINTER(iovec), ctypes.c_ulong, POINTER(iovec), ctypes.c_ulong, ctypes.c_ulong]
|
||||||
|
|
||||||
|
|
||||||
|
# The following code is a port of aldelaro5's Dolphin memory access methods
|
||||||
|
# for Linux into Python+ctypes.
|
||||||
|
# https://github.com/aldelaro5/Dolphin-memory-engine
|
||||||
|
|
||||||
|
"""
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Copyright (c) 2017 aldelaro5
|
||||||
|
|
||||||
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
of this software and associated documentation files (the "Software"), to deal
|
||||||
|
in the Software without restriction, including without limitation the rights
|
||||||
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||||
|
copies of the Software, and to permit persons to whom the Software is
|
||||||
|
furnished to do so, subject to the following conditions:
|
||||||
|
|
||||||
|
The above copyright notice and this permission notice shall be included in all
|
||||||
|
copies or substantial portions of the Software.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||||
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||||
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||||
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||||
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||||
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||||
|
SOFTWARE."""
|
||||||
|
|
||||||
|
class Dolphin(object):
|
||||||
|
def __init__(self):
|
||||||
|
self.pid = -1
|
||||||
|
self.handle = -1
|
||||||
|
|
||||||
|
self.address_start = 0
|
||||||
|
self.mem1_start = 0
|
||||||
|
self.mem2_start = 0
|
||||||
|
self.mem2_exists = False
|
||||||
|
|
||||||
|
def find_dolphin(self):
|
||||||
|
try:
|
||||||
|
if check_output(["pidof", "dolphin-emu"]) != '\n':
|
||||||
|
self.pid = int(check_output(["pidof", "dolphin-emu"]))
|
||||||
|
if check_output(["pidof", "dolphin-emu-qt2"]) != '\n':
|
||||||
|
self.pid = int(check_output(["pidof", "dolphin-emu-qt2"]))
|
||||||
|
if check_output(["pidof", "dolphin-emu-wx"]) != '\n':
|
||||||
|
self.pid = int(check_output(["pidof", "dolphin-emu-wx"]))
|
||||||
|
except Exception: #subprocess.CalledProcessError
|
||||||
|
# Do nothing because self.pid cant be modified until a successful run of pidof
|
||||||
|
pass
|
||||||
|
|
||||||
|
if self.pid == -1:
|
||||||
|
return False
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def get_emu_info(self):
|
||||||
|
MEM1_found = False
|
||||||
|
try:
|
||||||
|
maps_file = open("/proc/{}/maps".format(self.pid), 'r')
|
||||||
|
except IOError:
|
||||||
|
print("Cant open maps for process {}".format(self.pid))
|
||||||
|
heap_info = None
|
||||||
|
for line in maps_file:
|
||||||
|
foundDevShmDolphin = False
|
||||||
|
if '/dev/shm/dolphinmem' in line:
|
||||||
|
heap_info = line.split()
|
||||||
|
if '/dev/shm/dolphin-emu' in line:
|
||||||
|
heap_info = line.split()
|
||||||
|
if heap_info is None:
|
||||||
|
continue
|
||||||
|
else:
|
||||||
|
offset = 0
|
||||||
|
offset_str = "0x" + str(heap_info[2])
|
||||||
|
offset = int(offset_str, 16)
|
||||||
|
if offset != 0 and offset != 0x2000000:
|
||||||
|
continue
|
||||||
|
first_address = 0
|
||||||
|
second_address = 0
|
||||||
|
index_dash = heap_info[0].find('-')
|
||||||
|
|
||||||
|
first_address_str = "0x" + str(heap_info[0][: index_dash])
|
||||||
|
second_address_str = "0x" + str(heap_info[0][(index_dash + 1):])
|
||||||
|
|
||||||
|
first_address = int(first_address_str, 16)
|
||||||
|
second_address = int(second_address_str, 16)
|
||||||
|
|
||||||
|
if (second_address - first_address) == 0x4000000 and offset == 0x2000000:
|
||||||
|
self.mem2_start = first_address
|
||||||
|
self.mem2_exists = True
|
||||||
|
if (second_address - first_address) == 0x2000000 and offset == 0x0:
|
||||||
|
self.address_start = first_address
|
||||||
|
|
||||||
|
if self.address_start == 0:
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def read_ram(self, offset, size):
|
||||||
|
buffer_ = (ctypes.c_char*size)()
|
||||||
|
nread = ctypes.c_size_t
|
||||||
|
local = (iovec*1)()
|
||||||
|
remote = (iovec*1)()
|
||||||
|
local[0].iov_base = ctypes.addressof(buffer_)
|
||||||
|
local[0].iov_len = size
|
||||||
|
remote[0].iov_base = ctypes.c_void_p(self.address_start + offset)
|
||||||
|
remote[0].iov_len = size
|
||||||
|
nread = vm(self.pid, local, 1, remote, 1, 0)
|
||||||
|
if nread != size:
|
||||||
|
return False, buffer_
|
||||||
|
return True, buffer_
|
||||||
|
|
||||||
|
def write_ram(self, offset, data):
|
||||||
|
buffer_ = (ctypes.c_char*len(data))(*data)
|
||||||
|
nwrote = ctypes.c_size_t
|
||||||
|
local = (iovec*1)()
|
||||||
|
remote = (iovec*1)()
|
||||||
|
local[0].iov_base = ctypes.addressof(buffer_)
|
||||||
|
local[0].iov_len = len(data)
|
||||||
|
remote[0].iov_base = ctypes.c_void_p(self.address_start + offset)
|
||||||
|
remote[0].iov_len = len(data)
|
||||||
|
nwrote = vmwrite(self.pid, local, 1, remote, 1, 0)
|
||||||
|
if nwrote != len(data):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def read_uint32(self, addr):
|
||||||
|
assert addr >= 0x80000000
|
||||||
|
success, value = self.read_ram(addr-0x80000000, 4)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
return struct.unpack(">I", value)[0]
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def read_float(self, addr):
|
||||||
|
assert addr >= 0x80000000
|
||||||
|
success, value = self.read_ram(addr - 0x80000000, 4)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
return struct.unpack(">f", value)[0]
|
||||||
|
else:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def write_float(self, addr, val):
|
||||||
|
assert addr >= 0x80000000
|
||||||
|
return self.write_ram(addr - 0x80000000, struct.pack(">f", val))
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
dolphin = Dolphin()
|
||||||
|
|
||||||
|
if dolphin.find_dolphin():
|
||||||
|
|
||||||
|
print("Found Dolphin! ")
|
||||||
|
else:
|
||||||
|
print("Didn't find Dolphin")
|
||||||
|
|
||||||
|
print(dolphin.pid)
|
||||||
|
|
||||||
|
if dolphin.get_emu_info():
|
||||||
|
print("We found MEM1 and/or MEM2!", dolphin.address_start, dolphin.mem2_start)
|
||||||
|
else:
|
||||||
|
print("We didn't find it...")
|
||||||
|
print(dolphin.write_ram(0, b"GMS"))
|
||||||
|
success, result = dolphin.read_ram(0, 8)
|
||||||
|
print(result[0:8])
|
||||||
|
|
||||||
|
print(dolphin.write_ram(0, b"AWA"))
|
||||||
|
success, result = dolphin.read_ram(0, 8)
|
||||||
|
print(result[0:8])
|
4
run.sh
Executable file
4
run.sh
Executable file
|
@ -0,0 +1,4 @@
|
||||||
|
#!/bin/bash
|
||||||
|
echo "dolphin-memory-lib requires sudo permission to read and write to the emulator process memory."
|
||||||
|
sudo python memtest_lin.py
|
||||||
|
read -n1 -r
|
Reference in a new issue