/* * Registry Functions * * Copyright 1996 Marcus Meissner * Copyright 1998 Matthew Becker * Copyright 1999 Sylvain St-Germain * * December 21, 1997 - Kevin Cozens * Fixed bugs in the _w95_loadreg() function. Added extra information * regarding the format of the Windows '95 registry files. * * NOTES * When changing this file, please re-run the regtest program to ensure * the conditions are handled properly. * * TODO * Security access * Option handling * Time for RegEnumKey*, RegQueryInfoKey* */ #include "config.h" #include #include #include #include #include #include #include #include #include "winerror.h" #include "file.h" #include "heap.h" #include "debugtools.h" #include "options.h" #include "winreg.h" #include "server.h" #ifdef HAVE_SYS_MMAN_H # include #endif #include "winnt.h" DEFAULT_DEBUG_CHANNEL(reg); /* FIXME: following defines should be configured global */ #define SAVE_GLOBAL_REGBRANCH_USER_DEFAULT ETCDIR"/wine.userreg" #define SAVE_GLOBAL_REGBRANCH_LOCAL_MACHINE ETCDIR"/wine.systemreg" /* relative in ~user/.wine/ : */ #define SAVE_LOCAL_REGBRANCH_CURRENT_USER "user.reg" #define SAVE_LOCAL_REGBRANCH_USER_DEFAULT "userdef.reg" #define SAVE_LOCAL_REGBRANCH_LOCAL_MACHINE "system.reg" /* _xmalloc [Internal] */ static void *_xmalloc( size_t size ) { void *res; res = malloc (size ? size : 1); if (res == NULL) { WARN("Virtual memory exhausted.\n"); exit (1); } return res; } /* _strdupnA [Internal] */ static LPSTR _strdupnA(LPCSTR str,size_t len) { LPSTR ret; if (!str) return NULL; ret = _xmalloc( len + 1 ); memcpy( ret, str, len ); ret[len] = 0x00; return ret; } /* convert ansi string to unicode [Internal] */ static LPWSTR _strdupnAtoW(LPCSTR strA,size_t lenA) { LPWSTR ret; size_t lenW; if (!strA) return NULL; lenW = MultiByteToWideChar(CP_ACP,0,strA,lenA,NULL,0); ret = _xmalloc(lenW*sizeof(WCHAR)+sizeof(WCHAR)); MultiByteToWideChar(CP_ACP,0,strA,lenA,ret,lenW); ret[lenW] = 0; return ret; } /* dump a Unicode string with proper escaping [Internal] */ /* FIXME: this code duplicates server/unicode.c */ static int _dump_strW(const WCHAR *str,size_t len,FILE *f,char escape[2]) { static const char escapes[32] = ".......abtnvfr.............e...."; char buffer[256]; LPSTR pos = buffer; int count = 0; for (; len; str++, len--) { if (pos > buffer + sizeof(buffer) - 8) { fwrite( buffer, pos - buffer, 1, f ); count += pos - buffer; pos = buffer; } if (*str > 127) /* hex escape */ { if (len > 1 && str[1] < 128 && isxdigit((char)str[1])) pos += sprintf( pos, "\\x%04x", *str ); else pos += sprintf( pos, "\\x%x", *str ); continue; } if (*str < 32) /* octal or C escape */ { if (!*str && len == 1) continue; /* do not output terminating NULL */ if (escapes[*str] != '.') pos += sprintf( pos, "\\%c", escapes[*str] ); else if (len > 1 && str[1] >= '0' && str[1] <= '7') pos += sprintf( pos, "\\%03o", *str ); else pos += sprintf( pos, "\\%o", *str ); continue; } if (*str == '\\' || *str == escape[0] || *str == escape[1]) *pos++ = '\\'; *pos++ = *str; } fwrite( buffer, pos - buffer, 1, f ); count += pos - buffer; return count; } /* convert ansi string to unicode and dump with proper escaping [Internal] */ static int _dump_strAtoW(LPCSTR strA,size_t len,FILE *f,char escape[2]) { WCHAR *strW; int ret; if (strA == NULL) return 0; strW = _strdupnAtoW(strA,len); ret = _dump_strW(strW,len,f,escape); free(strW); return ret; } /* a key value */ /* FIXME: this code duplicates server/registry.c */ struct key_value { WCHAR *nameW; /* value name */ int type; /* value type */ size_t len; /* value data length in bytes */ void *data; /* pointer to value data */ }; /* dump a value to a text file */ /* FIXME: this code duplicates server/registry.c */ static void _dump_value(struct key_value *value,FILE *f) { int i, count; if (value->nameW[0]) { fputc( '\"', f ); count = 1 + _dump_strW(value->nameW,strlenW(value->nameW),f,"\"\""); count += fprintf( f, "\"=" ); } else count = fprintf( f, "@=" ); switch(value->type) { case REG_SZ: case REG_EXPAND_SZ: case REG_MULTI_SZ: if (value->type != REG_SZ) fprintf( f, "str(%d):", value->type ); fputc( '\"', f ); if (value->data) _dump_strW(value->data,value->len/sizeof(WCHAR),f,"\"\""); fputc( '\"', f ); break; case REG_DWORD: if (value->len == sizeof(DWORD)) { DWORD dw; memcpy( &dw, value->data, sizeof(DWORD) ); fprintf( f, "dword:%08lx", dw ); break; } /* else fall through */ default: if (value->type == REG_BINARY) count += fprintf( f, "hex:" ); else count += fprintf( f, "hex(%x):", value->type ); for (i = 0; i < value->len; i++) { count += fprintf( f, "%02x", *((unsigned char *)value->data + i) ); if (i < value->len-1) { fputc( ',', f ); if (++count > 76) { fprintf( f, "\\\n " ); count = 2; } } } break; } fputc( '\n', f ); } /******************************************************************/ /* WINDOWS 31 REGISTRY LOADER, supplied by Tor Sjøwall, tor@sn.no */ /* reghack - windows 3.11 registry data format demo program. The reg.dat file has 3 parts, a header, a table of 8-byte entries that is a combined hash table and tree description, and finally a text table. The header is obvious from the struct header. The taboff1 and taboff2 fields are always 0x20, and their usage is unknown. The 8-byte entry table has various entry types. tabent[0] is a root index. The second word has the index of the root of the directory. tabent[1..hashsize] is a hash table. The first word in the hash entry is the index of the key/value that has that hash. Data with the same hash value are on a circular list. The other three words in the hash entry are always zero. tabent[hashsize..tabcnt] is the tree structure. There are two kinds of entry: dirent and keyent/valent. They are identified by context. tabent[freeidx] is the first free entry. The first word in a free entry is the index of the next free entry. The last has 0 as a link. The other three words in the free list are probably irrelevant. Entries in text table are preceded by a word at offset-2. This word has the value (2*index)+1, where index is the referring keyent/valent entry in the table. I have no suggestion for the 2* and the +1. Following the word, there are N bytes of data, as per the keyent/valent entry length. The offset of the keyent/valent entry is from the start of the text table to the first data byte. This information is not available from Microsoft. The data format is deduced from the reg.dat file by me. Mistakes may have been made. I claim no rights and give no guarantees for this program. Tor Sjøwall, tor@sn.no */ /* reg.dat header format */ struct _w31_header { char cookie[8]; /* 'SHCC3.10' */ unsigned long taboff1; /* offset of hash table (??) = 0x20 */ unsigned long taboff2; /* offset of index table (??) = 0x20 */ unsigned long tabcnt; /* number of entries in index table */ unsigned long textoff; /* offset of text part */ unsigned long textsize; /* byte size of text part */ unsigned short hashsize; /* hash size */ unsigned short freeidx; /* free index */ }; /* generic format of table entries */ struct _w31_tabent { unsigned short w0, w1, w2, w3; }; /* directory tabent: */ struct _w31_dirent { unsigned short sibling_idx; /* table index of sibling dirent */ unsigned short child_idx; /* table index of child dirent */ unsigned short key_idx; /* table index of key keyent */ unsigned short value_idx; /* table index of value valent */ }; /* key tabent: */ struct _w31_keyent { unsigned short hash_idx; /* hash chain index for string */ unsigned short refcnt; /* reference count */ unsigned short length; /* length of string */ unsigned short string_off; /* offset of string in text table */ }; /* value tabent: */ struct _w31_valent { unsigned short hash_idx; /* hash chain index for string */ unsigned short refcnt; /* reference count */ unsigned short length; /* length of string */ unsigned short string_off; /* offset of string in text table */ }; /* recursive helper function to display a directory tree [Internal] */ void _w31_dumptree(unsigned short idx,unsigned char *txt,struct _w31_tabent *tab,struct _w31_header *head,HKEY hkey,time_t lastmodified, int level) { struct _w31_dirent *dir; struct _w31_keyent *key; struct _w31_valent *val; HKEY subkey = 0; static char tail[400]; while (idx!=0) { dir=(struct _w31_dirent*)&tab[idx]; if (dir->key_idx) { key = (struct _w31_keyent*)&tab[dir->key_idx]; memcpy(tail,&txt[key->string_off],key->length); tail[key->length]='\0'; /* all toplevel entries AND the entries in the * toplevel subdirectory belong to \SOFTWARE\Classes */ if (!level && !strcmp(tail,".classes")) { _w31_dumptree(dir->child_idx,txt,tab,head,hkey,lastmodified,level+1); idx=dir->sibling_idx; continue; } if (subkey) RegCloseKey( subkey ); if (RegCreateKeyA( hkey, tail, &subkey ) != ERROR_SUCCESS) subkey = 0; /* only add if leaf node or valued node */ if (dir->value_idx!=0||dir->child_idx==0) { if (dir->value_idx) { val=(struct _w31_valent*)&tab[dir->value_idx]; memcpy(tail,&txt[val->string_off],val->length); tail[val->length]='\0'; RegSetValueA( subkey, NULL, REG_SZ, tail, 0 ); } } } else TRACE("strange: no directory key name, idx=%04x\n", idx); _w31_dumptree(dir->child_idx,txt,tab,head,subkey,lastmodified,level+1); idx=dir->sibling_idx; } if (subkey) RegCloseKey( subkey ); } /****************************************************************************** * _w31_loadreg [Internal] */ void _w31_loadreg(void) { HFILE hf; struct _w31_header head; struct _w31_tabent *tab; unsigned char *txt; unsigned int len; OFSTRUCT ofs; BY_HANDLE_FILE_INFORMATION hfinfo; time_t lastmodified; TRACE("(void)\n"); hf = OpenFile("reg.dat",&ofs,OF_READ); if (hf==HFILE_ERROR) return; /* read & dump header */ if (sizeof(head)!=_lread(hf,&head,sizeof(head))) { ERR("reg.dat is too short.\n"); _lclose(hf); return; } if (memcmp(head.cookie, "SHCC3.10", sizeof(head.cookie))!=0) { ERR("reg.dat has bad signature.\n"); _lclose(hf); return; } len = head.tabcnt * sizeof(struct _w31_tabent); /* read and dump index table */ tab = _xmalloc(len); if (len!=_lread(hf,tab,len)) { ERR("couldn't read %d bytes.\n",len); free(tab); _lclose(hf); return; } /* read text */ txt = _xmalloc(head.textsize); if (-1==_llseek(hf,head.textoff,SEEK_SET)) { ERR("couldn't seek to textblock.\n"); free(tab); free(txt); _lclose(hf); return; } if (head.textsize!=_lread(hf,txt,head.textsize)) { ERR("textblock too short (%d instead of %ld).\n",len,head.textsize); free(tab); free(txt); _lclose(hf); return; } if (!GetFileInformationByHandle(hf,&hfinfo)) { ERR("GetFileInformationByHandle failed?.\n"); free(tab); free(txt); _lclose(hf); return; } lastmodified = DOSFS_FileTimeToUnixTime(&hfinfo.ftLastWriteTime,NULL); _w31_dumptree(tab[0].w1,txt,tab,&head,HKEY_CLASSES_ROOT,lastmodified,0); free(tab); free(txt); _lclose(hf); return; } /***********************************************************************************/ /* windows 95 registry loader */ /***********************************************************************************/ /* SECTION 1: main header * * once at offset 0 */ #define W95_REG_CREG_ID 0x47455243 typedef struct { DWORD id; /* "CREG" = W95_REG_CREG_ID */ DWORD version; /* ???? 0x00010000 */ DWORD rgdb_off; /* 0x08 Offset of 1st RGDB-block */ DWORD uk2; /* 0x0c */ WORD rgdb_num; /* 0x10 # of RGDB-blocks */ WORD uk3; DWORD uk[3]; /* rgkn */ } _w95creg; /* SECTION 2: Directory information (tree structure) * * once on offset 0x20 * * structure: [rgkn][dke]* (repeat till last_dke is reached) */ #define W95_REG_RGKN_ID 0x4e4b4752 typedef struct { DWORD id; /*"RGKN" = W95_REG_RGKN_ID */ DWORD size; /* Size of the RGKN-block */ DWORD root_off; /* Rel. Offset of the root-record */ DWORD last_dke; /* Offset to last DKE ? */ DWORD uk[4]; } _w95rgkn; /* Disk Key Entry Structure * * the 1st entry in a "usual" registry file is a nul-entry with subkeys: the * hive itself. It looks the same like other keys. Even the ID-number can * be any value. * * The "hash"-value is a value representing the key's name. Windows will not * search for the name, but for a matching hash-value. if it finds one, it * will compare the actual string info, otherwise continue with the next key. * To calculate the hash initialize a D-Word with 0 and add all ASCII-values * of the string which are smaller than 0x80 (128) to this D-Word. * * If you want to modify key names, also modify the hash-values, since they * cannot be found again (although they would be displayed in REGEDIT) * End of list-pointers are filled with 0xFFFFFFFF * * Disk keys are layed out flat ... But, sometimes, nrLS and nrMS are both * 0xFFFF, which means skipping over nextkeyoffset bytes (including this * structure) and reading another RGDB_section. * * The last DKE (see field last_dke in _w95_rgkn) has only 3 DWORDs with * 0x80000000 (EOL indicator ?) as x1, the hash value and 0xFFFFFFFF as x3. * The remaining space between last_dke and the offset calculated from * rgkn->size seems to be free for use for more dke:s. * So it seems if more dke:s are added, they are added to that space and * last_dke is grown, and in case that "free" space is out, the space * gets grown and rgkn->size gets adjusted. * * there is a one to one relationship between dke and dkh */ /* key struct, once per key */ typedef struct { DWORD x1; /* Free entry indicator(?) */ DWORD hash; /* sum of bytes of keyname */ DWORD x3; /* Root key indicator? usually 0xFFFFFFFF */ DWORD prevlvl; /* offset of previous key */ DWORD nextsub; /* offset of child key */ DWORD next; /* offset of sibling key */ WORD nrLS; /* id inside the rgdb block */ WORD nrMS; /* number of the rgdb block */ } _w95dke; /* SECTION 3: key information, values and data * * structure: * section: [blocks]* (repeat creg->rgdb_num times) * blocks: [rgdb] [subblocks]* (repeat till block size reached ) * subblocks: [dkh] [dkv]* (repeat dkh->values times ) * * An interesting relationship exists in RGDB_section. The DWORD value * at offset 0x10 equals the one at offset 0x04 minus the one at offset 0x08. * I have no idea at the moment what this means. (Kevin Cozens) */ /* block header, once per block */ #define W95_REG_RGDB_ID 0x42444752 typedef struct { DWORD id; /* 0x00 'RGDB' = W95_REG_RGDB_ID */ DWORD size; /* 0x04 */ DWORD uk1; /* 0x08 */ DWORD uk2; /* 0x0c */ DWORD uk3; /* 0x10 */ DWORD uk4; /* 0x14 */ DWORD uk5; /* 0x18 */ DWORD uk6; /* 0x1c */ /* dkh */ } _w95rgdb; /* Disk Key Header structure (RGDB part), once per key */ typedef struct { DWORD nextkeyoff; /* 0x00 offset to next dkh */ WORD nrLS; /* 0x04 id inside the rgdb block */ WORD nrMS; /* 0x06 number of the rgdb block */ DWORD bytesused; /* 0x08 */ WORD keynamelen; /* 0x0c len of name */ WORD values; /* 0x0e number of values */ DWORD xx1; /* 0x10 */ char name[1]; /* 0x14 */ /* dkv */ /* 0x14 + keynamelen */ } _w95dkh; /* Disk Key Value structure, once per value */ typedef struct { DWORD type; /* 0x00 */ DWORD x1; /* 0x04 */ WORD valnamelen; /* 0x08 length of name, 0 is default key */ WORD valdatalen; /* 0x0A length of data */ char name[1]; /* 0x0c */ /* raw data */ /* 0x0c + valnamelen */ } _w95dkv; /****************************************************************************** * _w95_lookup_dkh [Internal] * * seeks the dkh belonging to a dke */ static _w95dkh *_w95_lookup_dkh(_w95creg *creg,int nrLS,int nrMS) { _w95rgdb * rgdb; _w95dkh * dkh; int i; /* get the beginning of the rgdb datastore */ rgdb = (_w95rgdb*)((char*)creg+creg->rgdb_off); /* check: requested block < last_block) */ if (creg->rgdb_num <= nrMS) { ERR("registry file corrupt! requested block no. beyond end.\n"); goto error; } /* find the right block */ for(i=0; iid != W95_REG_RGDB_ID) { /* check the magic */ ERR("registry file corrupt! bad magic 0x%08lx\n", rgdb->id); goto error; } rgdb = (_w95rgdb*) ((char*)rgdb+rgdb->size); /* find next block */ } dkh = (_w95dkh*)(rgdb + 1); /* first sub block within the rgdb */ do { if(nrLS==dkh->nrLS ) return dkh; dkh = (_w95dkh*)((char*)dkh + dkh->nextkeyoff); /* find next subblock */ } while ((char *)dkh < ((char*)rgdb+rgdb->size)); error: return NULL; } /****************************************************************************** * _w95_dump_dkv [Internal] */ static int _w95_dump_dkv(_w95dkh *dkh,int nrLS,int nrMS,FILE *f) { _w95dkv * dkv; int i; /* first value block */ dkv = (_w95dkv*)((char*)dkh+dkh->keynamelen+0x14); /* loop trought the values */ for (i=0; i< dkh->values; i++) { struct key_value value; WCHAR *pdata; value.nameW = _strdupnAtoW(dkv->name,dkv->valnamelen); value.type = dkv->type; value.len = dkv->valdatalen; value.data = &(dkv->name[dkv->valnamelen]); pdata = NULL; if ( (value.type==REG_SZ) || (value.type==REG_EXPAND_SZ) || (value.type==REG_MULTI_SZ) ) { pdata = _strdupnAtoW(value.data,value.len); value.len *= 2; } if (pdata != NULL) value.data = pdata; _dump_value(&value,f); free(value.nameW); if (pdata != NULL) free(pdata); /* next value */ dkv = (_w95dkv*)((char*)dkv+dkv->valnamelen+dkv->valdatalen+0x0c); } return TRUE; } /****************************************************************************** * _w95_dump_dke [Internal] */ static int _w95_dump_dke(LPSTR key_name,_w95creg *creg,_w95rgkn *rgkn,_w95dke *dke,FILE *f,int level) { _w95dkh * dkh; LPSTR new_key_name = NULL; /* special root key */ if (dke->nrLS == 0xffff || dke->nrMS==0xffff) /* eg. the root key has no name */ { /* parse the one subkey */ if (dke->nextsub != 0xffffffff) return _w95_dump_dke(key_name, creg, rgkn, (_w95dke*)((char*)rgkn+dke->nextsub),f,level); /* has no sibling keys */ return FALSE; } /* search subblock */ if (!(dkh = _w95_lookup_dkh(creg, dke->nrLS, dke->nrMS))) { ERR("dke pointing to missing dkh !\n"); return FALSE; } if (level <= 0) { /* create new subkey name */ new_key_name = _strdupnA(key_name,strlen(key_name)+dkh->keynamelen+1); if (strcmp(new_key_name,"") != 0) strcat(new_key_name,"\\"); strncat(new_key_name,dkh->name,dkh->keynamelen); /* walk sibling keys */ if (dke->next != 0xffffffff ) { if (!_w95_dump_dke(key_name, creg, rgkn, (_w95dke*)((char*)rgkn+dke->next),f,level)) { free(new_key_name); return FALSE; } } /* write the key path (something like [Software\\Microsoft\\..]) only if: 1) key has some values 2) key has no values and no subkeys */ if (dkh->values > 0) { /* there are some values */ fprintf(f,"\n["); _dump_strAtoW(new_key_name,strlen(new_key_name),f,"[]"); fprintf(f,"]\n"); if (!_w95_dump_dkv(dkh, dke->nrLS, dke->nrMS,f)) { free(new_key_name); return FALSE; } } if ((dke->nextsub == 0xffffffff) && (dkh->values == 0)) { /* no subkeys and no values */ fprintf(f,"\n["); _dump_strAtoW(new_key_name,strlen(new_key_name),f,"[]"); fprintf(f,"]\n"); } } else new_key_name = _strdupnA(key_name,strlen(key_name)); /* next sub key */ if (dke->nextsub != 0xffffffff) { if (!_w95_dump_dke(new_key_name, creg, rgkn, (_w95dke*)((char*)rgkn+dke->nextsub),f,level-1)) { free(new_key_name); return FALSE; } } free(new_key_name); return TRUE; } /* end windows 95 loader */ /***********************************************************************************/ /* windows NT registry loader */ /***********************************************************************************/ /* NT REGISTRY LOADER */ #ifdef HAVE_SYS_MMAN_H # include #endif #ifndef MAP_FAILED #define MAP_FAILED ((LPVOID)-1) #endif #define NT_REG_BLOCK_SIZE 0x1000 #define NT_REG_HEADER_BLOCK_ID 0x66676572 /* regf */ #define NT_REG_POOL_BLOCK_ID 0x6E696268 /* hbin */ #define NT_REG_KEY_BLOCK_ID 0x6b6e /* nk */ #define NT_REG_VALUE_BLOCK_ID 0x6b76 /* vk */ /* subblocks of nk */ #define NT_REG_HASH_BLOCK_ID 0x666c /* lf */ #define NT_REG_NOHASH_BLOCK_ID 0x696c /* li */ #define NT_REG_RI_BLOCK_ID 0x6972 /* ri */ #define NT_REG_KEY_BLOCK_TYPE 0x20 #define NT_REG_ROOT_KEY_BLOCK_TYPE 0x2c typedef struct { DWORD id; /* 0x66676572 'regf'*/ DWORD uk1; /* 0x04 */ DWORD uk2; /* 0x08 */ FILETIME DateModified; /* 0x0c */ DWORD uk3; /* 0x14 */ DWORD uk4; /* 0x18 */ DWORD uk5; /* 0x1c */ DWORD uk6; /* 0x20 */ DWORD RootKeyBlock; /* 0x24 */ DWORD BlockSize; /* 0x28 */ DWORD uk7[116]; DWORD Checksum; /* at offset 0x1FC */ } nt_regf; typedef struct { DWORD blocksize; BYTE data[1]; } nt_hbin_sub; typedef struct { DWORD id; /* 0x6E696268 'hbin' */ DWORD off_prev; DWORD off_next; DWORD uk1; DWORD uk2; /* 0x10 */ DWORD uk3; /* 0x14 */ DWORD uk4; /* 0x18 */ DWORD size; /* 0x1C */ nt_hbin_sub hbin_sub; /* 0x20 */ } nt_hbin; /* * the value_list consists of offsets to the values (vk) */ typedef struct { WORD SubBlockId; /* 0x00 0x6B6E */ WORD Type; /* 0x02 for the root-key: 0x2C, otherwise 0x20*/ FILETIME writetime; /* 0x04 */ DWORD uk1; /* 0x0C */ DWORD parent_off; /* 0x10 Offset of Owner/Parent key */ DWORD nr_subkeys; /* 0x14 number of sub-Keys */ DWORD uk8; /* 0x18 */ DWORD lf_off; /* 0x1C Offset of the sub-key lf-Records */ DWORD uk2; /* 0x20 */ DWORD nr_values; /* 0x24 number of values */ DWORD valuelist_off; /* 0x28 Offset of the Value-List */ DWORD off_sk; /* 0x2c Offset of the sk-Record */ DWORD off_class; /* 0x30 Offset of the Class-Name */ DWORD uk3; /* 0x34 */ DWORD uk4; /* 0x38 */ DWORD uk5; /* 0x3c */ DWORD uk6; /* 0x40 */ DWORD uk7; /* 0x44 */ WORD name_len; /* 0x48 name-length */ WORD class_len; /* 0x4a class-name length */ char name[1]; /* 0x4c key-name */ } nt_nk; typedef struct { DWORD off_nk; /* 0x00 */ DWORD name; /* 0x04 */ } hash_rec; typedef struct { WORD id; /* 0x00 0x666c */ WORD nr_keys; /* 0x06 */ hash_rec hash_rec[1]; } nt_lf; /* list of subkeys without hash li --+-->nk | +-->nk */ typedef struct { WORD id; /* 0x00 0x696c */ WORD nr_keys; DWORD off_nk[1]; } nt_li; /* this is a intermediate node ri --+-->li--+-->nk | + | +-->nk | +-->li--+-->nk + +-->nk */ typedef struct { WORD id; /* 0x00 0x6972 */ WORD nr_li; /* 0x02 number off offsets */ DWORD off_li[1]; /* 0x04 points to li */ } nt_ri; typedef struct { WORD id; /* 0x00 'vk' */ WORD nam_len; DWORD data_len; DWORD data_off; DWORD type; WORD flag; WORD uk1; char name[1]; } nt_vk; /* * gets a value * * vk->flag: * 0 value is a default value * 1 the value has a name * * vk->data_len * len of the whole data block * - reg_sz (unicode) * bytes including the terminating \0 = 2*(number_of_chars+1) * - reg_dword, reg_binary: * if highest bit of data_len is set data_off contains the value */ static int _nt_dump_vk(LPSTR key_name, char *base, nt_vk *vk,FILE *f) { BYTE *pdata = (BYTE *)(base+vk->data_off+4); /* start of data */ struct key_value value; if (vk->id != NT_REG_VALUE_BLOCK_ID) { ERR("unknown block found (0x%04x), please report!\n", vk->id); return FALSE; } value.nameW = _strdupnAtoW(vk->name,vk->nam_len); value.type = vk->type; value.len = (vk->data_len & 0x7fffffff); value.data = (vk->data_len & 0x80000000) ? (LPBYTE)&(vk->data_off): pdata; _dump_value(&value,f); free(value.nameW); return TRUE; } /* it's called from _nt_dump_lf() */ static int _nt_dump_nk(LPSTR key_name,char *base,nt_nk *nk,FILE *f,int level); /* * get the subkeys * * this structure contains the hash of a keyname and points to all * subkeys * * exception: if the id is 'il' there are no hash values and every * dword is a offset */ static int _nt_dump_lf(LPSTR key_name, char *base, int subkeys, nt_lf *lf, FILE *f, int level) { int i; if (lf->id == NT_REG_HASH_BLOCK_ID) { if (subkeys != lf->nr_keys) goto error1; for (i=0; inr_keys; i++) if (!_nt_dump_nk(key_name, base, (nt_nk*)(base+lf->hash_rec[i].off_nk+4), f, level)) goto error; } else if (lf->id == NT_REG_NOHASH_BLOCK_ID) { nt_li * li = (nt_li*)lf; if (subkeys != li->nr_keys) goto error1; for (i=0; inr_keys; i++) if (!_nt_dump_nk(key_name, base, (nt_nk*)(base+li->off_nk[i]+4), f, level)) goto error; } else if (lf->id == NT_REG_RI_BLOCK_ID) { /* ri */ nt_ri * ri = (nt_ri*)lf; int li_subkeys = 0; /* count all subkeys */ for (i=0; inr_li; i++) { nt_li * li = (nt_li*)(base+ri->off_li[i]+4); if(li->id != NT_REG_NOHASH_BLOCK_ID) goto error2; li_subkeys += li->nr_keys; } /* check number */ if (subkeys != li_subkeys) goto error1; /* loop through the keys */ for (i=0; inr_li; i++) { nt_li *li = (nt_li*)(base+ri->off_li[i]+4); if (!_nt_dump_lf(key_name, base, li->nr_keys, (nt_lf*)li, f, level)) goto error; } } else goto error2; return TRUE; error2: ERR("unknown node id 0x%04x, please report!\n", lf->id); return TRUE; error1: ERR("registry file corrupt! (inconsistent number of subkeys)\n"); return FALSE; error: ERR("error reading lf block\n"); return FALSE; } /* _nt_dump_nk [Internal] */ static int _nt_dump_nk(LPSTR key_name,char *base,nt_nk *nk,FILE *f,int level) { unsigned int n; DWORD *vl; LPSTR new_key_name = NULL; if (nk->SubBlockId != NT_REG_KEY_BLOCK_ID) { ERR("unknown node id 0x%04x, please report!\n", nk->SubBlockId); return FALSE; } if ((nk->Type!=NT_REG_ROOT_KEY_BLOCK_TYPE) && (((nt_nk*)(base+nk->parent_off+4))->SubBlockId != NT_REG_KEY_BLOCK_ID)) { ERR("registry file corrupt!\n"); return FALSE; } /* create the new key */ if (level <= 0) { /* create new subkey name */ new_key_name = _strdupnA(key_name,strlen(key_name)+nk->name_len+1); if (strcmp(new_key_name,"") != 0) strcat(new_key_name,"\\"); strncat(new_key_name,nk->name,nk->name_len); /* write the key path (something like [Software\\Microsoft\\..]) only if: 1) key has some values 2) key has no values and no subkeys */ if (nk->nr_values > 0) { /* there are some values */ fprintf(f,"\n["); _dump_strAtoW(new_key_name,strlen(new_key_name),f,"[]"); fprintf(f,"]\n"); } if ((nk->nr_subkeys == 0) && (nk->nr_values == 0)) { /* no subkeys and no values */ fprintf(f,"\n["); _dump_strAtoW(new_key_name,strlen(new_key_name),f,"[]"); fprintf(f,"]\n"); } /* loop trough the value list */ vl = (DWORD *)(base+nk->valuelist_off+4); for (n=0; nnr_values; n++) { nt_vk * vk = (nt_vk*)(base+vl[n]+4); if (!_nt_dump_vk(new_key_name, base, vk, f)) { free(new_key_name); return FALSE; } } } else new_key_name = _strdupnA(key_name,strlen(key_name)); /* loop through the subkeys */ if (nk->nr_subkeys) { nt_lf *lf = (nt_lf*)(base+nk->lf_off+4); if (!_nt_dump_lf(new_key_name, base, nk->nr_subkeys, lf, f, level-1)) { free(new_key_name); return FALSE; } } free(new_key_name); return TRUE; } /* end nt loader */ /********************************************************************************** * _set_registry_levels [Internal] * * set level to 0 for loading system files * set level to 1 for loading user files */ static void _set_registry_levels(int level,int saving,int period) { SERVER_START_REQ { struct set_registry_levels_request *req = server_alloc_req( sizeof(*req), 0 ); req->current = level; req->saving = saving; req->period = period; server_call( REQ_SET_REGISTRY_LEVELS ); } SERVER_END_REQ; } /* _save_at_exit [Internal] */ static void _save_at_exit(HKEY hkey,LPCSTR path) { LPCSTR confdir = get_config_dir(); size_t len = strlen(confdir) + strlen(path) + 2; if (len > REQUEST_MAX_VAR_SIZE) { ERR( "config dir '%s' too long\n", confdir ); return; } SERVER_START_REQ { struct save_registry_atexit_request *req = server_alloc_req( sizeof(*req), len ); sprintf( server_data_ptr(req), "%s/%s", confdir, path ); req->hkey = hkey; server_call( REQ_SAVE_REGISTRY_ATEXIT ); } SERVER_END_REQ; } /* configure save files and start the periodic saving timer [Internal] */ static void _init_registry_saving( HKEY hkey_users_default ) { int all; int period; all = PROFILE_GetWineIniBool("registry","SaveOnlyUpdatedKeys",1); period = PROFILE_GetWineIniInt("registry","PeriodicSave",0); /* set saving level (0 for saving everything, 1 for saving only modified keys) */ _set_registry_levels(1,!all,period*1000); if (PROFILE_GetWineIniBool("registry","WritetoHomeRegistryFiles",1)) { _save_at_exit(HKEY_CURRENT_USER,SAVE_LOCAL_REGBRANCH_CURRENT_USER ); _save_at_exit(HKEY_LOCAL_MACHINE,SAVE_LOCAL_REGBRANCH_LOCAL_MACHINE); _save_at_exit(hkey_users_default,SAVE_LOCAL_REGBRANCH_USER_DEFAULT); } } /****************************************************************************** * _allocate_default_keys [Internal] * Registry initialisation, allocates some default keys. */ static void _allocate_default_keys(void) { HKEY hkey; char buf[200]; TRACE("(void)\n"); RegCreateKeyA(HKEY_DYN_DATA,"PerfStats\\StatData",&hkey); RegCloseKey(hkey); /* This was an Open, but since it is called before the real registries are loaded, it was changed to a Create - MTB 980507*/ RegCreateKeyA(HKEY_LOCAL_MACHINE,"HARDWARE\\DESCRIPTION\\System",&hkey); RegSetValueExA(hkey,"Identifier",0,REG_SZ,"SystemType WINE",strlen("SystemType WINE")); RegCloseKey(hkey); /* \\SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion * CurrentVersion * CurrentBuildNumber * CurrentType * string RegisteredOwner * string RegisteredOrganization * */ /* System\\CurrentControlSet\\Services\\SNMP\\Parameters\\RFC1156Agent * string SysContact * string SysLocation * SysServices */ if (-1!=gethostname(buf,200)) { RegCreateKeyA(HKEY_LOCAL_MACHINE,"System\\CurrentControlSet\\Control\\ComputerName\\ComputerName",&hkey); RegSetValueExA(hkey,"ComputerName",0,REG_SZ,buf,strlen(buf)+1); RegCloseKey(hkey); } RegCreateKeyA(HKEY_USERS,".Default",&hkey); RegCloseKey(hkey); } #define REG_DONTLOAD -1 #define REG_WIN31 0 #define REG_WIN95 1 #define REG_WINNT 2 /* return the type of native registry [Internal] */ static int _get_reg_type(void) { char windir[MAX_PATHNAME_LEN]; char tmp[MAX_PATHNAME_LEN]; int ret = REG_WIN31; GetWindowsDirectoryA(windir,MAX_PATHNAME_LEN); /* test %windir%/system32/config/system --> winnt */ strcpy(tmp, windir); strncat(tmp, "\\system32\\config\\system", MAX_PATHNAME_LEN - strlen(tmp) - 1); if(GetFileAttributesA(tmp) != (DWORD)-1) { ret = REG_WINNT; } else { /* test %windir%/system.dat --> win95 */ strcpy(tmp, windir); strncat(tmp, "\\system.dat", MAX_PATHNAME_LEN - strlen(tmp) - 1); if(GetFileAttributesA(tmp) != (DWORD)-1) { ret = REG_WIN95; } } if ((ret == REG_WINNT) && (!PROFILE_GetWineIniString( "Wine", "Profile", "", tmp, MAX_PATHNAME_LEN))) { MESSAGE("When you are running with a native NT directory specify\n"); MESSAGE("'Profile=' or disable loading of Windows\n"); MESSAGE("registry (LoadWindowsRegistryFiles=N)\n"); ret = REG_DONTLOAD; } return ret; } #define WINE_REG_VER_ERROR -1 #define WINE_REG_VER_1 0 #define WINE_REG_VER_2 1 #define WINE_REG_VER_OLD 2 #define WINE_REG_VER_UNKNOWN 3 /* return the version of wine registry file [Internal] */ static int _get_wine_registry_file_format_version(LPCSTR fn) { FILE *f; char tmp[50]; int ver; if ((f=fopen(fn,"rt")) == NULL) { WARN("Couldn't open %s for reading: %s\n",fn,strerror(errno)); return WINE_REG_VER_ERROR; } if (fgets(tmp,50,f) == NULL) { WARN("Error reading %s: %s\n",fn,strerror(errno)); fclose(f); return WINE_REG_VER_ERROR; } fclose(f); if (sscanf(tmp,"WINE REGISTRY Version %d",&ver) != 1) return WINE_REG_VER_UNKNOWN; switch (ver) { case 1: return WINE_REG_VER_1; break; case 2: return WINE_REG_VER_2; break; default: return WINE_REG_VER_UNKNOWN; } } /* load the registry file in wine format [Internal] */ static void load_wine_registry(HKEY hkey,LPCSTR fn) { int file_format; file_format = _get_wine_registry_file_format_version(fn); switch (file_format) { case WINE_REG_VER_1: WARN("Unable to load registry file %s: old format which is no longer supported.\n",fn); break; case WINE_REG_VER_2: { HANDLE file; if ((file = FILE_CreateFile( fn, GENERIC_READ, 0, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0, TRUE ))) { SERVER_START_REQ { struct load_registry_request *req = server_alloc_req( sizeof(*req), 0 ); req->hkey = hkey; req->file = file; server_call( REQ_LOAD_REGISTRY ); } SERVER_END_REQ; CloseHandle( file ); } break; } case WINE_REG_VER_UNKNOWN: WARN("Unable to load registry file %s: unknown format.\n",fn); break; case WINE_REG_VER_ERROR: break; } } /* generate and return the name of the tmp file and associated stream [Internal] */ static LPSTR _get_tmp_fn(FILE **f) { LPSTR ret; int tmp_fd,count; ret = _xmalloc(50); for (count = 0;;) { sprintf(ret,"/tmp/reg%lx%04x.tmp",(long)getpid(),count++); if ((tmp_fd = open(ret,O_CREAT | O_EXCL | O_WRONLY,0666)) != -1) break; if (errno != EEXIST) { ERR("Unexpected error while open() call: %s\n",strerror(errno)); free(ret); *f = NULL; return NULL; } } if ((*f = fdopen(tmp_fd,"w")) == NULL) { ERR("Unexpected error while fdopen() call: %s\n",strerror(errno)); close(tmp_fd); free(ret); return NULL; } return ret; } /* convert win95 native registry file to wine format [Internal] */ static LPSTR _convert_win95_registry_to_wine_format(LPCSTR fn,int level) { int fd; FILE *f; DOS_FULL_NAME full_name; void *base; LPSTR ret = NULL; struct stat st; _w95creg *creg; _w95rgkn *rgkn; _w95dke *dke, *root_dke; if (!DOSFS_GetFullName( fn, 0, &full_name )) return NULL; /* map the registry into the memory */ if ((fd = open(full_name.long_name, O_RDONLY | O_NONBLOCK)) == -1) return NULL; if ((fstat(fd, &st) == -1)) goto error1; if (!st.st_size) goto error1; if ((base = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) goto error1; /* control signature */ if (*(LPDWORD)base != W95_REG_CREG_ID) { ERR("unable to load native win95 registry file %s: unknown signature.\n",fn); goto error; } creg = base; /* load the header (rgkn) */ rgkn = (_w95rgkn*)(creg + 1); if (rgkn->id != W95_REG_RGKN_ID) { ERR("second IFF header not RGKN, but %lx\n", rgkn->id); goto error; } if (rgkn->root_off != 0x20) { ERR("rgkn->root_off not 0x20, please report !\n"); goto error; } if (rgkn->last_dke > rgkn->size) { ERR("registry file corrupt! last_dke > size!\n"); goto error; } /* verify last dke */ dke = (_w95dke*)((char*)rgkn + rgkn->last_dke); if (dke->x1 != 0x80000000) { /* wrong magic */ ERR("last dke invalid !\n"); goto error; } if (rgkn->size > creg->rgdb_off) { ERR("registry file corrupt! rgkn size > rgdb_off !\n"); goto error; } root_dke = (_w95dke*)((char*)rgkn + rgkn->root_off); if ( (root_dke->prevlvl != 0xffffffff) || (root_dke->next != 0xffffffff) ) { ERR("registry file corrupt! invalid root dke !\n"); goto error; } if ( (ret = _get_tmp_fn(&f)) == NULL) goto error; fprintf(f,"WINE REGISTRY Version 2"); _w95_dump_dke("",creg,rgkn,root_dke,f,level); fclose(f); error: if(ret == NULL) { ERR("Unable to load native win95 registry file %s.\n",fn); ERR("Please report to a.mohr@mailto.de.\n"); ERR("Make a backup of the file, run a good reg cleaner program and try again!\n"); } munmap(base, st.st_size); error1: close(fd); return ret; } /* convert winnt native registry file to wine format [Internal] */ static LPSTR _convert_winnt_registry_to_wine_format(LPCSTR fn,int level) { int fd; FILE *f; DOS_FULL_NAME full_name; void *base; LPSTR ret = NULL; struct stat st; nt_regf *regf; nt_hbin *hbin; nt_hbin_sub *hbin_sub; nt_nk *nk; if (!DOSFS_GetFullName( fn, 0, &full_name )) return NULL; /* map the registry into the memory */ if ((fd = open(full_name.long_name, O_RDONLY | O_NONBLOCK)) == -1) return NULL; if ((fstat(fd, &st) == -1)) goto error1; if (!st.st_size) goto error1; if ((base = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0)) == MAP_FAILED) goto error1; /* control signature */ if (*(LPDWORD)base != NT_REG_HEADER_BLOCK_ID) { ERR("unable to load native winnt registry file %s: unknown signature.\n",fn); goto error; } /* start block */ regf = base; /* hbin block */ hbin = (nt_hbin*)((char*) base + 0x1000); if (hbin->id != NT_REG_POOL_BLOCK_ID) { ERR( "hbin block invalid\n"); goto error; } /* hbin_sub block */ hbin_sub = (nt_hbin_sub*)&(hbin->hbin_sub); if ((hbin_sub->data[0] != 'n') || (hbin_sub->data[1] != 'k')) { ERR( "hbin_sub block invalid\n"); goto error; } /* nk block */ nk = (nt_nk*)&(hbin_sub->data[0]); if (nk->Type != NT_REG_ROOT_KEY_BLOCK_TYPE) { ERR( "special nk block not found\n"); goto error; } if ( (ret = _get_tmp_fn(&f)) == NULL) goto error; fprintf(f,"WINE REGISTRY Version 2"); _nt_dump_nk("",(char*)base+0x1000,nk,f,level); fclose(f); error: munmap(base,st.st_size); error1: close(fd); return ret; } /* convert native native registry to wine format and load it via server call [Internal] */ static void _convert_and_load_native_registry(LPCSTR fn,HKEY hkey,int reg_type,int level) { LPSTR tmp = NULL; switch (reg_type) { case REG_WINNT: /* FIXME: following function doesn't really convert yet */ tmp = _convert_winnt_registry_to_wine_format(fn,level); break; case REG_WIN95: tmp = _convert_win95_registry_to_wine_format(fn,level); break; case REG_WIN31: ERR("Don't know how to convert native 3.1 registry yet.\n"); break; default: ERR("Unknown registry format parameter (%d)\n",reg_type); break; } if (tmp != NULL) { load_wine_registry(hkey,tmp); TRACE("File %s successfuly converted to %s and loaded to registry.\n",fn,tmp); unlink(tmp); } else WARN("Unable to convert %s (not exist?)\n",fn); free(tmp); } /* load all native windows registry files [Internal] */ static void _load_windows_registry( HKEY hkey_users_default ) { int reg_type; char windir[MAX_PATHNAME_LEN]; char path[MAX_PATHNAME_LEN]; GetWindowsDirectoryA(windir,MAX_PATHNAME_LEN); reg_type = _get_reg_type(); switch (reg_type) { case REG_WINNT: { HKEY hkey; /* user specific ntuser.dat */ if (PROFILE_GetWineIniString( "Wine", "Profile", "", path, MAX_PATHNAME_LEN)) { strcat(path,"\\ntuser.dat"); _convert_and_load_native_registry(path,HKEY_CURRENT_USER,REG_WINNT,1); } /* default user.dat */ if (hkey_users_default) { strcpy(path,windir); strcat(path,"\\system32\\config\\default"); _convert_and_load_native_registry(path,hkey_users_default,REG_WINNT,1); } /* * FIXME * map HLM\System\ControlSet001 to HLM\System\CurrentControlSet */ if (!RegCreateKeyA(HKEY_LOCAL_MACHINE, "SYSTEM", &hkey)) { strcpy(path,windir); strcat(path,"\\system32\\config\\system"); _convert_and_load_native_registry(path,hkey,REG_WINNT,1); RegCloseKey(hkey); } if (!RegCreateKeyA(HKEY_LOCAL_MACHINE, "SOFTWARE", &hkey)) { strcpy(path,windir); strcat(path,"\\system32\\config\\software"); _convert_and_load_native_registry(path,hkey,REG_WINNT,1); RegCloseKey(hkey); } strcpy(path,windir); strcat(path,"\\system32\\config\\sam"); _convert_and_load_native_registry(path,HKEY_LOCAL_MACHINE,REG_WINNT,0); strcpy(path,windir); strcat(path,"\\system32\\config\\security"); _convert_and_load_native_registry(path,HKEY_LOCAL_MACHINE,REG_WINNT,0); /* this key is generated when the nt-core booted successfully */ if (!RegCreateKeyA(HKEY_LOCAL_MACHINE,"System\\Clone",&hkey)) RegCloseKey(hkey); break; } case REG_WIN95: _convert_and_load_native_registry("c:\\system.1st",HKEY_LOCAL_MACHINE,REG_WIN95,0); strcpy(path,windir); strcat(path,"\\system.dat"); _convert_and_load_native_registry(path,HKEY_LOCAL_MACHINE,REG_WIN95,0); if (PROFILE_GetWineIniString("Wine","Profile","",path,MAX_PATHNAME_LEN)) { /* user specific user.dat */ strncat(path, "\\user.dat", MAX_PATHNAME_LEN - strlen(path) - 1); _convert_and_load_native_registry(path,HKEY_CURRENT_USER,REG_WIN95,1); /* default user.dat */ if (hkey_users_default) { strcpy(path,windir); strcat(path,"\\user.dat"); _convert_and_load_native_registry(path,hkey_users_default,REG_WIN95,1); } } else { strcpy(path,windir); strcat(path,"\\user.dat"); _convert_and_load_native_registry(path,HKEY_CURRENT_USER,REG_WIN95,1); } break; case REG_WIN31: /* FIXME: here we should convert to *.reg file supported by server and call REQ_LOAD_REGISTRY, see REG_WIN95 case */ _w31_loadreg(); break; case REG_DONTLOAD: TRACE("REG_DONTLOAD\n"); break; default: ERR("switch: no match (%d)\n",reg_type); break; } } /* load global registry files (stored in /etc/wine) [Internal] */ static void _load_global_registry(void) { TRACE("(void)\n"); /* Load the global HKU hive directly from sysconfdir */ load_wine_registry( HKEY_USERS, SAVE_GLOBAL_REGBRANCH_USER_DEFAULT ); /* Load the global machine defaults directly from sysconfdir */ load_wine_registry( HKEY_LOCAL_MACHINE, SAVE_GLOBAL_REGBRANCH_LOCAL_MACHINE ); } /* load home registry files (stored in ~/.wine) [Internal] */ static void _load_home_registry( HKEY hkey_users_default ) { LPCSTR confdir = get_config_dir(); LPSTR tmp = _xmalloc(strlen(confdir)+20); strcpy(tmp,confdir); strcat(tmp,"/" SAVE_LOCAL_REGBRANCH_USER_DEFAULT); load_wine_registry(hkey_users_default,tmp); strcpy(tmp,confdir); strcat(tmp,"/" SAVE_LOCAL_REGBRANCH_CURRENT_USER); load_wine_registry(HKEY_CURRENT_USER,tmp); strcpy(tmp,confdir); strcat(tmp,"/" SAVE_LOCAL_REGBRANCH_LOCAL_MACHINE); load_wine_registry(HKEY_LOCAL_MACHINE,tmp); free(tmp); } /* load all registry (native and global and home) */ void SHELL_LoadRegistry( void ) { HKEY hkey_users_default; TRACE("(void)\n"); if (!CLIENT_IsBootThread()) return; /* already loaded */ if (RegCreateKeyA(HKEY_USERS,".Default",&hkey_users_default)) { ERR("Cannot create HKEY_USERS/.Default\n" ); ExitProcess(1); } _allocate_default_keys(); _set_registry_levels(0,0,0); if (PROFILE_GetWineIniBool("Registry","LoadWindowsRegistryFiles",1)) _load_windows_registry( hkey_users_default ); if (PROFILE_GetWineIniBool("Registry","LoadGlobalRegistryFiles",1)) _load_global_registry(); _set_registry_levels(1,0,0); if (PROFILE_GetWineIniBool("Registry","LoadHomeRegistryFiles",1)) _load_home_registry( hkey_users_default ); _init_registry_saving( hkey_users_default ); RegCloseKey(hkey_users_default); } /***************************************************************************/ /* API FUNCTIONS */ /***************************************************************************/ /****************************************************************************** * RegFlushKey [KERNEL.227] [ADVAPI32.@] * Immediately writes key to registry. * Only returns after data has been written to disk. * * FIXME: does it really wait until data is written ? * * PARAMS * hkey [I] Handle of key to write * * RETURNS * Success: ERROR_SUCCESS * Failure: Error code */ DWORD WINAPI RegFlushKey( HKEY hkey ) { FIXME( "(%x): stub\n", hkey ); return ERROR_SUCCESS; } /****************************************************************************** * RegUnLoadKeyA [ADVAPI32.@] */ LONG WINAPI RegUnLoadKeyA( HKEY hkey, LPCSTR lpSubKey ) { FIXME("(%x,%s): stub\n",hkey, debugstr_a(lpSubKey)); return ERROR_SUCCESS; } /****************************************************************************** * RegRestoreKeyW [ADVAPI32.@] * * PARAMS * hkey [I] Handle of key where restore begins * lpFile [I] Address of filename containing saved tree * dwFlags [I] Optional flags */ LONG WINAPI RegRestoreKeyW( HKEY hkey, LPCWSTR lpFile, DWORD dwFlags ) { TRACE("(%x,%s,%ld)\n",hkey,debugstr_w(lpFile),dwFlags); /* It seems to do this check before the hkey check */ if (!lpFile || !*lpFile) return ERROR_INVALID_PARAMETER; FIXME("(%x,%s,%ld): stub\n",hkey,debugstr_w(lpFile),dwFlags); /* Check for file existence */ return ERROR_SUCCESS; } /****************************************************************************** * RegRestoreKeyA [ADVAPI32.@] */ LONG WINAPI RegRestoreKeyA( HKEY hkey, LPCSTR lpFile, DWORD dwFlags ) { LPWSTR lpFileW = HEAP_strdupAtoW( GetProcessHeap(), 0, lpFile ); LONG ret = RegRestoreKeyW( hkey, lpFileW, dwFlags ); HeapFree( GetProcessHeap(), 0, lpFileW ); return ret; } /****************************************************************************** * RegReplaceKeyA [ADVAPI32.@] */ LONG WINAPI RegReplaceKeyA( HKEY hkey, LPCSTR lpSubKey, LPCSTR lpNewFile, LPCSTR lpOldFile ) { FIXME("(%x,%s,%s,%s): stub\n", hkey, debugstr_a(lpSubKey), debugstr_a(lpNewFile),debugstr_a(lpOldFile)); return ERROR_SUCCESS; } /* 16-bit functions */ /* 0 and 1 are valid rootkeys in win16 shell.dll and are used by * some programs. Do not remove those cases. -MM */ static inline void fix_win16_hkey( HKEY *hkey ) { if (*hkey == 0 || *hkey == 1) *hkey = HKEY_CLASSES_ROOT; } /****************************************************************************** * RegEnumKey16 [KERNEL.216] [SHELL.7] */ DWORD WINAPI RegEnumKey16( HKEY hkey, DWORD index, LPSTR name, DWORD name_len ) { fix_win16_hkey( &hkey ); return RegEnumKeyA( hkey, index, name, name_len ); } /****************************************************************************** * RegOpenKey16 [KERNEL.217] [SHELL.1] */ DWORD WINAPI RegOpenKey16( HKEY hkey, LPCSTR name, LPHKEY retkey ) { fix_win16_hkey( &hkey ); return RegOpenKeyA( hkey, name, retkey ); } /****************************************************************************** * RegCreateKey16 [KERNEL.218] [SHELL.2] */ DWORD WINAPI RegCreateKey16( HKEY hkey, LPCSTR name, LPHKEY retkey ) { fix_win16_hkey( &hkey ); return RegCreateKeyA( hkey, name, retkey ); } /****************************************************************************** * RegDeleteKey16 [KERNEL.219] [SHELL.4] */ DWORD WINAPI RegDeleteKey16( HKEY hkey, LPCSTR name ) { fix_win16_hkey( &hkey ); return RegDeleteKeyA( hkey, name ); } /****************************************************************************** * RegCloseKey16 [KERNEL.220] [SHELL.3] */ DWORD WINAPI RegCloseKey16( HKEY hkey ) { fix_win16_hkey( &hkey ); return RegCloseKey( hkey ); } /****************************************************************************** * RegSetValue16 [KERNEL.221] [SHELL.5] */ DWORD WINAPI RegSetValue16( HKEY hkey, LPCSTR name, DWORD type, LPCSTR data, DWORD count ) { fix_win16_hkey( &hkey ); return RegSetValueA( hkey, name, type, data, count ); } /****************************************************************************** * RegDeleteValue16 [KERNEL.222] */ DWORD WINAPI RegDeleteValue16( HKEY hkey, LPSTR name ) { fix_win16_hkey( &hkey ); return RegDeleteValueA( hkey, name ); } /****************************************************************************** * RegEnumValue16 [KERNEL.223] */ DWORD WINAPI RegEnumValue16( HKEY hkey, DWORD index, LPSTR value, LPDWORD val_count, LPDWORD reserved, LPDWORD type, LPBYTE data, LPDWORD count ) { fix_win16_hkey( &hkey ); return RegEnumValueA( hkey, index, value, val_count, reserved, type, data, count ); } /****************************************************************************** * RegQueryValue16 [KERNEL.224] [SHELL.6] * * NOTES * Is this HACK still applicable? * * HACK * The 16bit RegQueryValue doesn't handle selectorblocks anyway, so we just * mask out the high 16 bit. This (not so much incidently) hopefully fixes * Aldus FH4) */ DWORD WINAPI RegQueryValue16( HKEY hkey, LPCSTR name, LPSTR data, LPDWORD count ) { fix_win16_hkey( &hkey ); if (count) *count &= 0xffff; return RegQueryValueA( hkey, name, data, count ); } /****************************************************************************** * RegQueryValueEx16 [KERNEL.225] */ DWORD WINAPI RegQueryValueEx16( HKEY hkey, LPCSTR name, LPDWORD reserved, LPDWORD type, LPBYTE data, LPDWORD count ) { fix_win16_hkey( &hkey ); return RegQueryValueExA( hkey, name, reserved, type, data, count ); } /****************************************************************************** * RegSetValueEx16 [KERNEL.226] */ DWORD WINAPI RegSetValueEx16( HKEY hkey, LPCSTR name, DWORD reserved, DWORD type, CONST BYTE *data, DWORD count ) { fix_win16_hkey( &hkey ); if (!count && (type==REG_SZ)) count = strlen(data); return RegSetValueExA( hkey, name, reserved, type, data, count ); } /****************************************************************************** * RegFlushKey16 [KERNEL.227] */ DWORD WINAPI RegFlushKey16( HKEY hkey ) { fix_win16_hkey( &hkey ); return RegFlushKey( hkey ); }