env_windows.odin 1.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. //+private
  2. package os2
  3. import win32 "core:sys/windows"
  4. import "base:runtime"
  5. _lookup_env :: proc(key: string, allocator: runtime.Allocator) -> (value: string, found: bool) {
  6. if key == "" {
  7. return
  8. }
  9. wkey := win32.utf8_to_wstring(key)
  10. n := win32.GetEnvironmentVariableW(wkey, nil, 0)
  11. if n == 0 {
  12. err := win32.GetLastError()
  13. if err == win32.ERROR_ENVVAR_NOT_FOUND {
  14. return "", false
  15. }
  16. return "", true
  17. }
  18. _TEMP_ALLOCATOR_GUARD()
  19. b := make([]u16, n+1, _temp_allocator())
  20. n = win32.GetEnvironmentVariableW(wkey, raw_data(b), u32(len(b)))
  21. if n == 0 {
  22. err := win32.GetLastError()
  23. if err == win32.ERROR_ENVVAR_NOT_FOUND {
  24. return "", false
  25. }
  26. return "", false
  27. }
  28. value = win32.utf16_to_utf8(b[:n], allocator) or_else ""
  29. found = true
  30. return
  31. }
  32. _set_env :: proc(key, value: string) -> bool {
  33. k := win32.utf8_to_wstring(key)
  34. v := win32.utf8_to_wstring(value)
  35. return bool(win32.SetEnvironmentVariableW(k, v))
  36. }
  37. _unset_env :: proc(key: string) -> bool {
  38. k := win32.utf8_to_wstring(key)
  39. return bool(win32.SetEnvironmentVariableW(k, nil))
  40. }
  41. _clear_env :: proc() {
  42. _TEMP_ALLOCATOR_GUARD()
  43. envs := environ(_temp_allocator())
  44. for env in envs {
  45. for j in 1..<len(env) {
  46. if env[j] == '=' {
  47. unset_env(env[0:j])
  48. break
  49. }
  50. }
  51. }
  52. }
  53. _environ :: proc(allocator: runtime.Allocator) -> []string {
  54. envs := win32.GetEnvironmentStringsW()
  55. if envs == nil {
  56. return nil
  57. }
  58. defer win32.FreeEnvironmentStringsW(envs)
  59. n := 0
  60. for from, i, p := 0, 0, envs; true; i += 1 {
  61. c := ([^]u16)(p)[i]
  62. if c == 0 {
  63. if i <= from {
  64. break
  65. }
  66. n += 1
  67. from = i + 1
  68. }
  69. }
  70. r := make([dynamic]string, 0, n, allocator)
  71. for from, i, p := 0, 0, envs; true; i += 1 {
  72. c := ([^]u16)(p)[i]
  73. if c == 0 {
  74. if i <= from {
  75. break
  76. }
  77. w := ([^]u16)(p)[from:i]
  78. append(&r, win32.utf16_to_utf8(w, allocator) or_else "")
  79. from = i + 1
  80. }
  81. }
  82. return r[:]
  83. }