forked from kirillkovalenko/nssm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
env.cpp
259 lines (219 loc) · 7.06 KB
/
env.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
#include "nssm.h"
/*
Environment block is of the form:
KEY1=VALUE1 NULL
KEY2=VALUE2 NULL
NULL
A single variable KEY=VALUE has length 15:
KEY=VALUE (13) NULL (1)
NULL (1)
Environment variable names are case-insensitive!
*/
/* Find the length in characters of an environment block. */
size_t environment_length(TCHAR *env) {
size_t len = 0;
TCHAR *s;
for (s = env; ; s++) {
len++;
if (*s == _T('\0')) {
if (*(s + 1) == _T('\0')) {
len++;
break;
}
}
}
return len;
}
/* Copy an environment block. */
TCHAR *copy_environment_block(TCHAR *env) {
TCHAR *newenv;
if (copy_double_null(env, (unsigned long) environment_length(env), &newenv)) return 0;
return newenv;
}
/*
The environment block starts with variables of the form
=C:=C:\Windows\System32 which we ignore.
*/
TCHAR *useful_environment(TCHAR *rawenv) {
TCHAR *env = rawenv;
if (env) {
while (*env == _T('=')) {
for ( ; *env; env++);
env++;
}
}
return env;
}
/* Expand an environment variable. Must call HeapFree() on the result. */
TCHAR *expand_environment_string(TCHAR *string) {
unsigned long len;
len = ExpandEnvironmentStrings(string, 0, 0);
if (! len) {
log_event(EVENTLOG_ERROR_TYPE, NSSM_EVENT_EXPANDENVIRONMENTSTRINGS_FAILED, string, error_string(GetLastError()), 0);
return 0;
}
TCHAR *ret = (TCHAR *) HeapAlloc(GetProcessHeap(), 0, len * sizeof(TCHAR));
if (! ret) {
log_event(EVENTLOG_ERROR_TYPE, NSSM_EVENT_OUT_OF_MEMORY, _T("ExpandEnvironmentStrings()"), _T("expand_environment_string"), 0);
return 0;
}
if (! ExpandEnvironmentStrings(string, ret, len)) {
log_event(EVENTLOG_ERROR_TYPE, NSSM_EVENT_EXPANDENVIRONMENTSTRINGS_FAILED, string, error_string(GetLastError()), 0);
HeapFree(GetProcessHeap(), 0, ret);
return 0;
}
return ret;
}
/*
Set all the environment variables from an environment block in the current
environment or remove all the variables in the block from the current
environment.
*/
static int set_environment_block(TCHAR *env, bool set) {
int ret = 0;
TCHAR *s, *t;
for (s = env; *s; s++) {
for (t = s; *t && *t != _T('='); t++);
if (*t == _T('=')) {
*t = _T('\0');
if (set) {
TCHAR *expanded = expand_environment_string(++t);
if (expanded) {
if (! SetEnvironmentVariable(s, expanded)) ret++;
HeapFree(GetProcessHeap(), 0, expanded);
}
else {
if (! SetEnvironmentVariable(s, t)) ret++;
}
}
else {
if (! SetEnvironmentVariable(s, NULL)) ret++;
}
for (t++; *t; t++);
}
s = t;
}
return ret;
}
int set_environment_block(TCHAR *env) {
return set_environment_block(env, true);
}
static int unset_environment_block(TCHAR *env) {
return set_environment_block(env, false);
}
/* Remove all variables from the process environment. */
int clear_environment() {
TCHAR *rawenv = GetEnvironmentStrings();
TCHAR *env = useful_environment(rawenv);
int ret = unset_environment_block(env);
if (rawenv) FreeEnvironmentStrings(rawenv);
return ret;
}
/* Set the current environment to exactly duplicate an environment block. */
int duplicate_environment(TCHAR *rawenv) {
int ret = clear_environment();
TCHAR *env = useful_environment(rawenv);
ret += set_environment_block(env);
return ret;
}
/*
Verify an environment block.
Returns: 1 if environment is invalid.
0 if environment is OK.
-1 on error.
*/
int test_environment(TCHAR *env) {
TCHAR *path = (TCHAR *) nssm_imagepath();
STARTUPINFO si;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
PROCESS_INFORMATION pi;
ZeroMemory(&pi, sizeof(pi));
unsigned long flags = CREATE_SUSPENDED;
#ifdef UNICODE
flags |= CREATE_UNICODE_ENVIRONMENT;
#endif
/*
Try to relaunch ourselves but with the candidate environment set.
Assuming no solar flare activity, the only reason this would fail is if
the environment were invalid.
*/
if (CreateProcess(0, path, 0, 0, 0, flags, env, 0, &si, &pi)) {
TerminateProcess(pi.hProcess, 0);
}
else {
unsigned long error = GetLastError();
if (error == ERROR_INVALID_PARAMETER) return 1;
else return -1;
}
return 0;
}
/*
Duplicate an environment block returned by GetEnvironmentStrings().
Since such a block is by definition readonly, and duplicate_environment()
modifies its inputs, this function takes a copy of the input and operates
on that.
*/
void duplicate_environment_strings(TCHAR *env) {
TCHAR *newenv = copy_environment_block(env);
if (! newenv) return;
duplicate_environment(newenv);
HeapFree(GetProcessHeap(), 0, newenv);
}
/* Safely get a copy of the current environment. */
TCHAR *copy_environment() {
TCHAR *rawenv = GetEnvironmentStrings();
if (! rawenv) return NULL;
TCHAR *env = copy_environment_block(rawenv);
FreeEnvironmentStrings(rawenv);
return env;
}
/*
Create a new block with all the strings of the first block plus a new string.
If the key is already present its value will be overwritten in place.
If the key is blank or empty the new block will still be allocated and have
non-zero length.
*/
int append_to_environment_block(TCHAR *env, unsigned long envlen, TCHAR *string, TCHAR **newenv, unsigned long *newlen) {
size_t keylen = 0;
if (string && string[0]) {
for (; string[keylen]; keylen++) {
if (string[keylen] == _T('=')) {
keylen++;
break;
}
}
}
return append_to_double_null(env, envlen, newenv, newlen, string, keylen, false);
}
/*
Create a new block with all the strings of the first block minus the given
string.
If the key is not present the new block will be a copy of the original.
If the string is KEY=VALUE the key will only be removed if its value is
VALUE.
If the string is just KEY the key will unconditionally be removed.
If removing the string results in an empty list the new block will still be
allocated and have non-zero length.
*/
int remove_from_environment_block(TCHAR *env, unsigned long envlen, TCHAR *string, TCHAR **newenv, unsigned long *newlen) {
if (! string || ! string[0] || string[0] == _T('=')) return 1;
TCHAR *key = 0;
size_t len = _tcslen(string);
size_t i;
for (i = 0; i < len; i++) if (string[i] == _T('=')) break;
/* Rewrite KEY to KEY= but leave KEY=VALUE alone. */
size_t keylen = len;
if (i == len) keylen++;
key = (TCHAR *) HeapAlloc(GetProcessHeap(), 0, (keylen + 1) * sizeof(TCHAR));
if (! key) {
log_event(EVENTLOG_ERROR_TYPE, NSSM_EVENT_OUT_OF_MEMORY, _T("key"), _T("remove_from_environment_block()"), 0);
return 2;
}
memmove(key, string, len * sizeof(TCHAR));
if (keylen > len) key[keylen - 1] = _T('=');
key[keylen] = _T('\0');
int ret = remove_from_double_null(env, envlen, newenv, newlen, key, keylen, false);
HeapFree(GetProcessHeap(), 0, key);
return ret;
}