kernel32/tests: Use todo_wine_if() in tests.
[wine.git] / programs / regedit / regedit.c
blobacff7010719b676e25070aad5f11925e63c8b9d3
1 /*
2 * Windows regedit.exe registry editor implementation.
4 * Copyright 2002 Andriy Palamarchuk
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include <ctype.h>
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <windows.h>
25 #include "regproc.h"
27 static const char *usage =
28 "Usage:\n"
29 " regedit filename\n"
30 " regedit /E filename [regpath]\n"
31 " regedit /D regpath\n"
32 "\n"
33 "filename - registry file name\n"
34 "regpath - name of the registry key\n"
35 "\n"
36 "When called without any switches, adds the content of the specified\n"
37 "file to the registry\n"
38 "\n"
39 "Switches:\n"
40 " /E - exports contents of the specified registry key to the specified\n"
41 " file. Exports the whole registry if no key is specified.\n"
42 " /D - deletes specified registry key\n"
43 " /S - silent execution, can be used with any other switch.\n"
44 " Default. The only existing mode, exists for compatibility with Windows regedit.\n"
45 " /V - advanced mode, can be used with any other switch.\n"
46 " Ignored, exists for compatibility with Windows regedit.\n"
47 " /L - location of system.dat file. Can be used with any other switch.\n"
48 " Ignored. Exists for compatibility with Windows regedit.\n"
49 " /R - location of user.dat file. Can be used with any other switch.\n"
50 " Ignored. Exists for compatibility with Windows regedit.\n"
51 " /? - print this help. Any other switches are ignored.\n"
52 " /C - create registry from file. Not implemented.\n"
53 "\n"
54 "The switches are case-insensitive, can be prefixed either by '-' or '/'.\n"
55 "This program is command-line compatible with Microsoft Windows\n"
56 "regedit.\n";
58 typedef enum {
59 ACTION_UNDEF, ACTION_ADD, ACTION_EXPORT, ACTION_DELETE
60 } REGEDIT_ACTION;
63 const CHAR *getAppName(void)
65 return "regedit";
68 /******************************************************************************
69 * Copies file name from command line string to the buffer.
70 * Rewinds the command line string pointer to the next non-space character
71 * after the file name.
72 * Buffer contains an empty string if no filename was found;
74 * params:
75 * command_line - command line current position pointer
76 * where *s[0] is the first symbol of the file name.
77 * file_name - buffer to write the file name to.
79 static void get_file_name(CHAR **command_line, CHAR *file_name)
81 CHAR *s = *command_line;
82 int pos = 0; /* position of pointer "s" in *command_line */
83 file_name[0] = 0;
85 if (!s[0]) {
86 return;
89 if (s[0] == '"') {
90 s++;
91 (*command_line)++;
92 while(s[0] != '"') {
93 if (!s[0]) {
94 fprintf(stderr,"%s: Unexpected end of file name!\n",
95 getAppName());
96 exit(1);
98 s++;
99 pos++;
101 } else {
102 while(s[0] && !isspace(s[0])) {
103 s++;
104 pos++;
107 memcpy(file_name, *command_line, pos * sizeof((*command_line)[0]));
108 /* remove the last backslash */
109 if (file_name[pos - 1] == '\\') {
110 file_name[pos - 1] = '\0';
111 } else {
112 file_name[pos] = '\0';
115 if (s[0]) {
116 s++;
117 pos++;
119 while(s[0] && isspace(s[0])) {
120 s++;
121 pos++;
123 (*command_line) += pos;
126 static BOOL PerformRegAction(REGEDIT_ACTION action, LPSTR s)
128 switch (action) {
129 case ACTION_ADD: {
130 CHAR filename[MAX_PATH];
131 FILE *reg_file;
133 get_file_name(&s, filename);
134 if (!filename[0]) {
135 fprintf(stderr,"%s: No file name was specified\n", getAppName());
136 fprintf(stderr,usage);
137 exit(1);
140 while(filename[0]) {
141 char* realname = NULL;
143 if (strcmp(filename, "-") == 0)
145 reg_file = stdin;
147 else
149 int size;
151 size = SearchPathA(NULL, filename, NULL, 0, NULL, NULL);
152 if (size > 0)
154 realname = HeapAlloc(GetProcessHeap(), 0, size);
155 size = SearchPathA(NULL, filename, NULL, size, realname, NULL);
157 if (size == 0)
159 fprintf(stderr, "%s: File not found \"%s\" (%d)\n",
160 getAppName(), filename, GetLastError());
161 exit(1);
163 reg_file = fopen(realname, "rb");
164 if (reg_file == NULL)
166 perror("");
167 fprintf(stderr, "%s: Can't open file \"%s\"\n", getAppName(), filename);
168 exit(1);
171 import_registry_file(reg_file);
172 if (realname)
174 HeapFree(GetProcessHeap(),0,realname);
175 fclose(reg_file);
177 get_file_name(&s, filename);
179 break;
181 case ACTION_DELETE: {
182 CHAR reg_key_name[KEY_MAX_LEN];
184 get_file_name(&s, reg_key_name);
185 if (!reg_key_name[0]) {
186 fprintf(stderr,"%s: No registry key was specified for removal\n",
187 getAppName());
188 fprintf(stderr,usage);
189 exit(1);
190 } else
192 WCHAR* reg_key_nameW = GetWideString(reg_key_name);
193 delete_registry_key(reg_key_nameW);
194 HeapFree(GetProcessHeap(), 0, reg_key_nameW);
196 break;
198 case ACTION_EXPORT: {
199 CHAR filename[MAX_PATH];
200 WCHAR* filenameW;
202 filename[0] = '\0';
203 get_file_name(&s, filename);
204 if (!filename[0]) {
205 fprintf(stderr,"%s: No file name was specified\n", getAppName());
206 fprintf(stderr,usage);
207 exit(1);
210 filenameW = GetWideString(filename);
211 if (s[0]) {
212 CHAR reg_key_name[KEY_MAX_LEN];
213 WCHAR* reg_key_nameW;
215 get_file_name(&s, reg_key_name);
216 reg_key_nameW = GetWideString(reg_key_name);
217 export_registry_key(filenameW, reg_key_nameW, REG_FORMAT_4);
218 HeapFree(GetProcessHeap(), 0, reg_key_nameW);
219 } else {
220 export_registry_key(filenameW, NULL, REG_FORMAT_4);
222 HeapFree(GetProcessHeap(), 0, filenameW);
223 break;
225 default:
226 fprintf(stderr,"%s: Unhandled action!\n", getAppName());
227 exit(1);
228 break;
230 return TRUE;
234 * Process unknown switch.
236 * Params:
237 * chu - the switch character in upper-case.
238 * s - the command line string where s points to the switch character.
240 static void error_unknown_switch(char chu, char *s)
242 if (isalpha(chu)) {
243 fprintf(stderr,"%s: Undefined switch /%c!\n", getAppName(), chu);
244 } else {
245 fprintf(stderr,"%s: Alphabetic character is expected after '%c' "
246 "in switch specification\n", getAppName(), *(s - 1));
248 exit(1);
251 BOOL ProcessCmdLine(LPSTR lpCmdLine)
253 REGEDIT_ACTION action = ACTION_UNDEF;
254 LPSTR s = lpCmdLine; /* command line pointer */
255 CHAR ch = *s; /* current character */
257 while (ch && ((ch == '-') || (ch == '/'))) {
258 char chu;
259 char ch2;
261 s++;
262 ch = *s;
263 if (!ch || isspace(ch))
265 /* '-' is a file name. It indicates we should use stdin */
266 s--;
267 break;
269 ch2 = *(s+1);
270 chu = toupper(ch);
271 if (!ch2 || isspace(ch2)) {
272 if (chu == 'S' || chu == 'V') {
273 /* ignore these switches */
274 } else {
275 switch (chu) {
276 case 'D':
277 action = ACTION_DELETE;
278 break;
279 case 'E':
280 action = ACTION_EXPORT;
281 break;
282 case '?':
283 fprintf(stderr,usage);
284 exit(0);
285 break;
286 default:
287 error_unknown_switch(chu, s);
288 break;
291 s++;
292 } else {
293 if (ch2 == ':') {
294 switch (chu) {
295 case 'L':
296 /* fall through */
297 case 'R':
298 s += 2;
299 while (*s && !isspace(*s)) {
300 s++;
302 break;
303 default:
304 error_unknown_switch(chu, s);
305 break;
307 } else {
308 /* this is a file name, starting from '/' */
309 s--;
310 break;
313 /* skip spaces to the next parameter */
314 ch = *s;
315 while (ch && isspace(ch)) {
316 s++;
317 ch = *s;
321 if (*s && action == ACTION_UNDEF)
322 action = ACTION_ADD;
324 if (action == ACTION_UNDEF)
325 return FALSE;
327 return PerformRegAction(action, s);