From 50d211883e35fb186a049e6a24daa3dac48e384a Mon Sep 17 00:00:00 2001 From: rockcavera Date: Wed, 5 Jan 2022 11:04:23 -0300 Subject: [PATCH 1/5] Update osenv.nim --- lib/pure/includes/osenv.nim | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/lib/pure/includes/osenv.nim b/lib/pure/includes/osenv.nim index 1a01ab9bcd3e3..1c7b9a8a97251 100644 --- a/lib/pure/includes/osenv.nim +++ b/lib/pure/includes/osenv.nim @@ -45,7 +45,7 @@ when not defined(nimscript): proc c_getenv(env: cstring): cstring {. importc: "getenv", header: "".} when defined(windows): - proc c_putenv_s(envname: cstring, envval: cstring): cint {.importc: "_putenv_s", header: "".} + proc c_putenv(envstring: cstring): cint {.importc: "_putenv", header: "".} from std/private/win_setenv import setEnvImpl else: proc c_setenv(envname: cstring, envval: cstring, overwrite: cint): cint {.importc: "setenv", header: "".} @@ -121,7 +121,8 @@ when not defined(nimscript): ]# if key.len == 0 or '=' in key: raise newException(OSError, "invalid key, got: " & key) - if c_putenv_s(key, "") != 0'i32: bail + let envToDel: cstring = key & "=" + if c_putenv(envToDel) != 0'i32: bail else: if c_unsetenv(key) != 0'i32: bail From 987c5e00451852d064381922a8574b03c1bf5295 Mon Sep 17 00:00:00 2001 From: rockcavera Date: Wed, 5 Jan 2022 11:04:49 -0300 Subject: [PATCH 2/5] Update win_setenv.nim --- lib/std/private/win_setenv.nim | 42 ++++++++++++++++++---------------- 1 file changed, 22 insertions(+), 20 deletions(-) diff --git a/lib/std/private/win_setenv.nim b/lib/std/private/win_setenv.nim index 91c3f15b0b4c1..1f310d1f2e2f2 100644 --- a/lib/std/private/win_setenv.nim +++ b/lib/std/private/win_setenv.nim @@ -30,44 +30,46 @@ else: # same as winlean.setEnvironmentVariableA proc c_getenv(env: cstring): cstring {.importc: "getenv", header: "".} - proc c_putenv_s(envname: cstring, envval: cstring): cint {.importc: "_putenv_s", header: "".} + proc c_putenv(envstring: cstring): cint {.importc: "_putenv", header: "".} proc c_wgetenv(varname: ptr wchar_t): ptr wchar_t {.importc: "_wgetenv", header: "".} var errno {.importc, header: "".}: cint - var gWenviron {.importc:"_wenviron".}: ptr ptr wchar_t + var gWenviron {.importc: "_wenviron".}: ptr ptr wchar_t # xxx `ptr UncheckedArray[WideCString]` did not work - proc mbstowcs_s(pReturnValue: ptr csize_t, wcstr: ptr wchar_t, sizeInWords: csize_t, mbstr: cstring, count: csize_t): cint {.importc: "mbstowcs_s", header: "".} + proc mbstowcs(wcstr: ptr wchar_t, mbstr: cstring, count: csize_t): csize_t {.importc: "mbstowcs", header: "".} # xxx cint vs errno_t? - proc setEnvImpl*(name: cstring, value: cstring, overwrite: cint): cint = + proc setEnvImpl*(name: string, value: string, overwrite: cint): cint = const EINVAL = cint(22) - const MAX_ENV = 32767 - # xxx get it from: `var MAX_ENV {.importc: "_MAX_ENV", header:"".}: cint` - if overwrite == 0 and c_getenv(name) != nil: return 0 - if value[0] != '\0': - let e = c_putenv_s(name, value) + let cname = cstring(name) + if overwrite == 0 and c_getenv(cname) != nil: return 0 + if value != "": + let cenvstring = cstring(name & "=" & value) + let e = c_putenv(cenvstring) if e != 0: - errno = e + errno = EINVAL return -1 return 0 #[ - We are trying to set the value to an empty string, but `_putenv_s` deletes + We are trying to set the value to an empty string, but `_putenv` deletes entries if the value is an empty string, and just calling SetEnvironmentVariableA doesn't update `_environ`, so we have to do these terrible things. ]# - if c_putenv_s(name, " ") != 0: + let cenvstring = cstring(name & "= ") + if c_putenv(cenvstring) != 0: errno = EINVAL return -1 # Here lies the documentation we blatently ignore to make this work. - var s = c_getenv(name) + var s = c_getenv(cname) s[0] = '\0' #[ This would result in a double null termination, which normally signifies the end of the environment variable list, so we stick a completely empty environment variable into the list instead. ]# + s = c_getenv(cname) s[1] = '=' #[ If gWenviron is null, the wide environment has not been initialized @@ -77,19 +79,19 @@ else: ]# if gWenviron != nil: # var buf: array[MAX_ENV + 1, WideCString] - var buf: array[MAX_ENV + 1, Utf16Char] + let requiredSize = mbstowcs(nil, cname, 0).int + var buf = newSeq[Utf16Char](requiredSize + 1) let buf2 = cast[ptr wchar_t](buf[0].addr) - var len: csize_t - if mbstowcs_s(len.addr, buf2, buf.len.csize_t, name, MAX_ENV) != 0: + if mbstowcs(buf2, cname, csize_t(requiredSize + 1)) == csize_t(high(uint)): errno = EINVAL return -1 - let ptrToEnv = cast[WideCString](c_wgetenv(buf2)) + var ptrToEnv = cast[WideCString](c_wgetenv(buf2)) ptrToEnv[0] = '\0'.Utf16Char - let ptrToEnv2 = cast[WideCString](c_wgetenv(buf2)) - ptrToEnv2[1] = '='.Utf16Char + ptrToEnv = cast[WideCString](c_wgetenv(buf2)) + ptrToEnv[1] = '='.Utf16Char # And now, we have to update the outer environment to have a proper empty value. - if setEnvironmentVariableA(name, value) == 0: + if setEnvironmentVariableA(cname, value) == 0: errno = EINVAL return -1 return 0 From 5b4eaccae274fc98726be9ff259d4edb5725403d Mon Sep 17 00:00:00 2001 From: Andreas Rumpf Date: Thu, 6 Jan 2022 12:15:04 +0100 Subject: [PATCH 3/5] Update lib/pure/includes/osenv.nim --- lib/pure/includes/osenv.nim | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/pure/includes/osenv.nim b/lib/pure/includes/osenv.nim index 1c7b9a8a97251..9963efa2269a1 100644 --- a/lib/pure/includes/osenv.nim +++ b/lib/pure/includes/osenv.nim @@ -121,7 +121,7 @@ when not defined(nimscript): ]# if key.len == 0 or '=' in key: raise newException(OSError, "invalid key, got: " & key) - let envToDel: cstring = key & "=" + let envToDel = key & "=" if c_putenv(envToDel) != 0'i32: bail else: if c_unsetenv(key) != 0'i32: bail From cac08c1a23ec3376f78dc8d713cfde4817f1f30a Mon Sep 17 00:00:00 2001 From: Andreas Rumpf Date: Thu, 6 Jan 2022 12:15:08 +0100 Subject: [PATCH 4/5] Update lib/pure/includes/osenv.nim --- lib/pure/includes/osenv.nim | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/pure/includes/osenv.nim b/lib/pure/includes/osenv.nim index 9963efa2269a1..4a776eb78756c 100644 --- a/lib/pure/includes/osenv.nim +++ b/lib/pure/includes/osenv.nim @@ -122,7 +122,7 @@ when not defined(nimscript): if key.len == 0 or '=' in key: raise newException(OSError, "invalid key, got: " & key) let envToDel = key & "=" - if c_putenv(envToDel) != 0'i32: bail + if c_putenv(cstring envToDel) != 0'i32: bail else: if c_unsetenv(key) != 0'i32: bail From 717d77a0321abd431a2686ed81cab1fc51c5d154 Mon Sep 17 00:00:00 2001 From: rockcavera Date: Thu, 6 Jan 2022 10:00:17 -0300 Subject: [PATCH 5/5] fixing cstring --- lib/std/private/win_setenv.nim | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/lib/std/private/win_setenv.nim b/lib/std/private/win_setenv.nim index 1f310d1f2e2f2..0dfe0ed46dc08 100644 --- a/lib/std/private/win_setenv.nim +++ b/lib/std/private/win_setenv.nim @@ -42,11 +42,10 @@ else: proc setEnvImpl*(name: string, value: string, overwrite: cint): cint = const EINVAL = cint(22) - let cname = cstring(name) - if overwrite == 0 and c_getenv(cname) != nil: return 0 + if overwrite == 0 and c_getenv(cstring(name)) != nil: return 0 if value != "": - let cenvstring = cstring(name & "=" & value) - let e = c_putenv(cenvstring) + let envstring = name & "=" & value + let e = c_putenv(cstring(envstring)) if e != 0: errno = EINVAL return -1 @@ -57,19 +56,19 @@ else: SetEnvironmentVariableA doesn't update `_environ`, so we have to do these terrible things. ]# - let cenvstring = cstring(name & "= ") - if c_putenv(cenvstring) != 0: + let envstring = name & "= " + if c_putenv(cstring(envstring)) != 0: errno = EINVAL return -1 # Here lies the documentation we blatently ignore to make this work. - var s = c_getenv(cname) + var s = c_getenv(cstring(name)) s[0] = '\0' #[ This would result in a double null termination, which normally signifies the end of the environment variable list, so we stick a completely empty environment variable into the list instead. ]# - s = c_getenv(cname) + s = c_getenv(cstring(name)) s[1] = '=' #[ If gWenviron is null, the wide environment has not been initialized @@ -79,10 +78,10 @@ else: ]# if gWenviron != nil: # var buf: array[MAX_ENV + 1, WideCString] - let requiredSize = mbstowcs(nil, cname, 0).int + let requiredSize = mbstowcs(nil, cstring(name), 0).int var buf = newSeq[Utf16Char](requiredSize + 1) let buf2 = cast[ptr wchar_t](buf[0].addr) - if mbstowcs(buf2, cname, csize_t(requiredSize + 1)) == csize_t(high(uint)): + if mbstowcs(buf2, cstring(name), csize_t(requiredSize + 1)) == csize_t(high(uint)): errno = EINVAL return -1 var ptrToEnv = cast[WideCString](c_wgetenv(buf2)) @@ -91,7 +90,7 @@ else: ptrToEnv[1] = '='.Utf16Char # And now, we have to update the outer environment to have a proper empty value. - if setEnvironmentVariableA(cname, value) == 0: + if setEnvironmentVariableA(cstring(name), cstring(value)) == 0: errno = EINVAL return -1 return 0