Win32: fix environment memory leaks
authorKarsten Blees <blees@dcon.de>
Thu, 17 Jul 2014 15:37:57 +0000 (17:37 +0200)
committerJunio C Hamano <gitster@pobox.com>
Mon, 21 Jul 2014 16:32:49 +0000 (09:32 -0700)
All functions that modify the environment have memory leaks.

Disable gitunsetenv in the Makefile and use env_setenv (via mingw_putenv)
instead (this frees removed environment entries).

Move xstrdup from env_setenv to make_augmented_environ, so that
mingw_putenv no longer copies the environment entries (according to POSIX
[1], "the string [...] shall become part of the environment"). This also
fixes the memory leak in gitsetenv, which expects a POSIX compliant putenv.

[1] http://pubs.opengroup.org/onlinepubs/009695399/functions/putenv.html

Note: This patch depends on taking control of char **environ and having
our own mingw_putenv (both introduced in "Win32: Unicode environment
(incoming)").

Signed-off-by: Karsten Blees <blees@dcon.de>
Signed-off-by: Stepan Kasal <kasal@ucw.cz>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
compat/mingw.c
compat/mingw.h
config.mak.uname
index cb0914af6c681f6d4c9514e41b42068dddc7db03..f6c7ad7fd834560432001d1dc46b81dfee99e2de 100644 (file)
@@ -1220,14 +1220,14 @@ static char **env_setenv(char **env, const char *name)
                        for (i = 0; env[i]; i++)
                                ;
                        env = xrealloc(env, (i+2)*sizeof(*env));
-                       env[i] = xstrdup(name);
+                       env[i] = (char*) name;
                        env[i+1] = NULL;
                }
        }
        else {
                free(env[i]);
                if (*eq)
-                       env[i] = xstrdup(name);
+                       env[i] = (char*) name;
                else
                        for (; env[i]; i++)
                                env[i] = env[i+1];
@@ -1242,8 +1242,10 @@ char **make_augmented_environ(const char *const *vars)
 {
        char **env = copy_environ();
 
-       while (*vars)
-               env = env_setenv(env, *vars++);
+       while (*vars) {
+               const char *v = *vars++;
+               env = env_setenv(env, strchr(v, '=') ? xstrdup(v) : v);
+       }
        return env;
 }
 
index ca80be1bbea5301f38a893179202073dc16e4aac..828d97760c7dd3e9394710d0da55cb4bdac0ac0c 100644 (file)
@@ -209,6 +209,7 @@ char *mingw_getenv(const char *name);
 #define getenv mingw_getenv
 int mingw_putenv(const char *namevalue);
 #define putenv mingw_putenv
+#define unsetenv mingw_putenv
 
 int mingw_gethostname(char *host, int namelen);
 #define gethostname mingw_gethostname
index 8131c81985d97c23d19c7ef489f0db988f6fbbc7..462988e8ab44f964e093707c9838ee84a7b86e11 100644 (file)
@@ -326,7 +326,6 @@ ifeq ($(uname_S),Windows)
        NO_IPV6 = YesPlease
        NO_UNIX_SOCKETS = YesPlease
        NO_SETENV = YesPlease
-       NO_UNSETENV = YesPlease
        NO_STRCASESTR = YesPlease
        NO_STRLCPY = YesPlease
        NO_MEMMEM = YesPlease
@@ -479,7 +478,6 @@ ifneq (,$(findstring MINGW,$(uname_S)))
        NO_SYMLINK_HEAD = YesPlease
        NO_UNIX_SOCKETS = YesPlease
        NO_SETENV = YesPlease
-       NO_UNSETENV = YesPlease
        NO_STRCASESTR = YesPlease
        NO_STRLCPY = YesPlease
        NO_MEMMEM = YesPlease