-
-
Notifications
You must be signed in to change notification settings - Fork 423
Expand file tree
/
Copy pathalternate.c
More file actions
102 lines (92 loc) · 2.74 KB
/
alternate.c
File metadata and controls
102 lines (92 loc) · 2.74 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
#include <stdlib.h>
#include <stddef.h>
#include "alternate.h"
#include "custommem.h"
#include "khash.h"
#ifdef DYNAREC
#include "box64context.h"
#include "bridge.h"
#endif
#include "debug.h"
// Alternate address handling
typedef struct {
void* addr;
#ifdef HAVE_ALTJUMP
uintptr_t jump;
#endif
} my_alternate_t;
KHASH_MAP_INIT_INT64(alternate, my_alternate_t)
static kh_alternate_t *my_alternates = NULL;
// Per-address suppression: when set, getAlternate/hasAlternate/getAlternateJump
// will return "no alternate" for this specific address. Used by hooks that need
// to call through to the original emulated function via RunFunctionFmt without
// triggering infinite recursion (EmuRun applies getAlternate every iteration).
static uintptr_t alternate_suppress = 0;
int hasAlternate(void* addr) {
if(!my_alternates)
return 0;
if((uintptr_t)addr == alternate_suppress)
return 0;
khint_t k = kh_get(alternate, my_alternates, (uintptr_t)addr);
if(k==kh_end(my_alternates))
return 0;
return 1;
}
void* getAlternate(void* addr) {
if(!my_alternates)
return addr;
if((uintptr_t)addr == alternate_suppress)
return addr;
khint_t k = kh_get(alternate, my_alternates, (uintptr_t)addr);
if(k!=kh_end(my_alternates))
return kh_value(my_alternates, k).addr;
return addr;
}
void addAlternate(void* addr, void* alt) {
if(!my_alternates) {
my_alternates = kh_init(alternate);
}
if(addr==alt)
return; // nothing to do, alt is same as addr, but it's certainly an issue somewere!
int ret;
khint_t k = kh_put(alternate, my_alternates, (uintptr_t)addr, &ret);
if(!ret) // already there
return;
kh_value(my_alternates, k).addr = alt;
#ifdef HAVE_ALTJUMP
kh_value(my_alternates, k).jump = AddAltJump(my_context->alternates, (uintptr_t)addr, (uintptr_t)alt);
#endif
}
void suppressAlternate(void* addr) {
alternate_suppress = (uintptr_t)addr;
}
void unsuppressAlternate(void* addr) {
(void)addr;
alternate_suppress = 0;
}
void addCheckAlternate(void* addr, void* alt) {
if(!hasAlternate(addr))
addAlternate(addr, alt);
}
void cleanAlternate() {
if(my_alternates) {
kh_destroy(alternate, my_alternates);
my_alternates = NULL;
}
}
#ifdef HAVE_ALTJUMP
#include "bridge_private.h"
uintptr_t getAlternateJump(void* addr, int is32bits) {
if(!my_alternates)
return 0;
if((uintptr_t)addr == alternate_suppress)
return 0;
khint_t k = kh_get(alternate, my_alternates, (uintptr_t)addr);
if(k!=kh_end(my_alternates)) {
uintptr_t ret = kh_value(my_alternates, k).jump;
if(is32bits) ret += offsetof(onebridge_t, FF_2);
return ret;
}
return 0;
}
#endif