Commit db58ab96 authored by Russ Cox's avatar Russ Cox

runtime: more C to Go conversion adjustments

Mostly NOSPLIT additions.
Had to rewrite atomic_arm.c in Go because it calls lock,
and lock is too complex.

With this CL, I find no Go -> C calls that can split the stack
on any system except Solaris and Windows.

Solaris and Windows need more work and will be done separately.

LGTM=iant, dave
R=golang-codereviews, bradfitz, iant, dave
CC=dvyukov, golang-codereviews, khr, r
https://golang.org/cl/137160043
parent 1a14b5ba
...@@ -411,6 +411,7 @@ func (w *Walker) parseFile(dir, file string) (*ast.File, error) { ...@@ -411,6 +411,7 @@ func (w *Walker) parseFile(dir, file string) (*ast.File, error) {
"); " + "); " +
"const (" + "const (" +
" cb_max = 2000;" + " cb_max = 2000;" +
" _CacheLineSize = 64;" +
" _Gidle = 1;" + " _Gidle = 1;" +
" _Grunnable = 2;" + " _Grunnable = 2;" +
" _Grunning = 3;" + " _Grunning = 3;" +
......
...@@ -114,7 +114,7 @@ TEXT runtime·asminit(SB),NOSPLIT,$0-0 ...@@ -114,7 +114,7 @@ TEXT runtime·asminit(SB),NOSPLIT,$0-0
// void gosave(Gobuf*) // void gosave(Gobuf*)
// save state in Gobuf; setjmp // save state in Gobuf; setjmp
TEXT runtime·gosave(SB), NOSPLIT, $-4-4 TEXT runtime·gosave(SB),NOSPLIT,$-4-4
MOVW 0(FP), R0 // gobuf MOVW 0(FP), R0 // gobuf
MOVW SP, gobuf_sp(R0) MOVW SP, gobuf_sp(R0)
MOVW LR, gobuf_pc(R0) MOVW LR, gobuf_pc(R0)
...@@ -127,7 +127,7 @@ TEXT runtime·gosave(SB), NOSPLIT, $-4-4 ...@@ -127,7 +127,7 @@ TEXT runtime·gosave(SB), NOSPLIT, $-4-4
// void gogo(Gobuf*) // void gogo(Gobuf*)
// restore state from Gobuf; longjmp // restore state from Gobuf; longjmp
TEXT runtime·gogo(SB), NOSPLIT, $-4-4 TEXT runtime·gogo(SB),NOSPLIT,$-4-4
MOVW 0(FP), R1 // gobuf MOVW 0(FP), R1 // gobuf
MOVW gobuf_g(R1), g MOVW gobuf_g(R1), g
MOVW 0(g), R2 // make sure g != nil MOVW 0(g), R2 // make sure g != nil
...@@ -151,7 +151,7 @@ TEXT runtime·gogo(SB), NOSPLIT, $-4-4 ...@@ -151,7 +151,7 @@ TEXT runtime·gogo(SB), NOSPLIT, $-4-4
// Switch to m->g0's stack, call fn(g). // Switch to m->g0's stack, call fn(g).
// Fn must never return. It should gogo(&g->sched) // Fn must never return. It should gogo(&g->sched)
// to keep running g. // to keep running g.
TEXT runtime·mcall(SB), NOSPLIT, $-4-4 TEXT runtime·mcall(SB),NOSPLIT,$-4-4
// Save caller state in g->sched. // Save caller state in g->sched.
MOVW SP, (g_sched+gobuf_sp)(g) MOVW SP, (g_sched+gobuf_sp)(g)
MOVW LR, (g_sched+gobuf_pc)(g) MOVW LR, (g_sched+gobuf_pc)(g)
...@@ -184,13 +184,13 @@ TEXT runtime·mcall(SB), NOSPLIT, $-4-4 ...@@ -184,13 +184,13 @@ TEXT runtime·mcall(SB), NOSPLIT, $-4-4
// lives at the bottom of the G stack from the one that lives // lives at the bottom of the G stack from the one that lives
// at the top of the M stack because the one at the top of // at the top of the M stack because the one at the top of
// the M stack terminates the stack walk (see topofstack()). // the M stack terminates the stack walk (see topofstack()).
TEXT runtime·switchtoM(SB), NOSPLIT, $0-4 TEXT runtime·switchtoM(SB),NOSPLIT,$0-4
MOVW $0, R0 MOVW $0, R0
BL (R0) // clobber lr to ensure push {lr} is kept BL (R0) // clobber lr to ensure push {lr} is kept
RET RET
// func onM(fn func()) // func onM(fn func())
TEXT runtime·onM(SB), NOSPLIT, $0-4 TEXT runtime·onM(SB),NOSPLIT,$0-4
MOVW fn+0(FP), R0 // R0 = fn MOVW fn+0(FP), R0 // R0 = fn
MOVW g_m(g), R1 // R1 = m MOVW g_m(g), R1 // R1 = m
...@@ -300,7 +300,7 @@ TEXT runtime·morestack_noctxt(SB),NOSPLIT,$-4-0 ...@@ -300,7 +300,7 @@ TEXT runtime·morestack_noctxt(SB),NOSPLIT,$-4-0
// with the desired args running the desired function. // with the desired args running the desired function.
// //
// func call(fn *byte, arg *byte, argsize uint32). // func call(fn *byte, arg *byte, argsize uint32).
TEXT runtime·newstackcall(SB), NOSPLIT, $-4-12 TEXT runtime·newstackcall(SB),NOSPLIT,$-4-12
// Save our caller's state as the PC and SP to // Save our caller's state as the PC and SP to
// restore when returning from f. // restore when returning from f.
MOVW g_m(g), R8 MOVW g_m(g), R8
...@@ -348,7 +348,7 @@ TEXT runtime·newstackcall(SB), NOSPLIT, $-4-12 ...@@ -348,7 +348,7 @@ TEXT runtime·newstackcall(SB), NOSPLIT, $-4-12
MOVW $NAME(SB), R1; \ MOVW $NAME(SB), R1; \
B (R1) B (R1)
TEXT reflect·call(SB), NOSPLIT, $-4-16 TEXT reflect·call(SB),NOSPLIT,$-4-16
MOVW argsize+8(FP), R0 MOVW argsize+8(FP), R0
DISPATCH(runtime·call16, 16) DISPATCH(runtime·call16, 16)
DISPATCH(runtime·call32, 32) DISPATCH(runtime·call32, 32)
...@@ -459,7 +459,7 @@ CALLFN(runtime·call1073741824, 1073741824) ...@@ -459,7 +459,7 @@ CALLFN(runtime·call1073741824, 1073741824)
// //
// Lessstack can appear in stack traces for the same reason // Lessstack can appear in stack traces for the same reason
// as morestack; in that context, it has 0 arguments. // as morestack; in that context, it has 0 arguments.
TEXT runtime·lessstack(SB), NOSPLIT, $-4-0 TEXT runtime·lessstack(SB),NOSPLIT,$-4-0
// Save return value in m->cret // Save return value in m->cret
MOVW g_m(g), R8 MOVW g_m(g), R8
MOVW R0, m_cret(R8) MOVW R0, m_cret(R8)
...@@ -478,7 +478,7 @@ TEXT runtime·lessstack(SB), NOSPLIT, $-4-0 ...@@ -478,7 +478,7 @@ TEXT runtime·lessstack(SB), NOSPLIT, $-4-0
// to load all registers simultaneously, so that a profiling // to load all registers simultaneously, so that a profiling
// interrupt can never see mismatched SP/LR/PC. // interrupt can never see mismatched SP/LR/PC.
// (And double-check that pop is atomic in that way.) // (And double-check that pop is atomic in that way.)
TEXT runtime·jmpdefer(SB), NOSPLIT, $0-8 TEXT runtime·jmpdefer(SB),NOSPLIT,$0-8
MOVW 0(SP), LR MOVW 0(SP), LR
MOVW $-4(LR), LR // BL deferreturn MOVW $-4(LR), LR // BL deferreturn
MOVW fv+0(FP), R7 MOVW fv+0(FP), R7
...@@ -638,7 +638,7 @@ havem: ...@@ -638,7 +638,7 @@ havem:
RET RET
// void setg(G*); set g. for use by needm. // void setg(G*); set g. for use by needm.
TEXT runtime·setg(SB), NOSPLIT, $0-4 TEXT runtime·setg(SB),NOSPLIT,$0-4
MOVW gg+0(FP), g MOVW gg+0(FP), g
// Save g to thread-local storage. // Save g to thread-local storage.
...@@ -715,13 +715,13 @@ casfail: ...@@ -715,13 +715,13 @@ casfail:
MOVB R0, ret+12(FP) MOVB R0, ret+12(FP)
RET RET
TEXT runtime·casuintptr(SB), NOSPLIT, $0-13 TEXT runtime·casuintptr(SB),NOSPLIT,$0-13
B runtime·cas(SB) B runtime·cas(SB)
TEXT runtime·atomicloaduintptr(SB), NOSPLIT, $0-8 TEXT runtime·atomicloaduintptr(SB),NOSPLIT,$0-8
B runtime·atomicload(SB) B runtime·atomicload(SB)
TEXT runtime·atomicloaduint(SB), NOSPLIT, $0-8 TEXT runtime·atomicloaduint(SB),NOSPLIT,$0-8
B runtime·atomicload(SB) B runtime·atomicload(SB)
TEXT runtime·stackguard(SB),NOSPLIT,$0-8 TEXT runtime·stackguard(SB),NOSPLIT,$0-8
...@@ -874,7 +874,7 @@ _sib_notfound: ...@@ -874,7 +874,7 @@ _sib_notfound:
MOVW R0, ret+12(FP) MOVW R0, ret+12(FP)
RET RET
TEXT runtime·timenow(SB), NOSPLIT, $0-0 TEXT runtime·timenow(SB),NOSPLIT,$0-0
B time·now(SB) B time·now(SB)
// A Duff's device for zeroing memory. // A Duff's device for zeroing memory.
...@@ -885,7 +885,7 @@ TEXT runtime·timenow(SB), NOSPLIT, $0-0 ...@@ -885,7 +885,7 @@ TEXT runtime·timenow(SB), NOSPLIT, $0-0
// R0: zero // R0: zero
// R1: ptr to memory to be zeroed // R1: ptr to memory to be zeroed
// R1 is updated as a side effect. // R1 is updated as a side effect.
TEXT runtime·duffzero(SB), NOSPLIT, $0-0 TEXT runtime·duffzero(SB),NOSPLIT,$0-0
MOVW.P R0, 4(R1) MOVW.P R0, 4(R1)
MOVW.P R0, 4(R1) MOVW.P R0, 4(R1)
MOVW.P R0, 4(R1) MOVW.P R0, 4(R1)
...@@ -1026,7 +1026,7 @@ TEXT runtime·duffzero(SB), NOSPLIT, $0-0 ...@@ -1026,7 +1026,7 @@ TEXT runtime·duffzero(SB), NOSPLIT, $0-0
// R1: ptr to source memory // R1: ptr to source memory
// R2: ptr to destination memory // R2: ptr to destination memory
// R1 and R2 are updated as a side effect // R1 and R2 are updated as a side effect
TEXT runtime·duffcopy(SB), NOSPLIT, $0-0 TEXT runtime·duffcopy(SB),NOSPLIT,$0-0
MOVW.P 4(R1), R0 MOVW.P 4(R1), R0
MOVW.P R0, 4(R2) MOVW.P R0, 4(R2)
MOVW.P 4(R1), R0 MOVW.P 4(R1), R0
...@@ -1285,7 +1285,7 @@ TEXT runtime·duffcopy(SB), NOSPLIT, $0-0 ...@@ -1285,7 +1285,7 @@ TEXT runtime·duffcopy(SB), NOSPLIT, $0-0
MOVW.P R0, 4(R2) MOVW.P R0, 4(R2)
RET RET
TEXT runtime·fastrand1(SB), NOSPLIT, $-4-4 TEXT runtime·fastrand1(SB),NOSPLIT,$-4-4
MOVW g_m(g), R1 MOVW g_m(g), R1
MOVW m_fastrand(R1), R0 MOVW m_fastrand(R1), R0
ADD.S R0, R0 ADD.S R0, R0
...@@ -1294,9 +1294,19 @@ TEXT runtime·fastrand1(SB), NOSPLIT, $-4-4 ...@@ -1294,9 +1294,19 @@ TEXT runtime·fastrand1(SB), NOSPLIT, $-4-4
MOVW R0, ret+0(FP) MOVW R0, ret+0(FP)
RET RET
TEXT runtime·gocputicks(SB), NOSPLIT, $0 TEXT runtime·gocputicks(SB),NOSPLIT,$0
B runtime·cputicks(SB) B runtime·cputicks(SB)
TEXT runtime·return0(SB), NOSPLIT, $0 TEXT runtime·return0(SB),NOSPLIT,$0
MOVW $0, R0 MOVW $0, R0
RET RET
TEXT runtime·procyield(SB),NOSPLIT,$-4
MOVW cycles+0(FP), R1
MOVW $0, R0
yieldloop:
CMP R0, R1
B.NE 2(PC)
RET
SUB $1, R1
B yieldloop
// Copyright 2014 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// +build !arm
package runtime
import "unsafe"
//go:noescape
func xadd(ptr *uint32, delta int32) uint32
//go:noescape
func xadd64(ptr *uint64, delta int64) uint64
//go:noescape
func xchg(ptr *uint32, new uint32) uint32
//go:noescape
func xchg64(ptr *uint64, new uint64) uint64
//go:noescape
func xchgp(ptr unsafe.Pointer, new unsafe.Pointer) unsafe.Pointer
//go:noescape
func xchguintptr(ptr *uintptr, new uintptr) uintptr
//go:noescape
func atomicload(ptr *uint32) uint32
//go:noescape
func atomicload64(ptr *uint64) uint64
//go:noescape
func atomicloadp(ptr unsafe.Pointer) unsafe.Pointer
//go:noescape
func atomicor8(ptr *uint8, val uint8)
//go:noescape
func cas64(ptr *uint64, old, new uint64) bool
//go:noescape
func atomicstore(ptr *uint32, val uint32)
//go:noescape
func atomicstore64(ptr *uint64, val uint64)
//go:noescape
func atomicstorep(ptr unsafe.Pointer, val unsafe.Pointer)
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
#include "runtime.h"
#include "arch_GOARCH.h"
#include "../../cmd/ld/textflag.h"
static struct {
Mutex l;
byte pad[CacheLineSize-sizeof(Mutex)];
} locktab[57];
#define LOCK(addr) (&locktab[((uintptr)(addr)>>3)%nelem(locktab)].l)
// Atomic add and return new value.
#pragma textflag NOSPLIT
uint32
runtime·xadd(uint32 volatile *val, int32 delta)
{
uint32 oval, nval;
for(;;){
oval = *val;
nval = oval + delta;
if(runtime·cas(val, oval, nval))
return nval;
}
}
#pragma textflag NOSPLIT
uint32
runtime·xchg(uint32 volatile* addr, uint32 v)
{
uint32 old;
for(;;) {
old = *addr;
if(runtime·cas(addr, old, v))
return old;
}
}
#pragma textflag NOSPLIT
void*
runtime·xchgp(void* volatile* addr, void* v)
{
void *old;
for(;;) {
old = *addr;
if(runtime·casp(addr, old, v))
return old;
}
}
#pragma textflag NOSPLIT
void*
runtime·xchguintptr(void* volatile* addr, void* v)
{
return (void*)runtime·xchg((uint32*)addr, (uint32)v);
}
#pragma textflag NOSPLIT
void
runtime·procyield(uint32 cnt)
{
uint32 volatile i;
for(i = 0; i < cnt; i++) {
}
}
#pragma textflag NOSPLIT
uint32
runtime·atomicload(uint32 volatile* addr)
{
return runtime·xadd(addr, 0);
}
#pragma textflag NOSPLIT
void*
runtime·atomicloadp(void* volatile* addr)
{
return (void*)runtime·xadd((uint32 volatile*)addr, 0);
}
#pragma textflag NOSPLIT
void
runtime·atomicstorep(void* volatile* addr, void* v)
{
void *old;
for(;;) {
old = *addr;
if(runtime·casp(addr, old, v))
return;
}
}
#pragma textflag NOSPLIT
void
runtime·atomicstore(uint32 volatile* addr, uint32 v)
{
uint32 old;
for(;;) {
old = *addr;
if(runtime·cas(addr, old, v))
return;
}
}
#pragma textflag NOSPLIT
bool
runtime·cas64(uint64 volatile *addr, uint64 old, uint64 new)
{
bool res;
runtime·lock(LOCK(addr));
if(*addr == old) {
*addr = new;
res = true;
} else {
res = false;
}
runtime·unlock(LOCK(addr));
return res;
}
#pragma textflag NOSPLIT
uint64
runtime·xadd64(uint64 volatile *addr, int64 delta)
{
uint64 res;
runtime·lock(LOCK(addr));
res = *addr + delta;
*addr = res;
runtime·unlock(LOCK(addr));
return res;
}
#pragma textflag NOSPLIT
uint64
runtime·xchg64(uint64 volatile *addr, uint64 v)
{
uint64 res;
runtime·lock(LOCK(addr));
res = *addr;
*addr = v;
runtime·unlock(LOCK(addr));
return res;
}
#pragma textflag NOSPLIT
uint64
runtime·atomicload64(uint64 volatile *addr)
{
uint64 res;
runtime·lock(LOCK(addr));
res = *addr;
runtime·unlock(LOCK(addr));
return res;
}
#pragma textflag NOSPLIT
void
runtime·atomicstore64(uint64 volatile *addr, uint64 v)
{
runtime·lock(LOCK(addr));
*addr = v;
runtime·unlock(LOCK(addr));
}
#pragma textflag NOSPLIT
void
runtime·atomicor8(byte volatile *addr, byte v)
{
uint32 *addr32, old, word;
// Align down to 4 bytes and use 32-bit CAS.
addr32 = (uint32*)((uintptr)addr & ~3);
word = ((uint32)v) << (((uintptr)addr & 3) * 8);
for(;;) {
old = *addr32;
if(runtime·cas(addr32, old, old|word))
break;
}
}
// Copyright 2009 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package runtime
import "unsafe"
var locktab [57]struct {
l mutex
pad [_CacheLineSize - unsafe.Sizeof(mutex{})]byte
}
func addrLock(addr *uint64) *mutex {
return &locktab[(uintptr(unsafe.Pointer(addr))>>3)%uintptr(len(locktab))].l
}
// Atomic add and return new value.
//go:nosplit
func xadd(val *uint32, delta int32) uint32 {
for {
oval := *val
nval := oval + uint32(delta)
if cas(val, oval, nval) {
return nval
}
}
}
//go:nosplit
func xchg(addr *uint32, v uint32) uint32 {
for {
old := *addr
if cas(addr, old, v) {
return old
}
}
}
//go:nosplit
func xchgp(addr *unsafe.Pointer, v unsafe.Pointer) unsafe.Pointer {
for {
old := *addr
if casp(addr, old, v) {
return old
}
}
}
//go:nosplit
func xchguintptr(addr *uintptr, v uintptr) uintptr {
return uintptr(xchg((*uint32)(unsafe.Pointer(addr)), uint32(v)))
}
//go:nosplit
func atomicload(addr *uint32) uint32 {
return xadd(addr, 0)
}
//go:nosplit
func atomicloadp(addr unsafe.Pointer) unsafe.Pointer {
return unsafe.Pointer(uintptr(xadd((*uint32)(addr), 0)))
}
//go:nosplit
func atomicstorep(addr unsafe.Pointer, v unsafe.Pointer) {
for {
old := *(*unsafe.Pointer)(addr)
if casp((*unsafe.Pointer)(addr), old, v) {
return
}
}
}
//go:nosplit
func atomicstore(addr *uint32, v uint32) {
for {
old := *addr
if cas(addr, old, v) {
return
}
}
}
//go:nosplit
func cas64(addr *uint64, old, new uint64) bool {
var ok bool
onM(func() {
lock(addrLock(addr))
if *addr == old {
*addr = new
ok = true
}
unlock(addrLock(addr))
})
return ok
}
//go:nosplit
func xadd64(addr *uint64, delta int64) uint64 {
var r uint64
onM(func() {
lock(addrLock(addr))
r = *addr + uint64(delta)
*addr = r
unlock(addrLock(addr))
})
return r
}
//go:nosplit
func xchg64(addr *uint64, v uint64) uint64 {
var r uint64
onM(func() {
lock(addrLock(addr))
r = *addr
*addr = v
unlock(addrLock(addr))
})
return r
}
//go:nosplit
func atomicload64(addr *uint64) uint64 {
var r uint64
onM(func() {
lock(addrLock(addr))
r = *addr
unlock(addrLock(addr))
})
return r
}
//go:nosplit
func atomicstore64(addr *uint64, v uint64) {
onM(func() {
lock(addrLock(addr))
*addr = v
unlock(addrLock(addr))
})
}
//go:nosplit
func atomicor8(addr *uint8, v uint8) {
// Align down to 4 bytes and use 32-bit CAS.
uaddr := uintptr(unsafe.Pointer(addr))
addr32 := (*uint32)(unsafe.Pointer(uaddr &^ 3))
word := uint32(v) << ((uaddr & 3) * 8) // little endian
for {
old := *addr32
if cas(addr32, old, old|word) {
return
}
}
}
...@@ -87,6 +87,7 @@ runtime·mlookup(void *v, byte **base, uintptr *size, MSpan **sp) ...@@ -87,6 +87,7 @@ runtime·mlookup(void *v, byte **base, uintptr *size, MSpan **sp)
return 1; return 1;
} }
#pragma textflag NOSPLIT
void void
runtime·purgecachedstats(MCache *c) runtime·purgecachedstats(MCache *c)
{ {
......
...@@ -40,13 +40,38 @@ getncpu(void) ...@@ -40,13 +40,38 @@ getncpu(void)
return 1; return 1;
} }
static void futexsleep(void);
#pragma textflag NOSPLIT #pragma textflag NOSPLIT
void void
runtime·futexsleep(uint32 *addr, uint32 val, int64 ns) runtime·futexsleep(uint32 *addr, uint32 val, int64 ns)
{ {
void (*fn)(void);
g->m->ptrarg[0] = addr;
g->m->scalararg[0] = val;
g->m->ptrarg[1] = &ns;
fn = futexsleep;
runtime·onM(&fn);
}
static void
futexsleep(void)
{
uint32 *addr;
uint32 val;
int64 ns;
int32 timeout = 0; int32 timeout = 0;
int32 ret; int32 ret;
addr = g->m->ptrarg[0];
val = g->m->scalararg[0];
ns = *(int64*)g->m->ptrarg[1];
g->m->ptrarg[0] = nil;
g->m->scalararg[0] = 0;
g->m->ptrarg[1] = nil;
if(ns >= 0) { if(ns >= 0) {
// The timeout is specified in microseconds - ensure that we // The timeout is specified in microseconds - ensure that we
// do not end up dividing to zero, which would put us to sleep // do not end up dividing to zero, which would put us to sleep
......
...@@ -42,13 +42,38 @@ getncpu(void) ...@@ -42,13 +42,38 @@ getncpu(void)
// FreeBSD's umtx_op syscall is effectively the same as Linux's futex, and // FreeBSD's umtx_op syscall is effectively the same as Linux's futex, and
// thus the code is largely similar. See linux/thread.c and lock_futex.c for comments. // thus the code is largely similar. See linux/thread.c and lock_futex.c for comments.
static void futexsleep(void);
#pragma textflag NOSPLIT #pragma textflag NOSPLIT
void void
runtime·futexsleep(uint32 *addr, uint32 val, int64 ns) runtime·futexsleep(uint32 *addr, uint32 val, int64 ns)
{ {
void (*fn)(void);
g->m->ptrarg[0] = addr;
g->m->scalararg[0] = val;
g->m->ptrarg[1] = &ns;
fn = futexsleep;
runtime·onM(&fn);
}
static void
futexsleep(void)
{
uint32 *addr;
uint32 val;
int64 ns;
int32 ret; int32 ret;
Timespec ts; Timespec ts;
addr = g->m->ptrarg[0];
val = g->m->scalararg[0];
ns = *(int64*)g->m->ptrarg[1];
g->m->ptrarg[0] = nil;
g->m->scalararg[0] = 0;
g->m->ptrarg[1] = nil;
if(ns < 0) { if(ns < 0) {
ret = runtime·sys_umtx_op(addr, UMTX_OP_WAIT_UINT_PRIVATE, val, nil, nil); ret = runtime·sys_umtx_op(addr, UMTX_OP_WAIT_UINT_PRIVATE, val, nil, nil);
if(ret >= 0 || ret == -EINTR) if(ret >= 0 || ret == -EINTR)
......
...@@ -57,18 +57,23 @@ getncpu(void) ...@@ -57,18 +57,23 @@ getncpu(void)
return 1; return 1;
} }
#pragma textflag NOSPLIT
uintptr uintptr
runtime·semacreate(void) runtime·semacreate(void)
{ {
return 1; return 1;
} }
#pragma textflag NOSPLIT static void
int32 semasleep(void)
runtime·semasleep(int64 ns)
{ {
int64 ns;
Timespec ts; Timespec ts;
ns = (int64)(uint32)g->m->scalararg[0] | (int64)(uint32)g->m->scalararg[1]<<32;
g->m->scalararg[0] = 0;
g->m->scalararg[1] = 0;
// spin-mutex lock // spin-mutex lock
while(runtime·xchg(&g->m->waitsemalock, 1)) while(runtime·xchg(&g->m->waitsemalock, 1))
runtime·osyield(); runtime·osyield();
...@@ -115,7 +120,8 @@ runtime·semasleep(int64 ns) ...@@ -115,7 +120,8 @@ runtime·semasleep(int64 ns)
g->m->waitsemacount--; g->m->waitsemacount--;
// spin-mutex unlock // spin-mutex unlock
runtime·atomicstore(&g->m->waitsemalock, 0); runtime·atomicstore(&g->m->waitsemalock, 0);
return 0; // semaphore acquired g->m->scalararg[0] = 0; // semaphore acquired
return;
} }
// semaphore not available. // semaphore not available.
...@@ -128,13 +134,36 @@ runtime·semasleep(int64 ns) ...@@ -128,13 +134,36 @@ runtime·semasleep(int64 ns)
// lock held but giving up // lock held but giving up
// spin-mutex unlock // spin-mutex unlock
runtime·atomicstore(&g->m->waitsemalock, 0); runtime·atomicstore(&g->m->waitsemalock, 0);
return -1; g->m->scalararg[0] = -1;
return;
}
#pragma textflag NOSPLIT
int32
runtime·semasleep(int64 ns)
{
int32 r;
void (*fn)(void);
g->m->scalararg[0] = (uint32)ns;
g->m->scalararg[1] = (uint32)(ns>>32);
fn = semasleep;
runtime·onM(&fn);
r = g->m->scalararg[0];
g->m->scalararg[0] = 0;
return r;
} }
static void badsemawakeup(void);
#pragma textflag NOSPLIT
void void
runtime·semawakeup(M *mp) runtime·semawakeup(M *mp)
{ {
uint32 ret; uint32 ret;
void (*fn)(void);
void *oldptr;
uintptr oldscalar;
// spin-mutex lock // spin-mutex lock
while(runtime·xchg(&mp->waitsemalock, 1)) while(runtime·xchg(&mp->waitsemalock, 1))
...@@ -143,12 +172,39 @@ runtime·semawakeup(M *mp) ...@@ -143,12 +172,39 @@ runtime·semawakeup(M *mp)
// TODO(jsing) - potential deadlock, see semasleep() for details. // TODO(jsing) - potential deadlock, see semasleep() for details.
// Confirm that LWP is parked before unparking... // Confirm that LWP is parked before unparking...
ret = runtime·lwp_unpark(mp->procid, &mp->waitsemacount); 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); // semawakeup can be called on signal stack.
// Save old ptrarg/scalararg so we can restore them.
oldptr = g->m->ptrarg[0];
oldscalar = g->m->scalararg[0];
g->m->ptrarg[0] = mp;
g->m->scalararg[0] = ret;
fn = badsemawakeup;
if(g == g->m->gsignal)
fn();
else
runtime·onM(&fn);
g->m->ptrarg[0] = oldptr;
g->m->scalararg[0] = oldscalar;
}
// spin-mutex unlock // spin-mutex unlock
runtime·atomicstore(&mp->waitsemalock, 0); runtime·atomicstore(&mp->waitsemalock, 0);
} }
static void
badsemawakeup(void)
{
M *mp;
int32 ret;
mp = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
ret = g->m->scalararg[0];
g->m->scalararg[0] = 0;
runtime·printf("thrwakeup addr=%p sem=%d ret=%d\n", &mp->waitsemacount, mp->waitsemacount, ret);
}
void void
runtime·newosproc(M *mp, void *stk) runtime·newosproc(M *mp, void *stk)
{ {
......
...@@ -54,6 +54,7 @@ getncpu(void) ...@@ -54,6 +54,7 @@ getncpu(void)
return 1; return 1;
} }
#pragma textflag NOSPLIT
uintptr uintptr
runtime·semacreate(void) runtime·semacreate(void)
{ {
...@@ -111,22 +112,55 @@ runtime·semasleep(int64 ns) ...@@ -111,22 +112,55 @@ runtime·semasleep(int64 ns)
return -1; return -1;
} }
static void badsemawakeup(void);
#pragma textflag NOSPLIT
void void
runtime·semawakeup(M *mp) runtime·semawakeup(M *mp)
{ {
uint32 ret; uint32 ret;
void *oldptr;
uint32 oldscalar;
void (*fn)(void);
// spin-mutex lock // spin-mutex lock
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); ret = runtime·thrwakeup(&mp->waitsemacount, 1);
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); // semawakeup can be called on signal stack.
// Save old ptrarg/scalararg so we can restore them.
oldptr = g->m->ptrarg[0];
oldscalar = g->m->scalararg[0];
g->m->ptrarg[0] = mp;
g->m->scalararg[0] = ret;
fn = badsemawakeup;
if(g == g->m->gsignal)
fn();
else
runtime·onM(&fn);
g->m->ptrarg[0] = oldptr;
g->m->scalararg[0] = oldscalar;
}
// spin-mutex unlock // spin-mutex unlock
runtime·atomicstore(&mp->waitsemalock, 0); runtime·atomicstore(&mp->waitsemalock, 0);
} }
static void
badsemawakeup(void)
{
M *mp;
int32 ret;
mp = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
ret = g->m->scalararg[0];
g->m->scalararg[0] = 0;
runtime·printf("thrwakeup addr=%p sem=%d ret=%d\n", &mp->waitsemacount, mp->waitsemacount, ret);
}
void void
runtime·newosproc(M *mp, void *stk) runtime·newosproc(M *mp, void *stk)
{ {
......
...@@ -160,6 +160,7 @@ runtime·nanotime(void) ...@@ -160,6 +160,7 @@ runtime·nanotime(void)
return ns; return ns;
} }
#pragma textflag NOSPLIT
void void
time·now(int64 sec, int32 nsec) time·now(int64 sec, int32 nsec)
{ {
...@@ -172,6 +173,7 @@ time·now(int64 sec, int32 nsec) ...@@ -172,6 +173,7 @@ time·now(int64 sec, int32 nsec)
FLUSH(&nsec); FLUSH(&nsec);
} }
#pragma textflag NOSPLIT
void void
runtime·itoa(int32 n, byte *p, uint32 len) runtime·itoa(int32 n, byte *p, uint32 len)
{ {
...@@ -252,12 +254,29 @@ runtime·postnote(int32 pid, int8* msg) ...@@ -252,12 +254,29 @@ runtime·postnote(int32 pid, int8* msg)
return 0; return 0;
} }
static void exit(void);
#pragma textflag NOSPLIT
void void
runtime·exit(int32 e) runtime·exit(int32 e)
{ {
void (*fn)(void);
g->m->scalararg[0] = e;
fn = exit;
runtime·onM(&fn);
}
static void
exit(void)
{
int32 e;
byte tmp[16]; byte tmp[16];
int8 *status; int8 *status;
e = g->m->scalararg[0];
g->m->scalararg[0] = 0;
if(e == 0) if(e == 0)
status = ""; status = "";
else { else {
...@@ -283,6 +302,7 @@ runtime·newosproc(M *mp, void *stk) ...@@ -283,6 +302,7 @@ runtime·newosproc(M *mp, void *stk)
runtime·throw("newosproc: rfork failed"); runtime·throw("newosproc: rfork failed");
} }
#pragma textflag NOSPLIT
uintptr uintptr
runtime·semacreate(void) runtime·semacreate(void)
{ {
...@@ -312,6 +332,7 @@ runtime·semasleep(int64 ns) ...@@ -312,6 +332,7 @@ runtime·semasleep(int64 ns)
return 0; // success return 0; // success
} }
#pragma textflag NOSPLIT
void void
runtime·semawakeup(M *mp) runtime·semawakeup(M *mp)
{ {
...@@ -396,12 +417,14 @@ runtime·sigpanic(void) ...@@ -396,12 +417,14 @@ runtime·sigpanic(void)
} }
} }
#pragma textflag NOSPLIT
int32 int32
runtime·read(int32 fd, void *buf, int32 nbytes) runtime·read(int32 fd, void *buf, int32 nbytes)
{ {
return runtime·pread(fd, buf, nbytes, -1LL); return runtime·pread(fd, buf, nbytes, -1LL);
} }
#pragma textflag NOSPLIT
int32 int32
runtime·write(uintptr fd, void *buf, int32 nbytes) runtime·write(uintptr fd, void *buf, int32 nbytes)
{ {
......
...@@ -2623,20 +2623,6 @@ runtime·mcount(void) ...@@ -2623,20 +2623,6 @@ runtime·mcount(void)
return runtime·sched.mcount; return runtime·sched.mcount;
} }
void
runtime·badmcall(void (*fn)(G*)) // called from assembly
{
USED(fn); // TODO: print fn?
runtime·throw("runtime: mcall called on m->g0 stack");
}
void
runtime·badmcall2(void (*fn)(G*)) // called from assembly
{
USED(fn);
runtime·throw("runtime: mcall function returned");
}
void void
runtime·badreflectcall(void) // called from assembly runtime·badreflectcall(void) // called from assembly
{ {
......
...@@ -91,3 +91,12 @@ func releaseSudog(s *sudog) { ...@@ -91,3 +91,12 @@ func releaseSudog(s *sudog) {
func funcPC(f interface{}) uintptr { func funcPC(f interface{}) uintptr {
return **(**uintptr)(add(unsafe.Pointer(&f), ptrSize)) return **(**uintptr)(add(unsafe.Pointer(&f), ptrSize))
} }
// called from assembly
func badmcall(fn func(*g)) {
gothrow("runtime: mcall called on m->g0 stack")
}
func badmcall2(fn func(*g)) {
gothrow("runtime: mcall function returned")
}
...@@ -606,20 +606,43 @@ struct Sfregs ...@@ -606,20 +606,43 @@ struct Sfregs
uint32 cspr; uint32 cspr;
}; };
static void sfloat2(void);
#pragma textflag NOSPLIT #pragma textflag NOSPLIT
uint32* uint32*
runtime·_sfloat2(uint32 *lr, Sfregs regs) runtime·_sfloat2(uint32 *lr, Sfregs regs)
{ {
void (*fn)(void);
g->m->ptrarg[0] = lr;
g->m->ptrarg[1] = &regs;
fn = sfloat2;
runtime·onM(&fn);
lr = g->m->ptrarg[0];
g->m->ptrarg[0] = nil;
return lr;
}
static void
sfloat2(void)
{
uint32 *lr;
Sfregs *regs;
uint32 skip; uint32 skip;
skip = stepflt(lr, (uint32*)&regs.r0); lr = g->m->ptrarg[0];
regs = g->m->ptrarg[1];
g->m->ptrarg[0] = nil;
g->m->ptrarg[1] = nil;
skip = stepflt(lr, (uint32*)&regs->r0);
if(skip == 0) { if(skip == 0) {
runtime·printf("sfloat2 %p %x\n", lr, *lr); runtime·printf("sfloat2 %p %x\n", lr, *lr);
fabort(); // not ok to fail first instruction fabort(); // not ok to fail first instruction
} }
lr += skip; lr += skip;
while(skip = stepflt(lr, (uint32*)&regs.r0)) while(skip = stepflt(lr, (uint32*)&regs->r0))
lr += skip; lr += skip;
return lr; g->m->ptrarg[0] = lr;
} }
...@@ -174,8 +174,8 @@ func munmap(addr unsafe.Pointer, n uintptr) ...@@ -174,8 +174,8 @@ func munmap(addr unsafe.Pointer, n uintptr)
func madvise(addr unsafe.Pointer, n uintptr, flags int32) func madvise(addr unsafe.Pointer, n uintptr, flags int32)
func newstackcall(fv *funcval, addr unsafe.Pointer, size uint32) func newstackcall(fv *funcval, addr unsafe.Pointer, size uint32)
func reflectcall(fn, arg unsafe.Pointer, n uint32, retoffset uint32) func reflectcall(fn, arg unsafe.Pointer, n uint32, retoffset uint32)
func procyield(cycles uint32)
func osyield() func osyield()
func procyield(cycles uint32)
func cgocallback_gofunc(fv *funcval, frame unsafe.Pointer, framesize uintptr) func cgocallback_gofunc(fv *funcval, frame unsafe.Pointer, framesize uintptr)
func readgogc() int32 func readgogc() int32
func purgecachedstats(c *mcache) func purgecachedstats(c *mcache)
...@@ -188,63 +188,21 @@ func write(fd uintptr, p unsafe.Pointer, n int32) int32 ...@@ -188,63 +188,21 @@ func write(fd uintptr, p unsafe.Pointer, n int32) int32
//go:noescape //go:noescape
func cas(ptr *uint32, old, new uint32) bool func cas(ptr *uint32, old, new uint32) bool
//go:noescape
func cas64(ptr *uint64, old, new uint64) bool
//go:noescape //go:noescape
func casp(ptr *unsafe.Pointer, old, new unsafe.Pointer) bool func casp(ptr *unsafe.Pointer, old, new unsafe.Pointer) bool
//go:noescape //go:noescape
func casuintptr(ptr *uintptr, old, new uintptr) bool func casuintptr(ptr *uintptr, old, new uintptr) bool
//go:noescape
func xadd(ptr *uint32, delta int32) uint32
//go:noescape
func xadd64(ptr *uint64, delta int64) uint64
//go:noescape
func xchg(ptr *uint32, new uint32) uint32
//go:noescape
func xchg64(ptr *uint64, new uint64) uint64
//go:noescape
func xchgp(ptr unsafe.Pointer, new unsafe.Pointer) unsafe.Pointer
//go:noescape
func xchguintptr(ptr *uintptr, new uintptr) uintptr
//go:noescape
func atomicstore(ptr *uint32, val uint32)
//go:noescape
func atomicstore64(ptr *uint64, val uint64)
//go:noescape
func atomicstorep(ptr unsafe.Pointer, val unsafe.Pointer)
//go:noescape //go:noescape
func atomicstoreuintptr(ptr *uintptr, new uintptr) func atomicstoreuintptr(ptr *uintptr, new uintptr)
//go:noescape
func atomicload(ptr *uint32) uint32
//go:noescape
func atomicload64(ptr *uint64) uint64
//go:noescape
func atomicloadp(ptr unsafe.Pointer) unsafe.Pointer
//go:noescape //go:noescape
func atomicloaduintptr(ptr *uintptr) uintptr func atomicloaduintptr(ptr *uintptr) uintptr
//go:noescape //go:noescape
func atomicloaduint(ptr *uint) uint func atomicloaduint(ptr *uint) uint
//go:noescape
func atomicor8(ptr *uint8, val uint8)
//go:noescape //go:noescape
func setcallerpc(argp unsafe.Pointer, pc uintptr) func setcallerpc(argp unsafe.Pointer, pc uintptr)
......
...@@ -413,6 +413,7 @@ _modv(Vlong n, Vlong d) ...@@ -413,6 +413,7 @@ _modv(Vlong n, Vlong d)
return r; return r;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_rshav(Vlong a, int b) _rshav(Vlong a, int b)
{ {
...@@ -440,6 +441,7 @@ _rshav(Vlong a, int b) ...@@ -440,6 +441,7 @@ _rshav(Vlong a, int b)
return r; return r;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_rshlv(Vlong a, int b) _rshlv(Vlong a, int b)
{ {
...@@ -487,6 +489,7 @@ _lshv(Vlong a, int b) ...@@ -487,6 +489,7 @@ _lshv(Vlong a, int b)
return (Vlong){t<<b, (t >> (32-b)) | (a.hi << b)}; return (Vlong){t<<b, (t >> (32-b)) | (a.hi << b)};
} }
#pragma textflag NOSPLIT
Vlong Vlong
_andv(Vlong a, Vlong b) _andv(Vlong a, Vlong b)
{ {
...@@ -497,6 +500,7 @@ _andv(Vlong a, Vlong b) ...@@ -497,6 +500,7 @@ _andv(Vlong a, Vlong b)
return r; return r;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_orv(Vlong a, Vlong b) _orv(Vlong a, Vlong b)
{ {
...@@ -507,6 +511,7 @@ _orv(Vlong a, Vlong b) ...@@ -507,6 +511,7 @@ _orv(Vlong a, Vlong b)
return r; return r;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_xorv(Vlong a, Vlong b) _xorv(Vlong a, Vlong b)
{ {
...@@ -529,6 +534,7 @@ _vpp(Vlong *r) ...@@ -529,6 +534,7 @@ _vpp(Vlong *r)
return l; return l;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_vmm(Vlong *r) _vmm(Vlong *r)
{ {
...@@ -541,6 +547,7 @@ _vmm(Vlong *r) ...@@ -541,6 +547,7 @@ _vmm(Vlong *r)
return l; return l;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_ppv(Vlong *r) _ppv(Vlong *r)
{ {
...@@ -551,6 +558,7 @@ _ppv(Vlong *r) ...@@ -551,6 +558,7 @@ _ppv(Vlong *r)
return *r; return *r;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_mmv(Vlong *r) _mmv(Vlong *r)
{ {
...@@ -642,6 +650,7 @@ _vasop(void *lv, Vlong fn(Vlong, Vlong), int type, Vlong rv) ...@@ -642,6 +650,7 @@ _vasop(void *lv, Vlong fn(Vlong, Vlong), int type, Vlong rv)
return u; return u;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_p2v(void *p) _p2v(void *p)
{ {
...@@ -654,6 +663,7 @@ _p2v(void *p) ...@@ -654,6 +663,7 @@ _p2v(void *p)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_sl2v(long sl) _sl2v(long sl)
{ {
...@@ -666,6 +676,7 @@ _sl2v(long sl) ...@@ -666,6 +676,7 @@ _sl2v(long sl)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_ul2v(ulong ul) _ul2v(ulong ul)
{ {
...@@ -685,6 +696,7 @@ _si2v(int si) ...@@ -685,6 +696,7 @@ _si2v(int si)
return (Vlong){si, si>>31}; return (Vlong){si, si>>31};
} }
#pragma textflag NOSPLIT
Vlong Vlong
_ui2v(uint ui) _ui2v(uint ui)
{ {
...@@ -697,6 +709,7 @@ _ui2v(uint ui) ...@@ -697,6 +709,7 @@ _ui2v(uint ui)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_sh2v(long sh) _sh2v(long sh)
{ {
...@@ -709,6 +722,7 @@ _sh2v(long sh) ...@@ -709,6 +722,7 @@ _sh2v(long sh)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_uh2v(ulong ul) _uh2v(ulong ul)
{ {
...@@ -721,6 +735,7 @@ _uh2v(ulong ul) ...@@ -721,6 +735,7 @@ _uh2v(ulong ul)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_sc2v(long uc) _sc2v(long uc)
{ {
...@@ -733,6 +748,7 @@ _sc2v(long uc) ...@@ -733,6 +748,7 @@ _sc2v(long uc)
return ret; return ret;
} }
#pragma textflag NOSPLIT
Vlong Vlong
_uc2v(ulong ul) _uc2v(ulong ul)
{ {
...@@ -745,6 +761,7 @@ _uc2v(ulong ul) ...@@ -745,6 +761,7 @@ _uc2v(ulong ul)
return ret; return ret;
} }
#pragma textflag NOSPLIT
long long
_v2sc(Vlong rv) _v2sc(Vlong rv)
{ {
...@@ -754,6 +771,7 @@ _v2sc(Vlong rv) ...@@ -754,6 +771,7 @@ _v2sc(Vlong rv)
return (t << 24) >> 24; return (t << 24) >> 24;
} }
#pragma textflag NOSPLIT
long long
_v2uc(Vlong rv) _v2uc(Vlong rv)
{ {
...@@ -761,6 +779,7 @@ _v2uc(Vlong rv) ...@@ -761,6 +779,7 @@ _v2uc(Vlong rv)
return rv.lo & 0xff; return rv.lo & 0xff;
} }
#pragma textflag NOSPLIT
long long
_v2sh(Vlong rv) _v2sh(Vlong rv)
{ {
...@@ -770,6 +789,7 @@ _v2sh(Vlong rv) ...@@ -770,6 +789,7 @@ _v2sh(Vlong rv)
return (t << 16) >> 16; return (t << 16) >> 16;
} }
#pragma textflag NOSPLIT
long long
_v2uh(Vlong rv) _v2uh(Vlong rv)
{ {
...@@ -777,6 +797,7 @@ _v2uh(Vlong rv) ...@@ -777,6 +797,7 @@ _v2uh(Vlong rv)
return rv.lo & 0xffff; return rv.lo & 0xffff;
} }
#pragma textflag NOSPLIT
long long
_v2sl(Vlong rv) _v2sl(Vlong rv)
{ {
...@@ -784,6 +805,7 @@ _v2sl(Vlong rv) ...@@ -784,6 +805,7 @@ _v2sl(Vlong rv)
return rv.lo; return rv.lo;
} }
#pragma textflag NOSPLIT
long long
_v2ul(Vlong rv) _v2ul(Vlong rv)
{ {
...@@ -798,6 +820,7 @@ _v2si(Vlong rv) ...@@ -798,6 +820,7 @@ _v2si(Vlong rv)
return rv.lo; return rv.lo;
} }
#pragma textflag NOSPLIT
long long
_v2ui(Vlong rv) _v2ui(Vlong rv)
{ {
...@@ -805,24 +828,28 @@ _v2ui(Vlong rv) ...@@ -805,24 +828,28 @@ _v2ui(Vlong rv)
return rv.lo; return rv.lo;
} }
#pragma textflag NOSPLIT
int int
_testv(Vlong rv) _testv(Vlong rv)
{ {
return rv.lo || rv.hi; return rv.lo || rv.hi;
} }
#pragma textflag NOSPLIT
int int
_eqv(Vlong lv, Vlong rv) _eqv(Vlong lv, Vlong rv)
{ {
return lv.lo == rv.lo && lv.hi == rv.hi; return lv.lo == rv.lo && lv.hi == rv.hi;
} }
#pragma textflag NOSPLIT
int int
_nev(Vlong lv, Vlong rv) _nev(Vlong lv, Vlong rv)
{ {
return lv.lo != rv.lo || lv.hi != rv.hi; return lv.lo != rv.lo || lv.hi != rv.hi;
} }
#pragma textflag NOSPLIT
int int
_ltv(Vlong lv, Vlong rv) _ltv(Vlong lv, Vlong rv)
{ {
...@@ -830,6 +857,7 @@ _ltv(Vlong lv, Vlong rv) ...@@ -830,6 +857,7 @@ _ltv(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo < rv.lo); (lv.hi == rv.hi && lv.lo < rv.lo);
} }
#pragma textflag NOSPLIT
int int
_lev(Vlong lv, Vlong rv) _lev(Vlong lv, Vlong rv)
{ {
...@@ -837,6 +865,7 @@ _lev(Vlong lv, Vlong rv) ...@@ -837,6 +865,7 @@ _lev(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo <= rv.lo); (lv.hi == rv.hi && lv.lo <= rv.lo);
} }
#pragma textflag NOSPLIT
int int
_gtv(Vlong lv, Vlong rv) _gtv(Vlong lv, Vlong rv)
{ {
...@@ -852,6 +881,7 @@ _gev(Vlong lv, Vlong rv) ...@@ -852,6 +881,7 @@ _gev(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo >= rv.lo); (lv.hi == rv.hi && lv.lo >= rv.lo);
} }
#pragma textflag NOSPLIT
int int
_lov(Vlong lv, Vlong rv) _lov(Vlong lv, Vlong rv)
{ {
...@@ -859,6 +889,7 @@ _lov(Vlong lv, Vlong rv) ...@@ -859,6 +889,7 @@ _lov(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo < rv.lo); (lv.hi == rv.hi && lv.lo < rv.lo);
} }
#pragma textflag NOSPLIT
int int
_lsv(Vlong lv, Vlong rv) _lsv(Vlong lv, Vlong rv)
{ {
...@@ -866,6 +897,7 @@ _lsv(Vlong lv, Vlong rv) ...@@ -866,6 +897,7 @@ _lsv(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo <= rv.lo); (lv.hi == rv.hi && lv.lo <= rv.lo);
} }
#pragma textflag NOSPLIT
int int
_hiv(Vlong lv, Vlong rv) _hiv(Vlong lv, Vlong rv)
{ {
...@@ -873,6 +905,7 @@ _hiv(Vlong lv, Vlong rv) ...@@ -873,6 +905,7 @@ _hiv(Vlong lv, Vlong rv)
(lv.hi == rv.hi && lv.lo > rv.lo); (lv.hi == rv.hi && lv.lo > rv.lo);
} }
#pragma textflag NOSPLIT
int int
_hsv(Vlong lv, Vlong rv) _hsv(Vlong lv, Vlong rv)
{ {
......
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