Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 1 | // Copyright 2014 The Go Authors. All rights reserved. |
| 2 | // Use of this source code is governed by a BSD-style |
| 3 | // license that can be found in the LICENSE file. |
| 4 | |
Russ Cox | 09d92b6 | 2014-12-05 19:13:20 -0500 | [diff] [blame] | 5 | // +build ppc64 ppc64le |
Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 6 | // +build linux |
| 7 | |
| 8 | package runtime |
| 9 | |
| 10 | import "unsafe" |
| 11 | |
Russ Cox | 09d92b6 | 2014-12-05 19:13:20 -0500 | [diff] [blame] | 12 | // On ppc64, Linux limits the user address space to 46 bits (see |
Austin Clements | b76e836 | 2014-11-19 11:30:58 -0500 | [diff] [blame] | 13 | // TASK_SIZE_USER64 in the Linux kernel). This has grown over time, |
| 14 | // so here we allow 48 bit addresses. |
| 15 | // |
| 16 | // In addition to the 16 bits taken from the top, we can take 3 from the |
| 17 | // bottom, because node must be pointer-aligned, giving a total of 19 bits |
Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 18 | // of count. |
Austin Clements | b76e836 | 2014-11-19 11:30:58 -0500 | [diff] [blame] | 19 | const ( |
| 20 | addrBits = 48 |
| 21 | cntBits = 64 - addrBits + 3 |
| 22 | ) |
Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 23 | |
| 24 | func lfstackPack(node *lfnode, cnt uintptr) uint64 { |
Austin Clements | b76e836 | 2014-11-19 11:30:58 -0500 | [diff] [blame] | 25 | return uint64(uintptr(unsafe.Pointer(node)))<<(64-addrBits) | uint64(cnt&(1<<cntBits-1)) |
Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 26 | } |
| 27 | |
| 28 | func lfstackUnpack(val uint64) (node *lfnode, cnt uintptr) { |
Austin Clements | b76e836 | 2014-11-19 11:30:58 -0500 | [diff] [blame] | 29 | node = (*lfnode)(unsafe.Pointer(uintptr(val >> cntBits << 3))) |
| 30 | cnt = uintptr(val & (1<<cntBits - 1)) |
Russ Cox | 5fce15a | 2014-11-14 12:55:23 -0500 | [diff] [blame] | 31 | return |
| 32 | } |