Commit 6ef7794b authored by Alex Brainman's avatar Alex Brainman Committed by Brad Fitzpatrick

all: fix most of the remaining windows -d=checkptr violations

This change replaces

buf := [HUGE_CONST]*T)(unsafe.Pointer(p))[:]

with

buf := [HUGE_CONST]*T)(unsafe.Pointer(p))[:n:n]

Pointer p points to n of T elements. New unsafe pointer conversion
logic verifies that both first and last elements point into the same
Go variable.

This change replaces [:] with [:n:n] to please pointer checker.
According to @mdempsky, compiler specially recognizes when you
combine a pointer conversion with a full slice operation in a single
expression and makes an exception.

After this, only one failure in net remains when running:

go test -a -short -gcflags=all=-d=checkptr std cmd

Updates #34972

Change-Id: I2c8731650c856264bc788e4e07fa0530f7c250fa
Reviewed-on: https://go-review.googlesource.com/c/go/+/208617
Run-TryBot: Brad Fitzpatrick <bradfitz@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
Reviewed-by: default avatarMatthew Dempsky <mdempsky@google.com>
parent b81e01dc
...@@ -151,7 +151,7 @@ var pkgDeps = map[string][]string{ ...@@ -151,7 +151,7 @@ var pkgDeps = map[string][]string{
"syscall/js": {"L0"}, "syscall/js": {"L0"},
"internal/oserror": {"L0"}, "internal/oserror": {"L0"},
"internal/syscall/unix": {"L0", "syscall"}, "internal/syscall/unix": {"L0", "syscall"},
"internal/syscall/windows": {"L0", "syscall", "internal/syscall/windows/sysdll"}, "internal/syscall/windows": {"L0", "syscall", "internal/syscall/windows/sysdll", "unicode/utf16"},
"internal/syscall/windows/registry": {"L0", "syscall", "internal/syscall/windows/sysdll", "unicode/utf16"}, "internal/syscall/windows/registry": {"L0", "syscall", "internal/syscall/windows/sysdll", "unicode/utf16"},
"time": { "time": {
// "L0" without the "io" package: // "L0" without the "io" package:
......
...@@ -60,8 +60,9 @@ type SymbolicLinkReparseBuffer struct { ...@@ -60,8 +60,9 @@ type SymbolicLinkReparseBuffer struct {
// Path returns path stored in rb. // Path returns path stored in rb.
func (rb *SymbolicLinkReparseBuffer) Path() string { func (rb *SymbolicLinkReparseBuffer) Path() string {
p := (*[0xffff]uint16)(unsafe.Pointer(&rb.PathBuffer[0])) n1 := rb.SubstituteNameOffset / 2
return syscall.UTF16ToString(p[rb.SubstituteNameOffset/2 : (rb.SubstituteNameOffset+rb.SubstituteNameLength)/2]) n2 := (rb.SubstituteNameOffset + rb.SubstituteNameLength) / 2
return syscall.UTF16ToString((*[0xffff]uint16)(unsafe.Pointer(&rb.PathBuffer[0]))[n1:n2:n2])
} }
type MountPointReparseBuffer struct { type MountPointReparseBuffer struct {
...@@ -83,6 +84,7 @@ type MountPointReparseBuffer struct { ...@@ -83,6 +84,7 @@ type MountPointReparseBuffer struct {
// Path returns path stored in rb. // Path returns path stored in rb.
func (rb *MountPointReparseBuffer) Path() string { func (rb *MountPointReparseBuffer) Path() string {
p := (*[0xffff]uint16)(unsafe.Pointer(&rb.PathBuffer[0])) n1 := rb.SubstituteNameOffset / 2
return syscall.UTF16ToString(p[rb.SubstituteNameOffset/2 : (rb.SubstituteNameOffset+rb.SubstituteNameLength)/2]) n2 := (rb.SubstituteNameOffset + rb.SubstituteNameLength) / 2
return syscall.UTF16ToString((*[0xffff]uint16)(unsafe.Pointer(&rb.PathBuffer[0]))[n1:n2:n2])
} }
...@@ -7,9 +7,29 @@ package windows ...@@ -7,9 +7,29 @@ package windows
import ( import (
"sync" "sync"
"syscall" "syscall"
"unicode/utf16"
"unsafe" "unsafe"
) )
// UTF16PtrToString is like UTF16ToString, but takes *uint16
// as a parameter instead of []uint16.
// max is how many times p can be advanced looking for the null terminator.
// If max is hit, the string is truncated at that point.
func UTF16PtrToString(p *uint16, max int) string {
if p == nil {
return ""
}
// Find NUL terminator.
end := unsafe.Pointer(p)
n := 0
for *(*uint16)(end) != 0 && n < max {
end = unsafe.Pointer(uintptr(end) + unsafe.Sizeof(*p))
n++
}
s := (*[(1 << 30) - 1]uint16)(unsafe.Pointer(p))[:n:n]
return string(utf16.Decode(s))
}
const ( const (
ERROR_SHARING_VIOLATION syscall.Errno = 32 ERROR_SHARING_VIOLATION syscall.Errno = 32
ERROR_LOCK_VIOLATION syscall.Errno = 33 ERROR_LOCK_VIOLATION syscall.Errno = 33
......
...@@ -58,7 +58,7 @@ func interfaceTable(ifindex int) ([]Interface, error) { ...@@ -58,7 +58,7 @@ func interfaceTable(ifindex int) ([]Interface, error) {
if ifindex == 0 || ifindex == int(index) { if ifindex == 0 || ifindex == int(index) {
ifi := Interface{ ifi := Interface{
Index: int(index), Index: int(index),
Name: syscall.UTF16ToString((*(*[10000]uint16)(unsafe.Pointer(aa.FriendlyName)))[:]), Name: windows.UTF16PtrToString(aa.FriendlyName, 10000),
} }
if aa.OperStatus == windows.IfOperStatusUp { if aa.OperStatus == windows.IfOperStatusUp {
ifi.Flags |= FlagUp ifi.Flags |= FlagUp
......
...@@ -6,6 +6,7 @@ package net ...@@ -6,6 +6,7 @@ package net
import ( import (
"context" "context"
"internal/syscall/windows"
"os" "os"
"runtime" "runtime"
"syscall" "syscall"
...@@ -233,7 +234,7 @@ func (*Resolver) lookupCNAME(ctx context.Context, name string) (string, error) { ...@@ -233,7 +234,7 @@ func (*Resolver) lookupCNAME(ctx context.Context, name string) (string, error) {
defer syscall.DnsRecordListFree(r, 1) defer syscall.DnsRecordListFree(r, 1)
resolved := resolveCNAME(syscall.StringToUTF16Ptr(name), r) resolved := resolveCNAME(syscall.StringToUTF16Ptr(name), r)
cname := syscall.UTF16ToString((*[256]uint16)(unsafe.Pointer(resolved))[:]) cname := windows.UTF16PtrToString(resolved, 256)
return absDomainName([]byte(cname)), nil return absDomainName([]byte(cname)), nil
} }
...@@ -277,7 +278,7 @@ func (*Resolver) lookupMX(ctx context.Context, name string) ([]*MX, error) { ...@@ -277,7 +278,7 @@ func (*Resolver) lookupMX(ctx context.Context, name string) ([]*MX, error) {
mxs := make([]*MX, 0, 10) mxs := make([]*MX, 0, 10)
for _, p := range validRecs(r, syscall.DNS_TYPE_MX, name) { for _, p := range validRecs(r, syscall.DNS_TYPE_MX, name) {
v := (*syscall.DNSMXData)(unsafe.Pointer(&p.Data[0])) v := (*syscall.DNSMXData)(unsafe.Pointer(&p.Data[0]))
mxs = append(mxs, &MX{absDomainName([]byte(syscall.UTF16ToString((*[256]uint16)(unsafe.Pointer(v.NameExchange))[:]))), v.Preference}) mxs = append(mxs, &MX{absDomainName([]byte(windows.UTF16PtrToString(v.NameExchange, 256))), v.Preference})
} }
byPref(mxs).sort() byPref(mxs).sort()
return mxs, nil return mxs, nil
...@@ -317,8 +318,8 @@ func (*Resolver) lookupTXT(ctx context.Context, name string) ([]string, error) { ...@@ -317,8 +318,8 @@ func (*Resolver) lookupTXT(ctx context.Context, name string) ([]string, error) {
for _, p := range validRecs(r, syscall.DNS_TYPE_TEXT, name) { for _, p := range validRecs(r, syscall.DNS_TYPE_TEXT, name) {
d := (*syscall.DNSTXTData)(unsafe.Pointer(&p.Data[0])) d := (*syscall.DNSTXTData)(unsafe.Pointer(&p.Data[0]))
s := "" s := ""
for _, v := range (*[1 << 10]*uint16)(unsafe.Pointer(&(d.StringArray[0])))[:d.StringCount] { for _, v := range (*[1 << 10]*uint16)(unsafe.Pointer(&(d.StringArray[0])))[:d.StringCount:d.StringCount] {
s += syscall.UTF16ToString((*[1 << 20]uint16)(unsafe.Pointer(v))[:]) s += windows.UTF16PtrToString(v, 1<<20)
} }
txts = append(txts, s) txts = append(txts, s)
} }
...@@ -343,7 +344,7 @@ func (*Resolver) lookupAddr(ctx context.Context, addr string) ([]string, error) ...@@ -343,7 +344,7 @@ func (*Resolver) lookupAddr(ctx context.Context, addr string) ([]string, error)
ptrs := make([]string, 0, 10) ptrs := make([]string, 0, 10)
for _, p := range validRecs(r, syscall.DNS_TYPE_PTR, arpa) { for _, p := range validRecs(r, syscall.DNS_TYPE_PTR, arpa) {
v := (*syscall.DNSPTRData)(unsafe.Pointer(&p.Data[0])) v := (*syscall.DNSPTRData)(unsafe.Pointer(&p.Data[0]))
ptrs = append(ptrs, absDomainName([]byte(syscall.UTF16ToString((*[256]uint16)(unsafe.Pointer(v.Host))[:])))) ptrs = append(ptrs, absDomainName([]byte(windows.UTF16PtrToString(v.Host, 256))))
} }
return ptrs, nil return ptrs, nil
} }
......
...@@ -23,16 +23,20 @@ func environForSysProcAttr(sys *syscall.SysProcAttr) (env []string, err error) { ...@@ -23,16 +23,20 @@ func environForSysProcAttr(sys *syscall.SysProcAttr) (env []string, err error) {
defer windows.DestroyEnvironmentBlock(block) defer windows.DestroyEnvironmentBlock(block)
blockp := uintptr(unsafe.Pointer(block)) blockp := uintptr(unsafe.Pointer(block))
for { for {
entry := (*[(1 << 30) - 1]uint16)(unsafe.Pointer(blockp))[:]
for i, v := range entry { // find NUL terminator
if v == 0 { end := unsafe.Pointer(blockp)
entry = entry[:i] for *(*uint16)(end) != 0 {
break end = unsafe.Pointer(uintptr(end) + 2)
}
} }
if len(entry) == 0 {
n := (uintptr(end) - uintptr(unsafe.Pointer(blockp))) / 2
if n == 0 {
// environment block ends with empty string
break break
} }
entry := (*[(1 << 30) - 1]uint16)(unsafe.Pointer(blockp))[:n:n]
env = append(env, string(utf16.Decode(entry))) env = append(env, string(utf16.Decode(entry)))
blockp += 2 * (uintptr(len(entry)) + 1) blockp += 2 * (uintptr(len(entry)) + 1)
} }
......
...@@ -6,11 +6,11 @@ package os ...@@ -6,11 +6,11 @@ package os
import ( import (
"errors" "errors"
"internal/syscall/windows"
"runtime" "runtime"
"sync/atomic" "sync/atomic"
"syscall" "syscall"
"time" "time"
"unsafe"
) )
func (p *Process) wait() (ps *ProcessState, err error) { func (p *Process) wait() (ps *ProcessState, err error) {
...@@ -98,8 +98,7 @@ func findProcess(pid int) (p *Process, err error) { ...@@ -98,8 +98,7 @@ func findProcess(pid int) (p *Process, err error) {
} }
func init() { func init() {
p := syscall.GetCommandLine() cmd := windows.UTF16PtrToString(syscall.GetCommandLine(), 0xffff)
cmd := syscall.UTF16ToString((*[0xffff]uint16)(unsafe.Pointer(p))[:])
if len(cmd) == 0 { if len(cmd) == 0 {
arg0, _ := Executable() arg0, _ := Executable()
Args = []string{arg0} Args = []string{arg0}
......
...@@ -263,7 +263,8 @@ func createMountPoint(link string, target *reparseData) error { ...@@ -263,7 +263,8 @@ func createMountPoint(link string, target *reparseData) error {
buf.SubstituteNameLength = target.substituteName.length buf.SubstituteNameLength = target.substituteName.length
buf.PrintNameOffset = target.printName.offset buf.PrintNameOffset = target.printName.offset
buf.PrintNameLength = target.printName.length buf.PrintNameLength = target.printName.length
copy((*[2048]uint16)(unsafe.Pointer(&buf.PathBuffer[0]))[:], target.pathBuf) pbuflen := len(target.pathBuf)
copy((*[2048]uint16)(unsafe.Pointer(&buf.PathBuffer[0]))[:pbuflen:pbuflen], target.pathBuf)
var rdb _REPARSE_DATA_BUFFER var rdb _REPARSE_DATA_BUFFER
rdb.header.ReparseTag = windows.IO_REPARSE_TAG_MOUNT_POINT rdb.header.ReparseTag = windows.IO_REPARSE_TAG_MOUNT_POINT
...@@ -356,7 +357,8 @@ func createSymbolicLink(link string, target *reparseData, isrelative bool) error ...@@ -356,7 +357,8 @@ func createSymbolicLink(link string, target *reparseData, isrelative bool) error
if isrelative { if isrelative {
buf.Flags = windows.SYMLINK_FLAG_RELATIVE buf.Flags = windows.SYMLINK_FLAG_RELATIVE
} }
copy((*[2048]uint16)(unsafe.Pointer(&buf.PathBuffer[0]))[:], target.pathBuf) pbuflen := len(target.pathBuf)
copy((*[2048]uint16)(unsafe.Pointer(&buf.PathBuffer[0]))[:pbuflen:pbuflen], target.pathBuf)
var rdb _REPARSE_DATA_BUFFER var rdb _REPARSE_DATA_BUFFER
rdb.header.ReparseTag = syscall.IO_REPARSE_TAG_SYMLINK rdb.header.ReparseTag = syscall.IO_REPARSE_TAG_SYMLINK
...@@ -714,7 +716,7 @@ func TestReadStdin(t *testing.T) { ...@@ -714,7 +716,7 @@ func TestReadStdin(t *testing.T) {
if n > consoleSize { if n > consoleSize {
n = consoleSize n = consoleSize
} }
n = copy((*[10000]uint16)(unsafe.Pointer(buf))[:n], s16) n = copy((*[10000]uint16)(unsafe.Pointer(buf))[:n:n], s16)
s16 = s16[n:] s16 = s16[n:]
*read = uint32(n) *read = uint32(n)
t.Logf("read %d -> %d", toread, *read) t.Logf("read %d -> %d", toread, *read)
......
...@@ -44,11 +44,7 @@ func lookupFullNameServer(servername, username string) (string, error) { ...@@ -44,11 +44,7 @@ func lookupFullNameServer(servername, username string) (string, error) {
} }
defer syscall.NetApiBufferFree(p) defer syscall.NetApiBufferFree(p)
i := (*syscall.UserInfo10)(unsafe.Pointer(p)) i := (*syscall.UserInfo10)(unsafe.Pointer(p))
if i.FullName == nil { return windows.UTF16PtrToString(i.FullName, 1024), nil
return "", nil
}
name := syscall.UTF16ToString((*[1024]uint16)(unsafe.Pointer(i.FullName))[:])
return name, nil
} }
func lookupFullName(domain, username, domainAndUser string) (string, error) { func lookupFullName(domain, username, domainAndUser string) (string, error) {
...@@ -165,14 +161,13 @@ func listGroupsForUsernameAndDomain(username, domain string) ([]string, error) { ...@@ -165,14 +161,13 @@ func listGroupsForUsernameAndDomain(username, domain string) ([]string, error) {
if entriesRead == 0 { if entriesRead == 0 {
return nil, fmt.Errorf("listGroupsForUsernameAndDomain: NetUserGetLocalGroups() returned an empty list for domain: %s, username: %s", domain, username) return nil, fmt.Errorf("listGroupsForUsernameAndDomain: NetUserGetLocalGroups() returned an empty list for domain: %s, username: %s", domain, username)
} }
entries := (*[1024]windows.LocalGroupUserInfo0)(unsafe.Pointer(p0))[:entriesRead] entries := (*[1024]windows.LocalGroupUserInfo0)(unsafe.Pointer(p0))[:entriesRead:entriesRead]
var sids []string var sids []string
for _, entry := range entries { for _, entry := range entries {
if entry.Name == nil { if entry.Name == nil {
continue continue
} }
name := syscall.UTF16ToString((*[1024]uint16)(unsafe.Pointer(entry.Name))[:]) sid, err := lookupGroupName(windows.UTF16PtrToString(entry.Name, 1024))
sid, err := lookupGroupName(name)
if err != nil { if err != nil {
return nil, err return nil, err
} }
......
...@@ -163,7 +163,7 @@ func (sid *SID) String() (string, error) { ...@@ -163,7 +163,7 @@ func (sid *SID) String() (string, error) {
return "", e return "", e
} }
defer LocalFree((Handle)(unsafe.Pointer(s))) defer LocalFree((Handle)(unsafe.Pointer(s)))
return UTF16ToString((*[256]uint16)(unsafe.Pointer(s))[:]), nil return utf16PtrToString(s, 256), nil
} }
// Len returns the length, in bytes, of a valid security identifier sid. // Len returns the length, in bytes, of a valid security identifier sid.
......
...@@ -57,6 +57,25 @@ func UTF16ToString(s []uint16) string { ...@@ -57,6 +57,25 @@ func UTF16ToString(s []uint16) string {
return string(utf16.Decode(s)) return string(utf16.Decode(s))
} }
// utf16PtrToString is like UTF16ToString, but takes *uint16
// as a parameter instead of []uint16.
// max is how many times p can be advanced looking for the null terminator.
// If max is hit, the string is truncated at that point.
func utf16PtrToString(p *uint16, max int) string {
if p == nil {
return ""
}
// Find NUL terminator.
end := unsafe.Pointer(p)
n := 0
for *(*uint16)(end) != 0 && n < max {
end = unsafe.Pointer(uintptr(end) + unsafe.Sizeof(*p))
n++
}
s := (*[(1 << 30) - 1]uint16)(unsafe.Pointer(p))[:n:n]
return string(utf16.Decode(s))
}
// StringToUTF16Ptr returns pointer to the UTF-16 encoding of // StringToUTF16Ptr returns pointer to the UTF-16 encoding of
// the UTF-8 string s, with a terminating NUL added. If s // the UTF-8 string s, with a terminating NUL added. If s
// contains a NUL byte this function panics instead of // contains a NUL byte this function panics instead of
...@@ -769,7 +788,7 @@ func (rsa *RawSockaddrAny) Sockaddr() (Sockaddr, error) { ...@@ -769,7 +788,7 @@ func (rsa *RawSockaddrAny) Sockaddr() (Sockaddr, error) {
for n < len(pp.Path) && pp.Path[n] != 0 { for n < len(pp.Path) && pp.Path[n] != 0 {
n++ n++
} }
bytes := (*[len(pp.Path)]byte)(unsafe.Pointer(&pp.Path[0]))[0:n] bytes := (*[len(pp.Path)]byte)(unsafe.Pointer(&pp.Path[0]))[0:n:n]
sa.Name = string(bytes) sa.Name = string(bytes)
return sa, nil return sa, nil
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment