32b1d51f16
This is a step toward updating libgo to 1.13. This adds the 1.13 version of the osinit function to Go code, and removes the corresponding code from the C runtime. This should simplify future updates. Some additional 1.13 code was brought in to simplify this change. Reviewed-on: https://go-review.googlesource.com/c/gofrontend/+/191717 From-SVN: r275010
147 lines
2.9 KiB
Go
147 lines
2.9 KiB
Go
// Copyright 2018 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 aix
|
|
|
|
package runtime
|
|
|
|
import (
|
|
"internal/cpu"
|
|
"unsafe"
|
|
)
|
|
|
|
//extern sysconf
|
|
func sysconf(int32) _C_long
|
|
|
|
//extern getsystemcfg
|
|
func getsystemcfg(int32) uint64
|
|
|
|
type mOS struct {
|
|
waitsema uintptr // semaphore for parking on locks
|
|
}
|
|
|
|
//extern malloc
|
|
func libc_malloc(uintptr) unsafe.Pointer
|
|
|
|
//go:noescape
|
|
//extern sem_init
|
|
func sem_init(sem *semt, pshared int32, value uint32) int32
|
|
|
|
//go:noescape
|
|
//extern sem_wait
|
|
func sem_wait(sem *semt) int32
|
|
|
|
//go:noescape
|
|
//extern sem_post
|
|
func sem_post(sem *semt) int32
|
|
|
|
//go:noescape
|
|
//extern sem_timedwait
|
|
func sem_timedwait(sem *semt, timeout *timespec) int32
|
|
|
|
//go:noescape
|
|
//extern clock_gettime
|
|
func clock_gettime(clock_id int64, timeout *timespec) int32
|
|
|
|
//go:nosplit
|
|
func semacreate(mp *m) {
|
|
if mp.mos.waitsema != 0 {
|
|
return
|
|
}
|
|
|
|
var sem *semt
|
|
|
|
// Call libc's malloc rather than malloc. This will
|
|
// allocate space on the C heap. We can't call malloc
|
|
// here because it could cause a deadlock.
|
|
sem = (*semt)(libc_malloc(unsafe.Sizeof(*sem)))
|
|
if sem_init(sem, 0, 0) != 0 {
|
|
throw("sem_init")
|
|
}
|
|
mp.mos.waitsema = uintptr(unsafe.Pointer(sem))
|
|
}
|
|
|
|
//go:nosplit
|
|
func semasleep(ns int64) int32 {
|
|
_m_ := getg().m
|
|
if ns >= 0 {
|
|
var ts timespec
|
|
|
|
if clock_gettime(_CLOCK_REALTIME, &ts) != 0 {
|
|
throw("clock_gettime")
|
|
}
|
|
|
|
sec := int64(ts.tv_sec) + ns/1e9
|
|
nsec := int64(ts.tv_nsec) + ns%1e9
|
|
if nsec >= 1e9 {
|
|
sec++
|
|
nsec -= 1e9
|
|
}
|
|
if sec != int64(timespec_sec_t(sec)) {
|
|
// Handle overflows (timespec_sec_t is 32-bit in 32-bit applications)
|
|
sec = 1<<31 - 1
|
|
}
|
|
ts.tv_sec = timespec_sec_t(sec)
|
|
ts.tv_nsec = timespec_nsec_t(nsec)
|
|
|
|
if sem_timedwait((*semt)(unsafe.Pointer(_m_.mos.waitsema)), &ts) != 0 {
|
|
err := errno()
|
|
if err == _ETIMEDOUT || err == _EAGAIN || err == _EINTR {
|
|
return -1
|
|
}
|
|
println("sem_timedwait err ", err, " ts.tv_sec ", ts.tv_sec, " ts.tv_nsec ", ts.tv_nsec, " ns ", ns, " id ", _m_.id)
|
|
throw("sem_timedwait")
|
|
}
|
|
return 0
|
|
}
|
|
for {
|
|
r1 := sem_wait((*semt)(unsafe.Pointer(_m_.mos.waitsema)))
|
|
if r1 == 0 {
|
|
break
|
|
}
|
|
if errno() == _EINTR {
|
|
continue
|
|
}
|
|
throw("sem_wait")
|
|
}
|
|
return 0
|
|
}
|
|
|
|
//go:nosplit
|
|
func semawakeup(mp *m) {
|
|
if sem_post((*semt)(unsafe.Pointer(mp.mos.waitsema))) != 0 {
|
|
throw("sem_post")
|
|
}
|
|
}
|
|
|
|
func osinit() {
|
|
ncpu = int32(sysconf(__SC_NPROCESSORS_ONLN))
|
|
physPageSize = uintptr(sysconf(__SC_PAGE_SIZE))
|
|
setupSystemConf()
|
|
}
|
|
|
|
const (
|
|
_CLOCK_REALTIME = 9
|
|
_CLOCK_MONOTONIC = 10
|
|
)
|
|
|
|
const (
|
|
// getsystemcfg constants
|
|
_SC_IMPL = 2
|
|
_IMPL_POWER8 = 0x10000
|
|
_IMPL_POWER9 = 0x20000
|
|
)
|
|
|
|
// setupSystemConf retrieves information about the CPU and updates
|
|
// cpu.HWCap variables.
|
|
func setupSystemConf() {
|
|
impl := getsystemcfg(_SC_IMPL)
|
|
if impl&_IMPL_POWER8 != 0 {
|
|
cpu.HWCap2 |= cpu.PPC_FEATURE2_ARCH_2_07
|
|
}
|
|
if impl&_IMPL_POWER9 != 0 {
|
|
cpu.HWCap2 |= cpu.PPC_FEATURE2_ARCH_3_00
|
|
}
|
|
}
|