| // 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. | 
 |  | 
 | //go: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) | 
 | 	} | 
 | } |