debian/rules: Use `git' potty wrapper.
[qmail] / env.c
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(i) 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(s,len) 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(s) 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(s) 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(s) 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(s,t) 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 }