s/qmail  3.3.23
Next generation secure email transport
env.c
Go to the documentation of this file.
1 /* env.c, envread.c, env.h: environ library
2 Daniel J. Bernstein, djb@silverton.berkeley.edu.
3 Depends on str.h, alloc.h.
4 Requires environ.
5 19960113: rewrite. warning: interface is different.
6 No known patent problems.
7 */
8 
9 #include "str.h"
10 #include "alloc.h"
11 #include "env.h"
12 
13 int env_isinit = 0; /* if env_isinit: */
14 static int ea; /* environ is a pointer to ea+1 char*'s. */
15 static int en; /* the first en of those are ALLOCATED. environ[en] is 0. */
16 
17 static void env_goodbye(int i)
18 {
19  alloc_free(environ[i]);
20  environ[i] = environ[--en];
21  environ[en] = 0;
22 }
23 
24 static char *null = 0;
25 
26 void env_clear()
27 {
28  if (env_isinit) while (en) env_goodbye(0);
29  else environ = &null;
30 }
31 
32 static void env_unsetlen(char *s, int len)
33 {
34  int i;
35  for (i = en - 1;i >= 0;--i)
36  if (!str_diffn(s,environ[i],len))
37  if (environ[i][len] == '=')
38  env_goodbye(i);
39 }
40 
41 int env_unset(char *s)
42 {
43  if (!env_isinit) if (!env_init()) return 0;
44  env_unsetlen(s,str_len(s));
45  return 1;
46 }
47 
48 static int env_add(char *s)
49 {
50  char *t;
51  t = env_findeq(s);
52  if (t) env_unsetlen(s,t - s);
53  if (en == ea)
54  {
55  ea += 30;
56  if (!alloc_re(&environ,(en + 1) * sizeof(char *),(ea + 1) * sizeof(char *)))
57  { ea = en; return 0; }
58  }
59  environ[en++] = s;
60  environ[en] = 0;
61  return 1;
62 }
63 
64 int env_put(char *s)
65 {
66  char *u;
67  if (!env_isinit) if (!env_init()) return 0;
68  u = alloc(str_len(s) + 1);
69  if (!u) return 0;
70  str_copy(u,s);
71  if (!env_add(u)) { alloc_free(u); return 0; }
72  return 1;
73 }
74 
75 int env_put2(char *s, char *t)
76 {
77  char *u;
78  int slen;
79  if (!env_isinit) if (!env_init()) return 0;
80  slen = str_len(s);
81  u = alloc(slen + str_len(t) + 2);
82  if (!u) return 0;
83  str_copy(u,s);
84  u[slen] = '=';
85  str_copy(u + slen + 1,t);
86  if (!env_add(u)) { alloc_free(u); return 0; }
87  return 1;
88 }
89 
90 int env_init()
91 {
92  char **newenviron;
93  int i;
94  for (en = 0;environ[en];++en) ;
95  ea = en + 10;
96  newenviron = (char **) alloc((ea + 1) * sizeof(char *));
97  if (!newenviron) return 0;
98  for (en = 0;environ[en];++en)
99  {
100  newenviron[en] = alloc(str_len(environ[en]) + 1);
101  if (!newenviron[en])
102  {
103  for (i = 0;i < en;++i) alloc_free(newenviron[i]);
104  alloc_free(newenviron);
105  return 0;
106  }
107  str_copy(newenviron[en],environ[en]);
108  }
109  newenviron[en] = 0;
110  environ = newenviron;
111  env_isinit = 1;
112  return 1;
113 }
char * alloc(unsigned int n)
Definition: alloc.c:16
unsigned int str_len(char *)
unsigned int str_copy(char *, char *)
char ** environ
int str_diffn(char *, char *, unsigned int)
unsigned len
Definition: matchup.c:36
unsigned i
Definition: matchup.c:36
int env_unset(char *s)
Definition: env.c:41
int env_isinit
Definition: env.c:13
int env_put(char *s)
Definition: env.c:64
void env_clear()
Definition: env.c:26
char * env_findeq(char *)
Definition: envread.c:22
int env_put2(char *s, char *t)
Definition: env.c:75
void alloc_free(char *x)
Definition: alloc.c:26
int alloc_re(char **, unsigned int, unsigned int)
Definition: alloc_re.c:4
int env_init()
Definition: env.c:90
unsigned u
Definition: matchup.c:36