aboutsummaryrefslogtreecommitdiffstats
path: root/coroutine/win64/Context.h
diff options
context:
space:
mode:
Diffstat (limited to 'coroutine/win64/Context.h')
-rw-r--r--coroutine/win64/Context.h24
1 files changed, 17 insertions, 7 deletions
diff --git a/coroutine/win64/Context.h b/coroutine/win64/Context.h
index 32d6e1038b..db003a2ee6 100644
--- a/coroutine/win64/Context.h
+++ b/coroutine/win64/Context.h
@@ -17,13 +17,16 @@ extern "C" {
#define COROUTINE __declspec(noreturn) void
const size_t COROUTINE_REGISTERS = 8;
+const size_t COROUTINE_XMM_REGISTERS = 1+10*2;
struct coroutine_context
{
void **stack_pointer;
};
-typedef COROUTINE(* coroutine_start)(coroutine_context *from, coroutine_context *self);
+typedef void(* coroutine_start)(coroutine_context *from, coroutine_context *self);
+
+void coroutine_trampoline();
static inline void coroutine_initialize(
coroutine_context *context,
@@ -31,7 +34,8 @@ static inline void coroutine_initialize(
void *stack_pointer,
size_t stack_size
) {
- context->stack_pointer = (void**)stack_pointer;
+ /* Force 16-byte alignment */
+ context->stack_pointer = (void**)((uintptr_t)stack_pointer & ~0xF);
if (!start) {
assert(!context->stack_pointer);
@@ -39,16 +43,22 @@ static inline void coroutine_initialize(
return;
}
- /* Windows Thread Information Block */
- *--context->stack_pointer = 0; /* gs:[0x00] */
- *--context->stack_pointer = stack_pointer + stack_size; /* gs:[0x08] */
- *--context->stack_pointer = (void*)stack_pointer; /* gs:[0x10] */
-
+ /* Win64 ABI requires space for arguments */
+ context->stack_pointer -= 4;
+ /* Return address */
+ *--context->stack_pointer = 0;
*--context->stack_pointer = (void*)start;
+ *--context->stack_pointer = (void*)coroutine_trampoline;
+
+ /* Windows Thread Information Block */
+ /* *--context->stack_pointer = 0; */ /* gs:[0x00] is not used */
+ *--context->stack_pointer = (void*)stack_pointer; /* gs:[0x08] */
+ *--context->stack_pointer = (void*)((char *)stack_pointer - stack_size); /* gs:[0x10] */
context->stack_pointer -= COROUTINE_REGISTERS;
memset(context->stack_pointer, 0, sizeof(void*) * COROUTINE_REGISTERS);
+ memset(context->stack_pointer - COROUTINE_XMM_REGISTERS, 0, sizeof(void*) * COROUTINE_XMM_REGISTERS);
}
coroutine_context * coroutine_transfer(coroutine_context * current, coroutine_context * target);