mirror of
				https://github.com/yuzu-emu/unicorn.git
				synced 2025-10-26 20:37:11 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			782 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			782 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
| 
 | |
| Java bindings for the Unicorn Emulator Engine
 | |
| 
 | |
| Copyright(c) 2015 Chris Eagle
 | |
| 
 | |
| This program is free software; you can redistribute it and/or
 | |
| modify it under the terms of the GNU General Public License
 | |
| version 2 as published by the Free Software Foundation.
 | |
| 
 | |
| This program is distributed in the hope that it will be useful,
 | |
| but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| GNU General Public License for more details.
 | |
| 
 | |
| You should have received a copy of the GNU General Public License
 | |
| along with this program; if not, write to the Free Software
 | |
| Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 | |
| 
 | |
| */
 | |
| 
 | |
| #include <sys/types.h>
 | |
| #include "unicorn/platform.h"
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include <unicorn/unicorn.h>
 | |
| #include <unicorn/x86.h>
 | |
| #include "unicorn_Unicorn.h"
 | |
| 
 | |
| //cache jmethodID values as we look them up
 | |
| static jmethodID invokeBlockCallbacks = 0;
 | |
| static jmethodID invokeInterruptCallbacks = 0;
 | |
| static jmethodID invokeCodeCallbacks = 0;
 | |
| 
 | |
| static jmethodID invokeEventMemCallbacks = 0;
 | |
| static jmethodID invokeReadCallbacks = 0;
 | |
| static jmethodID invokeWriteCallbacks = 0;
 | |
| static jmethodID invokeInCallbacks = 0;
 | |
| static jmethodID invokeOutCallbacks = 0;
 | |
| static jmethodID invokeSyscallCallbacks = 0;
 | |
| 
 | |
| static JavaVM* cachedJVM;
 | |
| 
 | |
| JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *jvm, void *reserved) {
 | |
|    cachedJVM = jvm;
 | |
|    return JNI_VERSION_1_6;
 | |
| }
 | |
| 
 | |
| // Callback function for tracing code (UC_HOOK_CODE & UC_HOOK_BLOCK)
 | |
| // @address: address where the code is being executed
 | |
| // @size: size of machine instruction being executed
 | |
| // @user_data: user data passed to tracing APIs.
 | |
| static void cb_hookcode(uc_engine *eng, uint64_t address, uint32_t size, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    (*env)->CallStaticVoidMethod(env, clz, invokeCodeCallbacks, (jlong)eng, (jlong)address, (int)size);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // Callback function for tracing code (UC_HOOK_CODE & UC_HOOK_BLOCK)
 | |
| // @address: address where the code is being executed
 | |
| // @size: size of machine instruction being executed
 | |
| // @user_data: user data passed to tracing APIs.
 | |
| static void cb_hookblock(uc_engine *eng, uint64_t address, uint32_t size, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    (*env)->CallStaticVoidMethod(env, clz, invokeBlockCallbacks, (jlong)eng, (jlong)address, (int)size);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // Callback function for tracing interrupts (for uc_hook_intr())
 | |
| // @intno: interrupt number
 | |
| // @user_data: user data passed to tracing APIs.
 | |
| static void cb_hookintr(uc_engine *eng, uint32_t intno, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    (*env)->CallStaticVoidMethod(env, clz, invokeInterruptCallbacks, (jlong)eng, (int)intno);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // Callback function for tracing IN instruction of X86
 | |
| // @port: port number
 | |
| // @size: data size (1/2/4) to be read from this port
 | |
| // @user_data: user data passed to tracing APIs.
 | |
| static uint32_t cb_insn_in(uc_engine *eng, uint32_t port, int size, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    uint32_t res = 0;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return 0;
 | |
|    }
 | |
|    res = (uint32_t)(*env)->CallStaticIntMethod(env, clz, invokeInCallbacks, (jlong)eng, (jint)port, (jint)size);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
|    return res;
 | |
| }
 | |
| 
 | |
| // x86's handler for OUT
 | |
| // @port: port number
 | |
| // @size: data size (1/2/4) to be written to this port
 | |
| // @value: data value to be written to this port
 | |
| static void cb_insn_out(uc_engine *eng, uint32_t port, int size, uint32_t value, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    (*env)->CallStaticVoidMethod(env, clz, invokeOutCallbacks, (jlong)eng, (jint)port, (jint)size, (jint)value);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // x86's handler for SYSCALL/SYSENTER
 | |
| static void cb_insn_syscall(uc_engine *eng, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    (*env)->CallStaticVoidMethod(env, clz, invokeSyscallCallbacks, (jlong)eng);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // Callback function for hooking memory (UC_HOOK_MEM_*)
 | |
| // @type: this memory is being READ, or WRITE
 | |
| // @address: address where the code is being executed
 | |
| // @size: size of data being read or written
 | |
| // @value: value of data being written to memory, or irrelevant if type = READ.
 | |
| // @user_data: user data passed to tracing APIs
 | |
| static void cb_hookmem(uc_engine *eng, uc_mem_type type,
 | |
|         uint64_t address, int size, int64_t value, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
|    switch (type) {
 | |
|       case UC_MEM_READ:
 | |
|          (*env)->CallStaticVoidMethod(env, clz, invokeReadCallbacks, (jlong)eng, (jlong)address, (int)size);
 | |
|          break;
 | |
|       case UC_MEM_WRITE:
 | |
|          (*env)->CallStaticVoidMethod(env, clz, invokeWriteCallbacks, (jlong)eng, (jlong)address, (int)size, (jlong)value);
 | |
|          break;
 | |
|    }
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
| }
 | |
| 
 | |
| // Callback function for handling memory events (for UC_HOOK_MEM_UNMAPPED)
 | |
| // @type: this memory is being READ, or WRITE
 | |
| // @address: address where the code is being executed
 | |
| // @size: size of data being read or written
 | |
| // @value: value of data being written to memory, or irrelevant if type = READ.
 | |
| // @user_data: user data passed to tracing APIs
 | |
| // @return: return true to continue, or false to stop program (due to invalid memory).
 | |
| static bool cb_eventmem(uc_engine *eng, uc_mem_type type,
 | |
|                         uint64_t address, int size, int64_t value, void *user_data) {
 | |
|    JNIEnv *env;
 | |
|    (*cachedJVM)->AttachCurrentThread(cachedJVM, (void **)&env, NULL);
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/Unicorn");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return false;
 | |
|    }
 | |
|    jboolean res = (*env)->CallStaticBooleanMethod(env, clz, invokeEventMemCallbacks, (jlong)eng, (int)type, (jlong)address, (int)size, (jlong)value);
 | |
|    (*cachedJVM)->DetachCurrentThread(cachedJVM);
 | |
|    return res;
 | |
| }
 | |
| 
 | |
| static void throwException(JNIEnv *env, uc_err err) {
 | |
|    //throw exception
 | |
|    jclass clazz = (*env)->FindClass(env, "unicorn/UnicornException");
 | |
|    if (err != UC_ERR_OK) {
 | |
|       const char *msg = uc_strerror(err);
 | |
|       (*env)->ThrowNew(env, clazz, msg);
 | |
|    }
 | |
| }
 | |
| 
 | |
| static uc_engine *getEngine(JNIEnv *env, jobject self) {
 | |
|    static int haveFid = 0;
 | |
|    static jfieldID fid;
 | |
|    if (haveFid == 0) {
 | |
|       //cache the field id
 | |
|       jclass clazz = (*env)->GetObjectClass(env, self);
 | |
|       fid = (*env)->GetFieldID(env, clazz, "eng", "J");
 | |
|       haveFid = 1;
 | |
|    }
 | |
|    return (uc_engine *)(*env)->GetLongField(env, self, fid);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_write_num
 | |
|  * Signature: (ILjava/lang/Number;)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_reg_1write_1num
 | |
|   (JNIEnv *env, jobject self, jint regid, jobject value) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    jclass clz = (*env)->FindClass(env, "java/lang/Number");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
| 
 | |
|    jmethodID longValue = (*env)->GetMethodID(env, clz, "longValue", "()J");
 | |
|    jlong longVal = (*env)->CallLongMethod(env, value, longValue);
 | |
|    uc_err err = uc_reg_write(eng, regid, &longVal);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_write_mmr
 | |
|  * Signature: (ILunicorn/X86_MMR;)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_reg_1write_1mmr
 | |
|   (JNIEnv *env, jobject self, jint regid, jobject value) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    uc_x86_mmr mmr;
 | |
| 
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/X86_MMR");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return;
 | |
|    }
 | |
| 
 | |
|    jfieldID fid = (*env)->GetFieldID(env, clz, "base", "J");
 | |
|    mmr.base = (uint64_t)(*env)->GetLongField(env, value, fid);
 | |
| 
 | |
|    fid = (*env)->GetFieldID(env, clz, "limit", "I");
 | |
|    mmr.limit = (uint32_t)(*env)->GetLongField(env, value, fid);
 | |
| 
 | |
|    fid = (*env)->GetFieldID(env, clz, "flags", "I");
 | |
|    mmr.flags = (uint32_t)(*env)->GetLongField(env, value, fid);
 | |
| 
 | |
|    fid = (*env)->GetFieldID(env, clz, "selector", "S");
 | |
|    mmr.selector = (uint16_t)(*env)->GetLongField(env, value, fid);
 | |
| 
 | |
|    uc_err err = uc_reg_write(eng, regid, &mmr);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_read_num
 | |
|  * Signature: (I)Ljava/lang/Number;
 | |
|  */
 | |
| JNIEXPORT jobject JNICALL Java_unicorn_Unicorn_reg_1read_1num
 | |
|   (JNIEnv *env, jobject self, jint regid) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    jclass clz = (*env)->FindClass(env, "java/lang/Long");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return NULL;
 | |
|    }
 | |
| 
 | |
|    jlong longVal;
 | |
|    uc_err err = uc_reg_read(eng, regid, &longVal);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| 
 | |
|    jmethodID cons = (*env)->GetMethodID(env, clz, "<init>", "(J)V");
 | |
|    jobject result = (*env)->NewObject(env, clz, cons, longVal);
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return NULL;
 | |
|    }
 | |
|    return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_read_mmr
 | |
|  * Signature: (I)Ljava/lang/Number;
 | |
|  */
 | |
| JNIEXPORT jobject JNICALL Java_unicorn_Unicorn_reg_1read_1mmr
 | |
|   (JNIEnv *env, jobject self, jint regid) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/X86_MMR");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return NULL;
 | |
|    }
 | |
| 
 | |
|    uc_x86_mmr mmr;
 | |
|    uc_err err = uc_reg_read(eng, regid, &mmr);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| 
 | |
|    jmethodID cons = (*env)->GetMethodID(env, clz, "<init>", "(JIIS)V");
 | |
|    jobject result = (*env)->NewObject(env, clz, cons, mmr.base, mmr.limit, mmr.flags, mmr.selector);
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return NULL;
 | |
|    }
 | |
|    return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    open
 | |
|  * Signature: (II)J
 | |
|  */
 | |
| JNIEXPORT jlong JNICALL Java_unicorn_Unicorn_open
 | |
|   (JNIEnv *env, jobject self, jint arch, jint mode) {
 | |
|    uc_engine *eng = NULL;
 | |
|    uc_err err = uc_open((uc_arch)arch, (uc_mode)mode, &eng);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    return (jlong)eng;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    version
 | |
|  * Signature: ()I
 | |
|  */
 | |
| JNIEXPORT jint JNICALL Java_unicorn_Unicorn_version
 | |
|   (JNIEnv *env, jclass clz) {
 | |
|     return (jint)uc_version(NULL, NULL);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    arch_supported
 | |
|  * Signature: (I)Z
 | |
|  */
 | |
| JNIEXPORT jboolean JNICALL Java_unicorn_Unicorn_arch_1supported
 | |
|   (JNIEnv *env, jclass clz, jint arch) {
 | |
|     return (jboolean)(uc_arch_supported((uc_arch)arch) != 0);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    close
 | |
|  * Signature: ()V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_close
 | |
|   (JNIEnv *env, jobject self) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    uc_err err = uc_close(eng);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    //We also need to ReleaseByteArrayElements for any regions that
 | |
|    //were mapped with uc_mem_map_ptr
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    query
 | |
|  * Signature: (I)I
 | |
|  */
 | |
| JNIEXPORT jint JNICALL Java_unicorn_Unicorn_query
 | |
|   (JNIEnv *env, jobject self, jint type) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    size_t result;
 | |
|    uc_err err = uc_query(eng, type, &result);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    return (jint)result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    errno
 | |
|  * Signature: ()I
 | |
|  */
 | |
| JNIEXPORT jint JNICALL Java_unicorn_Unicorn_errno
 | |
|   (JNIEnv *env, jobject self) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    return (jint)uc_errno(eng);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    strerror
 | |
|  * Signature: (I)Ljava/lang/String;
 | |
|  */
 | |
| JNIEXPORT jstring JNICALL Java_unicorn_Unicorn_strerror
 | |
|   (JNIEnv *env, jclass clz, jint code) {
 | |
|    const char *err = uc_strerror((int)code);
 | |
|    jstring s = (*env)->NewStringUTF(env, err);
 | |
|    return s;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_write
 | |
|  * Signature: (I[B)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_reg_1write
 | |
|   (JNIEnv *env, jobject self, jint regid, jbyteArray value) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    jbyte *array = (*env)->GetByteArrayElements(env, value, NULL);
 | |
|    uc_err err = uc_reg_write(eng, (int)regid, (void *)array);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    (*env)->ReleaseByteArrayElements(env, value, array, JNI_ABORT);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    reg_read
 | |
|  * Signature: (II)[B
 | |
|  */
 | |
| JNIEXPORT jbyteArray JNICALL Java_unicorn_Unicorn_reg_1read
 | |
|   (JNIEnv *env, jobject self, jint regid, jint regsz) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    jbyteArray regval = (*env)->NewByteArray(env, (jsize)regsz);
 | |
|    jbyte *array = (*env)->GetByteArrayElements(env, regval, NULL);
 | |
|    uc_err err = uc_reg_read(eng, (int)regid, (void *)array);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    (*env)->ReleaseByteArrayElements(env, regval, array, 0);
 | |
|    return regval;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_write
 | |
|  * Signature: (J[B)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_mem_1write
 | |
|   (JNIEnv *env , jobject self, jlong address, jbyteArray bytes) {
 | |
| 
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    jbyte *array = (*env)->GetByteArrayElements(env, bytes, NULL);
 | |
|    jsize size = (*env)->GetArrayLength(env, bytes);
 | |
|    uc_err err = uc_mem_write(eng, (uint64_t)address, array, (size_t)size);
 | |
| 
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| 
 | |
|    (*env)->ReleaseByteArrayElements(env, bytes, array, JNI_ABORT);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_read
 | |
|  * Signature: (JJ)[B
 | |
|  */
 | |
| JNIEXPORT jbyteArray JNICALL Java_unicorn_Unicorn_mem_1read
 | |
|   (JNIEnv *env, jobject self, jlong address, jlong size) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    jbyteArray bytes = (*env)->NewByteArray(env, (jsize)size);
 | |
|    jbyte *array = (*env)->GetByteArrayElements(env, bytes, NULL);
 | |
|    uc_err err = uc_mem_read(eng, (uint64_t)address, array, (size_t)size);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    (*env)->ReleaseByteArrayElements(env, bytes, array, 0);
 | |
|    return bytes;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    emu_start
 | |
|  * Signature: (JJJJ)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_emu_1start
 | |
|   (JNIEnv *env, jobject self, jlong begin, jlong until, jlong timeout, jlong count) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_err err = uc_emu_start(eng, (uint64_t)begin, (uint64_t)until, (uint64_t)timeout, (size_t)count);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    emu_stop
 | |
|  * Signature: ()V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_emu_1stop
 | |
|   (JNIEnv *env, jobject self) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_err err = uc_emu_stop(eng);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    registerHook
 | |
|  * Signature: (JI)J
 | |
|  */
 | |
| JNIEXPORT jlong JNICALL Java_unicorn_Unicorn_registerHook__JI
 | |
|   (JNIEnv *env, jclass clz, jlong eng, jint type) {
 | |
|    uc_hook hh = 0;
 | |
|    uc_err err = 0;
 | |
|    switch (type) {
 | |
|       case UC_HOOK_INTR:           // Hook all interrupt events
 | |
|          if (invokeInterruptCallbacks == 0) {
 | |
|             invokeInterruptCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeInterruptCallbacks", "(JI)V");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_hookintr, env, 1, 0);
 | |
|          break;
 | |
|       case UC_HOOK_MEM_FETCH_UNMAPPED:    // Hook for all invalid memory access events
 | |
|       case UC_HOOK_MEM_READ_UNMAPPED:    // Hook for all invalid memory access events
 | |
|       case UC_HOOK_MEM_WRITE_UNMAPPED:    // Hook for all invalid memory access events
 | |
|       case UC_HOOK_MEM_FETCH_PROT:    // Hook for all invalid memory access events
 | |
|       case UC_HOOK_MEM_READ_PROT:    // Hook for all invalid memory access events
 | |
|       case UC_HOOK_MEM_WRITE_PROT:    // Hook for all invalid memory access events
 | |
|          if (invokeEventMemCallbacks == 0) {
 | |
|             invokeEventMemCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeEventMemCallbacks", "(JIJIJ)Z");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_eventmem, env, 1, 0);
 | |
|          break;
 | |
|    }
 | |
|    return (jlong)hh;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    registerHook
 | |
|  * Signature: (JII)J
 | |
|  */
 | |
| JNIEXPORT jlong JNICALL Java_unicorn_Unicorn_registerHook__JII
 | |
|   (JNIEnv *env, jclass clz, jlong eng, jint type, jint arg1) {
 | |
|    uc_hook hh = 0;
 | |
|    uc_err err = 0;
 | |
|    switch (type) {
 | |
|       case UC_HOOK_INSN:           // Hook a particular instruction
 | |
|          switch (arg1) {
 | |
|             case UC_X86_INS_OUT:
 | |
|                if (invokeOutCallbacks == 0) {
 | |
|                   invokeOutCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeOutCallbacks", "(JIII)V");
 | |
|                }
 | |
|                err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_insn_out, env, 1, 0, arg1);
 | |
|             case UC_X86_INS_IN:
 | |
|                if (invokeInCallbacks == 0) {
 | |
|                   invokeInCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeInCallbacks", "(JII)I");
 | |
|                }
 | |
|                err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_insn_in, env, 1, 0, arg1);
 | |
|             case UC_X86_INS_SYSENTER:
 | |
|             case UC_X86_INS_SYSCALL:
 | |
|                if (invokeSyscallCallbacks == 0) {
 | |
|                   invokeSyscallCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeSyscallCallbacks", "(J)V");
 | |
|                }
 | |
|                err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_insn_syscall, env, 1, 0, arg1);
 | |
|          }
 | |
|          break;
 | |
|    }
 | |
|    return (jlong)hh;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    registerHook
 | |
|  * Signature: (JIJJ)J
 | |
|  */
 | |
| JNIEXPORT jlong JNICALL Java_unicorn_Unicorn_registerHook__JIJJ
 | |
|   (JNIEnv *env, jclass clz, jlong eng, jint type, jlong arg1, jlong arg2) {
 | |
|    uc_hook hh = 0;
 | |
|    uc_err err = 0;
 | |
|    switch (type) {
 | |
|       case UC_HOOK_CODE:           // Hook a range of code
 | |
|          if (invokeCodeCallbacks == 0) {
 | |
|             invokeCodeCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeCodeCallbacks", "(JJI)V");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_hookcode, env, 1, 0, arg1, arg2);
 | |
|          break;
 | |
|       case UC_HOOK_BLOCK:          // Hook basic blocks
 | |
|          if (invokeBlockCallbacks == 0) {
 | |
|             invokeBlockCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeBlockCallbacks", "(JJI)V");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_hookblock, env, 1, 0, arg1, arg2);
 | |
|          break;
 | |
|       case UC_HOOK_MEM_READ:       // Hook all memory read events.
 | |
|          if (invokeReadCallbacks == 0) {
 | |
|             invokeReadCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeReadCallbacks", "(JJI)V");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_hookmem, env, 1, 0, arg1, arg2);
 | |
|          break;
 | |
|       case UC_HOOK_MEM_WRITE:      // Hook all memory write events.
 | |
|          if (invokeWriteCallbacks == 0) {
 | |
|             invokeWriteCallbacks = (*env)->GetStaticMethodID(env, clz, "invokeWriteCallbacks", "(JJIJ)V");
 | |
|          }
 | |
|          err = uc_hook_add((uc_engine*)eng, &hh, (uc_hook_type)type, cb_hookmem, env, 1, 0, arg1, arg2);
 | |
|          break;
 | |
|    }
 | |
|    return (jlong)hh;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    hook_del
 | |
|  * Signature: (J)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_hook_1del
 | |
|   (JNIEnv *env, jobject self, jlong hh) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    //**** TODO remove hook from any internal hook tables as well
 | |
| 
 | |
|    uc_err err = uc_hook_del(eng, (uc_hook)hh);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_map
 | |
|  * Signature: (JJI)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_mem_1map
 | |
|   (JNIEnv *env, jobject self, jlong address, jlong size, jint perms) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_err err = uc_mem_map(eng, (uint64_t)address, (size_t)size, (uint32_t)perms);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_map_ptr
 | |
|  * Signature: (JJI[B)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_mem_1map_1ptr
 | |
|   (JNIEnv *env, jobject self, jlong address, jlong size, jint perms, jbyteArray block) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    jbyte *array = (*env)->GetByteArrayElements(env, block, NULL);
 | |
|    uc_err err = uc_mem_map_ptr(eng, (uint64_t)address, (size_t)size, (uint32_t)perms, (void*)array);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    //Need to track address/block/array so that we can ReleaseByteArrayElements when the
 | |
|    //block gets unmapped or when uc_close gets called
 | |
|    //(*env)->ReleaseByteArrayElements(env, block, array, JNI_ABORT);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_unmap
 | |
|  * Signature: (JJ)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_mem_1unmap
 | |
|   (JNIEnv *env, jobject self, jlong address, jlong size) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_err err = uc_mem_unmap(eng, (uint64_t)address, (size_t)size);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| 
 | |
|    //If a region was mapped using uc_mem_map_ptr, we also need to
 | |
|    //ReleaseByteArrayElements for that region
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_protect
 | |
|  * Signature: (JJI)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_mem_1protect
 | |
|   (JNIEnv *env, jobject self, jlong address, jlong size, jint perms) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_err err = uc_mem_protect(eng, (uint64_t)address, (size_t)size, (uint32_t)perms);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    mem_regions
 | |
|  * Signature: ()[Lunicorn/MemRegion;
 | |
|  */
 | |
| JNIEXPORT jobjectArray JNICALL Java_unicorn_Unicorn_mem_1regions
 | |
|   (JNIEnv *env, jobject self) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
| 
 | |
|    uc_mem_region *regions = NULL;
 | |
|    uint32_t count = 0;
 | |
|    uint32_t i;
 | |
| 
 | |
|    uc_err err = uc_mem_regions(eng, ®ions, &count);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    jclass clz = (*env)->FindClass(env, "unicorn/MemRegion");
 | |
|    if ((*env)->ExceptionCheck(env)) {
 | |
|       return NULL;
 | |
|    }
 | |
|    jobjectArray result = (*env)->NewObjectArray(env, (jsize)count, clz, NULL);
 | |
|    jmethodID cons = (*env)->GetMethodID(env, clz, "<init>", "(JJI)V");
 | |
|    for (i = 0; i < count; i++) {
 | |
|       jobject mr = (*env)->NewObject(env, clz, cons, regions[i].begin, regions[i].end, regions[i].perms);
 | |
|       (*env)->SetObjectArrayElement(env, result, (jsize)i, mr);
 | |
|    }
 | |
|    uc_free(regions);
 | |
| 
 | |
|    return result;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    context_alloc
 | |
|  * Signature: ()J
 | |
|  */
 | |
| JNIEXPORT jlong JNICALL Java_unicorn_Unicorn_context_1alloc
 | |
|   (JNIEnv *env, jobject self) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    uc_context *ctx;
 | |
|    uc_err err = uc_context_alloc(eng, &ctx);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
|    return (jlong)(uint64_t)ctx;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    free
 | |
|  * Signature: (J)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_free
 | |
|   (JNIEnv *env, jobject self, jlong ctx) {
 | |
|    uc_err err = uc_free((void *)ctx);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    context_save
 | |
|  * Signature: (J)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_context_1save
 | |
|   (JNIEnv *env, jobject self, jlong ctx) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    uc_err err = uc_context_save(eng, (uc_context*)ctx);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * Class:     unicorn_Unicorn
 | |
|  * Method:    context_restore
 | |
|  * Signature: (J)V
 | |
|  */
 | |
| JNIEXPORT void JNICALL Java_unicorn_Unicorn_context_1restore
 | |
|   (JNIEnv *env, jobject self, jlong ctx) {
 | |
|    uc_engine *eng = getEngine(env, self);
 | |
|    uc_err err = uc_context_restore(eng, (uc_context*)ctx);
 | |
|    if (err != UC_ERR_OK) {
 | |
|       throwException(env, err);
 | |
|    }
 | |
| }
 |