runtime: Support cgo callbacks from threads started by C.

This adjusts the extram support to work with gccgo.  There are
some corresponding changes to cgo in
https://codereview.appspot.com/11406047/ .

From-SVN: r201179
This commit is contained in:
Ian Lance Taylor 2013-07-23 20:26:09 +00:00
parent fb48aadc78
commit 7acd2b86bf
5 changed files with 131 additions and 20 deletions

View File

@ -35,6 +35,9 @@ syscall_cgocall ()
M* m; M* m;
G* g; G* g;
if (runtime_needextram && runtime_cas (&runtime_needextram, 1, 0))
runtime_newextram ();
m = runtime_m (); m = runtime_m ();
++m->ncgocall; ++m->ncgocall;
g = runtime_g (); g = runtime_g ();
@ -71,7 +74,24 @@ syscall_cgocalldone ()
void void
syscall_cgocallback () syscall_cgocallback ()
{ {
M *mp;
mp = runtime_m ();
if (mp == NULL)
{
runtime_needm ();
mp = runtime_m ();
mp->dropextram = true;
}
runtime_exitsyscall (); runtime_exitsyscall ();
mp = runtime_m ();
if (mp->needextram)
{
mp->needextram = 0;
runtime_newextram ();
}
} }
/* Prepare to return to C/C++ code from a callback to Go code. */ /* Prepare to return to C/C++ code from a callback to Go code. */
@ -79,7 +99,15 @@ syscall_cgocallback ()
void void
syscall_cgocallbackdone () syscall_cgocallbackdone ()
{ {
M *mp;
runtime_entersyscall (); runtime_entersyscall ();
mp = runtime_m ();
if (mp->dropextram && runtime_g ()->ncgo == 0)
{
mp->dropextram = false;
runtime_dropm ();
}
} }
/* Allocate memory and save it in a list visible to the Go garbage /* Allocate memory and save it in a list visible to the Go garbage

View File

@ -42,6 +42,7 @@ __go_undefer (_Bool *frame)
{ {
struct __go_defer_stack *d; struct __go_defer_stack *d;
void (*pfn) (void *); void (*pfn) (void *);
M *m;
d = g->defer; d = g->defer;
pfn = d->__pfn; pfn = d->__pfn;
@ -51,7 +52,14 @@ __go_undefer (_Bool *frame)
(*pfn) (d->__arg); (*pfn) (d->__arg);
g->defer = d->__next; g->defer = d->__next;
__go_free (d);
/* This may be called by a cgo callback routine to defer the
call to syscall.CgocallBackDone, in which case we will not
have a memory context. Don't try to free anything in that
case--the GC will release it later. */
m = runtime_m ();
if (m != NULL && m->mcache != NULL)
__go_free (d);
/* Since we are executing a defer function here, we know we are /* Since we are executing a defer function here, we know we are
returning from the calling function. If the calling returning from the calling function. If the calling

View File

@ -54,6 +54,7 @@ __go_panic (struct __go_empty_interface arg)
{ {
struct __go_defer_stack *d; struct __go_defer_stack *d;
void (*pfn) (void *); void (*pfn) (void *);
M *m;
d = g->defer; d = g->defer;
if (d == NULL) if (d == NULL)
@ -95,7 +96,14 @@ __go_panic (struct __go_empty_interface arg)
} }
g->defer = d->__next; g->defer = d->__next;
__go_free (d);
/* This may be called by a cgo callback routine to defer the
call to syscall.CgocallBackDone, in which case we will not
have a memory context. Don't try to free anything in that
case--the GC will release it later. */
m = runtime_m ();
if (m != NULL && m->mcache != NULL)
__go_free (d);
} }
/* The panic was not recovered. */ /* The panic was not recovered. */

View File

@ -397,7 +397,8 @@ enum { MaxGomaxprocs = 1<<8 };
Sched runtime_sched; Sched runtime_sched;
int32 runtime_gomaxprocs; int32 runtime_gomaxprocs;
bool runtime_singleproc; bool runtime_singleproc;
bool runtime_iscgo; bool runtime_iscgo = true;
uint32 runtime_needextram = 1;
uint32 runtime_gcwaiting; uint32 runtime_gcwaiting;
M runtime_m0; M runtime_m0;
G runtime_g0; // idle goroutine for m0 G runtime_g0; // idle goroutine for m0
@ -901,8 +902,8 @@ runtime_mstart(void* mp)
#ifdef USING_SPLIT_STACK #ifdef USING_SPLIT_STACK
{ {
int dont_block_signals = 0; int dont_block_signals = 0;
__splitstack_block_signals(&dont_block_signals, nil); __splitstack_block_signals(&dont_block_signals, nil);
} }
#endif #endif
@ -944,7 +945,7 @@ struct CgoThreadStart
// Allocate a new m unassociated with any thread. // Allocate a new m unassociated with any thread.
// Can use p for allocation context if needed. // Can use p for allocation context if needed.
M* M*
runtime_allocm(P *p) runtime_allocm(P *p, int32 stacksize, byte** ret_g0_stack, size_t* ret_g0_stacksize)
{ {
M *mp; M *mp;
@ -961,7 +962,7 @@ runtime_allocm(P *p)
mp = runtime_mal(sizeof *mp); mp = runtime_mal(sizeof *mp);
mcommoninit(mp); mcommoninit(mp);
mp->g0 = runtime_malg(-1, nil, nil); mp->g0 = runtime_malg(stacksize, ret_g0_stack, ret_g0_stacksize);
if(p == m->p) if(p == m->p)
releasep(); releasep();
@ -1006,6 +1007,9 @@ static void unlockextra(M*);
// //
// When the callback is done with the m, it calls dropm to // When the callback is done with the m, it calls dropm to
// put the m back on the list. // put the m back on the list.
//
// Unlike the gc toolchain, we start running on curg, since we are
// just going to return and let the caller continue.
void void
runtime_needm(void) runtime_needm(void)
{ {
@ -1027,18 +1031,40 @@ runtime_needm(void)
mp->needextram = mp->schedlink == nil; mp->needextram = mp->schedlink == nil;
unlockextra(mp->schedlink); unlockextra(mp->schedlink);
// Install m and g (= m->g0) and set the stack bounds // Install m and g (= m->curg).
// to match the current stack. We don't actually know runtime_setmg(mp, mp->curg);
// how big the stack is, like we don't know how big any
// scheduling stack is, but we assume there's at least 32 kB,
// which is more than enough for us.
runtime_setmg(mp, mp->g0);
// We assume that the split stack support has been initialized // Initialize g's context as in mstart.
// for this new thread. initcontext();
g->status = Gsyscall;
g->entry = nil;
g->param = nil;
#ifdef USING_SPLIT_STACK
__splitstack_getcontext(&g->stack_context[0]);
#else
g->gcinitial_sp = &mp;
g->gcstack_size = 0;
g->gcnext_sp = &mp;
#endif
getcontext(&g->context);
if(g->entry != nil) {
// Got here from mcall.
void (*pfn)(G*) = (void (*)(G*))g->entry;
G* gp = (G*)g->param;
pfn(gp);
*(int*)0x22 = 0x22;
}
// Initialize this thread to use the m. // Initialize this thread to use the m.
runtime_minit(); runtime_minit();
#ifdef USING_SPLIT_STACK
{
int dont_block_signals = 0;
__splitstack_block_signals(&dont_block_signals, nil);
}
#endif
} }
// newextram allocates an m and puts it on the extra list. // newextram allocates an m and puts it on the extra list.
@ -1049,15 +1075,17 @@ runtime_newextram(void)
{ {
M *mp, *mnext; M *mp, *mnext;
G *gp; G *gp;
byte *g0_sp, *sp;
size_t g0_spsize, spsize;
// Create extra goroutine locked to extra m. // Create extra goroutine locked to extra m.
// The goroutine is the context in which the cgo callback will run. // The goroutine is the context in which the cgo callback will run.
// The sched.pc will never be returned to, but setting it to // The sched.pc will never be returned to, but setting it to
// runtime.goexit makes clear to the traceback routines where // runtime.goexit makes clear to the traceback routines where
// the goroutine stack ends. // the goroutine stack ends.
mp = runtime_allocm(nil); mp = runtime_allocm(nil, StackMin, &g0_sp, &g0_spsize);
gp = runtime_malg(StackMin, nil, nil); gp = runtime_malg(StackMin, &sp, &spsize);
gp->status = Gsyscall; gp->status = Gdead;
mp->curg = gp; mp->curg = gp;
mp->locked = LockInternal; mp->locked = LockInternal;
mp->lockedg = gp; mp->lockedg = gp;
@ -1072,6 +1100,16 @@ runtime_newextram(void)
runtime_unlock(&runtime_sched); runtime_unlock(&runtime_sched);
gp->goid = runtime_xadd64(&runtime_sched.goidgen, 1); gp->goid = runtime_xadd64(&runtime_sched.goidgen, 1);
// The context for gp will be set up in runtime_needm. But
// here we need to set up the context for g0.
getcontext(&mp->g0->context);
mp->g0->context.uc_stack.ss_sp = g0_sp;
#ifdef MAKECONTEXT_STACK_TOP
mp->g0->context.uc_stack.ss_sp += g0_spsize;
#endif
mp->g0->context.uc_stack.ss_size = g0_spsize;
makecontext(&mp->g0->context, kickoff, 0);
// Add m to the extra list. // Add m to the extra list.
mnext = lockextra(true); mnext = lockextra(true);
mp->schedlink = mnext; mp->schedlink = mnext;
@ -1114,6 +1152,8 @@ runtime_dropm(void)
mp = m; mp = m;
runtime_setmg(nil, nil); runtime_setmg(nil, nil);
mp->curg->status = Gdead;
mnext = lockextra(true); mnext = lockextra(true);
mp->schedlink = mnext; mp->schedlink = mnext;
unlockextra(mp); unlockextra(mp);
@ -1159,6 +1199,29 @@ unlockextra(M *mp)
runtime_atomicstorep(&runtime_extram, mp); runtime_atomicstorep(&runtime_extram, mp);
} }
static int32
countextra()
{
M *mp, *mc;
int32 c;
for(;;) {
mp = runtime_atomicloadp(&runtime_extram);
if(mp == MLOCKED) {
runtime_osyield();
continue;
}
if(!runtime_casp(&runtime_extram, mp, MLOCKED)) {
runtime_osyield();
continue;
}
c = 0;
for(mc = mp; mc != nil; mc = mc->schedlink)
c++;
runtime_atomicstorep(&runtime_extram, mp);
return c;
}
}
// Create a new m. It will start off with a call to fn, or else the scheduler. // Create a new m. It will start off with a call to fn, or else the scheduler.
static void static void
@ -1166,7 +1229,7 @@ newm(void(*fn)(void), P *p)
{ {
M *mp; M *mp;
mp = runtime_allocm(p); mp = runtime_allocm(p, -1, nil, nil);
mp->nextp = p; mp->nextp = p;
mp->mstartfn = fn; mp->mstartfn = fn;
@ -2348,7 +2411,7 @@ checkdead(void)
int32 run, grunning, s; int32 run, grunning, s;
// -1 for sysmon // -1 for sysmon
run = runtime_sched.mcount - runtime_sched.nmidle - runtime_sched.mlocked - 1; run = runtime_sched.mcount - runtime_sched.nmidle - runtime_sched.mlocked - 1 - countextra();
if(run > 0) if(run > 0)
return; return;
if(run < 0) { if(run < 0) {

View File

@ -273,6 +273,7 @@ struct M
GCStats gcstats; GCStats gcstats;
bool racecall; bool racecall;
bool needextram; bool needextram;
bool dropextram; // for gccgo: drop after call is done.
void* racepc; void* racepc;
void (*waitunlockf)(Lock*); void (*waitunlockf)(Lock*);
void* waitlock; void* waitlock;
@ -450,6 +451,7 @@ extern G* runtime_lastg;
extern M* runtime_allm; extern M* runtime_allm;
extern P** runtime_allp; extern P** runtime_allp;
extern int32 runtime_gomaxprocs; extern int32 runtime_gomaxprocs;
extern uint32 runtime_needextram;
extern bool runtime_singleproc; extern bool runtime_singleproc;
extern uint32 runtime_panicking; extern uint32 runtime_panicking;
extern uint32 runtime_gcwaiting; // gc is waiting to run extern uint32 runtime_gcwaiting; // gc is waiting to run
@ -518,6 +520,8 @@ G* runtime_malg(int32, byte**, size_t*);
void runtime_mpreinit(M*); void runtime_mpreinit(M*);
void runtime_minit(void); void runtime_minit(void);
void runtime_unminit(void); void runtime_unminit(void);
void runtime_needm(void);
void runtime_dropm(void);
void runtime_signalstack(byte*, int32); void runtime_signalstack(byte*, int32);
MCache* runtime_allocmcache(void); MCache* runtime_allocmcache(void);
void runtime_freemcache(MCache*); void runtime_freemcache(MCache*);