| /* |
| * Windows regedit.exe registry editor implementation. |
| * |
| * Copyright 2002 Andriy Palamarchuk |
| * |
| * 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 <ctype.h> |
| #include <stdio.h> |
| #include <windows.h> |
| #include "regproc.h" |
| |
| static const char *usage = |
| "Usage:\n" |
| " regedit filename\n" |
| " regedit /E filename [regpath]\n" |
| " regedit /D regpath\n" |
| "\n" |
| "filename - registry file name\n" |
| "regpath - name of the registry key\n" |
| "\n" |
| "When called without any switches, adds the content of the specified\n" |
| "file to the registry\n" |
| "\n" |
| "Switches:\n" |
| " /E - exports contents of the specified registry key to the specified\n" |
| " file. Exports the whole registry if no key is specified.\n" |
| " /D - deletes specified registry key\n" |
| " /S - silent execution, can be used with any other switch.\n" |
| " Default. The only existing mode, exists for compatibility with Windows regedit.\n" |
| " /V - advanced mode, can be used with any other switch.\n" |
| " Ignored, exists for compatibility with Windows regedit.\n" |
| " /L - location of system.dat file. Can be used with any other switch.\n" |
| " Ignored. Exists for compatibility with Windows regedit.\n" |
| " /R - location of user.dat file. Can be used with any other switch.\n" |
| " Ignored. Exists for compatibility with Windows regedit.\n" |
| " /? - print this help. Any other switches are ignored.\n" |
| " /C - create registry from file. Not implemented.\n" |
| "\n" |
| "The switches are case-insensitive, can be prefixed either by '-' or '/'.\n" |
| "This program is command-line compatible with Microsoft Windows\n" |
| "regedit.\n"; |
| |
| typedef enum { |
| ACTION_UNDEF, ACTION_ADD, ACTION_EXPORT, ACTION_DELETE |
| } REGEDIT_ACTION; |
| |
| |
| const CHAR *getAppName(void) |
| { |
| return "regedit"; |
| } |
| |
| /****************************************************************************** |
| * Copies file name from command line string to the buffer. |
| * Rewinds the command line string pointer to the next non-space character |
| * after the file name. |
| * Buffer contains an empty string if no filename was found; |
| * |
| * params: |
| * command_line - command line current position pointer |
| * where *s[0] is the first symbol of the file name. |
| * file_name - buffer to write the file name to. |
| */ |
| static void get_file_name(CHAR **command_line, CHAR *file_name) |
| { |
| CHAR *s = *command_line; |
| int pos = 0; /* position of pointer "s" in *command_line */ |
| file_name[0] = 0; |
| |
| if (!s[0]) { |
| return; |
| } |
| |
| if (s[0] == '"') { |
| s++; |
| (*command_line)++; |
| while(s[0] != '"') { |
| if (!s[0]) { |
| fprintf(stderr,"%s: Unexpected end of file name!\n", |
| getAppName()); |
| exit(1); |
| } |
| s++; |
| pos++; |
| } |
| } else { |
| while(s[0] && !isspace(s[0])) { |
| s++; |
| pos++; |
| } |
| } |
| memcpy(file_name, *command_line, pos * sizeof((*command_line)[0])); |
| /* remove the last backslash */ |
| if (file_name[pos - 1] == '\\') { |
| file_name[pos - 1] = '\0'; |
| } else { |
| file_name[pos] = '\0'; |
| } |
| |
| if (s[0]) { |
| s++; |
| pos++; |
| } |
| while(s[0] && isspace(s[0])) { |
| s++; |
| pos++; |
| } |
| (*command_line) += pos; |
| } |
| |
| static BOOL PerformRegAction(REGEDIT_ACTION action, LPSTR s) |
| { |
| switch (action) { |
| case ACTION_ADD: { |
| CHAR filename[MAX_PATH]; |
| FILE *reg_file; |
| |
| get_file_name(&s, filename); |
| if (!filename[0]) { |
| fprintf(stderr,"%s: No file name was specified\n", getAppName()); |
| fprintf(stderr,usage); |
| exit(1); |
| } |
| |
| while(filename[0]) { |
| char* realname = NULL; |
| |
| if (strcmp(filename, "-") == 0) |
| { |
| reg_file=stdin; |
| } |
| else |
| { |
| int size; |
| |
| size=SearchPath(NULL, filename, NULL,0, NULL, NULL); |
| if (size>0) |
| { |
| realname=HeapAlloc(GetProcessHeap(), 0, size); |
| size=SearchPath(NULL, filename, NULL, size, realname, NULL); |
| } |
| if (size==0) |
| { |
| fprintf(stderr, "%s: File not found \"%s\" (%d)\n", |
| getAppName(), filename, GetLastError()); |
| exit(1); |
| } |
| reg_file = fopen(realname, "r"); |
| if (reg_file==NULL) |
| { |
| perror(""); |
| fprintf(stderr,"%s: Can't open file \"%s\"\n", getAppName(), filename); |
| exit(1); |
| } |
| } |
| import_registry_file(reg_file); |
| if (realname) |
| { |
| HeapFree(GetProcessHeap(),0,realname); |
| fclose(reg_file); |
| } |
| get_file_name(&s, filename); |
| } |
| break; |
| } |
| case ACTION_DELETE: { |
| CHAR reg_key_name[KEY_MAX_LEN]; |
| |
| get_file_name(&s, reg_key_name); |
| if (!reg_key_name[0]) { |
| fprintf(stderr,"%s: No registry key was specified for removal\n", |
| getAppName()); |
| fprintf(stderr,usage); |
| exit(1); |
| } else |
| { |
| WCHAR* reg_key_nameW = GetWideString(reg_key_name); |
| delete_registry_key(reg_key_nameW); |
| HeapFree(GetProcessHeap(), 0, reg_key_nameW); |
| } |
| break; |
| } |
| case ACTION_EXPORT: { |
| CHAR filename[MAX_PATH]; |
| WCHAR* filenameW; |
| |
| filename[0] = '\0'; |
| get_file_name(&s, filename); |
| if (!filename[0]) { |
| fprintf(stderr,"%s: No file name was specified\n", getAppName()); |
| fprintf(stderr,usage); |
| exit(1); |
| } |
| |
| filenameW = GetWideString(filename); |
| if (s[0]) { |
| CHAR reg_key_name[KEY_MAX_LEN]; |
| WCHAR* reg_key_nameW; |
| |
| get_file_name(&s, reg_key_name); |
| reg_key_nameW = GetWideString(reg_key_name); |
| export_registry_key(filenameW, reg_key_nameW, REG_FORMAT_4); |
| HeapFree(GetProcessHeap(), 0, reg_key_nameW); |
| } else { |
| export_registry_key(filenameW, NULL, REG_FORMAT_4); |
| } |
| HeapFree(GetProcessHeap(), 0, filenameW); |
| break; |
| } |
| default: |
| fprintf(stderr,"%s: Unhandled action!\n", getAppName()); |
| exit(1); |
| break; |
| } |
| return TRUE; |
| } |
| |
| /** |
| * Process unknown switch. |
| * |
| * Params: |
| * chu - the switch character in upper-case. |
| * s - the command line string where s points to the switch character. |
| */ |
| static void error_unknown_switch(char chu, char *s) |
| { |
| if (isalpha(chu)) { |
| fprintf(stderr,"%s: Undefined switch /%c!\n", getAppName(), chu); |
| } else { |
| fprintf(stderr,"%s: Alphabetic character is expected after '%c' " |
| "in switch specification\n", getAppName(), *(s - 1)); |
| } |
| exit(1); |
| } |
| |
| BOOL ProcessCmdLine(LPSTR lpCmdLine) |
| { |
| REGEDIT_ACTION action = ACTION_UNDEF; |
| LPSTR s = lpCmdLine; /* command line pointer */ |
| CHAR ch = *s; /* current character */ |
| |
| while (ch && ((ch == '-') || (ch == '/'))) { |
| char chu; |
| char ch2; |
| |
| s++; |
| ch = *s; |
| if (!ch || isspace(ch)) |
| { |
| /* '-' is a file name. It indicates we should use stdin */ |
| s--; |
| break; |
| } |
| ch2 = *(s+1); |
| chu = toupper(ch); |
| if (!ch2 || isspace(ch2)) { |
| if (chu == 'S' || chu == 'V') { |
| /* ignore these switches */ |
| } else { |
| switch (chu) { |
| case 'D': |
| action = ACTION_DELETE; |
| break; |
| case 'E': |
| action = ACTION_EXPORT; |
| break; |
| case '?': |
| fprintf(stderr,usage); |
| exit(0); |
| break; |
| default: |
| error_unknown_switch(chu, s); |
| break; |
| } |
| } |
| s++; |
| } else { |
| if (ch2 == ':') { |
| switch (chu) { |
| case 'L': |
| /* fall through */ |
| case 'R': |
| s += 2; |
| while (*s && !isspace(*s)) { |
| s++; |
| } |
| break; |
| default: |
| error_unknown_switch(chu, s); |
| break; |
| } |
| } else { |
| /* this is a file name, starting from '/' */ |
| s--; |
| break; |
| } |
| } |
| /* skip spaces to the next parameter */ |
| ch = *s; |
| while (ch && isspace(ch)) { |
| s++; |
| ch = *s; |
| } |
| } |
| |
| if (*s && action == ACTION_UNDEF) |
| action = ACTION_ADD; |
| |
| if (action == ACTION_UNDEF) |
| return FALSE; |
| |
| return PerformRegAction(action, s); |
| } |