xref: /aosp_15_r20/external/musl/src/internal/fdpic_crt.h (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
1*c9945492SAndroid Build Coastguard Worker #include <stdint.h>
2*c9945492SAndroid Build Coastguard Worker #include <features.h>
3*c9945492SAndroid Build Coastguard Worker 
__fdpic_fixup(void * map,uintptr_t * a,uintptr_t * z)4*c9945492SAndroid Build Coastguard Worker hidden void *__fdpic_fixup(void *map, uintptr_t *a, uintptr_t *z)
5*c9945492SAndroid Build Coastguard Worker {
6*c9945492SAndroid Build Coastguard Worker 	/* If map is a null pointer, the program was loaded by a
7*c9945492SAndroid Build Coastguard Worker 	 * non-FDPIC-aware ELF loader, and fixups are not needed,
8*c9945492SAndroid Build Coastguard Worker 	 * but the value for the GOT pointer is. */
9*c9945492SAndroid Build Coastguard Worker 	if (!map) return (void *)z[-1];
10*c9945492SAndroid Build Coastguard Worker 
11*c9945492SAndroid Build Coastguard Worker 	struct {
12*c9945492SAndroid Build Coastguard Worker 		unsigned short version, nsegs;
13*c9945492SAndroid Build Coastguard Worker 		struct fdpic_loadseg {
14*c9945492SAndroid Build Coastguard Worker 			uintptr_t addr, p_vaddr, p_memsz;
15*c9945492SAndroid Build Coastguard Worker 		} segs[];
16*c9945492SAndroid Build Coastguard Worker 	} *lm = map;
17*c9945492SAndroid Build Coastguard Worker 	int nsegs = lm->nsegs, rseg = 0, vseg = 0;
18*c9945492SAndroid Build Coastguard Worker 	for (;;) {
19*c9945492SAndroid Build Coastguard Worker 		while (*a-lm->segs[rseg].p_vaddr >= lm->segs[rseg].p_memsz)
20*c9945492SAndroid Build Coastguard Worker 			if (++rseg == nsegs) rseg = 0;
21*c9945492SAndroid Build Coastguard Worker 		uintptr_t *r = (uintptr_t *)
22*c9945492SAndroid Build Coastguard Worker 			(*a + lm->segs[rseg].addr - lm->segs[rseg].p_vaddr);
23*c9945492SAndroid Build Coastguard Worker 		if (++a == z) return r;
24*c9945492SAndroid Build Coastguard Worker 		while (*r-lm->segs[vseg].p_vaddr >= lm->segs[vseg].p_memsz)
25*c9945492SAndroid Build Coastguard Worker 			if (++vseg == nsegs) vseg = 0;
26*c9945492SAndroid Build Coastguard Worker 		*r += lm->segs[vseg].addr - lm->segs[vseg].p_vaddr;
27*c9945492SAndroid Build Coastguard Worker 	}
28*c9945492SAndroid Build Coastguard Worker }
29