1 /* Copyright (C) 1992, 1995-2002, 2005-2021 Free Software Foundation, Inc.
2    This file is part of the GNU C Library.
3 
4    This program is free software: you can redistribute it and/or modify
5    it under the terms of the GNU Lesser 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.
8 
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 Lesser General Public License for more details.
13 
14    You should have received a copy of the GNU Lesser General Public License
15    along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
16 
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)
20 
21 #include <config.h>
22 
23 /* Specification.  */
24 #include <stdlib.h>
25 
26 #include <errno.h>
27 #if !_LIBC
28 # define __set_errno(ev) ((errno) = (ev))
29 #endif
30 
31 #include <string.h>
32 #include <unistd.h>
33 
34 #if !_LIBC
35 # define __environ      environ
36 #endif
37 
38 #if _LIBC
39 /* This lock protects against simultaneous modifications of 'environ'.  */
40 # include <bits/libc-lock.h>
__libc_lock_define_initialized(static,envlock)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
48 
49 /* In the GNU C library we must keep the namespace clean.  */
50 #ifdef _LIBC
51 # define unsetenv __unsetenv
52 #endif
53 
54 #if _LIBC || !HAVE_UNSETENV
55 
56 int
57 unsetenv (const char *name)
58 {
59   size_t len;
60   char **ep;
61 
62   if (name == NULL || *name == '\0' || strchr (name, '=') != NULL)
63     {
64       __set_errno (EINVAL);
65       return -1;
66     }
67 
68   len = strlen (name);
69 
70   LOCK;
71 
72   ep = __environ;
73   while (*ep != NULL)
74     if (!strncmp (*ep, name, len) && (*ep)[len] == '=')
75       {
76         /* Found it.  Remove this pointer by moving later ones back.  */
77         char **dp = ep;
78 
79         do
80           dp[0] = dp[1];
81         while (*dp++);
82         /* Continue the loop in case NAME appears again.  */
83       }
84     else
85       ++ep;
86 
87   UNLOCK;
88 
89   return 0;
90 }
91 
92 #ifdef _LIBC
93 # undef unsetenv
94 weak_alias (__unsetenv, unsetenv)
95 #endif
96 
97 #else /* HAVE_UNSETENV */
98 
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
107 
108 /* Call the underlying unsetenv, in case there is hidden bookkeeping
109    that needs updating beyond just modifying environ.  */
110 int
111 rpl_unsetenv (const char *name)
112 {
113   int result = 0;
114   if (!name || !*name || strchr (name, '='))
115     {
116       errno = EINVAL;
117       return -1;
118     }
119   while (getenv (name))
120 # if !VOID_UNSETENV
121     result =
122 # endif
123       unsetenv (name);
124   return result;
125 }
126 
127 #endif /* HAVE_UNSETENV */
128