| // run | 
 |  | 
 | // Copyright 2021 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. | 
 |  | 
 | package main | 
 |  | 
 | import ( | 
 | 	"math" | 
 | 	"unsafe" | 
 | ) | 
 |  | 
 | const maxUintptr = 1 << (8 * unsafe.Sizeof(uintptr(0))) | 
 |  | 
 | func main() { | 
 | 	var p [10]byte | 
 |  | 
 | 	// unsafe.Add | 
 | 	{ | 
 | 		p1 := unsafe.Pointer(&p[1]) | 
 | 		assert(unsafe.Add(p1, 1) == unsafe.Pointer(&p[2])) | 
 | 		assert(unsafe.Add(p1, -1) == unsafe.Pointer(&p[0])) | 
 | 	} | 
 |  | 
 | 	// unsafe.Slice | 
 | 	{ | 
 | 		s := unsafe.Slice(&p[0], len(p)) | 
 | 		assert(&s[0] == &p[0]) | 
 | 		assert(len(s) == len(p)) | 
 | 		assert(cap(s) == len(p)) | 
 |  | 
 | 		// nil pointer with zero length returns nil | 
 | 		assert(unsafe.Slice((*int)(nil), 0) == nil) | 
 |  | 
 | 		// nil pointer with positive length panics | 
 | 		mustPanic(func() { _ = unsafe.Slice((*int)(nil), 1) }) | 
 |  | 
 | 		// negative length | 
 | 		var neg int = -1 | 
 | 		mustPanic(func() { _ = unsafe.Slice(new(byte), neg) }) | 
 |  | 
 | 		// length too large | 
 | 		var tooBig uint64 = math.MaxUint64 | 
 | 		mustPanic(func() { _ = unsafe.Slice(new(byte), tooBig) }) | 
 |  | 
 | 		// size overflows address space | 
 | 		mustPanic(func() { _ = unsafe.Slice(new(uint64), maxUintptr/8) }) | 
 | 		mustPanic(func() { _ = unsafe.Slice(new(uint64), maxUintptr/8+1) }) | 
 |  | 
 | 		// sliced memory overflows address space | 
 | 		last := (*byte)(unsafe.Pointer(^uintptr(0))) | 
 | 		_ = unsafe.Slice(last, 1) | 
 | 		mustPanic(func() { _ = unsafe.Slice(last, 2) }) | 
 | 	} | 
 |  | 
 | 	// unsafe.String | 
 | 	{ | 
 | 		s := unsafe.String(&p[0], len(p)) | 
 | 		assert(s == string(p[:])) | 
 | 		assert(len(s) == len(p)) | 
 |  | 
 | 		// the empty string | 
 | 		assert(unsafe.String(nil, 0) == "") | 
 |  | 
 | 		// nil pointer with positive length panics | 
 | 		mustPanic(func() { _ = unsafe.String(nil, 1) }) | 
 |  | 
 | 		// negative length | 
 | 		var neg int = -1 | 
 | 		mustPanic(func() { _ = unsafe.String(new(byte), neg) }) | 
 |  | 
 | 		// length too large | 
 | 		var tooBig uint64 = math.MaxUint64 | 
 | 		mustPanic(func() { _ = unsafe.String(new(byte), tooBig) }) | 
 |  | 
 | 		// string memory overflows address space | 
 | 		last := (*byte)(unsafe.Pointer(^uintptr(0))) | 
 | 		_ = unsafe.String(last, 1) | 
 | 		mustPanic(func() { _ = unsafe.String(last, 2) }) | 
 | 	} | 
 |  | 
 | 	// unsafe.StringData | 
 | 	{ | 
 | 		var s = "string" | 
 | 		assert(string(unsafe.Slice(unsafe.StringData(s), len(s))) == s) | 
 | 	} | 
 |  | 
 | 	//unsafe.SliceData | 
 | 	{ | 
 | 		var s = []byte("slice") | 
 | 		assert(unsafe.String(unsafe.SliceData(s), len(s)) == string(s)) | 
 | 	} | 
 | } | 
 |  | 
 | func assert(ok bool) { | 
 | 	if !ok { | 
 | 		panic("FAIL") | 
 | 	} | 
 | } | 
 |  | 
 | func mustPanic(f func()) { | 
 | 	defer func() { | 
 | 		assert(recover() != nil) | 
 | 	}() | 
 | 	f() | 
 | } |