Commit afb22604 authored by Dmitriy Vyukov's avatar Dmitriy Vyukov

runtime: convert note to Go

Note is required for timers and heap scavenger.

LGTM=rsc
R=golang-codereviews, rsc
CC=golang-codereviews, khr, rlh
https://golang.org/cl/128620043
parent f4485784
...@@ -383,6 +383,7 @@ func (w *Walker) parseFile(dir, file string) (*ast.File, error) { ...@@ -383,6 +383,7 @@ func (w *Walker) parseFile(dir, file string) (*ast.File, error) {
" mspan struct{}; m struct{}; lock struct{}; slicetype struct{};" + " mspan struct{}; m struct{}; lock struct{}; slicetype struct{};" +
" iface struct{}; eface struct{}; interfacetype struct{}; itab struct{};" + " iface struct{}; eface struct{}; interfacetype struct{}; itab struct{};" +
" mcache struct{}; bucket struct{}; sudog struct{}; g struct{};" + " mcache struct{}; bucket struct{}; sudog struct{}; g struct{};" +
" note struct{};" +
")" ")"
f, err = parser.ParseFile(fset, filename, src, 0) f, err = parser.ParseFile(fset, filename, src, 0)
if err != nil { if err != nil {
......
...@@ -19,7 +19,6 @@ var Fintto64 = fintto64 ...@@ -19,7 +19,6 @@ var Fintto64 = fintto64
var F64toint = f64toint var F64toint = f64toint
func entersyscall() func entersyscall()
func exitsyscall()
func golockedOSThread() bool func golockedOSThread() bool
func stackguard() (sp, limit uintptr) func stackguard() (sp, limit uintptr)
......
...@@ -125,6 +125,16 @@ runtime·notewakeup(Note *n) ...@@ -125,6 +125,16 @@ runtime·notewakeup(Note *n)
runtime·futexwakeup((uint32*)&n->key, 1); runtime·futexwakeup((uint32*)&n->key, 1);
} }
void
runtime·notewakeup_m(void)
{
Note *n;
n = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
runtime·notewakeup(n);
}
void void
runtime·notesleep(Note *n) runtime·notesleep(Note *n)
{ {
...@@ -199,3 +209,19 @@ runtime·notetsleepg(Note *n, int64 ns) ...@@ -199,3 +209,19 @@ runtime·notetsleepg(Note *n, int64 ns)
runtime·exitsyscall(); runtime·exitsyscall();
return res; return res;
} }
void
runtime·notetsleepg_m(void)
{
Note *n;
int64 ns;
n = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
ns = g->m->scalararg[0] + ((int64)g->m->scalararg[1] << 32);
runtime·entersyscallblock_m(pc, sp);
notetsleep(n, ns, 0, 0);
// caller will call exitsyscall on g stack
runtime·gogo(&g->m->curg->sched);
}
...@@ -147,6 +147,16 @@ runtime·notewakeup(Note *n) ...@@ -147,6 +147,16 @@ runtime·notewakeup(Note *n)
} }
} }
void
runtime·notewakeup_m(void)
{
Note *n;
n = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
runtime·notewakeup(n);
}
void void
runtime·notesleep(Note *n) runtime·notesleep(Note *n)
{ {
...@@ -264,3 +274,22 @@ runtime·notetsleepg(Note *n, int64 ns) ...@@ -264,3 +274,22 @@ runtime·notetsleepg(Note *n, int64 ns)
runtime·exitsyscall(); runtime·exitsyscall();
return res; return res;
} }
void
runtime·notetsleepg_m(void)
{
Note *n;
int64 ns;
n = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
ns = g->m->scalararg[0] + ((int64)g->m->scalararg[1] << 32);
if(g->m->waitsema == 0)
g->m->waitsema = runtime·semacreate();
runtime·entersyscallblock_m();
notetsleep(n, ns, 0, nil);
// caller will call exitsyscall on g stack
runtime·gogo(&g->m->curg->sched);
}
...@@ -1630,6 +1630,31 @@ void ...@@ -1630,6 +1630,31 @@ void
g->m->locks--; g->m->locks--;
} }
// The same as runtime·entersyscallblock(), but called on g0 stack.
void
runtime·entersyscallblock_m(void)
{
G *gp;
gp = g->m->curg;
// sched.{g,pc,sp,lr} are already set by mcall.
gp->stackguard0 = StackPreempt; // we are on g0, the goroutine must not touch its stack until exitsyscall
gp->sched.ret = 0;
gp->sched.ctxt = 0;
gp->syscallsp = gp->sched.sp;
gp->syscallpc = gp->sched.pc;
gp->syscallstack = gp->stackbase;
gp->syscallguard = gp->stackguard;
gp->status = Gsyscall;
if(gp->syscallsp < gp->syscallguard-StackGuard || gp->syscallstack < gp->syscallsp) {
// runtime·printf("entersyscall inconsistent %p [%p,%p]\n",
// gp->syscallsp, gp->syscallguard-StackGuard, gp->syscallstack);
runtime·throw("entersyscall_m");
}
handoffp(releasep());
}
// The goroutine g exited its system call. // The goroutine g exited its system call.
// Arrange for it to run on a cpu again. // Arrange for it to run on a cpu again.
// This is called only from the go syscall library, not // This is called only from the go syscall library, not
......
...@@ -931,6 +931,7 @@ void runtime·asmcgocall(void (*fn)(void*), void*); ...@@ -931,6 +931,7 @@ void runtime·asmcgocall(void (*fn)(void*), void*);
void runtime·entersyscall(void); void runtime·entersyscall(void);
void runtime·entersyscallblock(void); void runtime·entersyscallblock(void);
void runtime·exitsyscall(void); void runtime·exitsyscall(void);
void runtime·entersyscallblock_m(void);
G* runtime·newproc1(FuncVal*, byte*, int32, int32, void*); G* runtime·newproc1(FuncVal*, byte*, int32, int32, void*);
bool runtime·sigsend(int32 sig); bool runtime·sigsend(int32 sig);
int32 runtime·callers(int32, uintptr*, int32); int32 runtime·callers(int32, uintptr*, int32);
......
...@@ -73,7 +73,9 @@ var ( ...@@ -73,7 +73,9 @@ var (
gosched_m, gosched_m,
ready_m, ready_m,
park_m, park_m,
blockevent_m mFunction blockevent_m,
notewakeup_m,
notetsleepg_m mFunction
) )
// memclr clears n bytes starting at ptr. // memclr clears n bytes starting at ptr.
...@@ -169,3 +171,26 @@ func noescape(p unsafe.Pointer) unsafe.Pointer { ...@@ -169,3 +171,26 @@ func noescape(p unsafe.Pointer) unsafe.Pointer {
func gopersistentalloc(n uintptr) unsafe.Pointer func gopersistentalloc(n uintptr) unsafe.Pointer
func gocputicks() int64 func gocputicks() int64
func gonoteclear(n *note) {
n.key = 0
}
func gonotewakeup(n *note) {
mp := acquirem()
mp.ptrarg[0] = unsafe.Pointer(n)
onM(&notewakeup_m)
releasem(mp)
}
func gonotetsleepg(n *note, t int64) {
mp := acquirem()
mp.ptrarg[0] = unsafe.Pointer(n)
mp.scalararg[0] = uint(uint32(t)) // low 32 bits
mp.scalararg[1] = uint(t >> 32) // high 32 bits
releasem(mp)
mcall(&notetsleepg_m)
exitsyscall()
}
func exitsyscall()
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment