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