// Copyright 2016 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 windows,race
|
|
package race_test
|
|
import (
|
"sync/atomic"
|
"syscall"
|
"testing"
|
"unsafe"
|
)
|
|
func TestAtomicMmap(t *testing.T) {
|
// Test that atomic operations work on "external" memory. Previously they crashed (#16206).
|
// Also do a sanity correctness check: under race detector atomic operations
|
// are implemented inside of race runtime.
|
kernel32 := syscall.NewLazyDLL("kernel32.dll")
|
VirtualAlloc := kernel32.NewProc("VirtualAlloc")
|
VirtualFree := kernel32.NewProc("VirtualFree")
|
const (
|
MEM_COMMIT = 0x00001000
|
MEM_RESERVE = 0x00002000
|
MEM_RELEASE = 0x8000
|
PAGE_READWRITE = 0x04
|
)
|
mem, _, err := syscall.Syscall6(VirtualAlloc.Addr(), 4, 0, 1<<20, MEM_COMMIT|MEM_RESERVE, PAGE_READWRITE, 0, 0)
|
if err != 0 {
|
t.Fatalf("VirtualAlloc failed: %v", err)
|
}
|
defer syscall.Syscall(VirtualFree.Addr(), 3, mem, 1<<20, MEM_RELEASE)
|
a := (*uint64)(unsafe.Pointer(mem))
|
if *a != 0 {
|
t.Fatalf("bad atomic value: %v, want 0", *a)
|
}
|
atomic.AddUint64(a, 1)
|
if *a != 1 {
|
t.Fatalf("bad atomic value: %v, want 1", *a)
|
}
|
atomic.AddUint64(a, 1)
|
if *a != 2 {
|
t.Fatalf("bad atomic value: %v, want 2", *a)
|
}
|
}
|