aboutsummaryrefslogtreecommitdiff
path: root/src/libthread/x86_64-ucontext.c
diff options
context:
space:
mode:
authorRuss Cox <rsc@swtch.com>2020-01-19 22:39:22 -0500
committerRuss Cox <rsc@swtch.com>2020-01-19 23:04:58 -0500
commit41b3e8b9893a8561af7e85ca98444bc284b4013d (patch)
treee9bf7fdeefbdb7d2e6c1ebc25c7888b81f0bb16b /src/libthread/x86_64-ucontext.c
parentac8042dfa9819f76ccfedd4aa36c1239322808b8 (diff)
downloadplan9port-41b3e8b9893a8561af7e85ca98444bc284b4013d.tar.gz
plan9port-41b3e8b9893a8561af7e85ca98444bc284b4013d.tar.bz2
plan9port-41b3e8b9893a8561af7e85ca98444bc284b4013d.zip
libthread: use consistent stack calculation code in makecontext
Also reduce duplication: makecontext is per-arch not per-os-arch. May fix #353.
Diffstat (limited to 'src/libthread/x86_64-ucontext.c')
-rw-r--r--src/libthread/x86_64-ucontext.c28
1 files changed, 28 insertions, 0 deletions
diff --git a/src/libthread/x86_64-ucontext.c b/src/libthread/x86_64-ucontext.c
new file mode 100644
index 00000000..5d1aaefc
--- /dev/null
+++ b/src/libthread/x86_64-ucontext.c
@@ -0,0 +1,28 @@
+#include "threadimpl.h"
+
+void
+makecontext(ucontext_t *uc, void (*fn)(void), int argc, ...)
+{
+ uintptr *sp;
+ va_list arg;
+
+ if(argc != 2)
+ sysfatal("libthread: makecontext misused");
+ va_start(arg, argc);
+ uc->mc.di = va_arg(arg, uint);
+ uc->mc.si = va_arg(arg, uint);
+ va_end(arg);
+
+ sp = USPALIGN(uc, 16);
+ *--sp = 0; // fn's return address
+ *--sp = (uintptr)fn; // return address of setcontext
+ uc->mc.sp = (uintptr)sp;
+}
+
+int
+swapcontext(ucontext_t *oucp, ucontext_t *ucp)
+{
+ if(getcontext(oucp) == 0)
+ setcontext(ucp);
+ return 0;
+}