mirror of
				https://github.com/tailscale/tailscale.git
				synced 2025-10-25 10:09:17 +00:00 
			
		
		
		
	 df86576989
			
		
	
	df86576989
	
	
	
		
			
			AllocateContiguousBuffer is for allocating structs with trailing buffers containing additional data. It is to be used for various Windows structures containing pointers to data located immediately after the struct. SetNTString performs in-place setting of windows.NTString and windows.NTUnicodeString. Updates #12383 Signed-off-by: Aaron Klotz <aaron@tailscale.com>
		
			
				
	
	
		
			135 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			135 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright (c) Tailscale Inc & AUTHORS
 | |
| // SPDX-License-Identifier: BSD-3-Clause
 | |
| 
 | |
| package winutil
 | |
| 
 | |
| import (
 | |
| 	"reflect"
 | |
| 	"testing"
 | |
| 	"unsafe"
 | |
| )
 | |
| 
 | |
| //lint:file-ignore U1000 Fields are unused but necessary for tests.
 | |
| 
 | |
| const (
 | |
| 	localSystemSID = "S-1-5-18"
 | |
| 	networkSID     = "S-1-5-2"
 | |
| )
 | |
| 
 | |
| func TestLookupPseudoUser(t *testing.T) {
 | |
| 	localSystem, err := LookupPseudoUser(localSystemSID)
 | |
| 	if err != nil {
 | |
| 		t.Errorf("LookupPseudoUser(%q) error: %v", localSystemSID, err)
 | |
| 	}
 | |
| 	if localSystem.Gid != localSystemSID {
 | |
| 		t.Errorf("incorrect Gid, got %q, want %q", localSystem.Gid, localSystemSID)
 | |
| 	}
 | |
| 	t.Logf("localSystem: %v", localSystem)
 | |
| 
 | |
| 	// networkSID is a built-in known group but not a pseudo-user.
 | |
| 	_, err = LookupPseudoUser(networkSID)
 | |
| 	if err == nil {
 | |
| 		t.Errorf("LookupPseudoUser(%q) unexpectedly succeeded", networkSID)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type testType interface {
 | |
| 	byte | uint16 | uint32 | uint64
 | |
| }
 | |
| 
 | |
| type noPointers[T testType] struct {
 | |
| 	foo byte
 | |
| 	bar T
 | |
| 	baz bool
 | |
| }
 | |
| 
 | |
| type hasPointer struct {
 | |
| 	foo byte
 | |
| 	bar uint32
 | |
| 	s1  *struct{}
 | |
| 	baz byte
 | |
| }
 | |
| 
 | |
| func checkContiguousBuffer[T any, BU BufUnit](t *testing.T, extra []BU, pt *T, ptLen uint32, slcs [][]BU) {
 | |
| 	szBU := int(unsafe.Sizeof(BU(0)))
 | |
| 	expectedAlign := max(reflect.TypeFor[T]().Align(), szBU)
 | |
| 	// Check that pointer is aligned
 | |
| 	if rem := uintptr(unsafe.Pointer(pt)) % uintptr(expectedAlign); rem != 0 {
 | |
| 		t.Errorf("pointer alignment got %d, want 0", rem)
 | |
| 	}
 | |
| 	// Check that alloc length is aligned
 | |
| 	if rem := int(ptLen) % expectedAlign; rem != 0 {
 | |
| 		t.Errorf("allocation length alignment got %d, want 0", rem)
 | |
| 	}
 | |
| 	expectedLen := int(unsafe.Sizeof(*pt))
 | |
| 	expectedLen = alignUp(expectedLen, szBU)
 | |
| 	expectedLen += len(extra) * szBU
 | |
| 	expectedLen = alignUp(expectedLen, expectedAlign)
 | |
| 	if gotLen := int(ptLen); gotLen != expectedLen {
 | |
| 		t.Errorf("allocation length got %d, want %d", gotLen, expectedLen)
 | |
| 	}
 | |
| 	if l := len(slcs); l != 1 {
 | |
| 		t.Errorf("len(slcs) got %d, want 1", l)
 | |
| 	}
 | |
| 	if len(extra) == 0 && slcs[0] != nil {
 | |
| 		t.Error("slcs[0] got non-nil, want nil")
 | |
| 	}
 | |
| 	if len(extra) != len(slcs[0]) {
 | |
| 		t.Errorf("len(slcs[0]) got %d, want %d", len(slcs[0]), len(extra))
 | |
| 	} else if rem := uintptr(unsafe.Pointer(unsafe.SliceData(slcs[0]))) % uintptr(szBU); rem != 0 {
 | |
| 		t.Errorf("additional data alignment got %d, want 0", rem)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestAllocateContiguousBuffer(t *testing.T) {
 | |
| 	t.Run("NoValues", testNoValues)
 | |
| 	t.Run("NoPointers", testNoPointers)
 | |
| 	t.Run("HasPointer", testHasPointer)
 | |
| }
 | |
| 
 | |
| func testNoValues(t *testing.T) {
 | |
| 	defer func() {
 | |
| 		if r := recover(); r == nil {
 | |
| 			t.Error("expected panic but didn't get one")
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	AllocateContiguousBuffer[hasPointer, byte]()
 | |
| }
 | |
| 
 | |
| const maxTestBufLen = 8
 | |
| 
 | |
| func testNoPointers(t *testing.T) {
 | |
| 	buf8 := make([]byte, maxTestBufLen)
 | |
| 	buf16 := make([]uint16, maxTestBufLen)
 | |
| 	for i := range maxTestBufLen {
 | |
| 		s8, sl, slcs8 := AllocateContiguousBuffer[noPointers[byte]](buf8[:i])
 | |
| 		checkContiguousBuffer(t, buf8[:i], s8, sl, slcs8)
 | |
| 		s16, sl, slcs8 := AllocateContiguousBuffer[noPointers[uint16]](buf8[:i])
 | |
| 		checkContiguousBuffer(t, buf8[:i], s16, sl, slcs8)
 | |
| 		s32, sl, slcs8 := AllocateContiguousBuffer[noPointers[uint32]](buf8[:i])
 | |
| 		checkContiguousBuffer(t, buf8[:i], s32, sl, slcs8)
 | |
| 		s64, sl, slcs8 := AllocateContiguousBuffer[noPointers[uint64]](buf8[:i])
 | |
| 		checkContiguousBuffer(t, buf8[:i], s64, sl, slcs8)
 | |
| 		s8, sl, slcs16 := AllocateContiguousBuffer[noPointers[byte]](buf16[:i])
 | |
| 		checkContiguousBuffer(t, buf16[:i], s8, sl, slcs16)
 | |
| 		s16, sl, slcs16 = AllocateContiguousBuffer[noPointers[uint16]](buf16[:i])
 | |
| 		checkContiguousBuffer(t, buf16[:i], s16, sl, slcs16)
 | |
| 		s32, sl, slcs16 = AllocateContiguousBuffer[noPointers[uint32]](buf16[:i])
 | |
| 		checkContiguousBuffer(t, buf16[:i], s32, sl, slcs16)
 | |
| 		s64, sl, slcs16 = AllocateContiguousBuffer[noPointers[uint64]](buf16[:i])
 | |
| 		checkContiguousBuffer(t, buf16[:i], s64, sl, slcs16)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func testHasPointer(t *testing.T) {
 | |
| 	buf8 := make([]byte, maxTestBufLen)
 | |
| 	buf16 := make([]uint16, maxTestBufLen)
 | |
| 	for i := range maxTestBufLen {
 | |
| 		s, sl, slcs8 := AllocateContiguousBuffer[hasPointer](buf8[:i])
 | |
| 		checkContiguousBuffer(t, buf8[:i], s, sl, slcs8)
 | |
| 		s, sl, slcs16 := AllocateContiguousBuffer[hasPointer](buf16[:i])
 | |
| 		checkContiguousBuffer(t, buf16[:i], s, sl, slcs16)
 | |
| 	}
 | |
| }
 |