mirror of
https://github.com/golang/go
synced 2024-11-18 05:24:47 -07:00
54 lines
1.4 KiB
Go
54 lines
1.4 KiB
Go
|
// run
|
||
|
|
||
|
// 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.
|
||
|
|
||
|
// Verify that we don't consider a Go'd function's
|
||
|
// arguments as pointers when they aren't.
|
||
|
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"unsafe"
|
||
|
)
|
||
|
|
||
|
var badPtr uintptr
|
||
|
|
||
|
var sink []byte
|
||
|
|
||
|
func init() {
|
||
|
// Allocate large enough to use largeAlloc.
|
||
|
b := make([]byte, 1<<16-1)
|
||
|
sink = b // force heap allocation
|
||
|
// Any space between the object and the end of page is invalid to point to.
|
||
|
badPtr = uintptr(unsafe.Pointer(&b[len(b)-1])) + 1
|
||
|
}
|
||
|
|
||
|
var throttle = make(chan struct{}, 10)
|
||
|
|
||
|
// There are 2 arg bitmaps for this function, each with 2 bits.
|
||
|
// In the first, p and q are both live, so that bitmap is 11.
|
||
|
// In the second, only p is live, so that bitmap is 10.
|
||
|
// Bitmaps are byte aligned, so if the first bitmap is interpreted as
|
||
|
// extending across the entire argument area, we incorrectly concatenate
|
||
|
// the bitmaps and end up using 110000001. That bad bitmap causes a6
|
||
|
// to be considered a pointer.
|
||
|
func noPointerArgs(p, q *byte, a0, a1, a2, a3, a4, a5, a6 uintptr) {
|
||
|
sink = make([]byte, 4096)
|
||
|
sinkptr = q
|
||
|
<-throttle
|
||
|
sinkptr = p
|
||
|
}
|
||
|
|
||
|
var sinkptr *byte
|
||
|
|
||
|
func main() {
|
||
|
const N = 1000
|
||
|
for i := 0; i < N; i++ {
|
||
|
throttle <- struct{}{}
|
||
|
go noPointerArgs(nil, nil, badPtr, badPtr, badPtr, badPtr, badPtr, badPtr, badPtr)
|
||
|
sink = make([]byte, 4096)
|
||
|
}
|
||
|
}
|