summaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
authormarcel <marcel@FreeBSD.org>2003-08-07 08:03:05 +0000
committermarcel <marcel@FreeBSD.org>2003-08-07 08:03:05 +0000
commit0dae1482723408fa760bf4b8cf973955a4b195a1 (patch)
treec94f88a3937bcdb6fa6a1e39de2d73a704df6905 /lib
parentb88da9d9aa359a2c9b535d0d5ba6cafdd00efcbe (diff)
downloadFreeBSD-src-0dae1482723408fa760bf4b8cf973955a4b195a1.zip
FreeBSD-src-0dae1482723408fa760bf4b8cf973955a4b195a1.tar.gz
Grok async contexts. When a thread is interrupted and an upcall
happens, the context of the interrupted thread is exported to userland. Unlike most contexts, it will be an async context and we cannot easily use our existing functions to set such a context. To avoid a lot of complexity that may possibly interfere with the common case, we simply let the kernel deal with it. However, we don't use the EPC based syscall path to invoke setcontext(2). No, we use the break-based syscall path. That way the trapframe will be compatible with the context we're trying to restore and we save the kernel a lot of trouble. The kind of trouble we did not want to go though ourselves... However, we also need to set the threads mailbox and there's no syscall to help us out. To avoid creating a new syscall, we use the context itself to pass the information to the kernel so that the kernel can update the mailbox. This involves setting a flag (_MC_FLAGS_KSE_SET_MBOX) and setting ifa (the address) and isr (the value).
Diffstat (limited to 'lib')
-rw-r--r--lib/libkse/arch/ia64/ia64/context.S14
-rw-r--r--lib/libkse/arch/ia64/include/pthread_md.h36
-rw-r--r--lib/libpthread/arch/ia64/ia64/context.S14
-rw-r--r--lib/libpthread/arch/ia64/include/pthread_md.h36
4 files changed, 82 insertions, 18 deletions
diff --git a/lib/libkse/arch/ia64/ia64/context.S b/lib/libkse/arch/ia64/ia64/context.S
index 6c0c082..49fe42e 100644
--- a/lib/libkse/arch/ia64/ia64/context.S
+++ b/lib/libkse/arch/ia64/ia64/context.S
@@ -27,6 +27,8 @@
#include <machine/asm.h>
__FBSDID("$FreeBSD$");
+#include <sys/syscall.h>
+
#define SIZEOF_SPECIAL (18*8)
/*
@@ -335,3 +337,15 @@ ENTRY(_ia64_save_context, 1)
;;
}
END(_ia64_save_context)
+
+/*
+ * void _ia64_break_setcontext(ucontext_t *ucp);
+ */
+ENTRY(_ia64_break_setcontext, 1)
+{ .mib
+ mov r15=SYS_setcontext
+ break 0x100000
+ br.ret.sptk rp
+ ;;
+}
+END(_ia64_break_setcontext)
diff --git a/lib/libkse/arch/ia64/include/pthread_md.h b/lib/libkse/arch/ia64/include/pthread_md.h
index d6aa130..95f1d2d 100644
--- a/lib/libkse/arch/ia64/include/pthread_md.h
+++ b/lib/libkse/arch/ia64/include/pthread_md.h
@@ -34,8 +34,14 @@
#include <ucontext.h>
#define THR_GETCONTEXT(ucp) _ia64_save_context(&(ucp)->uc_mcontext)
-#define THR_SETCONTEXT(ucp) _ia64_restore_context(&(ucp)->uc_mcontext, \
- 0, NULL)
+#define THR_SETCONTEXT(ucp) \
+ do { \
+ if ((ucp)->uc_mcontext.mc_flags & _MC_FLAGS_ASYNC_CONTEXT) \
+ _ia64_break_setcontext(ucp); \
+ else \
+ _ia64_restore_context(&(ucp)->uc_mcontext, 0, \
+ NULL); \
+ } while (0)
#define PER_THREAD
@@ -194,6 +200,7 @@ _get_curkse(void)
return (_tcb->tcb_curkcb->kcb_kse);
}
+void _ia64_break_setcontext(ucontext_t *ucp);
void _ia64_enter_uts(kse_func_t uts, struct kse_mailbox *km, void *stack,
size_t stacksz);
int _ia64_restore_context(mcontext_t *mc, intptr_t val, intptr_t *loc);
@@ -218,14 +225,25 @@ _thread_enter_uts(struct tcb *tcb, struct kcb *kcb)
static __inline int
_thread_switch(struct kcb *kcb, struct tcb *tcb, int setmbox)
{
+ mcontext_t *mc;
+
_tcb_set(kcb, tcb);
- if (setmbox != 0)
- _ia64_restore_context(&tcb->tcb_tmbx.tm_context.uc_mcontext,
- (intptr_t)&tcb->tcb_tmbx,
- (intptr_t *)&kcb->kcb_kmbx.km_curthread);
- else
- _ia64_restore_context(&tcb->tcb_tmbx.tm_context.uc_mcontext,
- 0, NULL);
+ mc = &tcb->tcb_tmbx.tm_context.uc_mcontext;
+ if (mc->mc_flags & _MC_FLAGS_ASYNC_CONTEXT) {
+ if (setmbox) {
+ mc->mc_flags |= _MC_FLAGS_KSE_SET_MBOX;
+ mc->mc_special.ifa =
+ (intptr_t)&kcb->kcb_kmbx.km_curthread;
+ mc->mc_special.isr = (intptr_t)&tcb->tcb_tmbx;
+ }
+ _ia64_break_setcontext(&tcb->tcb_tmbx.tm_context);
+ } else {
+ if (setmbox)
+ _ia64_restore_context(mc, (intptr_t)&tcb->tcb_tmbx,
+ (intptr_t *)&kcb->kcb_kmbx.km_curthread);
+ else
+ _ia64_restore_context(mc, 0, NULL);
+ }
/* We should not reach here. */
return (-1);
}
diff --git a/lib/libpthread/arch/ia64/ia64/context.S b/lib/libpthread/arch/ia64/ia64/context.S
index 6c0c082..49fe42e 100644
--- a/lib/libpthread/arch/ia64/ia64/context.S
+++ b/lib/libpthread/arch/ia64/ia64/context.S
@@ -27,6 +27,8 @@
#include <machine/asm.h>
__FBSDID("$FreeBSD$");
+#include <sys/syscall.h>
+
#define SIZEOF_SPECIAL (18*8)
/*
@@ -335,3 +337,15 @@ ENTRY(_ia64_save_context, 1)
;;
}
END(_ia64_save_context)
+
+/*
+ * void _ia64_break_setcontext(ucontext_t *ucp);
+ */
+ENTRY(_ia64_break_setcontext, 1)
+{ .mib
+ mov r15=SYS_setcontext
+ break 0x100000
+ br.ret.sptk rp
+ ;;
+}
+END(_ia64_break_setcontext)
diff --git a/lib/libpthread/arch/ia64/include/pthread_md.h b/lib/libpthread/arch/ia64/include/pthread_md.h
index d6aa130..95f1d2d 100644
--- a/lib/libpthread/arch/ia64/include/pthread_md.h
+++ b/lib/libpthread/arch/ia64/include/pthread_md.h
@@ -34,8 +34,14 @@
#include <ucontext.h>
#define THR_GETCONTEXT(ucp) _ia64_save_context(&(ucp)->uc_mcontext)
-#define THR_SETCONTEXT(ucp) _ia64_restore_context(&(ucp)->uc_mcontext, \
- 0, NULL)
+#define THR_SETCONTEXT(ucp) \
+ do { \
+ if ((ucp)->uc_mcontext.mc_flags & _MC_FLAGS_ASYNC_CONTEXT) \
+ _ia64_break_setcontext(ucp); \
+ else \
+ _ia64_restore_context(&(ucp)->uc_mcontext, 0, \
+ NULL); \
+ } while (0)
#define PER_THREAD
@@ -194,6 +200,7 @@ _get_curkse(void)
return (_tcb->tcb_curkcb->kcb_kse);
}
+void _ia64_break_setcontext(ucontext_t *ucp);
void _ia64_enter_uts(kse_func_t uts, struct kse_mailbox *km, void *stack,
size_t stacksz);
int _ia64_restore_context(mcontext_t *mc, intptr_t val, intptr_t *loc);
@@ -218,14 +225,25 @@ _thread_enter_uts(struct tcb *tcb, struct kcb *kcb)
static __inline int
_thread_switch(struct kcb *kcb, struct tcb *tcb, int setmbox)
{
+ mcontext_t *mc;
+
_tcb_set(kcb, tcb);
- if (setmbox != 0)
- _ia64_restore_context(&tcb->tcb_tmbx.tm_context.uc_mcontext,
- (intptr_t)&tcb->tcb_tmbx,
- (intptr_t *)&kcb->kcb_kmbx.km_curthread);
- else
- _ia64_restore_context(&tcb->tcb_tmbx.tm_context.uc_mcontext,
- 0, NULL);
+ mc = &tcb->tcb_tmbx.tm_context.uc_mcontext;
+ if (mc->mc_flags & _MC_FLAGS_ASYNC_CONTEXT) {
+ if (setmbox) {
+ mc->mc_flags |= _MC_FLAGS_KSE_SET_MBOX;
+ mc->mc_special.ifa =
+ (intptr_t)&kcb->kcb_kmbx.km_curthread;
+ mc->mc_special.isr = (intptr_t)&tcb->tcb_tmbx;
+ }
+ _ia64_break_setcontext(&tcb->tcb_tmbx.tm_context);
+ } else {
+ if (setmbox)
+ _ia64_restore_context(mc, (intptr_t)&tcb->tcb_tmbx,
+ (intptr_t *)&kcb->kcb_kmbx.km_curthread);
+ else
+ _ia64_restore_context(mc, 0, NULL);
+ }
/* We should not reach here. */
return (-1);
}
OpenPOWER on IntegriCloud