Bug#16045
[emacs.git] / lib / unsetenv.c
blobc58c82f4f44115d846117398696347f5e7f8771c
1 /* Copyright (C) 1992, 1995-2002, 2005-2013 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <http://www.gnu.org/licenses/>. */
17 /* Don't use __attribute__ __nonnull__ in this compilation unit. Otherwise gcc
18 optimizes away the name == NULL test below. */
19 #define _GL_ARG_NONNULL(params)
21 #include <config.h>
23 /* Specification. */
24 #include <stdlib.h>
26 #include <errno.h>
27 #if !_LIBC
28 # define __set_errno(ev) ((errno) = (ev))
29 #endif
31 #include <string.h>
32 #include <unistd.h>
34 #if !_LIBC
35 # define __environ environ
36 #endif
38 #if _LIBC
39 /* This lock protects against simultaneous modifications of 'environ'. */
40 # include <bits/libc-lock.h>
41 __libc_lock_define_initialized (static, envlock)
42 # define LOCK __libc_lock_lock (envlock)
43 # define UNLOCK __libc_lock_unlock (envlock)
44 #else
45 # define LOCK
46 # define UNLOCK
47 #endif
49 /* In the GNU C library we must keep the namespace clean. */
50 #ifdef _LIBC
51 # define unsetenv __unsetenv
52 #endif
54 #if _LIBC || !HAVE_UNSETENV
56 int
57 unsetenv (const char *name)
59 size_t len;
60 char **ep;
62 if (name == NULL || *name == '\0' || strchr (name, '=') != NULL)
64 __set_errno (EINVAL);
65 return -1;
68 len = strlen (name);
70 LOCK;
72 ep = __environ;
73 while (*ep != NULL)
74 if (!strncmp (*ep, name, len) && (*ep)[len] == '=')
76 /* Found it. Remove this pointer by moving later ones back. */
77 char **dp = ep;
80 dp[0] = dp[1];
81 while (*dp++);
82 /* Continue the loop in case NAME appears again. */
84 else
85 ++ep;
87 UNLOCK;
89 return 0;
92 #ifdef _LIBC
93 # undef unsetenv
94 weak_alias (__unsetenv, unsetenv)
95 #endif
97 #else /* HAVE_UNSETENV */
99 # undef unsetenv
100 # if !HAVE_DECL_UNSETENV
101 # if VOID_UNSETENV
102 extern void unsetenv (const char *);
103 # else
104 extern int unsetenv (const char *);
105 # endif
106 # endif
108 /* Call the underlying unsetenv, in case there is hidden bookkeeping
109 that needs updating beyond just modifying environ. */
111 rpl_unsetenv (const char *name)
113 int result = 0;
114 if (!name || !*name || strchr (name, '='))
116 errno = EINVAL;
117 return -1;
119 while (getenv (name))
120 # if !VOID_UNSETENV
121 result =
122 # endif
123 unsetenv (name);
124 return result;
127 #endif /* HAVE_UNSETENV */