| /* | 
 |  * 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); | 
 | } |