2021-06-10 19:27:00 +00:00
|
|
|
package winapi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
2021-07-20 17:59:04 +00:00
|
|
|
"reflect"
|
2021-06-10 19:27:00 +00:00
|
|
|
"syscall"
|
|
|
|
"unsafe"
|
2021-07-20 17:59:04 +00:00
|
|
|
|
|
|
|
"golang.org/x/sys/windows"
|
2021-06-10 19:27:00 +00:00
|
|
|
)
|
|
|
|
|
2021-07-20 17:59:04 +00:00
|
|
|
// Uint16BufferToSlice wraps a uint16 pointer-and-length into a slice
|
|
|
|
// for easier interop with Go APIs
|
|
|
|
func Uint16BufferToSlice(buffer *uint16, bufferLength int) (result []uint16) {
|
|
|
|
hdr := (*reflect.SliceHeader)(unsafe.Pointer(&result))
|
|
|
|
hdr.Data = uintptr(unsafe.Pointer(buffer))
|
|
|
|
hdr.Cap = bufferLength
|
|
|
|
hdr.Len = bufferLength
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-06-10 19:27:00 +00:00
|
|
|
type UnicodeString struct {
|
|
|
|
Length uint16
|
|
|
|
MaximumLength uint16
|
|
|
|
Buffer *uint16
|
|
|
|
}
|
|
|
|
|
|
|
|
//String converts a UnicodeString to a golang string
|
|
|
|
func (uni UnicodeString) String() string {
|
|
|
|
// UnicodeString is not guaranteed to be null terminated, therefore
|
|
|
|
// use the UnicodeString's Length field
|
2021-07-20 17:59:04 +00:00
|
|
|
return syscall.UTF16ToString(Uint16BufferToSlice(uni.Buffer, int(uni.Length/2)))
|
2021-06-10 19:27:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewUnicodeString allocates a new UnicodeString and copies `s` into
|
|
|
|
// the buffer of the new UnicodeString.
|
|
|
|
func NewUnicodeString(s string) (*UnicodeString, error) {
|
2021-07-20 17:59:04 +00:00
|
|
|
// Get length of original `s` to use in the UnicodeString since the `buf`
|
|
|
|
// created later will have an additional trailing null character
|
|
|
|
length := len(s)
|
|
|
|
if length > 32767 {
|
2021-06-10 19:27:00 +00:00
|
|
|
return nil, syscall.ENAMETOOLONG
|
|
|
|
}
|
|
|
|
|
2021-07-20 17:59:04 +00:00
|
|
|
buf, err := windows.UTF16FromString(s)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-06-10 19:27:00 +00:00
|
|
|
uni := &UnicodeString{
|
2021-07-20 17:59:04 +00:00
|
|
|
Length: uint16(length * 2),
|
|
|
|
MaximumLength: uint16(length * 2),
|
|
|
|
Buffer: &buf[0],
|
2021-06-10 19:27:00 +00:00
|
|
|
}
|
|
|
|
return uni, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConvertStringSetToSlice is a helper function used to convert the contents of
|
|
|
|
// `buf` into a string slice. `buf` contains a set of null terminated strings
|
|
|
|
// with an additional null at the end to indicate the end of the set.
|
|
|
|
func ConvertStringSetToSlice(buf []byte) ([]string, error) {
|
|
|
|
var results []string
|
|
|
|
prev := 0
|
|
|
|
for i := range buf {
|
|
|
|
if buf[i] == 0 {
|
|
|
|
if prev == i {
|
|
|
|
// found two null characters in a row, return result
|
|
|
|
return results, nil
|
|
|
|
}
|
|
|
|
results = append(results, string(buf[prev:i]))
|
|
|
|
prev = i + 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil, errors.New("string set malformed: missing null terminator at end of buffer")
|
|
|
|
}
|