2017-05-26 04:48:44 +08:00
|
|
|
/* Copyright (c) 2005-2006 Russ Cox, MIT; see COPYRIGHT */
|
|
|
|
|
|
|
|
#include "taskimpl.h"
|
|
|
|
|
|
|
|
#if defined(__APPLE__)
|
|
|
|
#if defined(__i386__)
|
|
|
|
#define NEEDX86MAKECONTEXT
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#elif defined(__x86_64__)
|
|
|
|
#define NEEDAMD64MAKECONTEXT
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#else
|
|
|
|
#define NEEDPOWERMAKECONTEXT
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__FreeBSD__) && defined(__i386__) && __FreeBSD__ < 5
|
|
|
|
#define NEEDX86MAKECONTEXT
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__OpenBSD__) && defined(__i386__)
|
|
|
|
#define NEEDX86MAKECONTEXT
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(__linux__) && defined(__arm__)
|
|
|
|
#define NEEDSWAPCONTEXT
|
|
|
|
#define NEEDARMMAKECONTEXT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NEEDPOWERMAKECONTEXT
|
2021-03-11 02:06:03 +08:00
|
|
|
void makecontext(ucontext_t* ucp, void (*func)(void), int argc, ...) {
|
2017-05-26 04:48:44 +08:00
|
|
|
ulong *sp, *tos;
|
|
|
|
va_list arg;
|
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
tos = (ulong*)ucp->uc_stack.ss_sp + ucp->uc_stack.ss_size / sizeof(ulong);
|
|
|
|
sp = tos - 16;
|
2017-05-26 04:48:44 +08:00
|
|
|
ucp->mc.pc = (long)func;
|
|
|
|
ucp->mc.sp = (long)sp;
|
|
|
|
va_start(arg, argc);
|
|
|
|
ucp->mc.r3 = va_arg(arg, long);
|
|
|
|
va_end(arg);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NEEDX86MAKECONTEXT
|
2021-03-11 02:06:03 +08:00
|
|
|
void makecontext(ucontext_t* ucp, void (*func)(void), int argc, ...) {
|
|
|
|
int* sp;
|
2017-05-26 04:48:44 +08:00
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
sp = (int*)ucp->uc_stack.ss_sp + ucp->uc_stack.ss_size / 4;
|
2017-05-26 04:48:44 +08:00
|
|
|
sp -= argc;
|
2021-03-11 02:06:03 +08:00
|
|
|
sp = (void*)((uintptr_t)sp - (uintptr_t)sp % 16); /* 16-align for OS X */
|
|
|
|
memmove(sp, &argc + 1, argc * sizeof(int));
|
2017-05-26 04:48:44 +08:00
|
|
|
|
2021-03-11 02:06:03 +08:00
|
|
|
*--sp = 0; /* return address */
|
2017-05-26 04:48:44 +08:00
|
|
|
ucp->uc_mcontext.mc_eip = (long)func;
|
|
|
|
ucp->uc_mcontext.mc_esp = (int)sp;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NEEDAMD64MAKECONTEXT
|
2021-03-11 02:06:03 +08:00
|
|
|
void makecontext(ucontext_t* ucp, void (*func)(void), int argc, ...) {
|
|
|
|
long* sp;
|
2017-05-26 04:48:44 +08:00
|
|
|
va_list va;
|
|
|
|
|
|
|
|
memset(&ucp->uc_mcontext, 0, sizeof ucp->uc_mcontext);
|
2021-03-11 02:06:03 +08:00
|
|
|
if (argc != 2)
|
2017-05-26 04:48:44 +08:00
|
|
|
__builtin_trap();
|
|
|
|
va_start(va, argc);
|
|
|
|
ucp->uc_mcontext.mc_rdi = va_arg(va, int);
|
|
|
|
ucp->uc_mcontext.mc_rsi = va_arg(va, int);
|
|
|
|
va_end(va);
|
2021-03-11 02:06:03 +08:00
|
|
|
sp = (long*)ucp->uc_stack.ss_sp + ucp->uc_stack.ss_size / sizeof(long);
|
2017-05-26 04:48:44 +08:00
|
|
|
sp -= argc;
|
2021-03-11 02:06:03 +08:00
|
|
|
sp = (void*)((uintptr_t)sp - (uintptr_t)sp % 16); /* 16-align for OS X */
|
|
|
|
*--sp = 0; /* return address */
|
2017-05-26 04:48:44 +08:00
|
|
|
ucp->uc_mcontext.mc_rip = (long)func;
|
|
|
|
ucp->uc_mcontext.mc_rsp = (long)sp;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NEEDARMMAKECONTEXT
|
2021-03-11 02:06:03 +08:00
|
|
|
void makecontext(ucontext_t* uc, void (*fn)(void), int argc, ...) {
|
2017-05-26 04:48:44 +08:00
|
|
|
int i, *sp;
|
|
|
|
va_list arg;
|
2021-03-11 02:06:03 +08:00
|
|
|
|
|
|
|
sp = (int*)uc->uc_stack.ss_sp + uc->uc_stack.ss_size / 4;
|
2017-05-26 04:48:44 +08:00
|
|
|
va_start(arg, argc);
|
2021-03-11 02:06:03 +08:00
|
|
|
for (i = 0; i < 4 && i < argc; i++)
|
2017-05-26 04:48:44 +08:00
|
|
|
uc->uc_mcontext.gregs[i] = va_arg(arg, uint);
|
|
|
|
va_end(arg);
|
|
|
|
uc->uc_mcontext.gregs[13] = (uint)sp;
|
|
|
|
uc->uc_mcontext.gregs[14] = (uint)fn;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef NEEDSWAPCONTEXT
|
2021-03-11 02:06:03 +08:00
|
|
|
int swapcontext(ucontext_t* oucp, const ucontext_t* ucp) {
|
|
|
|
if (getcontext(oucp) == 0)
|
2017-05-26 04:48:44 +08:00
|
|
|
setcontext(ucp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|