123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411 |
- /*
- * Copyright 2017 Hugh McMaster
- *
- * This library is free software; you can redistribute it and/or
- * modify it under the terms of the GNU Lesser General Public
- * License as published by the Free Software Foundation; either
- * version 2.1 of the License, or (at your option) any later version.
- *
- * This library is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser General Public
- * License along with this library; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
- */
- #include <windows.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <wine/heap.h>
- #include "reg.h"
- static void write_file(HANDLE hFile, const WCHAR *str)
- {
- DWORD written;
- WriteFile(hFile, str, lstrlenW(str) * sizeof(WCHAR), &written, NULL);
- }
- static WCHAR *escape_string(WCHAR *str, size_t str_len, size_t *line_len)
- {
- size_t i, escape_count, pos;
- WCHAR *buf;
- for (i = 0, escape_count = 0; i < str_len; i++)
- {
- WCHAR c = str[i];
- if (!c) break;
- if (c == '\r' || c == '\n' || c == '\\' || c == '"')
- escape_count++;
- }
- buf = heap_xalloc((str_len + escape_count + 1) * sizeof(WCHAR));
- for (i = 0, pos = 0; i < str_len; i++, pos++)
- {
- WCHAR c = str[i];
- if (!c) break;
- switch (c)
- {
- case '\r':
- buf[pos++] = '\\';
- buf[pos] = 'r';
- break;
- case '\n':
- buf[pos++] = '\\';
- buf[pos] = 'n';
- break;
- case '\\':
- buf[pos++] = '\\';
- buf[pos] = '\\';
- break;
- case '"':
- buf[pos++] = '\\';
- buf[pos] = '"';
- break;
- default:
- buf[pos] = c;
- }
- }
- buf[pos] = 0;
- *line_len = pos;
- return buf;
- }
- static size_t export_value_name(HANDLE hFile, WCHAR *name, size_t len)
- {
- static const WCHAR quoted_fmt[] = {'"','%','s','"','=',0};
- static const WCHAR default_name[] = {'@','=',0};
- size_t line_len;
- if (name && *name)
- {
- WCHAR *str = escape_string(name, len, &line_len);
- WCHAR *buf = heap_xalloc((line_len + 4) * sizeof(WCHAR));
- line_len = swprintf(buf, line_len + 4, quoted_fmt, str);
- write_file(hFile, buf);
- heap_free(buf);
- heap_free(str);
- }
- else
- {
- line_len = lstrlenW(default_name);
- write_file(hFile, default_name);
- }
- return line_len;
- }
- static void export_string_data(WCHAR **buf, WCHAR *data, size_t size)
- {
- size_t len = 0, line_len;
- WCHAR *str;
- static const WCHAR fmt[] = {'"','%','s','"',0};
- if (size)
- len = size / sizeof(WCHAR) - 1;
- str = escape_string(data, len, &line_len);
- *buf = heap_xalloc((line_len + 3) * sizeof(WCHAR));
- swprintf(*buf, line_len + 3, fmt, str);
- heap_free(str);
- }
- static void export_dword_data(WCHAR **buf, DWORD *data)
- {
- static const WCHAR fmt[] = {'d','w','o','r','d',':','%','0','8','x',0};
- *buf = heap_xalloc(15 * sizeof(WCHAR));
- swprintf(*buf, 15, fmt, *data);
- }
- static size_t export_hex_data_type(HANDLE hFile, DWORD type)
- {
- static const WCHAR hex[] = {'h','e','x',':',0};
- static const WCHAR hexp_fmt[] = {'h','e','x','(','%','x',')',':',0};
- size_t line_len;
- if (type == REG_BINARY)
- {
- line_len = lstrlenW(hex);
- write_file(hFile, hex);
- }
- else
- {
- WCHAR *buf = heap_xalloc(15 * sizeof(WCHAR));
- line_len = swprintf(buf, 15, hexp_fmt, type);
- write_file(hFile, buf);
- heap_free(buf);
- }
- return line_len;
- }
- #define MAX_HEX_CHARS 77
- static void export_hex_data(HANDLE hFile, WCHAR **buf, DWORD type,
- DWORD line_len, void *data, DWORD size)
- {
- static const WCHAR fmt[] = {'%','0','2','x',0};
- static const WCHAR hex_concat[] = {'\\','\r','\n',' ',' ',0};
- size_t num_commas, i, pos;
- line_len += export_hex_data_type(hFile, type);
- if (!size) return;
- num_commas = size - 1;
- *buf = heap_xalloc(size * 3 * sizeof(WCHAR));
- for (i = 0, pos = 0; i < size; i++)
- {
- pos += swprintf(*buf + pos, 3, fmt, ((BYTE *)data)[i]);
- if (i == num_commas) break;
- (*buf)[pos++] = ',';
- (*buf)[pos] = 0;
- line_len += 3;
- if (line_len >= MAX_HEX_CHARS)
- {
- write_file(hFile, *buf);
- write_file(hFile, hex_concat);
- line_len = 2;
- pos = 0;
- }
- }
- }
- static void export_newline(HANDLE hFile)
- {
- static const WCHAR newline[] = {'\r','\n',0};
- write_file(hFile, newline);
- }
- static void export_data(HANDLE hFile, WCHAR *value_name, DWORD value_len,
- DWORD type, void *data, size_t size)
- {
- WCHAR *buf = NULL;
- size_t line_len = export_value_name(hFile, value_name, value_len);
- switch (type)
- {
- case REG_SZ:
- export_string_data(&buf, data, size);
- break;
- case REG_DWORD:
- if (size)
- {
- export_dword_data(&buf, data);
- break;
- }
- /* fall through */
- case REG_NONE:
- case REG_EXPAND_SZ:
- case REG_BINARY:
- case REG_MULTI_SZ:
- default:
- export_hex_data(hFile, &buf, type, line_len, data, size);
- break;
- }
- if (size || type == REG_SZ)
- {
- write_file(hFile, buf);
- heap_free(buf);
- }
- export_newline(hFile);
- }
- static void export_key_name(HANDLE hFile, WCHAR *name)
- {
- static const WCHAR fmt[] = {'\r','\n','[','%','s',']','\r','\n',0};
- WCHAR *buf;
- buf = heap_xalloc((lstrlenW(name) + 7) * sizeof(WCHAR));
- swprintf(buf, lstrlenW(name) + 7, fmt, name);
- write_file(hFile, buf);
- heap_free(buf);
- }
- static int export_registry_data(HANDLE hFile, HKEY key, WCHAR *path)
- {
- LONG rc;
- DWORD max_value_len = 256, value_len;
- DWORD max_data_bytes = 2048, data_size;
- DWORD subkey_len;
- DWORD i, type, path_len;
- WCHAR *value_name, *subkey_name, *subkey_path;
- BYTE *data;
- HKEY subkey;
- export_key_name(hFile, path);
- value_name = heap_xalloc(max_value_len * sizeof(WCHAR));
- data = heap_xalloc(max_data_bytes);
- i = 0;
- for (;;)
- {
- value_len = max_value_len;
- data_size = max_data_bytes;
- rc = RegEnumValueW(key, i, value_name, &value_len, NULL, &type, data, &data_size);
- if (rc == ERROR_SUCCESS)
- {
- export_data(hFile, value_name, value_len, type, data, data_size);
- i++;
- }
- else if (rc == ERROR_MORE_DATA)
- {
- if (data_size > max_data_bytes)
- {
- max_data_bytes = data_size;
- data = heap_xrealloc(data, max_data_bytes);
- }
- else
- {
- max_value_len *= 2;
- value_name = heap_xrealloc(value_name, max_value_len * sizeof(WCHAR));
- }
- }
- else break;
- }
- heap_free(data);
- heap_free(value_name);
- subkey_name = heap_xalloc(MAX_SUBKEY_LEN * sizeof(WCHAR));
- path_len = lstrlenW(path);
- i = 0;
- for (;;)
- {
- subkey_len = MAX_SUBKEY_LEN;
- rc = RegEnumKeyExW(key, i, subkey_name, &subkey_len, NULL, NULL, NULL, NULL);
- if (rc == ERROR_SUCCESS)
- {
- subkey_path = build_subkey_path(path, path_len, subkey_name, subkey_len);
- if (!RegOpenKeyExW(key, subkey_name, 0, KEY_READ, &subkey))
- {
- export_registry_data(hFile, subkey, subkey_path);
- RegCloseKey(subkey);
- }
- heap_free(subkey_path);
- i++;
- }
- else break;
- }
- heap_free(subkey_name);
- return 0;
- }
- static void export_file_header(HANDLE hFile)
- {
- static const WCHAR header[] = { 0xfeff,'W','i','n','d','o','w','s',' ',
- 'R','e','g','i','s','t','r','y',' ','E','d','i','t','o','r',' ',
- 'V','e','r','s','i','o','n',' ','5','.','0','0','\r','\n',0};
- write_file(hFile, header);
- }
- static HANDLE create_file(const WCHAR *filename, DWORD action)
- {
- return CreateFileW(filename, GENERIC_WRITE, 0, NULL, action, FILE_ATTRIBUTE_NORMAL, NULL);
- }
- static HANDLE get_file_handle(WCHAR *filename, BOOL overwrite_file)
- {
- HANDLE hFile = create_file(filename, overwrite_file ? CREATE_ALWAYS : CREATE_NEW);
- if (hFile == INVALID_HANDLE_VALUE)
- {
- DWORD error = GetLastError();
- if (error == ERROR_FILE_EXISTS)
- {
- if (!ask_confirm(STRING_OVERWRITE_FILE, filename))
- {
- output_message(STRING_CANCELLED);
- exit(0);
- }
- hFile = create_file(filename, CREATE_ALWAYS);
- }
- else
- {
- WCHAR *str;
- FormatMessageW(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
- FORMAT_MESSAGE_IGNORE_INSERTS, NULL, error, 0, (WCHAR *)&str, 0, NULL);
- output_writeconsole(str, lstrlenW(str));
- LocalFree(str);
- exit(1);
- }
- }
- return hFile;
- }
- static BOOL is_overwrite_switch(const WCHAR *s)
- {
- if (lstrlenW(s) > 2)
- return FALSE;
- if ((s[0] == '/' || s[0] == '-') && (s[1] == 'y' || s[1] == 'Y'))
- return TRUE;
- return FALSE;
- }
- int reg_export(int argc, WCHAR *argv[])
- {
- HKEY root, hkey;
- WCHAR *path, *long_key;
- BOOL overwrite_file = FALSE;
- HANDLE hFile;
- int ret;
- if (argc == 3 || argc > 5)
- goto error;
- if (!parse_registry_key(argv[2], &root, &path, &long_key))
- return 1;
- if (argc == 5 && !(overwrite_file = is_overwrite_switch(argv[4])))
- goto error;
- if (RegOpenKeyExW(root, path, 0, KEY_READ, &hkey))
- {
- output_message(STRING_INVALID_KEY);
- return 1;
- }
- hFile = get_file_handle(argv[3], overwrite_file);
- export_file_header(hFile);
- ret = export_registry_data(hFile, hkey, long_key);
- export_newline(hFile);
- CloseHandle(hFile);
- RegCloseKey(hkey);
- return ret;
- error:
- output_message(STRING_INVALID_SYNTAX);
- output_message(STRING_FUNC_HELP, wcsupr(argv[1]));
- return 1;
- }
|