mirror of
https://github.com/golang/go
synced 2024-11-19 21:04:43 -07:00
runtime: make go work on netbsd/amd64
R=golang-dev, rsc, devon.odell CC=golang-dev https://golang.org/cl/6222044
This commit is contained in:
parent
e947eba291
commit
5a043de746
@ -16,9 +16,9 @@ void runtime·setsig(int32, void(*)(int32, Siginfo*, void*, G*), bool);
|
|||||||
void runtime·sighandler(int32 sig, Siginfo *info, void *context, G *gp);
|
void runtime·sighandler(int32 sig, Siginfo *info, void *context, G *gp);
|
||||||
void runtime·sigpanic(void);
|
void runtime·sigpanic(void);
|
||||||
|
|
||||||
|
void runtime·setitimer(int32, Itimerval*, Itimerval*);
|
||||||
void runtime·sigaction(int32, struct sigaction*, struct sigaction*);
|
void runtime·sigaction(int32, struct sigaction*, struct sigaction*);
|
||||||
void runtime·sigaltstack(Sigaltstack*, Sigaltstack*);
|
void runtime·sigaltstack(Sigaltstack*, Sigaltstack*);
|
||||||
void runtime·setitimer(int32, Itimerval*, Itimerval*);
|
|
||||||
void runtime·sigprocmask(int32, Sigset*, Sigset*);
|
void runtime·sigprocmask(int32, Sigset*, Sigset*);
|
||||||
int32 runtime·sysctl(uint32*, uint32, byte*, uintptr*, byte*, uintptr);
|
int32 runtime·sysctl(uint32*, uint32, byte*, uintptr*, byte*, uintptr);
|
||||||
|
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
#include "signals_GOOS.h"
|
#include "signals_GOOS.h"
|
||||||
#include "os_GOOS.h"
|
#include "os_GOOS.h"
|
||||||
|
|
||||||
|
extern void runtime·lwp_tramp(void);
|
||||||
extern void runtime·sigtramp(void);
|
extern void runtime·sigtramp(void);
|
||||||
|
|
||||||
typedef struct sigaction {
|
typedef struct sigaction {
|
||||||
@ -143,3 +144,14 @@ runtime·setsig(int32 i, void (*fn)(int32, Siginfo*, void*, G*), bool restart)
|
|||||||
sa._sa_u._sa_sigaction = (void*)fn;
|
sa._sa_u._sa_sigaction = (void*)fn;
|
||||||
runtime·sigaction(i, &sa, nil);
|
runtime·sigaction(i, &sa, nil);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
runtime·lwp_mcontext_init(McontextT *mc, void *stack, M *m, G *g, void (*fn)(void))
|
||||||
|
{
|
||||||
|
// Machine dependent mcontext initialisation for LWP.
|
||||||
|
mc->__gregs[REG_RIP] = (uint64)runtime·lwp_tramp;
|
||||||
|
mc->__gregs[REG_RSP] = (uint64)stack;
|
||||||
|
mc->__gregs[REG_R8] = (uint64)m;
|
||||||
|
mc->__gregs[REG_R9] = (uint64)g;
|
||||||
|
mc->__gregs[REG_R12] = (uint64)fn;
|
||||||
|
}
|
||||||
|
@ -8,42 +8,25 @@
|
|||||||
|
|
||||||
#include "zasm_GOOS_GOARCH.h"
|
#include "zasm_GOOS_GOARCH.h"
|
||||||
|
|
||||||
// int64 rfork_thread(int32 flags, void *stack, M *m, G *g, void (*fn)(void));
|
// int64 lwp_create(void *context, uintptr flags, void *lwpid)
|
||||||
TEXT runtime·rfork_thread(SB),7,$0
|
TEXT runtime·lwp_create(SB),7,$0
|
||||||
MOVL flags+8(SP), DI
|
|
||||||
MOVQ stack+16(SP), SI
|
|
||||||
|
|
||||||
// Copy m, g, fn off parent stack for use by child.
|
MOVQ context+0(FP), DI
|
||||||
MOVQ mm+24(SP), R8
|
MOVQ flags+8(FP), SI
|
||||||
MOVQ gg+32(SP), R9
|
MOVQ lwpid+16(FP), DX
|
||||||
MOVQ fn+40(SP), R12
|
MOVL $309, AX // sys__lwp_create
|
||||||
|
|
||||||
MOVL $251, AX // sys_rfork
|
|
||||||
SYSCALL
|
SYSCALL
|
||||||
|
JCC 2(PC)
|
||||||
// Return if rfork syscall failed
|
|
||||||
JCC 3(PC)
|
|
||||||
NEGL AX
|
NEGL AX
|
||||||
RET
|
RET
|
||||||
|
|
||||||
// In parent, return.
|
TEXT runtime·lwp_tramp(SB),7,$0
|
||||||
CMPL AX, $0
|
|
||||||
JEQ 2(PC)
|
|
||||||
RET
|
|
||||||
|
|
||||||
// In child, on new stack.
|
|
||||||
MOVQ SI, SP
|
|
||||||
|
|
||||||
// Initialize m->procid to thread ID
|
|
||||||
MOVL $299, AX // sys_getthrid
|
|
||||||
SYSCALL
|
|
||||||
MOVQ AX, m_procid(R8)
|
|
||||||
|
|
||||||
// Set FS to point at m->tls.
|
// Set FS to point at m->tls.
|
||||||
LEAQ m_tls(R8), DI
|
LEAQ m_tls(R8), DI
|
||||||
CALL runtime·settls(SB)
|
CALL runtime·settls(SB)
|
||||||
|
|
||||||
// In child, set up new stack
|
// Set up new stack.
|
||||||
get_tls(CX)
|
get_tls(CX)
|
||||||
MOVQ R8, m(CX)
|
MOVQ R8, m(CX)
|
||||||
MOVQ R9, g(CX)
|
MOVQ R9, g(CX)
|
||||||
@ -52,8 +35,8 @@ TEXT runtime·rfork_thread(SB),7,$0
|
|||||||
// Call fn
|
// Call fn
|
||||||
CALL R12
|
CALL R12
|
||||||
|
|
||||||
// It shouldn't return. If it does, exit
|
// It shouldn't return. If it does, exit.
|
||||||
MOVL $302, AX // sys_threxit
|
MOVL $310, AX // sys__lwp_exit
|
||||||
SYSCALL
|
SYSCALL
|
||||||
JMP -3(PC) // keep exiting
|
JMP -3(PC) // keep exiting
|
||||||
|
|
||||||
@ -62,19 +45,19 @@ TEXT runtime·osyield(SB),7,$0
|
|||||||
SYSCALL
|
SYSCALL
|
||||||
RET
|
RET
|
||||||
|
|
||||||
TEXT runtime·thrsleep(SB),7,$0
|
TEXT runtime·lwp_park(SB),7,$0
|
||||||
MOVQ 8(SP), DI // arg 1 - ident
|
MOVQ 8(SP), DI // arg 1 - abstime
|
||||||
MOVL 16(SP), SI // arg 2 - clock_id
|
MOVL 16(SP), SI // arg 2 - unpark
|
||||||
MOVQ 24(SP), DX // arg 3 - tp
|
MOVQ 24(SP), DX // arg 3 - hint
|
||||||
MOVQ 32(SP), R10 // arg 4 - lock
|
MOVQ 32(SP), R10 // arg 4 - unparkhint
|
||||||
MOVL $300, AX // sys_thrsleep
|
MOVL $434, AX // sys__lwp_park
|
||||||
SYSCALL
|
SYSCALL
|
||||||
RET
|
RET
|
||||||
|
|
||||||
TEXT runtime·thrwakeup(SB),7,$0
|
TEXT runtime·lwp_unpark(SB),7,$0
|
||||||
MOVQ 8(SP), DI // arg 1 - ident
|
MOVQ 8(SP), DI // arg 1 - lwp
|
||||||
MOVL 16(SP), SI // arg 2 - n
|
MOVL 16(SP), SI // arg 2 - hint
|
||||||
MOVL $301, AX // sys_thrwakeup
|
MOVL $321, AX // sys__lwp_unpark
|
||||||
SYSCALL
|
SYSCALL
|
||||||
RET
|
RET
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@ enum
|
|||||||
ESRCH = 3,
|
ESRCH = 3,
|
||||||
ENOTSUP = 91,
|
ENOTSUP = 91,
|
||||||
|
|
||||||
// From NetBSD's sys/time.h
|
// From NetBSD's <sys/time.h>
|
||||||
CLOCK_REALTIME = 0,
|
CLOCK_REALTIME = 0,
|
||||||
CLOCK_VIRTUAL = 1,
|
CLOCK_VIRTUAL = 1,
|
||||||
CLOCK_PROF = 2,
|
CLOCK_PROF = 2,
|
||||||
@ -20,9 +20,14 @@ enum
|
|||||||
|
|
||||||
extern SigTab runtime·sigtab[];
|
extern SigTab runtime·sigtab[];
|
||||||
|
|
||||||
extern int64 runtime·rfork_thread(int32 flags, void *stack, M *m, G *g, void (*fn)(void));
|
static Sigset sigset_all = { ~(uint32)0, ~(uint32)0, ~(uint32)0, ~(uint32)0, };
|
||||||
extern int32 runtime·thrsleep(void *ident, int32 clock_id, void *tsp, void *lock);
|
static Sigset sigset_none;
|
||||||
extern int32 runtime·thrwakeup(void *ident, int32 n);
|
|
||||||
|
extern void runtime·getcontext(UcontextT *context);
|
||||||
|
extern int32 runtime·lwp_create(UcontextT *context, uintptr flags, void *lwpid);
|
||||||
|
extern void runtime·lwp_mcontext_init(void *mc, void *stack, M *m, G *g, void (*fn)(void));
|
||||||
|
extern int32 runtime·lwp_park(Timespec *abstime, int32 unpark, void *hint, void *unparkhint);
|
||||||
|
extern int32 runtime·lwp_unpark(int32 lwp, void *hint);
|
||||||
|
|
||||||
// From NetBSD's <sys/sysctl.h>
|
// From NetBSD's <sys/sysctl.h>
|
||||||
#define CTL_HW 6
|
#define CTL_HW 6
|
||||||
@ -68,13 +73,30 @@ runtime·semasleep(int64 ns)
|
|||||||
if(m->waitsemacount == 0) {
|
if(m->waitsemacount == 0) {
|
||||||
// sleep until semaphore != 0 or timeout.
|
// sleep until semaphore != 0 or timeout.
|
||||||
// thrsleep unlocks m->waitsemalock.
|
// thrsleep unlocks m->waitsemalock.
|
||||||
if(ns < 0)
|
if(ns < 0) {
|
||||||
runtime·thrsleep(&m->waitsemacount, 0, nil, &m->waitsemalock);
|
// TODO(jsing) - potential deadlock!
|
||||||
else {
|
//
|
||||||
|
// There is a potential deadlock here since we
|
||||||
|
// have to release the waitsemalock mutex
|
||||||
|
// before we call lwp_park() to suspend the
|
||||||
|
// thread. This allows another thread to
|
||||||
|
// release the lock and call lwp_unpark()
|
||||||
|
// before the thread is actually suspended.
|
||||||
|
// If this occurs the current thread will end
|
||||||
|
// up sleeping indefinitely. Unfortunately
|
||||||
|
// the NetBSD kernel does not appear to provide
|
||||||
|
// a mechanism for unlocking the userspace
|
||||||
|
// mutex once the thread is actually parked.
|
||||||
|
runtime·atomicstore(&m->waitsemalock, 0);
|
||||||
|
runtime·lwp_park(nil, 0, &m->waitsemacount, nil);
|
||||||
|
} else {
|
||||||
ns += runtime·nanotime();
|
ns += runtime·nanotime();
|
||||||
ts.tv_sec = ns/1000000000LL;
|
ts.tv_sec = ns/1000000000LL;
|
||||||
ts.tv_nsec = ns%1000000000LL;
|
ts.tv_nsec = ns%1000000000LL;
|
||||||
runtime·thrsleep(&m->waitsemacount, CLOCK_REALTIME, &ts, &m->waitsemalock);
|
// TODO(jsing) - potential deadlock!
|
||||||
|
// See above for details.
|
||||||
|
runtime·atomicstore(&m->waitsemalock, 0);
|
||||||
|
runtime·lwp_park(&ts, 0, &m->waitsemacount, nil);
|
||||||
}
|
}
|
||||||
// reacquire lock
|
// reacquire lock
|
||||||
while(runtime·xchg(&m->waitsemalock, 1))
|
while(runtime·xchg(&m->waitsemalock, 1))
|
||||||
@ -112,28 +134,26 @@ runtime·semawakeup(M *mp)
|
|||||||
while(runtime·xchg(&mp->waitsemalock, 1))
|
while(runtime·xchg(&mp->waitsemalock, 1))
|
||||||
runtime·osyield();
|
runtime·osyield();
|
||||||
mp->waitsemacount++;
|
mp->waitsemacount++;
|
||||||
ret = runtime·thrwakeup(&mp->waitsemacount, 1);
|
// TODO(jsing) - potential deadlock, see semasleep() for details.
|
||||||
|
// Confirm that LWP is parked before unparking...
|
||||||
|
ret = runtime·lwp_unpark(mp->procid, &mp->waitsemacount);
|
||||||
if(ret != 0 && ret != ESRCH)
|
if(ret != 0 && ret != ESRCH)
|
||||||
runtime·printf("thrwakeup addr=%p sem=%d ret=%d\n", &mp->waitsemacount, mp->waitsemacount, ret);
|
runtime·printf("thrwakeup addr=%p sem=%d ret=%d\n", &mp->waitsemacount, mp->waitsemacount, ret);
|
||||||
// spin-mutex unlock
|
// spin-mutex unlock
|
||||||
runtime·atomicstore(&mp->waitsemalock, 0);
|
runtime·atomicstore(&mp->waitsemalock, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// From NetBSD's sys/param.h
|
// From NetBSD's <sys/ucontext.h>
|
||||||
#define RFPROC (1<<4) /* change child (else changes curproc) */
|
#define _UC_SIGMASK 0x01
|
||||||
#define RFMEM (1<<5) /* share `address space' */
|
#define _UC_CPU 0x04
|
||||||
#define RFNOWAIT (1<<6) /* parent need not wait() on child */
|
|
||||||
#define RFTHREAD (1<<13) /* create a thread, not a process */
|
|
||||||
|
|
||||||
void
|
void
|
||||||
runtime·newosproc(M *m, G *g, void *stk, void (*fn)(void))
|
runtime·newosproc(M *m, G *g, void *stk, void (*fn)(void))
|
||||||
{
|
{
|
||||||
int32 flags;
|
UcontextT uc;
|
||||||
int32 ret;
|
int32 ret;
|
||||||
|
|
||||||
flags = RFPROC | RFTHREAD | RFMEM | RFNOWAIT;
|
if(0) {
|
||||||
|
|
||||||
if (0) {
|
|
||||||
runtime·printf(
|
runtime·printf(
|
||||||
"newosproc stk=%p m=%p g=%p fn=%p id=%d/%d ostk=%p\n",
|
"newosproc stk=%p m=%p g=%p fn=%p id=%d/%d ostk=%p\n",
|
||||||
stk, m, g, fn, m->id, m->tls[0], &m);
|
stk, m, g, fn, m->id, m->tls[0], &m);
|
||||||
@ -141,10 +161,18 @@ runtime·newosproc(M *m, G *g, void *stk, void (*fn)(void))
|
|||||||
|
|
||||||
m->tls[0] = m->id; // so 386 asm can find it
|
m->tls[0] = m->id; // so 386 asm can find it
|
||||||
|
|
||||||
if((ret = runtime·rfork_thread(flags, stk, m, g, fn)) < 0) {
|
runtime·getcontext(&uc);
|
||||||
|
|
||||||
|
uc.uc_flags = _UC_SIGMASK | _UC_CPU;
|
||||||
|
uc.uc_link = nil;
|
||||||
|
uc.uc_sigmask = sigset_all;
|
||||||
|
|
||||||
|
runtime·lwp_mcontext_init(&uc.uc_mcontext, stk, m, g, fn);
|
||||||
|
|
||||||
|
ret = runtime·lwp_create(&uc, 0, &m->procid);
|
||||||
|
|
||||||
|
if(ret < 0) {
|
||||||
runtime·printf("runtime: failed to create new OS thread (have %d already; errno=%d)\n", runtime·mcount() - 1, -ret);
|
runtime·printf("runtime: failed to create new OS thread (have %d already; errno=%d)\n", runtime·mcount() - 1, -ret);
|
||||||
if (ret == -ENOTSUP)
|
|
||||||
runtime·printf("runtime: is kern.rthreads disabled?\n");
|
|
||||||
runtime·throw("runtime.newosproc");
|
runtime·throw("runtime.newosproc");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -153,6 +181,9 @@ void
|
|||||||
runtime·osinit(void)
|
runtime·osinit(void)
|
||||||
{
|
{
|
||||||
runtime·ncpu = getncpu();
|
runtime·ncpu = getncpu();
|
||||||
|
|
||||||
|
// Main thread is always LWP 1.
|
||||||
|
m->procid = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
@ -168,6 +199,7 @@ runtime·minit(void)
|
|||||||
// Initialize signal handling
|
// Initialize signal handling
|
||||||
m->gsignal = runtime·malg(32*1024);
|
m->gsignal = runtime·malg(32*1024);
|
||||||
runtime·signalstack(m->gsignal->stackguard - StackGuard, 32*1024);
|
runtime·signalstack(m->gsignal->stackguard - StackGuard, 32*1024);
|
||||||
|
runtime·sigprocmask(SIG_SETMASK, &sigset_none, nil);
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
Loading…
Reference in New Issue
Block a user