about summary refs log tree commit diff stats
path: root/src/libtools/sdl1rwops.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/libtools/sdl1rwops.c')
-rwxr-xr-xsrc/libtools/sdl1rwops.c182
1 files changed, 182 insertions, 0 deletions
diff --git a/src/libtools/sdl1rwops.c b/src/libtools/sdl1rwops.c
new file mode 100755
index 00000000..bdb26c83
--- /dev/null
+++ b/src/libtools/sdl1rwops.c
@@ -0,0 +1,182 @@
+#include <stdio.h>
+#include <stdlib.h>
+#include <stdint.h>
+#include <string.h>
+
+#include "sdl1rwops.h"
+#include "debug.h"
+#include "wrapper.h"
+#include "box64context.h"
+#include "x64run.h"
+#include "x64emu.h"
+#include "emu/x64emu_private.h"
+#include "librarian/library_private.h"
+#include "bridge.h"
+#include "callback.h"
+
+typedef struct SDL1_RWops_s SDL1_RWops_t;
+
+typedef int32_t (*sdl1_seek)(SDL1_RWops_t *context, int32_t offset, int32_t whence);
+typedef int32_t (*sdl1_read)(SDL1_RWops_t *context, void *ptr, int32_t size, int32_t maxnum);
+typedef int32_t (*sdl1_write)(SDL1_RWops_t *context, const void *ptr, int32_t size, int32_t num);
+typedef int32_t (*sdl1_close)(SDL1_RWops_t *context);
+
+#define BOX64RW 0xBECF4572 // random signature value
+
+typedef struct SDL1_RWops_s {
+    sdl1_seek  seek;
+    sdl1_read  read;
+    sdl1_write write;
+
+    sdl1_close close;
+
+    uint32_t type;
+    union {
+        struct {
+            int autoclose;
+            FILE *fp;
+        } stdio;
+        struct {
+            uint8_t *base;
+            uint8_t *here;
+            uint8_t *stop;
+        } mem;
+        struct {
+            void *data1;
+        } unknown;
+        struct {
+            SDL1_RWops_t *orig;
+            sdl1_freerw custom_free;
+        } my;
+    } hidden;
+} SDL1_RWops_t;
+
+
+EXPORT int32_t my_native_seek(SDL1_RWops_t *context, int32_t offset, int32_t whence)
+{
+    return context->hidden.my.orig->seek(context->hidden.my.orig, offset, whence);
+}
+EXPORT int32_t my_native_read(SDL1_RWops_t *context, void *ptr, int32_t size, int32_t maxnum)
+{
+    return context->hidden.my.orig->read(context->hidden.my.orig, ptr, size, maxnum);
+}
+EXPORT int32_t my_native_write(SDL1_RWops_t *context, const void *ptr, int32_t size, int32_t num)
+{
+    return context->hidden.my.orig->write(context->hidden.my.orig, ptr, size, num);
+}
+EXPORT int32_t my_native_close(SDL1_RWops_t *context)
+{
+    int32_t ret = context->hidden.my.orig->close(context->hidden.my.orig);
+    context->hidden.my.custom_free(context);
+    return ret;
+}
+EXPORT int32_t my_emulated_seek(SDL1_RWops_t *context, int32_t offset, int32_t whence)
+{
+    return (int32_t)RunFunction(my_context, (uintptr_t)context->hidden.my.orig->seek, 3, context->hidden.my.orig, offset, whence);
+}
+EXPORT int32_t my_emulated_read(SDL1_RWops_t *context, void *ptr, int32_t size, int32_t maxnum)
+{
+    return (int32_t)RunFunction(my_context, (uintptr_t)context->hidden.my.orig->read, 4, context->hidden.my.orig, ptr, size, maxnum);
+}
+
+EXPORT int32_t my_emulated_write(SDL1_RWops_t *context, const void *ptr, int32_t size, int32_t num)
+{
+    return (int32_t)RunFunction(my_context, (uintptr_t)context->hidden.my.orig->write, 4, context->hidden.my.orig, ptr, size, num);
+}
+EXPORT int32_t my_emulated_close(SDL1_RWops_t *context)
+{
+    return (int32_t)RunFunction(my_context, (uintptr_t)context->hidden.my.orig->close, 1, context->hidden.my.orig);
+}
+
+SDL1_RWops_t* AddNativeRW(x64emu_t* emu, SDL1_RWops_t* ops)
+{
+    if(!ops)
+        return NULL;
+
+    uintptr_t fnc;
+    bridge_t* system = emu->context->system;
+
+    sdl1_allocrw Alloc = (sdl1_allocrw)emu->context->sdl1allocrw;
+    sdl1_freerw Free = (sdl1_freerw)emu->context->sdl1freerw;
+
+    SDL1_RWops_t* newrw = Alloc();
+    newrw->type = BOX64RW;
+    newrw->hidden.my.orig = ops;
+    newrw->hidden.my.custom_free = Free;
+
+    // get or create wrapper, add it to map and change to the emulated one if rw
+    #define GO(A, W) \
+    fnc = AddCheckBridge(system, W, my_native_##A, 0); \
+    newrw->A = (sdl1_##A)fnc;
+
+    GO(seek, iFpii)
+    GO(read, iFppii)
+    GO(write, iFppii)
+    GO(close, iFp)
+
+    #undef GO
+
+    return newrw;
+}
+
+// put Native RW function, wrapping emulated (callback style) ones if needed
+SDL1_RWops_t* RWNativeStart(x64emu_t* emu, SDL1_RWops_t* ops)
+{
+    if(!ops)
+        return NULL;
+    if(ops->type == BOX64RW)
+        return ops->hidden.my.orig;
+
+    sdl1_allocrw Alloc = (sdl1_allocrw)emu->context->sdl1allocrw;
+
+    SDL1_RWops_t* newrw = Alloc();
+    newrw->type = BOX64RW;
+    newrw->hidden.my.orig = ops;
+    // create wrapper
+    #define GO(A, W) \
+    newrw->A = my_emulated_##A;
+
+    GO(seek, iFpii)
+    GO(read, iFppii)
+    GO(write, iFppii)
+    GO(close, iFp)
+
+    #undef GO
+
+    return newrw;
+}
+
+// put back emulated function back in place
+void RWNativeEnd(SDL1_RWops_t* ops)
+{
+    if(!ops)
+        return;
+
+    if(ops->type != BOX64RW)
+        return; // do nothing
+
+    ops->hidden.my.custom_free(ops);
+}
+
+int32_t RWNativeSeek(SDL1_RWops_t *ops, int32_t offset, int32_t whence)
+{
+    return ops->seek(ops, offset, whence);
+}
+uint32_t RWNativeRead(SDL1_RWops_t* ops, void* ptr, uint32_t size, uint32_t maxnum)
+{
+    return ops->read(ops, ptr, size, maxnum);
+}
+int32_t RWNativeWrite(SDL1_RWops_t *ops, const void *ptr, int32_t size, int32_t num)
+{
+    return ops->write(ops, ptr, size, num);
+}
+int32_t RWNativeClose(SDL1_RWops_t* ops)
+{
+    return ops->close(ops);
+}
+
+void RWSetType(SDL1_RWops_t* r, int awesome)
+{
+    if(r)
+        r->type = awesome;  // I like shoot'em up :D
+}
\ No newline at end of file