github.com/alash3al/go@v0.0.0-20150827002835-d497eeb00540/src/internal/syscall/windows/registry/value.go (about) 1 // Copyright 2015 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 // +build windows 6 7 package registry 8 9 import ( 10 "errors" 11 "io" 12 "syscall" 13 "unicode/utf16" 14 "unsafe" 15 ) 16 17 const ( 18 // Registry value types. 19 NONE = 0 20 SZ = 1 21 EXPAND_SZ = 2 22 BINARY = 3 23 DWORD = 4 24 DWORD_BIG_ENDIAN = 5 25 LINK = 6 26 MULTI_SZ = 7 27 RESOURCE_LIST = 8 28 FULL_RESOURCE_DESCRIPTOR = 9 29 RESOURCE_REQUIREMENTS_LIST = 10 30 QWORD = 11 31 ) 32 33 var ( 34 // ErrShortBuffer is returned when the buffer was too short for the operation. 35 ErrShortBuffer = syscall.ERROR_MORE_DATA 36 37 // ErrNotExist is returned when a registry key or value does not exist. 38 ErrNotExist = syscall.ERROR_FILE_NOT_FOUND 39 40 // ErrUnexpectedType is returned by Get*Value when the value's type was unexpected. 41 ErrUnexpectedType = errors.New("unexpected key value type") 42 ) 43 44 // GetValue retrieves the type and data for the specified value associated 45 // with an open key k. It fills up buffer buf and returns the retrieved 46 // byte count n. If buf is too small to fit the stored value it returns 47 // ErrShortBuffer error along with the required buffer size n. 48 // If no buffer is provided, it returns true and actual buffer size n. 49 // If no buffer is provided, GetValue returns the value's type only. 50 // If the value does not exist, the error returned is ErrNotExist. 51 // 52 // GetValue is a low level function. If value's type is known, use the appropriate 53 // Get*Value function instead. 54 func (k Key) GetValue(name string, buf []byte) (n int, valtype uint32, err error) { 55 pname, err := syscall.UTF16PtrFromString(name) 56 if err != nil { 57 return 0, 0, err 58 } 59 var pbuf *byte 60 if len(buf) > 0 { 61 pbuf = (*byte)(unsafe.Pointer(&buf[0])) 62 } 63 l := uint32(len(buf)) 64 err = syscall.RegQueryValueEx(syscall.Handle(k), pname, nil, &valtype, pbuf, &l) 65 if err != nil { 66 return int(l), valtype, err 67 } 68 return int(l), valtype, nil 69 } 70 71 func (k Key) getValue(name string, buf []byte) (date []byte, valtype uint32, err error) { 72 p, err := syscall.UTF16PtrFromString(name) 73 if err != nil { 74 return nil, 0, err 75 } 76 var t uint32 77 n := uint32(len(buf)) 78 for { 79 err = syscall.RegQueryValueEx(syscall.Handle(k), p, nil, &t, (*byte)(unsafe.Pointer(&buf[0])), &n) 80 if err == nil { 81 return buf[:n], t, nil 82 } 83 if err != syscall.ERROR_MORE_DATA { 84 return nil, 0, err 85 } 86 if n <= uint32(len(buf)) { 87 return nil, 0, err 88 } 89 buf = make([]byte, n) 90 } 91 } 92 93 // GetStringValue retrieves the string value for the specified 94 // value name associated with an open key k. It also returns the value's type. 95 // If value does not exist, GetStringValue returns ErrNotExist. 96 // If value is not SZ or EXPAND_SZ, it will return the correct value 97 // type and ErrUnexpectedType. 98 func (k Key) GetStringValue(name string) (val string, valtype uint32, err error) { 99 data, typ, err2 := k.getValue(name, make([]byte, 64)) 100 if err2 != nil { 101 return "", typ, err2 102 } 103 switch typ { 104 case SZ, EXPAND_SZ: 105 default: 106 return "", typ, ErrUnexpectedType 107 } 108 if len(data) == 0 { 109 return "", typ, nil 110 } 111 u := (*[1 << 10]uint16)(unsafe.Pointer(&data[0]))[:] 112 return syscall.UTF16ToString(u), typ, nil 113 } 114 115 // GetMUIStringValue retrieves the localized string value for 116 // the specified value name associated with an open key k. 117 // If the value name doesn't exist or the localized string value 118 // can't be resolved, GetMUIStringValue returns ErrNotExist. 119 // GetMUIStringValue panics if the system doesn't support 120 // regLoadMUIString; use LoadRegLoadMUIString to check if 121 // regLoadMUIString is supported before calling this function. 122 func (k Key) GetMUIStringValue(name string) (string, error) { 123 pname, err := syscall.UTF16PtrFromString(name) 124 if err != nil { 125 return "", err 126 } 127 128 buf := make([]uint16, 1024) 129 var buflen uint32 130 var pdir *uint16 131 132 err = regLoadMUIString(syscall.Handle(k), pname, &buf[0], uint32(len(buf)), &buflen, 0, pdir) 133 if err == syscall.ERROR_FILE_NOT_FOUND { // Try fallback path 134 var s string 135 s, err = ExpandString("%SystemRoot%\\system32\\") 136 if err != nil { 137 return "", err 138 } 139 pdir, err = syscall.UTF16PtrFromString(s) 140 if err != nil { 141 return "", err 142 } 143 144 err = regLoadMUIString(syscall.Handle(k), pname, &buf[0], uint32(len(buf)), &buflen, 0, pdir) 145 } 146 147 for err == syscall.ERROR_MORE_DATA { // Grow buffer if needed 148 if buflen <= uint32(len(buf)) { 149 break // Buffer not growing, assume race; break 150 } 151 buf = make([]uint16, buflen) 152 err = regLoadMUIString(syscall.Handle(k), pname, &buf[0], uint32(len(buf)), &buflen, 0, pdir) 153 } 154 155 if err != nil { 156 return "", err 157 } 158 159 return syscall.UTF16ToString(buf), nil 160 } 161 162 // ExpandString expands environment-variable strings and replaces 163 // them with the values defined for the current user. 164 // Use ExpandString to expand EXPAND_SZ strings. 165 func ExpandString(value string) (string, error) { 166 if value == "" { 167 return "", nil 168 } 169 p, err := syscall.UTF16PtrFromString(value) 170 if err != nil { 171 return "", err 172 } 173 r := make([]uint16, 100) 174 for { 175 n, err := expandEnvironmentStrings(p, &r[0], uint32(len(r))) 176 if err != nil { 177 return "", err 178 } 179 if n <= uint32(len(r)) { 180 u := (*[1 << 15]uint16)(unsafe.Pointer(&r[0]))[:] 181 return syscall.UTF16ToString(u), nil 182 } 183 r = make([]uint16, n) 184 } 185 } 186 187 // GetStringsValue retrieves the []string value for the specified 188 // value name associated with an open key k. It also returns the value's type. 189 // If value does not exist, GetStringsValue returns ErrNotExist. 190 // If value is not MULTI_SZ, it will return the correct value 191 // type and ErrUnexpectedType. 192 func (k Key) GetStringsValue(name string) (val []string, valtype uint32, err error) { 193 data, typ, err2 := k.getValue(name, make([]byte, 64)) 194 if err2 != nil { 195 return nil, typ, err2 196 } 197 if typ != MULTI_SZ { 198 return nil, typ, ErrUnexpectedType 199 } 200 if len(data) == 0 { 201 return nil, typ, nil 202 } 203 p := (*[1 << 24]uint16)(unsafe.Pointer(&data[0]))[:len(data)/2] 204 if len(p) == 0 { 205 return nil, typ, nil 206 } 207 if p[len(p)-1] == 0 { 208 p = p[:len(p)-1] // remove terminating null 209 } 210 val = make([]string, 0, 5) 211 from := 0 212 for i, c := range p { 213 if c == 0 { 214 val = append(val, string(utf16.Decode(p[from:i]))) 215 from = i + 1 216 } 217 } 218 return val, typ, nil 219 } 220 221 // GetIntegerValue retrieves the integer value for the specified 222 // value name associated with an open key k. It also returns the value's type. 223 // If value does not exist, GetIntegerValue returns ErrNotExist. 224 // If value is not DWORD or QWORD, it will return the correct value 225 // type and ErrUnexpectedType. 226 func (k Key) GetIntegerValue(name string) (val uint64, valtype uint32, err error) { 227 data, typ, err2 := k.getValue(name, make([]byte, 8)) 228 if err2 != nil { 229 return 0, typ, err2 230 } 231 switch typ { 232 case DWORD: 233 if len(data) != 4 { 234 return 0, typ, errors.New("DWORD value is not 4 bytes long") 235 } 236 return uint64(*(*uint32)(unsafe.Pointer(&data[0]))), DWORD, nil 237 case QWORD: 238 if len(data) != 8 { 239 return 0, typ, errors.New("QWORD value is not 8 bytes long") 240 } 241 return uint64(*(*uint64)(unsafe.Pointer(&data[0]))), QWORD, nil 242 default: 243 return 0, typ, ErrUnexpectedType 244 } 245 } 246 247 // GetBinaryValue retrieves the binary value for the specified 248 // value name associated with an open key k. It also returns the value's type. 249 // If value does not exist, GetBinaryValue returns ErrNotExist. 250 // If value is not BINARY, it will return the correct value 251 // type and ErrUnexpectedType. 252 func (k Key) GetBinaryValue(name string) (val []byte, valtype uint32, err error) { 253 data, typ, err2 := k.getValue(name, make([]byte, 64)) 254 if err2 != nil { 255 return nil, typ, err2 256 } 257 if typ != BINARY { 258 return nil, typ, ErrUnexpectedType 259 } 260 return data, typ, nil 261 } 262 263 func (k Key) setValue(name string, valtype uint32, data []byte) error { 264 p, err := syscall.UTF16PtrFromString(name) 265 if err != nil { 266 return err 267 } 268 if len(data) == 0 { 269 return regSetValueEx(syscall.Handle(k), p, 0, valtype, nil, 0) 270 } 271 return regSetValueEx(syscall.Handle(k), p, 0, valtype, &data[0], uint32(len(data))) 272 } 273 274 // SetDWordValue sets the data and type of a name value 275 // under key k to value and DWORD. 276 func (k Key) SetDWordValue(name string, value uint32) error { 277 return k.setValue(name, DWORD, (*[4]byte)(unsafe.Pointer(&value))[:]) 278 } 279 280 // SetQWordValue sets the data and type of a name value 281 // under key k to value and QWORD. 282 func (k Key) SetQWordValue(name string, value uint64) error { 283 return k.setValue(name, QWORD, (*[8]byte)(unsafe.Pointer(&value))[:]) 284 } 285 286 func (k Key) setStringValue(name string, valtype uint32, value string) error { 287 v, err := syscall.UTF16FromString(value) 288 if err != nil { 289 return err 290 } 291 buf := (*[1 << 10]byte)(unsafe.Pointer(&v[0]))[:len(v)*2] 292 return k.setValue(name, valtype, buf) 293 } 294 295 // SetStringValue sets the data and type of a name value 296 // under key k to value and SZ. The value must not contain a zero byte. 297 func (k Key) SetStringValue(name, value string) error { 298 return k.setStringValue(name, SZ, value) 299 } 300 301 // SetExpandStringValue sets the data and type of a name value 302 // under key k to value and EXPAND_SZ. The value must not contain a zero byte. 303 func (k Key) SetExpandStringValue(name, value string) error { 304 return k.setStringValue(name, EXPAND_SZ, value) 305 } 306 307 // SetStringsValue sets the data and type of a name value 308 // under key k to value and MULTI_SZ. The value strings 309 // must not contain a zero byte. 310 func (k Key) SetStringsValue(name string, value []string) error { 311 ss := "" 312 for _, s := range value { 313 for i := 0; i < len(s); i++ { 314 if s[i] == 0 { 315 return errors.New("string cannot have 0 inside") 316 } 317 } 318 ss += s + "\x00" 319 } 320 v := utf16.Encode([]rune(ss + "\x00")) 321 buf := (*[1 << 10]byte)(unsafe.Pointer(&v[0]))[:len(v)*2] 322 return k.setValue(name, MULTI_SZ, buf) 323 } 324 325 // SetBinaryValue sets the data and type of a name value 326 // under key k to value and BINARY. 327 func (k Key) SetBinaryValue(name string, value []byte) error { 328 return k.setValue(name, BINARY, value) 329 } 330 331 // DeleteValue removes a named value from the key k. 332 func (k Key) DeleteValue(name string) error { 333 return regDeleteValue(syscall.Handle(k), syscall.StringToUTF16Ptr(name)) 334 } 335 336 // ReadValueNames returns the value names of key k. 337 // The parameter n controls the number of returned names, 338 // analogous to the way os.File.Readdirnames works. 339 func (k Key) ReadValueNames(n int) ([]string, error) { 340 ki, err := k.Stat() 341 if err != nil { 342 return nil, err 343 } 344 names := make([]string, 0, ki.ValueCount) 345 buf := make([]uint16, ki.MaxValueNameLen+1) // extra room for terminating null character 346 loopItems: 347 for i := uint32(0); ; i++ { 348 if n > 0 { 349 if len(names) == n { 350 return names, nil 351 } 352 } 353 l := uint32(len(buf)) 354 for { 355 err := regEnumValue(syscall.Handle(k), i, &buf[0], &l, nil, nil, nil, nil) 356 if err == nil { 357 break 358 } 359 if err == syscall.ERROR_MORE_DATA { 360 // Double buffer size and try again. 361 l = uint32(2 * len(buf)) 362 buf = make([]uint16, l) 363 continue 364 } 365 if err == _ERROR_NO_MORE_ITEMS { 366 break loopItems 367 } 368 return names, err 369 } 370 names = append(names, syscall.UTF16ToString(buf[:l])) 371 } 372 if n > len(names) { 373 return names, io.EOF 374 } 375 return names, nil 376 }