Reactos
1/*
2 * Copyright 2017 Hugh McMaster
3 *
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
8 *
9 * This library 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 GNU
12 * Lesser General Public License for more details.
13 *
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
17 */
18
19#include <stdio.h>
20#include "reg.h"
21
22static void write_file(HANDLE hFile, const WCHAR *str)
23{
24 DWORD written;
25
26 WriteFile(hFile, str, lstrlenW(str) * sizeof(WCHAR), &written, NULL);
27}
28
29static WCHAR *escape_string(WCHAR *str, size_t str_len, size_t *line_len)
30{
31 size_t i, escape_count, pos;
32 WCHAR *buf;
33
34 for (i = 0, escape_count = 0; i < str_len; i++)
35 {
36 WCHAR c = str[i];
37
38 if (!c) break;
39
40 if (c == '\r' || c == '\n' || c == '\\' || c == '"')
41 escape_count++;
42 }
43
44 buf = malloc((str_len + escape_count + 1) * sizeof(WCHAR));
45
46 for (i = 0, pos = 0; i < str_len; i++, pos++)
47 {
48 WCHAR c = str[i];
49
50 if (!c) break;
51
52 switch (c)
53 {
54 case '\r':
55 buf[pos++] = '\\';
56 buf[pos] = 'r';
57 break;
58 case '\n':
59 buf[pos++] = '\\';
60 buf[pos] = 'n';
61 break;
62 case '\\':
63 buf[pos++] = '\\';
64 buf[pos] = '\\';
65 break;
66 case '"':
67 buf[pos++] = '\\';
68 buf[pos] = '"';
69 break;
70 default:
71 buf[pos] = c;
72 }
73 }
74
75 buf[pos] = 0;
76 *line_len = pos;
77 return buf;
78}
79
80static size_t export_value_name(HANDLE hFile, WCHAR *name, size_t len)
81{
82 static const WCHAR *default_name = L"@=";
83 size_t line_len;
84
85 if (name && *name)
86 {
87 WCHAR *str = escape_string(name, len, &line_len);
88 WCHAR *buf = malloc((line_len + 4) * sizeof(WCHAR));
89 line_len = swprintf(buf, L"\"%s\"=", str);
90 write_file(hFile, buf);
91 free(buf);
92 free(str);
93 }
94 else
95 {
96 line_len = lstrlenW(default_name);
97 write_file(hFile, default_name);
98 }
99
100 return line_len;
101}
102
103static void export_string_data(WCHAR **buf, WCHAR *data, size_t size)
104{
105 size_t len = 0, line_len;
106 WCHAR *str;
107
108 if (size)
109 len = size / sizeof(WCHAR) - 1;
110 str = escape_string(data, len, &line_len);
111 *buf = malloc((line_len + 3) * sizeof(WCHAR));
112 swprintf(*buf, L"\"%s\"", str);
113 free(str);
114}
115
116static void export_dword_data(WCHAR **buf, DWORD *data)
117{
118 *buf = malloc(15 * sizeof(WCHAR));
119 swprintf(*buf, L"dword:%08x", *data);
120}
121
122static size_t export_hex_data_type(HANDLE hFile, DWORD type)
123{
124 static const WCHAR *hex = L"hex:";
125 size_t line_len;
126
127 if (type == REG_BINARY)
128 {
129 line_len = lstrlenW(hex);
130 write_file(hFile, hex);
131 }
132 else
133 {
134 WCHAR *buf = malloc(15 * sizeof(WCHAR));
135 line_len = swprintf(buf, L"hex(%x):", type);
136 write_file(hFile, buf);
137 free(buf);
138 }
139
140 return line_len;
141}
142
143#define MAX_HEX_CHARS 77
144
145static void export_hex_data(HANDLE hFile, WCHAR **buf, DWORD type,
146 DWORD line_len, void *data, DWORD size)
147{
148 size_t num_commas, i, pos;
149
150 line_len += export_hex_data_type(hFile, type);
151
152 if (!size) return;
153
154 num_commas = size - 1;
155 *buf = malloc(size * 3 * sizeof(WCHAR));
156
157 for (i = 0, pos = 0; i < size; i++)
158 {
159 pos += swprintf(*buf + pos, L"%02x", ((BYTE *)data)[i]);
160 if (i == num_commas) break;
161 (*buf)[pos++] = ',';
162 (*buf)[pos] = 0;
163 line_len += 3;
164
165 if (line_len >= MAX_HEX_CHARS)
166 {
167 write_file(hFile, *buf);
168 write_file(hFile, L"\\\r\n ");
169 line_len = 2;
170 pos = 0;
171 }
172 }
173}
174
175static void export_newline(HANDLE hFile)
176{
177 static const WCHAR *newline = L"\r\n";
178
179 write_file(hFile, newline);
180}
181
182static void export_data(HANDLE hFile, WCHAR *value_name, DWORD value_len,
183 DWORD type, void *data, size_t size)
184{
185 WCHAR *buf = NULL;
186 size_t line_len = export_value_name(hFile, value_name, value_len);
187
188 switch (type)
189 {
190 case REG_SZ:
191 export_string_data(&buf, data, size);
192 break;
193 case REG_DWORD:
194 if (size)
195 {
196 export_dword_data(&buf, data);
197 break;
198 }
199 /* fall through */
200 case REG_NONE:
201 case REG_EXPAND_SZ:
202 case REG_BINARY:
203 case REG_MULTI_SZ:
204 default:
205 export_hex_data(hFile, &buf, type, line_len, data, size);
206 break;
207 }
208
209 if (size || type == REG_SZ)
210 {
211 write_file(hFile, buf);
212 free(buf);
213 }
214
215 export_newline(hFile);
216}
217
218static void export_key_name(HANDLE hFile, WCHAR *name)
219{
220 WCHAR *buf;
221
222 buf = malloc((lstrlenW(name) + 7) * sizeof(WCHAR));
223 swprintf(buf, L"\r\n[%s]\r\n", name);
224 write_file(hFile, buf);
225 free(buf);
226}
227
228static int export_registry_data(HANDLE hFile, HKEY hkey, WCHAR *path, REGSAM sam)
229{
230 LONG rc;
231 DWORD max_value_len = 256, value_len;
232 DWORD max_data_bytes = 2048, data_size;
233 DWORD subkey_len;
234 DWORD i, type, path_len;
235 WCHAR *value_name, *subkey_name, *subkey_path;
236 BYTE *data;
237 HKEY subkey;
238
239 export_key_name(hFile, path);
240
241 value_name = malloc(max_value_len * sizeof(WCHAR));
242 data = malloc(max_data_bytes);
243
244 i = 0;
245 for (;;)
246 {
247 value_len = max_value_len;
248 data_size = max_data_bytes;
249 rc = RegEnumValueW(hkey, i, value_name, &value_len, NULL, &type, data, &data_size);
250
251 if (rc == ERROR_SUCCESS)
252 {
253 export_data(hFile, value_name, value_len, type, data, data_size);
254 i++;
255 }
256 else if (rc == ERROR_MORE_DATA)
257 {
258 if (data_size > max_data_bytes)
259 {
260 max_data_bytes = data_size;
261 data = realloc(data, max_data_bytes);
262 }
263 else
264 {
265 max_value_len *= 2;
266 value_name = realloc(value_name, max_value_len * sizeof(WCHAR));
267 }
268 }
269 else break;
270 }
271
272 free(data);
273 free(value_name);
274
275 subkey_name = malloc(MAX_SUBKEY_LEN * sizeof(WCHAR));
276
277 path_len = lstrlenW(path);
278
279 i = 0;
280 for (;;)
281 {
282 subkey_len = MAX_SUBKEY_LEN;
283 rc = RegEnumKeyExW(hkey, i, subkey_name, &subkey_len, NULL, NULL, NULL, NULL);
284 if (rc == ERROR_SUCCESS)
285 {
286 subkey_path = build_subkey_path(path, path_len, subkey_name, subkey_len);
287 if (!RegOpenKeyExW(hkey, subkey_name, 0, KEY_READ|sam, &subkey))
288 {
289 export_registry_data(hFile, subkey, subkey_path, sam);
290 RegCloseKey(subkey);
291 }
292 free(subkey_path);
293 i++;
294 }
295 else break;
296 }
297
298 free(subkey_name);
299 return 0;
300}
301
302static void export_file_header(HANDLE hFile)
303{
304 static const WCHAR header[] = L"\xFEFFWindows Registry Editor Version 5.00\r\n";
305
306 write_file(hFile, header);
307}
308
309static HANDLE create_file(const WCHAR *filename, DWORD action)
310{
311 return CreateFileW(filename, GENERIC_WRITE, 0, NULL, action, FILE_ATTRIBUTE_NORMAL, NULL);
312}
313
314static HANDLE get_file_handle(WCHAR *filename, BOOL overwrite_file)
315{
316 HANDLE hFile = create_file(filename, overwrite_file ? CREATE_ALWAYS : CREATE_NEW);
317
318 if (hFile == INVALID_HANDLE_VALUE)
319 {
320 DWORD error = GetLastError();
321
322 if (error == ERROR_FILE_EXISTS)
323 {
324 if (!ask_confirm(STRING_OVERWRITE_FILE, filename))
325 {
326 output_message(STRING_CANCELLED);
327 exit(0);
328 }
329
330 hFile = create_file(filename, CREATE_ALWAYS);
331 }
332 else
333 {
334 WCHAR *str;
335
336 FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
337 FORMAT_MESSAGE_IGNORE_INSERTS, NULL, error, 0, (WCHAR *)&str, 0, NULL);
338 output_writeconsole(str, lstrlenW(str));
339 LocalFree(str);
340 exit(1);
341 }
342 }
343
344 return hFile;
345}
346
347int reg_export(int argc, WCHAR *argvW[])
348{
349 HKEY root, hkey;
350 WCHAR *path, *key_name;
351 BOOL overwrite_file = FALSE;
352 REGSAM sam = 0;
353 HANDLE hFile;
354 int i, ret;
355
356 if (argc < 4) goto invalid;
357
358 if (!parse_registry_key(argvW[2], &root, &path))
359 return 1;
360
361 for (i = 4; i < argc; i++)
362 {
363 WCHAR *str;
364
365 if (argvW[i][0] != '/' && argvW[i][0] != '-')
366 goto invalid;
367
368 str = &argvW[i][1];
369
370 if (is_char(*str, 'y') && !str[1])
371 overwrite_file = TRUE;
372 else if (!lstrcmpiW(str, L"reg:32"))
373 {
374 if (sam & KEY_WOW64_32KEY) goto invalid;
375 sam |= KEY_WOW64_32KEY;
376 continue;
377 }
378 else if (!lstrcmpiW(str, L"reg:64"))
379 {
380 if (sam & KEY_WOW64_64KEY) goto invalid;
381 sam |= KEY_WOW64_64KEY;
382 continue;
383 }
384 else
385 goto invalid;
386 }
387
388 if (sam == (KEY_WOW64_32KEY|KEY_WOW64_64KEY))
389 goto invalid;
390
391 if (RegOpenKeyExW(root, path, 0, KEY_READ|sam, &hkey))
392 {
393 output_message(STRING_KEY_NONEXIST);
394 return 1;
395 }
396
397 key_name = get_long_key(root, path);
398
399 hFile = get_file_handle(argvW[3], overwrite_file);
400 export_file_header(hFile);
401 ret = export_registry_data(hFile, hkey, key_name, sam);
402 export_newline(hFile);
403 CloseHandle(hFile);
404
405 RegCloseKey(hkey);
406
407 return ret;
408
409invalid:
410 output_message(STRING_INVALID_SYNTAX);
411 output_message(STRING_FUNC_HELP, _wcsupr(argvW[1]));
412 return 1;
413}