3173 lines
84 KiB
C
3173 lines
84 KiB
C
/*
|
|
* ntreg.c - NT Registry Hive access library
|
|
*
|
|
* 2010-jun: Patches from Frediano Ziglio adding support for wide characters
|
|
* and some bugfixes. Thank you!
|
|
* 2008-mar: Type QWORD (XP/Vista and newer) now recognized
|
|
* 2008-mar: Most functions accepting a path now also have a parameter specifying if
|
|
* the search should be exact or on first match basis
|
|
* 2008-mar: Fixed bug which skipped first indirect index table when deleting keys,
|
|
* usually leading to endless loop when recursive deleting.
|
|
* 2008-mar: Export to .reg file by Leo von Klenze, expanded a bit by me.
|
|
* 2008-mar: 64 bit compatible patch by Mike Doty, via Alon Bar-Lev
|
|
* http://bugs.gentoo.org/show_bug.cgi?id=185411
|
|
* 2007-sep: Verbosity/debug messages minor changes
|
|
* 2007-apr: LGPL license.
|
|
* 2004-aug: Deep indirect index support. NT351 support. Recursive delete.
|
|
* Debugged a lot in allocation routines. Still no expansion.
|
|
* 2004-jan: Verbosity updates
|
|
* 2003-jan: Allocation of new data, supports adding/deleting keys & stuff.
|
|
* Missing is expanding the file.
|
|
* 2003-jan: Seems there may be garbage pages at end of file, not zero pages
|
|
* now stops enumerating at first non 'hbin' page.
|
|
*
|
|
* NOTE: The API is not frozen. It can and will change every release.
|
|
*
|
|
*****
|
|
*
|
|
* NTREG - Window registry file reader / writer library
|
|
* Copyright (c) 1997-2010 Petter Nordahl-Hagen.
|
|
*
|
|
* 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;
|
|
* version 2.1 of the License.
|
|
*
|
|
* 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.
|
|
* See file LGPL.txt for the full license.
|
|
*
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <inttypes.h>
|
|
|
|
#include "ntreg.h"
|
|
|
|
/* Set to abort() and debug on more critical errors */
|
|
#define DOCORE 1
|
|
|
|
#define ZEROFILL 1 /* Fill blocks with zeroes when allocating and deallocating */
|
|
#define ZEROFILLONLOAD 0 /* Fill blocks marked as unused/deallocated with zeroes on load. FOR DEBUG */
|
|
|
|
const char ntreg_version[] = "ntreg lib routines, v0.94.1 100627, (c) Petter N Hagen";
|
|
|
|
const char *val_types[REG_MAX+1] = {
|
|
"REG_NONE", "REG_SZ", "REG_EXPAND_SZ", "REG_BINARY", "REG_DWORD", /* 0 - 4 */
|
|
"REG_DWORD_BIG_ENDIAN", "REG_LINK", /* 5 - 6 */
|
|
"REG_MULTI_SZ", "REG_RESOUCE_LIST", "REG_FULL_RES_DESC", "REG_RES_REQ", /* 7 - 10 */
|
|
"REG_QWORD", /* 11 */
|
|
};
|
|
|
|
static char * string_regw2prog(void *string, int len);
|
|
static char * string_rega2prog(void *string, int len);
|
|
static char * string_prog2regw(void *string, int len, int* out_len);
|
|
static void string_prog2rega(char *string, int len);
|
|
|
|
/* Utility routines */
|
|
char *str_dup( const char *str )
|
|
{
|
|
char *str_new;
|
|
|
|
if (!str)
|
|
return 0 ;
|
|
|
|
CREATE( str_new, char, strlen(str) + 1 );
|
|
strcpy( str_new, str );
|
|
return str_new;
|
|
}
|
|
|
|
int fmyinput(char *prmpt, char *ibuf, int maxlen)
|
|
{
|
|
|
|
printf("%s",prmpt);
|
|
|
|
fgets(ibuf,maxlen+1,stdin);
|
|
|
|
ibuf[strlen(ibuf)-1] = 0;
|
|
|
|
return(strlen(ibuf));
|
|
}
|
|
|
|
/* Print len number of hexbytes */
|
|
|
|
void hexprnt(char *s, unsigned char *bytes, int len)
|
|
{
|
|
int i;
|
|
|
|
printf("%s",s);
|
|
for (i = 0; i < len; i++) {
|
|
printf("%02x ",bytes[i]);
|
|
}
|
|
printf("\n");
|
|
}
|
|
|
|
/* HexDump all or a part of some buffer */
|
|
|
|
void hexdump(char *hbuf, int start, int stop, int ascii)
|
|
{
|
|
char c;
|
|
int diff,i;
|
|
|
|
while (start < stop ) {
|
|
|
|
diff = stop - start;
|
|
if (diff > 16) diff = 16;
|
|
|
|
printf(":%05X ",start);
|
|
|
|
for (i = 0; i < diff; i++) {
|
|
printf("%02X ",(unsigned char)*(hbuf+start+i));
|
|
}
|
|
if (ascii) {
|
|
for (i = diff; i < 16; i++) printf(" ");
|
|
for (i = 0; i < diff; i++) {
|
|
c = *(hbuf+start+i);
|
|
printf("%c", isprint(c) ? c : '.');
|
|
}
|
|
}
|
|
printf("\n");
|
|
start += 16;
|
|
}
|
|
}
|
|
|
|
/* General search routine, find something in something else */
|
|
int find_in_buf(char *buf, char *what, int sz, int len, int start)
|
|
{
|
|
int i;
|
|
|
|
for (; start < sz; start++) {
|
|
for (i = 0; i < len; i++) {
|
|
if (*(buf+start+i) != *(what+i)) break;
|
|
}
|
|
if (i == len) return(start);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/* Get INTEGER from memory. This is probably low-endian specific? */
|
|
int get_int( char *array )
|
|
{
|
|
return ((array[0]&0xff) + ((array[1]<<8)&0xff00) +
|
|
((array[2]<<16)&0xff0000) +
|
|
((array[3]<<24)&0xff000000));
|
|
}
|
|
|
|
|
|
/* Quick and dirty UNICODE to std. ascii */
|
|
|
|
void cheap_uni2ascii(char *src, char *dest, int l)
|
|
{
|
|
|
|
for (; l > 0; l -=2) {
|
|
*dest = *src;
|
|
dest++; src +=2;
|
|
}
|
|
*dest = 0;
|
|
}
|
|
|
|
|
|
/* Quick and dirty ascii to unicode */
|
|
|
|
void cheap_ascii2uni(char *src, char *dest, int l)
|
|
{
|
|
for (; l > 0; l--) {
|
|
*dest++ = *src++;
|
|
*dest++ = 0;
|
|
|
|
}
|
|
}
|
|
|
|
void skipspace(char **c)
|
|
{
|
|
while( **c == ' ' ) (*c)++;
|
|
}
|
|
|
|
int gethex(char **c)
|
|
{
|
|
int value;
|
|
|
|
skipspace(c);
|
|
|
|
if (!(**c)) return(0);
|
|
|
|
sscanf(*c,"%x",&value);
|
|
|
|
while( **c != ' ' && (**c)) (*c)++;
|
|
|
|
return(value);
|
|
}
|
|
|
|
/* Get a string of HEX bytes (space separated),
|
|
* or if first char is ' get an ASCII string instead.
|
|
*/
|
|
|
|
int gethexorstr(char **c, char *wb)
|
|
{
|
|
int l = 0;
|
|
|
|
skipspace(c);
|
|
|
|
if ( **c == '\'') {
|
|
(*c)++;
|
|
while ( **c ) {
|
|
*(wb++) = *((*c)++);
|
|
l++;
|
|
}
|
|
} else {
|
|
do {
|
|
*(wb++) = gethex(c);
|
|
l++;
|
|
skipspace(c);
|
|
} while ( **c );
|
|
}
|
|
return(l);
|
|
}
|
|
|
|
/* Simple buffer debugger, returns 1 if buffer dirty/edited */
|
|
|
|
int debugit(char *buf, int sz)
|
|
{
|
|
|
|
|
|
char inbuf[100],whatbuf[100],*bp;
|
|
|
|
int dirty=0,to,from,l,i,j,wlen,cofs = 0;
|
|
|
|
printf("Buffer debugger. '?' for help.\n");
|
|
|
|
while (1) {
|
|
l = fmyinput(".",inbuf,90);
|
|
bp = inbuf;
|
|
|
|
skipspace(&bp);
|
|
|
|
if (l > 0 && *bp) {
|
|
switch(*bp) {
|
|
case 'd' :
|
|
bp++;
|
|
if (*bp) {
|
|
from = gethex(&bp);
|
|
to = gethex(&bp);
|
|
} else {
|
|
from = cofs; to = 0;
|
|
}
|
|
if (to == 0) to = from + 0x100;
|
|
if (to > sz) to = sz;
|
|
hexdump(buf,from,to,1);
|
|
cofs = to;
|
|
break;
|
|
case 'a' :
|
|
bp++;
|
|
if (*bp) {
|
|
from = gethex(&bp);
|
|
to = gethex(&bp);
|
|
} else {
|
|
from = cofs; to = 0;
|
|
}
|
|
if (to == 0) to = from + 0x100;
|
|
if (to > sz) to = sz;
|
|
hexdump(buf,from,to,0);
|
|
cofs = to;
|
|
break;
|
|
#if 0
|
|
case 'k' :
|
|
bp++;
|
|
if (*bp) {
|
|
from = gethex(&bp);
|
|
} else {
|
|
from = cofs;
|
|
}
|
|
if (to > sz) to = sz;
|
|
parse_block(from,1);
|
|
cofs = to;
|
|
break;
|
|
#endif
|
|
#if 0
|
|
case 'l' :
|
|
bp++;
|
|
if (*bp) {
|
|
from = gethex(&bp);
|
|
} else {
|
|
from = cofs;
|
|
}
|
|
if (to > sz) to = sz;
|
|
nk_ls(from+4,0);
|
|
cofs = to;
|
|
break;
|
|
#endif
|
|
case 'q':
|
|
return(0);
|
|
break;
|
|
case 's':
|
|
if (!dirty) printf("Buffer has not changed, no need to write..\n");
|
|
return(dirty);
|
|
break;
|
|
case 'h':
|
|
bp++;
|
|
if (*bp == 'a') {
|
|
from = 0;
|
|
to = sz;
|
|
bp++;
|
|
} else {
|
|
from = gethex(&bp);
|
|
to = gethex(&bp);
|
|
}
|
|
wlen = gethexorstr(&bp,whatbuf);
|
|
if (to > sz) to = sz;
|
|
printf("from: %x, to: %x, wlen: %d\n",from,to,wlen);
|
|
for (i = from; i < to; i++) {
|
|
for (j = 0; j < wlen; j++) {
|
|
if ( *(buf+i+j) != *(whatbuf+j)) break;
|
|
}
|
|
if (j == wlen) printf("%06x ",i);
|
|
}
|
|
printf("\n");
|
|
break;
|
|
case ':':
|
|
bp++;
|
|
if (!*bp) break;
|
|
from = gethex(&bp);
|
|
wlen = gethexorstr(&bp,whatbuf);
|
|
|
|
printf("from: %x, wlen: %d\n",from,wlen);
|
|
|
|
memcpy(buf+from,whatbuf,wlen);
|
|
dirty = 1;
|
|
break;
|
|
#if 0
|
|
case 'p':
|
|
j = 0;
|
|
if (*(++bp) != 0) {
|
|
from = gethex(&bp);
|
|
}
|
|
if (*(++bp) != 0) {
|
|
j = gethex(&bp);
|
|
}
|
|
printf("from: %x, rid: %x\n",from,j);
|
|
seek_n_destroy(from,j,500,0);
|
|
break;
|
|
#endif
|
|
case '?':
|
|
printf("d [<from>] [<to>] - dump buffer within range\n");
|
|
printf("a [<from>] [<to>] - same as d, but without ascii-part (for cut'n'paste)\n");
|
|
printf(": <offset> <hexbyte> [<hexbyte> ...] - change bytes\n");
|
|
printf("h <from> <to> <hexbyte> [<hexbyte> ...] - hunt (search) for bytes\n");
|
|
printf("ha <hexbyte> [<hexbyte] - Hunt all (whole buffer)\n");
|
|
printf("s - save & quit\n");
|
|
printf("q - quit (no save)\n");
|
|
printf(" instead of <hexbyte> etc. you may give 'string to enter/search a string\n");
|
|
break;
|
|
default:
|
|
printf("?\n");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/* ========================================================================= */
|
|
|
|
/* The following routines are mostly for debugging, I used it
|
|
* much during discovery. the -t command line option uses it,
|
|
* also the 'st' and 's' from the editor & hexdebugger.
|
|
* All offsets shown in these are unadjusted (ie you must add
|
|
* headerpage (most often 0x1000) to get file offset)
|
|
*/
|
|
|
|
/* Parse the nk datablock
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_nk(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
|
|
struct nk_key *key;
|
|
int i;
|
|
|
|
printf("== nk at offset %0x\n",vofs);
|
|
|
|
/* #define D_OFFS2(o) ( (void *)&(key->o)-(void *)hdesc->buffer-vofs ) */
|
|
#define D_OFFS(o) ( (void *)&(key->o)-(void *)hdesc->buffer-vofs )
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx type = 0x%02x %s\n", D_OFFS(type) ,key->type,
|
|
(key->type == KEY_ROOT ? "ROOT_KEY" : "") );
|
|
printf("%04lx timestamp skipped\n", D_OFFS(timestamp) );
|
|
printf("%04lx parent key offset = 0x%0x\n", D_OFFS(ofs_parent) ,key->ofs_parent);
|
|
printf("%04lx number of subkeys = %d\n", D_OFFS(no_subkeys),key->no_subkeys);
|
|
printf("%04lx lf-record offset = 0x%0x\n",D_OFFS(ofs_lf),key->ofs_lf);
|
|
printf("%04lx number of values = %d\n", D_OFFS(no_values),key->no_values);
|
|
printf("%04lx val-list offset = 0x%0x\n",D_OFFS(ofs_vallist),key->ofs_vallist);
|
|
printf("%04lx sk-record offset = 0x%0x\n",D_OFFS(ofs_sk),key->ofs_sk);
|
|
printf("%04lx classname offset = 0x%0x\n",D_OFFS(ofs_classnam),key->ofs_classnam);
|
|
printf("%04lx *unused?* = 0x%0x\n",D_OFFS(dummy4),key->dummy4);
|
|
printf("%04lx name length = %d\n", D_OFFS(len_name),key->len_name);
|
|
printf("%04lx classname length = %d\n", D_OFFS(len_classnam),key->len_classnam);
|
|
|
|
printf("%04lx Key name: <",D_OFFS(keyname) );
|
|
for(i = 0; i < key->len_name; i++) putchar(key->keyname[i]);
|
|
printf(">\n== End of key info.\n");
|
|
|
|
}
|
|
|
|
/* Parse the vk datablock
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_vk(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct vk_key *key;
|
|
int i;
|
|
|
|
printf("== vk at offset %0x\n",vofs);
|
|
|
|
|
|
key = (struct vk_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx name length = %d (0x%0x)\n", D_OFFS(len_name),
|
|
key->len_name, key->len_name );
|
|
printf("%04lx length of data = %d (0x%0x)\n", D_OFFS(len_data),
|
|
key->len_data, key->len_data );
|
|
printf("%04lx data offset = 0x%0x\n",D_OFFS(ofs_data),key->ofs_data);
|
|
printf("%04lx value type = 0x%0x %s\n", D_OFFS(val_type), key->val_type,
|
|
(key->val_type <= REG_MAX ? val_types[key->val_type] : "(unknown)") ) ;
|
|
|
|
printf("%04lx flag = 0x%0x\n",D_OFFS(flag),key->flag);
|
|
printf("%04lx *unused?* = 0x%0x\n",D_OFFS(dummy1),key->dummy1);
|
|
|
|
printf("%04lx Key name: <",D_OFFS(keyname) );
|
|
for(i = 0; i < key->len_name; i++) putchar(key->keyname[i]);
|
|
printf(">\n== End of key info.\n");
|
|
|
|
}
|
|
|
|
/* Parse the sk datablock
|
|
* Gee, this is the security info. Who cares? *evil grin*
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_sk(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct sk_key *key;
|
|
/* int i; */
|
|
|
|
printf("== sk at offset %0x\n",vofs);
|
|
|
|
key = (struct sk_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx *unused?* = %d\n" , D_OFFS(dummy1), key->dummy1 );
|
|
printf("%04lx Offset to prev sk = 0x%0x\n", D_OFFS(ofs_prevsk), key->ofs_prevsk);
|
|
printf("%04lx Offset to next sk = 0x%0x\n", D_OFFS(ofs_nextsk), key->ofs_nextsk);
|
|
printf("%04lx Usage counter = %d (0x%0x)\n", D_OFFS(no_usage),
|
|
key->no_usage,key->no_usage);
|
|
printf("%04lx Security data len = %d (0x%0x)\n", D_OFFS(len_sk),
|
|
key->len_sk,key->len_sk);
|
|
|
|
printf("== End of key info.\n");
|
|
|
|
}
|
|
|
|
|
|
/* Parse the lf datablock (>4.0 'nk' offsets lookuptable)
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_lf(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct lf_key *key;
|
|
int i;
|
|
|
|
printf("== lf at offset %0x\n",vofs);
|
|
|
|
key = (struct lf_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx number of keys = %d\n", D_OFFS(no_keys), key->no_keys );
|
|
|
|
for(i = 0; i < key->no_keys; i++) {
|
|
printf("%04lx %3d Offset: 0x%0x - <%c%c%c%c>\n",
|
|
D_OFFS(hash[i].ofs_nk), i,
|
|
key->hash[i].ofs_nk,
|
|
key->hash[i].name[0],
|
|
key->hash[i].name[1],
|
|
key->hash[i].name[2],
|
|
key->hash[i].name[3] );
|
|
}
|
|
|
|
printf("== End of key info.\n");
|
|
|
|
}
|
|
|
|
/* Parse the lh datablock (WinXP offsets lookuptable)
|
|
* vofs = offset into struct (after size linkage)
|
|
* The hash is most likely a base 37 conversion of the name string
|
|
*/
|
|
void parse_lh(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct lf_key *key;
|
|
int i;
|
|
|
|
printf("== lh at offset %0x\n",vofs);
|
|
|
|
key = (struct lf_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx number of keys = %d\n", D_OFFS(no_keys), key->no_keys );
|
|
|
|
for(i = 0; i < key->no_keys; i++) {
|
|
printf("%04lx %3d Offset: 0x%0x - <hash: %08x>\n",
|
|
D_OFFS(lh_hash[i].ofs_nk), i,
|
|
key->lh_hash[i].ofs_nk,
|
|
key->lh_hash[i].hash );
|
|
}
|
|
|
|
printf("== End of key info.\n");
|
|
|
|
}
|
|
|
|
|
|
/* Parse the li datablock (3.x 'nk' offsets list)
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_li(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct li_key *key;
|
|
int i;
|
|
|
|
printf("== li at offset %0x\n",vofs);
|
|
|
|
/* #define D_OFFS(o) ( (void *)&(key->o)-(void *)hdesc->buffer-vofs ) */
|
|
|
|
key = (struct li_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx number of keys = %d\n", D_OFFS(no_keys), key->no_keys );
|
|
|
|
for(i = 0; i < key->no_keys; i++) {
|
|
printf("%04lx %3d Offset: 0x%0x\n",
|
|
D_OFFS(hash[i].ofs_nk), i,
|
|
key->hash[i].ofs_nk);
|
|
}
|
|
printf("== End of key info.\n");
|
|
|
|
}
|
|
|
|
/* Parse the ri subindex-datablock
|
|
* (Used to list li/lf/lh's when ~>500keys)
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
void parse_ri(struct hive *hdesc, int vofs, int blen)
|
|
{
|
|
struct ri_key *key;
|
|
int i;
|
|
|
|
printf("== ri at offset %0x\n",vofs);
|
|
|
|
/* #define D_OFFS(o) ( (void *)&(key->o)-(void *)hdesc->buffer-vofs ) */
|
|
|
|
key = (struct ri_key *)(hdesc->buffer + vofs);
|
|
printf("%04lx number of subindices = %d\n", D_OFFS(no_lis), key->no_lis );
|
|
|
|
for(i = 0; i < key->no_lis; i++) {
|
|
printf("%04lx %3d Offset: 0x%0x\n",
|
|
D_OFFS(hash[i].ofs_li), i,
|
|
key->hash[i].ofs_li);
|
|
}
|
|
printf("== End of key info.\n");
|
|
|
|
}
|
|
|
|
|
|
/* Parse the datablock
|
|
* vofs = offset into struct (after size linkage)
|
|
*/
|
|
|
|
int parse_block(struct hive *hdesc, int vofs,int verbose)
|
|
{
|
|
unsigned short id;
|
|
int seglen;
|
|
|
|
seglen = get_int(hdesc->buffer+vofs);
|
|
|
|
if (verbose || seglen == 0) {
|
|
printf("** Block at offset %0x\n",vofs);
|
|
printf("seglen: %d, %u, 0x%0x\n",seglen,seglen,seglen);
|
|
}
|
|
if (seglen == 0) {
|
|
printf("Whoops! FATAL! Zero data block size! (not registry or corrupt file?)\n");
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
return(0);
|
|
}
|
|
|
|
if (seglen < 0) {
|
|
seglen = -seglen;
|
|
hdesc->usetot += seglen;
|
|
hdesc->useblk++;
|
|
if (verbose) {
|
|
printf("USED BLOCK: %d, 0x%0x\n",seglen,seglen);
|
|
/* hexdump(hdesc->buffer,vofs,vofs+seglen+4,1); */
|
|
}
|
|
} else {
|
|
hdesc->unusetot += seglen;
|
|
hdesc->unuseblk++;
|
|
/* Useful to zero blocks we think are empty when debugging.. */
|
|
#if ZEROFILLONLOAD
|
|
bzero(hdesc->buffer+vofs+4,seglen-4);
|
|
#endif
|
|
|
|
if (verbose) {
|
|
printf("FREE BLOCK!\n");
|
|
/* hexdump(hdesc->buffer,vofs,vofs+seglen+4,1); */
|
|
}
|
|
}
|
|
|
|
|
|
/* printf("Seglen: 0x%02x\n",seglen & 0xff ); */
|
|
|
|
vofs += 4;
|
|
id = (*(hdesc->buffer + vofs)<<8) + *(hdesc->buffer+vofs+1);
|
|
|
|
if (verbose) {
|
|
switch (id) {
|
|
case 0x6e6b: /* nk */
|
|
parse_nk(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x766b: /* vk */
|
|
parse_vk(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x6c66: /* lf */
|
|
parse_lf(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x6c68: /* lh */
|
|
parse_lh(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x6c69: /* li */
|
|
parse_li(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x736b: /* sk */
|
|
parse_sk(hdesc, vofs, seglen);
|
|
break;
|
|
case 0x7269: /* ri */
|
|
parse_ri(hdesc, vofs, seglen);
|
|
break;
|
|
default:
|
|
printf("value data, or not handeled yet!\n");
|
|
break;
|
|
}
|
|
}
|
|
return(seglen);
|
|
}
|
|
|
|
/* ================================================================ */
|
|
/* Scan and allocation routines */
|
|
|
|
/* Find start of page given a current pointer into the buffer
|
|
* hdesc = hive
|
|
* vofs = offset pointer into buffer
|
|
* returns: offset to start of page (and page header)
|
|
*/
|
|
|
|
int find_page_start(struct hive *hdesc, int vofs)
|
|
{
|
|
int r,prev;
|
|
struct hbin_page *h;
|
|
|
|
/* Again, assume start at 0x1000 */
|
|
|
|
r = 0x1000;
|
|
while (r < hdesc->size) {
|
|
prev = r;
|
|
h = (struct hbin_page *)(hdesc->buffer + r);
|
|
if (h->id != 0x6E696268) return(0);
|
|
if (h->ofs_next == 0) {
|
|
printf("find_page_start: zero len or ofs_next found in page at 0x%x\n",r);
|
|
return(0);
|
|
}
|
|
r += h->ofs_next;
|
|
if (r > vofs) return (prev);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/* Find free space in page
|
|
* size = requested size in bytes
|
|
* pofs = offset to start of actual page header
|
|
* returns: offset to free block, or 0 for error
|
|
*/
|
|
|
|
#define FB_DEBUG 0
|
|
|
|
int find_free_blk(struct hive *hdesc, int pofs, int size)
|
|
{
|
|
int vofs = pofs + 0x20;
|
|
int seglen;
|
|
struct hbin_page *p;
|
|
|
|
p = (struct hbin_page *)(hdesc->buffer + pofs);
|
|
|
|
while (vofs-pofs < (p->ofs_next - HBIN_ENDFILL)) {
|
|
|
|
seglen = get_int(hdesc->buffer+vofs);
|
|
|
|
#if FB_DEBUG
|
|
printf("** Block at offset %0x\n",vofs);
|
|
printf("seglen: %d, %u, 0x%0x\n",seglen,seglen,seglen);
|
|
#endif
|
|
|
|
if (seglen == 0) {
|
|
printf("find_free_blk: FATAL! Zero data block size! (not registry or corrupt file?)\n");
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
return(0);
|
|
}
|
|
|
|
if (seglen < 0) {
|
|
seglen = -seglen;
|
|
#if FB_DEBUG
|
|
printf("USED BLOCK: %d, 0x%0x\n",seglen,seglen);
|
|
#endif
|
|
/* hexdump(hdesc->buffer,vofs,vofs+seglen+4,1); */
|
|
} else {
|
|
#if FB_DEBUG
|
|
printf("FREE BLOCK!\n");
|
|
#endif
|
|
/* hexdump(hdesc->buffer,vofs,vofs+seglen+4,1); */
|
|
if (seglen >= size) {
|
|
#if FB_DEBUG
|
|
printf("find_free_blk: found size %d block at 0x%x\n",seglen,vofs);
|
|
#endif
|
|
#if 0
|
|
if (vofs == 0x19fb8) {
|
|
printf("find_free_blk: vofs = %x, seglen = %x\n",vofs,seglen);
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
abort();
|
|
}
|
|
#endif
|
|
return(vofs);
|
|
}
|
|
}
|
|
vofs += seglen;
|
|
}
|
|
return(0);
|
|
|
|
}
|
|
|
|
#undef FB_DEBUG
|
|
|
|
/* Search pages from start to find free block
|
|
* hdesc - hive
|
|
* size - space requested, in bytes
|
|
* returns: offset to free block, 0 if not found or error
|
|
*/
|
|
|
|
int find_free(struct hive *hdesc, int size)
|
|
{
|
|
int r,blk;
|
|
struct hbin_page *h;
|
|
|
|
/* Align to 8 byte boundary */
|
|
if (size & 7) size += (8 - (size & 7));
|
|
|
|
/* Again, assume start at 0x1000 */
|
|
|
|
r = 0x1000;
|
|
while (r < hdesc->size) {
|
|
h = (struct hbin_page *)(hdesc->buffer + r);
|
|
if (h->id != 0x6E696268) return(0);
|
|
if (h->ofs_next == 0) {
|
|
printf("find_free: zero len or ofs_next found in page at 0x%x\n",r);
|
|
return(0);
|
|
}
|
|
blk = find_free_blk(hdesc,r,size);
|
|
if (blk) return (blk);
|
|
r += h->ofs_next;
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/* Allocate a block of requested size if possible
|
|
* hdesc - hive
|
|
* pofs - If >0 will try this page first (ptr may be inside page)
|
|
* size - number of bytes to allocate
|
|
* returns: 0 - failed, else pointer to allocated block.
|
|
* This function WILL CHANGE THE HIVE (change block linkage) if it
|
|
* succeeds.
|
|
*/
|
|
|
|
int alloc_block(struct hive *hdesc, int ofs, int size)
|
|
{
|
|
int pofs = 0;
|
|
int blk = 0;
|
|
int trail, trailsize, oldsz;
|
|
|
|
if (hdesc->state & HMODE_NOALLOC) {
|
|
printf("alloc_block: ERROR: Hive %s is in no allocation safe mode,"
|
|
"new space not allocated. Operation will fail!\n", hdesc->filename);
|
|
return(0);
|
|
}
|
|
|
|
size += 4; /* Add linkage */
|
|
if (size & 7) size += (8 - (size & 7));
|
|
|
|
/* Check current page first */
|
|
if (ofs) {
|
|
pofs = find_page_start(hdesc,ofs);
|
|
blk = find_free_blk(hdesc,pofs,size);
|
|
}
|
|
|
|
/* Then check whole hive */
|
|
if (!blk) {
|
|
blk = find_free(hdesc,size);
|
|
}
|
|
|
|
if (blk) { /* Got the space */
|
|
oldsz = get_int(hdesc->buffer+blk);
|
|
#if 0
|
|
printf("Block at : %x\n",blk);
|
|
printf("Old block size is: %x\n",oldsz);
|
|
printf("New block size is: %x\n",size);
|
|
#endif
|
|
trailsize = oldsz - size;
|
|
|
|
if (trailsize == 4) {
|
|
trailsize = 0;
|
|
size += 4;
|
|
}
|
|
|
|
#if 1
|
|
if (trailsize & 7) { /* Trail must be 8 aligned */
|
|
trailsize -= (8 - (trailsize & 7));
|
|
size += (8 - (trailsize & 7));
|
|
}
|
|
if (trailsize == 4) {
|
|
trailsize = 0;
|
|
size += 4;
|
|
}
|
|
#endif
|
|
|
|
#if 0
|
|
printf("trail after comp: %x\n",trailsize);
|
|
printf("size after comp: %x\n",size);
|
|
#endif
|
|
|
|
/* Now change pointers on this to reflect new size */
|
|
*(int *)((hdesc->buffer)+blk) = -(size);
|
|
/* If the fit was exact (unused block was same size as wee need)
|
|
* there is no need for more, else make free block after end
|
|
* of newly allocated one */
|
|
|
|
hdesc->useblk++;
|
|
hdesc->unuseblk--;
|
|
hdesc->usetot += size;
|
|
hdesc->unusetot -= size;
|
|
|
|
if (trailsize) {
|
|
trail = blk + size;
|
|
|
|
*(int *)((hdesc->buffer)+trail) = (int)trailsize;
|
|
|
|
hdesc->useblk++; /* This will keep blockcount */
|
|
hdesc->unuseblk--;
|
|
hdesc->usetot += 4; /* But account for more linkage bytes */
|
|
hdesc->unusetot -= 4;
|
|
|
|
}
|
|
/* Clear the block data, makes it easier to debug */
|
|
#if ZEROFILL
|
|
bzero( (void *)(hdesc->buffer+blk+4), size-4);
|
|
#endif
|
|
|
|
hdesc->state |= HMODE_DIRTY;
|
|
|
|
return(blk);
|
|
} else {
|
|
printf("alloc_block: failed to alloc %d bytes, and hive expansion not implemented yet!\n",size);
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
/* Free a block in registry
|
|
* hdesc - hive
|
|
* blk - offset of block, MUST POINT TO THE LINKAGE!
|
|
* returns bytes freed (incl linkage bytes) or 0 if fail
|
|
* Will CHANGE HIVE IF SUCCESSFUL (changes linkage)
|
|
*/
|
|
|
|
#define FB_DEBUG 0
|
|
|
|
int free_block(struct hive *hdesc, int blk)
|
|
{
|
|
int pofs,vofs,seglen,prev,next,nextsz,prevsz,size;
|
|
struct hbin_page *p;
|
|
|
|
if (hdesc->state & HMODE_NOALLOC) {
|
|
printf("free_block: ERROR: Hive %s is in no allocation safe mode,"
|
|
"space not freed. Operation will fail!\n", hdesc->filename);
|
|
return(0);
|
|
}
|
|
|
|
size = get_int(hdesc->buffer+blk);
|
|
if (size >= 0) {
|
|
printf("free_block: trying to free already free block!\n");
|
|
#ifdef DOCORE
|
|
printf("blk = %x\n",blk);
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
abort();
|
|
#endif
|
|
return(0);
|
|
}
|
|
size = -size;
|
|
|
|
/* So, we must find start of the block BEFORE us */
|
|
pofs = find_page_start(hdesc,blk);
|
|
if (!pofs) return(0);
|
|
|
|
p = (struct hbin_page *)(hdesc->buffer + pofs);
|
|
vofs = pofs + 0x20;
|
|
|
|
prevsz = -32;
|
|
|
|
if (vofs != blk) { /* Block is not at start of page? */
|
|
while (vofs-pofs < (p->ofs_next - HBIN_ENDFILL) ) {
|
|
|
|
seglen = get_int(hdesc->buffer+vofs);
|
|
|
|
if (seglen == 0) {
|
|
printf("free_block: EEEK! Zero data block size! (not registry or corrupt file?)\n");
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
return(0);
|
|
}
|
|
|
|
if (seglen < 0) {
|
|
seglen = -seglen;
|
|
/* hexdump(hdesc->buffer,vofs,vofs+seglen+4,1); */
|
|
}
|
|
prev = vofs;
|
|
vofs += seglen;
|
|
if (vofs == blk) break;
|
|
}
|
|
|
|
if (vofs != blk) {
|
|
printf("free_block: ran off end of page!?!? Error in chains?\n");
|
|
#ifdef DOCORE
|
|
printf("vofs = %x, pofs = %x, blk = %x\n",vofs,pofs,blk);
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
abort();
|
|
#endif
|
|
return(0);
|
|
}
|
|
|
|
prevsz = get_int(hdesc->buffer+prev);
|
|
|
|
}
|
|
|
|
/* We also need details on next block (unless at end of page) */
|
|
next = blk + size;
|
|
|
|
nextsz = 0;
|
|
if (next-pofs < (p->ofs_next - HBIN_ENDFILL) ) nextsz = get_int(hdesc->buffer+next);
|
|
|
|
#if 0
|
|
printf("offset prev : %x , blk: %x , next: %x\n",prev,blk,next);
|
|
printf("size prev : %x , blk: %x , next: %x\n",prevsz,size,nextsz);
|
|
#endif
|
|
|
|
/* Now check if next block is free, if so merge it with the one to be freed */
|
|
if ( nextsz > 0) {
|
|
#if 0
|
|
printf("Swallow next\n");
|
|
#endif
|
|
size += nextsz; /* Swallow it in current block */
|
|
hdesc->useblk--;
|
|
hdesc->usetot -= 4;
|
|
hdesc->unusetot -= 4; /* FIXME !??!?? */
|
|
}
|
|
|
|
/* Now free the block (possibly with ajusted size as above) */
|
|
#if ZEROFILL
|
|
bzero( (void *)(hdesc->buffer+blk), size);
|
|
#endif
|
|
|
|
*(int *)((hdesc->buffer)+blk) = (int)size;
|
|
hdesc->usetot -= size;
|
|
hdesc->unusetot -= size; /* FIXME !?!? */
|
|
hdesc->unuseblk--;
|
|
|
|
hdesc->state |= HMODE_DIRTY;
|
|
|
|
/* Check if previous block is also free, if so, merge.. */
|
|
if (prevsz > 0) {
|
|
#if 0
|
|
printf("Swallow prev\n");
|
|
#endif
|
|
hdesc->usetot -= prevsz;
|
|
hdesc->unusetot += prevsz;
|
|
prevsz += size;
|
|
/* And swallow current.. */
|
|
#if ZEROFILL
|
|
bzero( (void *)(hdesc->buffer+prev), prevsz);
|
|
#endif
|
|
*(int *)((hdesc->buffer)+prev) = (int)prevsz;
|
|
hdesc->useblk--;
|
|
return(prevsz);
|
|
}
|
|
return(size);
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* ================================================================ */
|
|
|
|
/* ** Registry manipulation routines ** */
|
|
|
|
|
|
|
|
/* "directory scan", return next name/pointer of a subkey on each call
|
|
* nkofs = offset to directory to scan
|
|
* lfofs = pointer to int to hold the current scan position,
|
|
* set position to 0 to start.
|
|
* sptr = pointer to struct to hold a single result
|
|
* returns: -1 = error. 0 = end of key. 1 = more subkeys to scan
|
|
* NOTE: caller must free the name-buffer (struct ex_data *name)
|
|
*/
|
|
int ex_next_n(struct hive *hdesc, int nkofs, int *count, int *countri, struct ex_data *sptr)
|
|
{
|
|
struct nk_key *key, *newnkkey;
|
|
int newnkofs;
|
|
struct lf_key *lfkey;
|
|
struct li_key *likey;
|
|
struct ri_key *rikey;
|
|
|
|
|
|
if (!nkofs) return(-1);
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
if (key->id != 0x6b6e) {
|
|
printf("ex_next error: Not a 'nk' node at 0x%0x\n",nkofs);
|
|
return(-1);
|
|
}
|
|
|
|
#define EXNDEBUG 0
|
|
|
|
lfkey = (struct lf_key *)(hdesc->buffer + key->ofs_lf + 0x1004);
|
|
rikey = (struct ri_key *)(hdesc->buffer + key->ofs_lf + 0x1004);
|
|
|
|
if (rikey->id == 0x6972) { /* Is it extended 'ri'-block? */
|
|
#if EXNDEBUG
|
|
printf("%d , %d\n",*countri,*count);
|
|
#endif
|
|
if (*countri < 0 || *countri >= rikey->no_lis) { /* End of ri's? */
|
|
return(0);
|
|
}
|
|
/* Get the li of lf-struct that's current based on countri */
|
|
likey = (struct li_key *)( hdesc->buffer + rikey->hash[*countri].ofs_li + 0x1004 ) ;
|
|
if (likey->id == 0x696c) {
|
|
newnkofs = likey->hash[*count].ofs_nk + 0x1000;
|
|
} else {
|
|
lfkey = (struct lf_key *)( hdesc->buffer + rikey->hash[*countri].ofs_li + 0x1004 ) ;
|
|
newnkofs = lfkey->hash[*count].ofs_nk + 0x1000;
|
|
}
|
|
|
|
/* Check if current li/lf is exhausted */
|
|
#if EXNDEBUG
|
|
printf("likey->no_keys = %d\n",likey->no_keys);
|
|
#endif
|
|
if (*count >= likey->no_keys-1) { /* Last legal entry in li list? */
|
|
(*countri)++; /* Bump up ri count so we take next ri entry next time */
|
|
(*count) = -1; /* Reset li traverse counter for next round, not used later here */
|
|
}
|
|
} else { /* Plain handler */
|
|
if (key->no_subkeys <= 0 || *count >= key->no_subkeys) {
|
|
return(0);
|
|
}
|
|
if (lfkey->id == 0x696c) { /* Is it 3.x 'li' instead? */
|
|
likey = (struct li_key *)(hdesc->buffer + key->ofs_lf + 0x1004);
|
|
newnkofs = likey->hash[*count].ofs_nk + 0x1000;
|
|
} else {
|
|
newnkofs = lfkey->hash[*count].ofs_nk + 0x1000;
|
|
}
|
|
}
|
|
|
|
sptr->nkoffs = newnkofs;
|
|
newnkkey = (struct nk_key *)(hdesc->buffer + newnkofs + 4);
|
|
sptr->nk = newnkkey;
|
|
|
|
if (newnkkey->id != 0x6b6e) {
|
|
printf("ex_next: ERROR: not 'nk' node at 0x%0x\n",newnkofs);
|
|
|
|
return(-1);
|
|
} else {
|
|
if (newnkkey->len_name <= 0) {
|
|
printf("ex_next: nk at 0x%0x has no name!\n",newnkofs);
|
|
} else if (newnkkey->type & 0x20) {
|
|
#if 0
|
|
printf("dummy1 %x\n", *((int*)newnkkey->dummy1));
|
|
printf("dummy2 %x\n", *((int*)newnkkey->dummy2));
|
|
printf("type %x\n", newnkkey->type);
|
|
printf("timestamp+8 %x\n", *((int*)(newnkkey->timestamp+8)));
|
|
printf("dummy3+0 %x\n", *((int*)(newnkkey->dummy3+0)));
|
|
printf("dummy3+4 %x\n", *((int*)(newnkkey->dummy3+4)));
|
|
printf("dummy3+8 %x\n", *((int*)(newnkkey->dummy3+8)));
|
|
printf("dummy3+12 %x\n", *((int*)(newnkkey->dummy3+12)));
|
|
printf("dummy4 %x\n", *((int*)&newnkkey->dummy4));
|
|
printf("len %d\n", newnkkey->len_name);
|
|
printf("len class %d\n", newnkkey->len_classnam);
|
|
fflush(stdout);
|
|
#endif
|
|
sptr->name = string_rega2prog(newnkkey->keyname, newnkkey->len_name);
|
|
} else {
|
|
sptr->name = string_regw2prog(newnkkey->keyname, newnkkey->len_name);
|
|
}
|
|
} /* if */
|
|
(*count)++;
|
|
return(1);
|
|
/* return( *count <= key->no_subkeys); */
|
|
}
|
|
|
|
/* "directory scan" for VALUES, return next name/pointer of a value on each call
|
|
* nkofs = offset to directory to scan
|
|
* lfofs = pointer to int to hold the current scan position,
|
|
* set position to 0 to start.
|
|
* sptr = pointer to struct to hold a single result
|
|
* returns: -1 = error. 0 = end of key. 1 = more values to scan
|
|
* NOTE: caller must free the name-buffer (struct vex_data *name)
|
|
*/
|
|
int ex_next_v(struct hive *hdesc, int nkofs, int *count, struct vex_data *sptr)
|
|
{
|
|
struct nk_key *key /* , *newnkkey */ ;
|
|
int vkofs,vlistofs;
|
|
int *vlistkey;
|
|
struct vk_key *vkkey;
|
|
|
|
|
|
if (!nkofs) return(-1);
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
if (key->id != 0x6b6e) {
|
|
printf("ex_next_v error: Not a 'nk' node at 0x%0x\n",nkofs);
|
|
return(-1);
|
|
}
|
|
|
|
if (key->no_values <= 0 || *count >= key->no_values) {
|
|
return(0);
|
|
}
|
|
|
|
vlistofs = key->ofs_vallist + 0x1004;
|
|
vlistkey = (int *)(hdesc->buffer + vlistofs);
|
|
|
|
vkofs = vlistkey[*count] + 0x1004;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
if (vkkey->id != 0x6b76) {
|
|
printf("ex_next_v: hit non valuekey (vk) node during scan at offs 0x%0x\n",vkofs);
|
|
return(-1);
|
|
}
|
|
|
|
/* parse_vk(hdesc, vkofs, 4); */
|
|
|
|
sptr->vk = vkkey;
|
|
sptr->vkoffs = vkofs;
|
|
sptr->name = 0;
|
|
sptr->size = (vkkey->len_data & 0x7fffffff);
|
|
|
|
if (vkkey->len_name >0) {
|
|
if (vkkey->flag & 1) {
|
|
sptr->name = string_rega2prog(vkkey->keyname, vkkey->len_name);
|
|
} else {
|
|
sptr->name = string_regw2prog(vkkey->keyname, vkkey->len_name);
|
|
}
|
|
} else {
|
|
sptr->name = str_dup("");
|
|
}
|
|
|
|
sptr->type = vkkey->val_type;
|
|
if (sptr->size) {
|
|
if (vkkey->val_type == REG_DWORD) {
|
|
if (vkkey->len_data & 0x80000000) {
|
|
sptr->val = (int)(vkkey->ofs_data);
|
|
}
|
|
}
|
|
} else if (vkkey->len_data == 0x80000000) {
|
|
/* Data SIZE is 0, high bit set: special inline case, data is DWORD and in TYPE field!! */
|
|
/* Used a lot in SAM, and maybe in SECURITY I think */
|
|
sptr->val = (int)(vkkey->val_type);
|
|
sptr->size = 4;
|
|
sptr->type = REG_DWORD;
|
|
} else {
|
|
sptr->val = 0;
|
|
sptr->size = 0;
|
|
}
|
|
|
|
(*count)++;
|
|
return( *count <= key->no_values );
|
|
}
|
|
|
|
/* traceback - trace nk's back to root,
|
|
* building path string as we go.
|
|
* nkofs = offset to nk-node
|
|
* path = pointer to pathstring-buffer
|
|
* maxlen = max length of path-buffer
|
|
* return: length of path string
|
|
*/
|
|
|
|
int get_abs_path(struct hive *hdesc, int nkofs, char *path, int maxlen)
|
|
{
|
|
/* int newnkofs; */
|
|
struct nk_key *key;
|
|
char tmp[ABSPATHLEN+1];
|
|
char *keyname;
|
|
int len_name;
|
|
|
|
maxlen = (maxlen < ABSPATHLEN ? maxlen : ABSPATHLEN);
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("get_abs_path: Not a 'nk' node!\n");
|
|
return(0);
|
|
}
|
|
|
|
if (key->type == KEY_ROOT) { /* We're at the root */
|
|
return(strlen(path));
|
|
}
|
|
|
|
strncpy(tmp,path,ABSPATHLEN-1);
|
|
|
|
if (key->type & 0x20)
|
|
keyname = string_rega2prog(key->keyname, key->len_name);
|
|
else
|
|
keyname = string_regw2prog(key->keyname, key->len_name);
|
|
len_name = strlen(keyname);
|
|
if ( (strlen(path) + len_name) >= maxlen-6) {
|
|
free(keyname);
|
|
snprintf(path,maxlen,"(...)%s",tmp);
|
|
return(strlen(path)); /* Stop trace when string exhausted */
|
|
}
|
|
*path = '\\';
|
|
memcpy(path+1,keyname,len_name);
|
|
free(keyname);
|
|
strncpy(path+len_name+1,tmp,maxlen-6-len_name);
|
|
return(get_abs_path(hdesc, key->ofs_parent+0x1004, path, maxlen)); /* go back one more */
|
|
}
|
|
|
|
|
|
/* Value index table lookup
|
|
* hdesc - hive as usual
|
|
* vlistofs - offset of table
|
|
* name - value name to look for
|
|
* returns index into table or -1 if err
|
|
*/
|
|
|
|
int vlist_find(struct hive *hdesc, int vlistofs, int numval, char *name, int type)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int i,vkofs,len;
|
|
int32_t *vlistkey;
|
|
|
|
len = strlen(name);
|
|
vlistkey = (int32_t *)(hdesc->buffer + vlistofs);
|
|
|
|
for (i = 0; i < numval; i++) {
|
|
vkofs = vlistkey[i] + 0x1004;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
if (vkkey->len_name == 0 && *name == '@') { /* @ is alias for nameless value */
|
|
return(i);
|
|
}
|
|
if ( !(type & TPF_EXACT) || vkkey->len_name == len ) {
|
|
if (!strncmp(name, vkkey->keyname, len)) { /* name match? */
|
|
return(i);
|
|
}
|
|
}
|
|
}
|
|
return(-1);
|
|
|
|
}
|
|
|
|
/* Recursevely follow 'nk'-nodes based on a path-string,
|
|
* returning offset of last 'nk' or 'vk'
|
|
* vofs - offset to start node
|
|
* path - null-terminated pathname (relative to vofs, \ is separator)
|
|
* type - type to return TPF_??, see ntreg.h
|
|
* return: offset to nk or vk (or NULL if not found)
|
|
*/
|
|
|
|
int trav_path(struct hive *hdesc, int vofs, char *path, int type)
|
|
{
|
|
struct nk_key *key, *newnkkey;
|
|
struct lf_key *lfkey;
|
|
struct li_key *likey;
|
|
struct ri_key *rikey;
|
|
|
|
int32_t *vlistkey;
|
|
int newnkofs, plen, i, lfofs, vlistofs, adjust, r, ricnt, subs;
|
|
char *buf;
|
|
char part[ABSPATHLEN+1];
|
|
char *partptr;
|
|
|
|
if (!hdesc) return(0);
|
|
buf = hdesc->buffer;
|
|
|
|
if (!vofs) vofs = hdesc->rootofs+4; /* No current key given , so start at root */
|
|
|
|
if (*path == '\\' && *(path+1) != '\\') { /* Start from root if path starts with \ */
|
|
path++;
|
|
vofs = hdesc->rootofs+4;
|
|
}
|
|
|
|
key = (struct nk_key *)(buf + vofs);
|
|
/* printf("check of nk at offset: 0x%0x\n",vofs); */
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("trav_path: Error: Not a 'nk' node!\n");
|
|
return(0);
|
|
}
|
|
|
|
/* Find \ delimiter or end of string, copying to name part buffer as we go,
|
|
rewriting double \\s */
|
|
partptr = part;
|
|
for(plen = 0; path[plen] && (path[plen] != '\\' || path[plen+1] == '\\'); plen++) {
|
|
if (path[plen] == '\\' && path[plen+1] == '\\') plen++; /* Skip one if double */
|
|
*partptr++ = path[plen];
|
|
}
|
|
*partptr = '\0';
|
|
|
|
/* printf("Name component: <%s>\n",part); */
|
|
|
|
adjust = (path[plen] == '\\' ) ? 1 : 0;
|
|
/* printf("Checking for <%s> with len %d\n",path,plen); */
|
|
if (!plen) return(vofs-4); /* Path has no lenght - we're there! */
|
|
if ( (plen == 1) && (*path == '.') && !(type & TPF_EXACT)) { /* Handle '.' current dir */
|
|
return(trav_path(hdesc,vofs,path+plen+adjust,type));
|
|
}
|
|
if ( !(type & TPF_EXACT) && (plen == 2) && !strncmp("..",path,2) ) { /* Get parent key */
|
|
newnkofs = key->ofs_parent + 0x1004;
|
|
/* Return parent (or only root if at the root) */
|
|
return(trav_path(hdesc, (key->type == KEY_ROOT ? vofs : newnkofs), path+plen+adjust, type));
|
|
}
|
|
|
|
/* at last name of path, and we want vk, and the nk has values */
|
|
if (!path[plen] && (type & TPF_VK) && key->no_values) {
|
|
/* printf("VK namematch for <%s>\n",part); */
|
|
vlistofs = key->ofs_vallist + 0x1004;
|
|
vlistkey = (int32_t *)(buf + vlistofs);
|
|
i = vlist_find(hdesc, vlistofs, key->no_values, part, type);
|
|
if (i != -1) {
|
|
return(vlistkey[i] + 0x1000);
|
|
}
|
|
}
|
|
|
|
if (key->no_subkeys > 0) { /* If it has subkeys, loop through the hash */
|
|
char *partw = NULL;
|
|
int partw_len, part_len;
|
|
|
|
lfofs = key->ofs_lf + 0x1004; /* lf (hash) record */
|
|
lfkey = (struct lf_key *)(buf + lfofs);
|
|
|
|
if (lfkey->id == 0x6972) { /* ri struct need special parsing */
|
|
/* Prime loop state */
|
|
|
|
rikey = (struct ri_key *)lfkey;
|
|
ricnt = rikey->no_lis;
|
|
r = 0;
|
|
likey = (struct li_key *)( hdesc->buffer + rikey->hash[r].ofs_li + 0x1004 ) ;
|
|
subs = likey->no_keys;
|
|
if (likey->id != 0x696c) { /* Bwah, not li anyway, XP uses lh usually which is actually smarter */
|
|
lfkey = (struct lf_key *)( hdesc->buffer + rikey->hash[r].ofs_li + 0x1004 ) ;
|
|
likey = NULL;
|
|
}
|
|
} else {
|
|
if (lfkey->id == 0x696c) { /* li? */
|
|
likey = (struct li_key *)(buf + lfofs);
|
|
} else {
|
|
likey = NULL;
|
|
}
|
|
rikey = NULL;
|
|
ricnt = 0; r = 0; subs = key->no_subkeys;
|
|
}
|
|
|
|
partw = string_prog2regw(part, partptr-part, &partw_len);
|
|
string_prog2rega(part, partptr-part);
|
|
part_len = strlen(part);
|
|
do {
|
|
for(i = 0; i < subs; i++) {
|
|
if (likey) newnkofs = likey->hash[i].ofs_nk + 0x1004;
|
|
else newnkofs = lfkey->hash[i].ofs_nk + 0x1004;
|
|
newnkkey = (struct nk_key *)(buf + newnkofs);
|
|
if (newnkkey->id != 0x6b6e) {
|
|
printf("ERROR: not 'nk' node! (strange?)\n");
|
|
} else {
|
|
if (newnkkey->len_name <= 0) {
|
|
printf("[No name]\n");
|
|
} else {
|
|
int cmp;
|
|
if (newnkkey->type & 0x20)
|
|
cmp = strncmp(part,newnkkey->keyname,part_len);
|
|
else
|
|
cmp = memcmp(partw, newnkkey->keyname, partw_len);
|
|
if (!cmp) {
|
|
/* printf("Key at 0x%0x matches! recursing!\n",newnkofs); */
|
|
free(partw);
|
|
return(trav_path(hdesc, newnkofs, path+plen+adjust, type));
|
|
}
|
|
}
|
|
} /* if id OK */
|
|
} /* hash loop */
|
|
r++;
|
|
if (ricnt && r < ricnt) {
|
|
newnkofs = rikey->hash[r].ofs_li;
|
|
likey = (struct li_key *)( hdesc->buffer + newnkofs + 0x1004 ) ;
|
|
subs = likey->no_keys;
|
|
if (likey->id != 0x696c) { /* Bwah, not li anyway, XP uses lh usually which is actually smarter */
|
|
lfkey = (struct lf_key *)( hdesc->buffer + rikey->hash[r].ofs_li + 0x1004 ) ;
|
|
likey = NULL;
|
|
}
|
|
}
|
|
} while (r < ricnt && ricnt);
|
|
free(partw);
|
|
|
|
} /* if subkeys */
|
|
|
|
/* Not found */
|
|
return(0);
|
|
}
|
|
|
|
|
|
/* ls - list a 'nk' nodes subkeys and values
|
|
* vofs - offset to start of data (skipping block linkage)
|
|
* type - 0 = full, 1 = keys only. 2 = values only
|
|
*/
|
|
void nk_ls(struct hive *hdesc, char *path, int vofs, int type)
|
|
{
|
|
struct nk_key *key;
|
|
int nkofs;
|
|
struct ex_data ex;
|
|
struct vex_data vex;
|
|
int count = 0, countri = 0;
|
|
|
|
|
|
nkofs = trav_path(hdesc, vofs, path, 0);
|
|
|
|
if(!nkofs) {
|
|
printf("nk_ls: Key <%s> not found\n",path);
|
|
return;
|
|
}
|
|
nkofs += 4;
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
VERBF(hdesc,"ls of node at offset 0x%0x\n",nkofs);
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("Error: Not a 'nk' node!\n");
|
|
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
|
|
}
|
|
|
|
printf("Node has %d subkeys and %d values",key->no_subkeys,key->no_values);
|
|
if (key->len_classnam) printf(", and class-data of %d bytes",key->len_classnam);
|
|
printf("\n");
|
|
|
|
if (key->no_subkeys) {
|
|
printf(" key name\n");
|
|
while ((ex_next_n(hdesc, nkofs, &count, &countri, &ex) > 0)) {
|
|
if (!(hdesc->state & HMODE_VERBOSE)) printf("%c <%s>\n", (ex.nk->len_classnam)?'*':' ',ex.name);
|
|
else printf("[%6x] %c <%s>\n", ex.nkoffs, (ex.nk->len_classnam)?'*':' ',ex.name);
|
|
FREE(ex.name);
|
|
}
|
|
}
|
|
count = 0;
|
|
if (key->no_values) {
|
|
printf(" size type value name [value if type DWORD]\n");
|
|
while ((ex_next_v(hdesc, nkofs, &count, &vex) > 0)) {
|
|
if (hdesc->state & HMODE_VERBOSE) printf("[%6x] %6d %-16s <%s>", vex.vkoffs, vex.size,
|
|
(vex.type < REG_MAX ? val_types[vex.type] : "(unknown)"), vex.name);
|
|
else
|
|
printf("%6d %-16s <%s>", vex.size,
|
|
(vex.type < REG_MAX ? val_types[vex.type] : "(unknown)"), vex.name);
|
|
|
|
if (vex.type == REG_DWORD) printf(" %*d [0x%x]",25-(int)strlen(vex.name),vex.val , vex.val);
|
|
printf("\n");
|
|
FREE(vex.name);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Get the type of a value */
|
|
int get_val_type(struct hive *hdesc, int vofs, char *path, int exact)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int vkofs;
|
|
|
|
vkofs = trav_path(hdesc, vofs,path,exact | TPF_VK);
|
|
if (!vkofs) {
|
|
return -1;
|
|
}
|
|
vkofs +=4;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
#if 0
|
|
if (vkkey->len_data & 0x80000000) return(REG_DWORD); /* Special case of INLINE storage */
|
|
#endif
|
|
return(vkkey->val_type);
|
|
}
|
|
|
|
|
|
/* Get len of a value, given current key + path */
|
|
int get_val_len(struct hive *hdesc, int vofs, char *path, int exact)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int vkofs;
|
|
int len;
|
|
|
|
vkofs = trav_path(hdesc, vofs,path,exact | TPF_VK);
|
|
if (!vkofs) {
|
|
return -1;
|
|
}
|
|
vkofs +=4;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
|
|
len = vkkey->len_data & 0x7fffffff;
|
|
|
|
if ( vkkey->len_data == 0x80000000 ) { /* Special inline case, return size of 4 (dword) */
|
|
len = 4;
|
|
}
|
|
|
|
return(len);
|
|
}
|
|
|
|
/* Get void-pointer to value-data, also if inline.
|
|
* If val_type != 0 a check for correct value type is done
|
|
* Caller must keep track of value's length (call function above to get it)
|
|
*/
|
|
void *get_val_data(struct hive *hdesc, int vofs, char *path, int val_type, int exact)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int vkofs;
|
|
|
|
vkofs = trav_path(hdesc,vofs,path,exact | TPF_VK);
|
|
if (!vkofs) {
|
|
return NULL;
|
|
}
|
|
vkofs +=4;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
|
|
|
|
if (vkkey->len_data == 0) return NULL;
|
|
if (vkkey->len_data == 0x80000000) { /* Special inline case (len = 0x80000000) */
|
|
return(&vkkey->val_type); /* Data (4 bytes?) in type field */
|
|
}
|
|
|
|
if (val_type && vkkey->val_type && (vkkey->val_type) != val_type) {
|
|
printf("Value <%s> is not of correct type!\n",path);
|
|
#if DOCORE
|
|
abort();
|
|
#endif
|
|
return NULL;
|
|
}
|
|
|
|
/* Negative len is inline, return ptr to offset-field which in
|
|
* this case contains the data itself
|
|
*/
|
|
if (vkkey->len_data & 0x80000000) return(&vkkey->ofs_data);
|
|
/* Normal return, return data pointer */
|
|
return(hdesc->buffer + vkkey->ofs_data + 0x1004);
|
|
}
|
|
|
|
|
|
/* Get and copy key data (if any) to buffer
|
|
* if kv==NULL will allocate needed return struct & buffer
|
|
* else will use buffer allocated for it (if it fits)
|
|
* return len+data or NULL if not found (or other error)
|
|
* NOTE: caller must deallocate buffer! a simple free(keyval) will suffice.
|
|
*/
|
|
struct keyval *get_val2buf(struct hive *hdesc, struct keyval *kv,
|
|
int vofs, char *path, int type, int exact )
|
|
{
|
|
int l;
|
|
struct keyval *kr;
|
|
void *keydataptr;
|
|
|
|
l = get_val_len(hdesc, vofs, path, exact);
|
|
if (l == -1) return(NULL); /* error */
|
|
if (kv && (kv->len < l)) return(NULL); /* Check for overflow of supplied buffer */
|
|
|
|
keydataptr = get_val_data(hdesc, vofs, path, type, exact);
|
|
/* if (!keydataptr) return(NULL); error */
|
|
|
|
/* Allocate space for data + header, or use supplied buffer */
|
|
if (kv) {
|
|
kr = kv;
|
|
} else {
|
|
ALLOC(kr,1,l+sizeof(int)+4);
|
|
}
|
|
|
|
kr->len = l;
|
|
memcpy(&(kr->data), keydataptr, l);
|
|
|
|
return(kr);
|
|
}
|
|
|
|
/* DWORDs are so common that I make a small function to get it easily */
|
|
|
|
int get_dword(struct hive *hdesc, int vofs, char *path, int exact)
|
|
{
|
|
struct keyval *v;
|
|
int dword;
|
|
|
|
v = get_val2buf(hdesc, NULL, vofs, path, REG_DWORD, exact | TPF_VK);
|
|
if (!v) return(-1); /* well... -1 COULD BE THE STORED VALUE TOO */
|
|
|
|
dword = (int)v->data;
|
|
|
|
FREE(v);
|
|
|
|
return(dword);
|
|
|
|
}
|
|
|
|
/* Sanity checker when transferring data into a block
|
|
* ofs = offset to data block, point to start of actual datablock linkage
|
|
* data = data to copy
|
|
* size = size of data to copy
|
|
*/
|
|
|
|
int fill_block(struct hive *hdesc, int ofs, void *data, int size)
|
|
{
|
|
int blksize;
|
|
|
|
blksize = get_int(hdesc->buffer + ofs);
|
|
blksize = -blksize;
|
|
|
|
#if 0
|
|
printf("fill_block: ofs = %x - %x, size = %x, blksize = %x\n",ofs,ofs+size,size,blksize);
|
|
#endif
|
|
/* if (blksize < size || ( (ofs & 0xfffff000) != ((ofs+size) & 0xfffff000) )) { */
|
|
if (blksize < size) {
|
|
printf("fill_block: ERROR: block to small for data: ofs = %x, size = %x, blksize = %x\n",ofs,size,blksize);
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
abort();
|
|
}
|
|
|
|
memcpy(hdesc->buffer + ofs + 4, data, size);
|
|
return(0);
|
|
}
|
|
|
|
|
|
/* Free actual data of a value, and update value descriptor
|
|
* hdesc - hive
|
|
* vofs - current key
|
|
* path - path to value
|
|
* we return offset of vk
|
|
*/
|
|
|
|
int free_val_data(struct hive *hdesc, int vofs, char *path, int exact)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int vkofs, inl;
|
|
|
|
vkofs = trav_path(hdesc,vofs,path,1);
|
|
if (!vkofs) {
|
|
return 0;
|
|
}
|
|
vkofs +=4;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
|
|
inl = (vkkey->len_data & 0x80000000);
|
|
|
|
if (!inl) {
|
|
free_block(hdesc, vkkey->ofs_data + 0x1000);
|
|
}
|
|
vkkey->len_data = 0;
|
|
vkkey->ofs_data = 0;
|
|
|
|
return(vkofs);
|
|
|
|
}
|
|
|
|
/* Allocate data for value, realloc if it already contains stuff
|
|
* hdesc - hive
|
|
* vofs - current key
|
|
* path - path to value
|
|
* size - size of data
|
|
* Returns: 0 - error, >0 pointer to actual dataspace
|
|
*/
|
|
|
|
int alloc_val_data(struct hive *hdesc, int vofs, char *path, int size,int exact)
|
|
{
|
|
struct vk_key *vkkey;
|
|
int vkofs, len;
|
|
int datablk;
|
|
|
|
vkofs = trav_path(hdesc,vofs,path,1);
|
|
if (!vkofs) {
|
|
return (0);
|
|
}
|
|
|
|
vkofs +=4;
|
|
vkkey = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
|
|
/* Allocate space for new data */
|
|
datablk = alloc_block(hdesc, vkofs, size);
|
|
if (!datablk) return(0);
|
|
|
|
len = vkkey->len_data & 0x7fffffff;
|
|
|
|
/* Then we dealloc if something was there before */
|
|
if (len) free_val_data(hdesc,vofs,path,exact);
|
|
|
|
/* Link in new datablock */
|
|
vkkey->ofs_data = datablk - 0x1000;
|
|
vkkey->len_data = size;
|
|
|
|
return(datablk + 4);
|
|
}
|
|
|
|
|
|
/* Add a value to a key.
|
|
* Just add the metadata (empty value), to put data into it, use
|
|
* put_buf2val afterwards
|
|
* hdesc - hive
|
|
* nkofs - current key
|
|
* name - name of value
|
|
* type - type of value
|
|
* returns: 0 err, >0 offset to value metadata
|
|
*/
|
|
|
|
struct vk_key *add_value(struct hive *hdesc, int nkofs, char *name, int type)
|
|
{
|
|
struct nk_key *nk;
|
|
int oldvlist = 0, newvlist, newvkofs;
|
|
struct vk_key *newvkkey;
|
|
char *blank="";
|
|
|
|
if (!name || !*name) return(NULL);
|
|
|
|
|
|
nk = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
if (nk->id != 0x6b6e) {
|
|
printf("add_value: Key pointer not to 'nk' node!\n");
|
|
return(NULL);
|
|
}
|
|
|
|
if (trav_path(hdesc, nkofs, name, 1)) {
|
|
printf("add_value: value %s already exists\n",name);
|
|
return(NULL);
|
|
}
|
|
|
|
if (!strcmp(name,"@")) name = blank;
|
|
|
|
if (nk->no_values) oldvlist = nk->ofs_vallist;
|
|
|
|
newvlist = alloc_block(hdesc, nkofs, nk->no_values * 4 + 4);
|
|
if (!newvlist) {
|
|
printf("add_value: failed to allocate new value list!\n");
|
|
return(NULL);
|
|
}
|
|
if (oldvlist) { /* Copy old data if any */
|
|
memcpy(hdesc->buffer + newvlist + 4, hdesc->buffer + oldvlist + 0x1004, nk->no_values * 4 + 4);
|
|
}
|
|
|
|
/* Allocate value descriptor including its name */
|
|
newvkofs = alloc_block(hdesc, newvlist, sizeof(struct vk_key) + strlen(name));
|
|
if (!newvkofs) {
|
|
printf("add_value: failed to allocate value descriptor\n");
|
|
free_block(hdesc, newvlist);
|
|
return(NULL);
|
|
}
|
|
|
|
/* Success, now fill in the metadata */
|
|
|
|
newvkkey = (struct vk_key *)(hdesc->buffer + newvkofs + 4);
|
|
|
|
/* Add pointer in value list */
|
|
*(int *)(hdesc->buffer + newvlist + 4 + (nk->no_values * 4)) = newvkofs - 0x1000;
|
|
|
|
/* Fill in vk struct */
|
|
newvkkey->id = 0x6b76;
|
|
newvkkey->len_name = strlen(name);
|
|
if (type == REG_DWORD || type == REG_DWORD_BIG_ENDIAN) {
|
|
newvkkey->len_data = 0x80000004; /* Prime the DWORD inline stuff */
|
|
} else {
|
|
newvkkey->len_data = 0x00000000;
|
|
}
|
|
newvkkey->ofs_data = 0;
|
|
newvkkey->val_type = type;
|
|
newvkkey->flag = 1; /* Don't really know what this is */
|
|
newvkkey->dummy1 = 0;
|
|
strcpy((char *)&newvkkey->keyname, name); /* And copy name */
|
|
|
|
/* Finally update the key and free the old valuelist */
|
|
nk->no_values++;
|
|
nk->ofs_vallist = newvlist - 0x1000;
|
|
if (oldvlist) free_block(hdesc,oldvlist + 0x1000);
|
|
|
|
return(newvkkey);
|
|
|
|
}
|
|
|
|
/* Remove a vk-struct incl dataspace if any
|
|
* Mostly for use by higher level stuff
|
|
* hdesc - hive
|
|
* vkofs - offset to vk
|
|
*/
|
|
|
|
void del_vk(struct hive *hdesc, int vkofs)
|
|
{
|
|
struct vk_key *vk;
|
|
|
|
vk = (struct vk_key *)(hdesc->buffer + vkofs);
|
|
if (vk->id != 0x6b76) {
|
|
printf("del_vk: Key pointer not to 'vk' node!\n");
|
|
return;
|
|
}
|
|
|
|
if ( !(vk->len_data & 0x80000000) && vk->ofs_data) {
|
|
free_block(hdesc, vk->ofs_data + 0x1000);
|
|
}
|
|
|
|
free_block(hdesc, vkofs - 4);
|
|
}
|
|
|
|
/* Delete all values from key (used in recursive delete)
|
|
* hdesc - yer usual hive
|
|
* nkofs - current keyoffset
|
|
*/
|
|
|
|
void del_allvalues(struct hive *hdesc, int nkofs)
|
|
{
|
|
int vlistofs, o, vkofs;
|
|
int32_t *vlistkey;
|
|
struct nk_key *nk;
|
|
|
|
nk = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
if (nk->id != 0x6b6e) {
|
|
printf("del_allvalues: Key pointer not to 'nk' node!\n");
|
|
return;
|
|
}
|
|
|
|
if (!nk->no_values) {
|
|
/* printf("del_avalues: Key has no values!\n"); */
|
|
return;
|
|
}
|
|
|
|
vlistofs = nk->ofs_vallist + 0x1004;
|
|
vlistkey = (int32_t *)(hdesc->buffer + vlistofs);
|
|
|
|
/* Loop through index and delete all vk's */
|
|
for (o = 0; o < nk->no_values; o++) {
|
|
vkofs = vlistkey[o] + 0x1004;
|
|
del_vk(hdesc, vkofs);
|
|
}
|
|
|
|
/* Then zap the index, and update nk */
|
|
free_block(hdesc, vlistofs-4);
|
|
nk->ofs_vallist = -1;
|
|
nk->no_values = 0;
|
|
}
|
|
|
|
|
|
/* Delete single value from key
|
|
* hdesc - yer usual hive
|
|
* nkofs - current keyoffset
|
|
* name - name of value to delete
|
|
* exact - NKF_EXACT to do exact match, else first match
|
|
* returns: 0 - ok, 1 - failed
|
|
*/
|
|
|
|
int del_value(struct hive *hdesc, int nkofs, char *name, int exact)
|
|
{
|
|
int vlistofs, slot, o, n, vkofs, newlistofs;
|
|
int32_t *vlistkey, *tmplist, *newlistkey;
|
|
struct nk_key *nk;
|
|
char *blank="";
|
|
|
|
if (!name || !*name) return(1);
|
|
|
|
if (!strcmp(name,"@")) name = blank;
|
|
|
|
nk = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
if (nk->id != 0x6b6e) {
|
|
printf("del_value: Key pointer not to 'nk' node!\n");
|
|
return(1);
|
|
}
|
|
|
|
if (!nk->no_values) {
|
|
printf("del_value: Key has no values!\n");
|
|
return(1);
|
|
}
|
|
|
|
vlistofs = nk->ofs_vallist + 0x1004;
|
|
vlistkey = (int32_t *)(hdesc->buffer + vlistofs);
|
|
|
|
slot = vlist_find(hdesc, vlistofs, nk->no_values, name, TPF_VK);
|
|
|
|
if (slot == -1) {
|
|
printf("del_value: value %s not found!\n",name);
|
|
return(1);
|
|
}
|
|
|
|
/* Delete vk and data */
|
|
vkofs = vlistkey[slot] + 0x1004;
|
|
del_vk(hdesc, vkofs);
|
|
|
|
/* Copy out old index list */
|
|
CREATE(tmplist,int32_t,nk->no_values);
|
|
memcpy(tmplist, vlistkey, nk->no_values * sizeof(int32_t));
|
|
|
|
free_block(hdesc,vlistofs-4); /* Get rid of old list */
|
|
|
|
nk->no_values--;
|
|
|
|
if (nk->no_values) {
|
|
newlistofs = alloc_block(hdesc, vlistofs, nk->no_values * sizeof(int32_t));
|
|
if (!newlistofs) {
|
|
printf("del_value: FATAL: Was not able to alloc new index list\n");
|
|
abort();
|
|
}
|
|
/* Now copy over, omitting deleted entry */
|
|
newlistkey = (int32_t *)(hdesc->buffer + newlistofs + 4);
|
|
for (n = 0, o = 0; o < nk->no_values+1; o++, n++) {
|
|
if (o == slot) o++;
|
|
newlistkey[n] = tmplist[o];
|
|
}
|
|
nk->ofs_vallist = newlistofs - 0x1000;
|
|
} else {
|
|
nk->ofs_vallist = -1;
|
|
}
|
|
return(0);
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Add a subkey to a key
|
|
* hdesc - usual..
|
|
* nkofs - offset of current nk
|
|
* name - name of key to add
|
|
* return: ptr to new keystruct, or NULL
|
|
*/
|
|
|
|
#define AKDEBUG 1
|
|
struct nk_key *add_key(struct hive *hdesc, int nkofs, char *name)
|
|
{
|
|
|
|
int slot, newlfofs = 0, oldlfofs = 0, newliofs = 0;
|
|
int oldliofs = 0;
|
|
int o, n, i, onkofs, newnkofs, cmp;
|
|
int rimax, rislot, riofs, namlen;
|
|
struct ri_key *ri = NULL;
|
|
struct lf_key *newlf = NULL, *oldlf;
|
|
struct li_key *newli = NULL, *oldli;
|
|
struct nk_key *key, *newnk, *onk;
|
|
int32_t hash;
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("add_key: current ptr not 'nk'\n");
|
|
return(NULL);
|
|
}
|
|
|
|
namlen = strlen(name);
|
|
|
|
slot = -1;
|
|
if (key->no_subkeys) { /* It already has subkeys */
|
|
|
|
oldlfofs = key->ofs_lf;
|
|
oldliofs = key->ofs_lf;
|
|
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
if (oldlf->id != 0x666c && oldlf->id != 0x686c && oldlf->id != 0x696c && oldlf->id != 0x6972) {
|
|
printf("add_key: index type not supported: 0x%04x\n",oldlf->id);
|
|
return(NULL);
|
|
}
|
|
|
|
rimax = 0; ri = NULL; riofs = 0; rislot = -1;
|
|
if (oldlf->id == 0x6972) { /* Indirect index 'ri', init loop */
|
|
riofs = key->ofs_lf;
|
|
ri = (struct ri_key *)(hdesc->buffer + riofs + 0x1004);
|
|
rimax = ri->no_lis-1;
|
|
|
|
#ifdef AKDEBUG
|
|
printf("add_key: entering 'ri' traverse, rimax = %d\n",rimax);
|
|
#endif
|
|
|
|
oldliofs = ri->hash[rislot+1].ofs_li;
|
|
oldlfofs = ri->hash[rislot+1].ofs_li;
|
|
|
|
}
|
|
|
|
do { /* 'ri' loop, at least run once if no 'ri' deep index */
|
|
|
|
if (ri) { /* Do next 'ri' slot */
|
|
rislot++;
|
|
oldliofs = ri->hash[rislot].ofs_li;
|
|
oldlfofs = ri->hash[rislot].ofs_li;
|
|
oldli = (struct li_key *)(hdesc->buffer + oldliofs + 0x1004);
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
}
|
|
|
|
oldli = (struct li_key *)(hdesc->buffer + oldliofs + 0x1004);
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
|
|
#ifdef AKDEBUG
|
|
printf("add_key: top of ri-loop: rislot = %d, rimax = %d\n",rislot,rimax);
|
|
#endif
|
|
slot = -1;
|
|
|
|
if (oldli->id == 0x696c) { /* li */
|
|
|
|
#ifdef AKDEBUG
|
|
printf("add_key: li slot allocate\n");
|
|
#endif
|
|
|
|
FREE(newli);
|
|
ALLOC(newli, 8 + 4*oldli->no_keys + 4, 1);
|
|
newli->no_keys = oldli->no_keys;
|
|
newli->id = oldli->id;
|
|
|
|
/* Now copy old, checking where to insert (alphabetically) */
|
|
for (o = 0, n = 0; o < oldli->no_keys; o++,n++) {
|
|
onkofs = oldli->hash[o].ofs_nk;
|
|
onk = (struct nk_key *)(onkofs + hdesc->buffer + 0x1004);
|
|
if (slot == -1) {
|
|
#if 1
|
|
printf("add_key: cmp <%s> with <%s>\n",name,onk->keyname);
|
|
#endif
|
|
|
|
cmp = strncasecmp(name, onk->keyname, (namlen > onk->len_name) ? namlen : onk->len_name);
|
|
if (!cmp) {
|
|
printf("add_key: key %s already exists!\n",name);
|
|
FREE(newli);
|
|
return(NULL);
|
|
}
|
|
if ( cmp < 0) {
|
|
slot = o;
|
|
rimax = rislot; /* Cause end of 'ri' search, too */
|
|
n++;
|
|
#ifdef AKDEBUG
|
|
printf("add_key: li-match: slot = %d\n",o);
|
|
#endif
|
|
}
|
|
}
|
|
newli->hash[n].ofs_nk = oldli->hash[o].ofs_nk;
|
|
}
|
|
if (slot == -1) slot = oldli->no_keys;
|
|
|
|
} else { /* lf or lh */
|
|
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
|
|
FREE(newlf);
|
|
ALLOC(newlf, 8 + 8*oldlf->no_keys + 8, 1);
|
|
newlf->no_keys = oldlf->no_keys;
|
|
newlf->id = oldlf->id;
|
|
#ifdef AKDEBUG
|
|
printf("add_key: new lf/lh no_keys: %d\n",newlf->no_keys);
|
|
#endif
|
|
|
|
/* Now copy old, checking where to insert (alphabetically) */
|
|
for (o = 0, n = 0; o < oldlf->no_keys; o++,n++) {
|
|
onkofs = oldlf->hash[o].ofs_nk;
|
|
onk = (struct nk_key *)(onkofs + hdesc->buffer + 0x1004);
|
|
if (slot == -1) {
|
|
|
|
#if 0
|
|
printf("add_key: cmp <%s> with <%s>\n",name,onk->keyname);
|
|
#endif
|
|
cmp = strncasecmp(name, onk->keyname, (namlen > onk->len_name) ? namlen : onk->len_name);
|
|
if (!cmp) {
|
|
printf("add_key: key %s already exists!\n",name);
|
|
FREE(newlf);
|
|
return(NULL);
|
|
}
|
|
if ( cmp < 0 ) {
|
|
slot = o;
|
|
rimax = rislot; /* Cause end of 'ri' search, too */
|
|
n++;
|
|
#ifdef AKDEBUG
|
|
printf("add_key: lf-match: slot = %d\n",o);
|
|
#endif
|
|
}
|
|
}
|
|
newlf->hash[n].ofs_nk = oldlf->hash[o].ofs_nk;
|
|
newlf->hash[n].name[0] = oldlf->hash[o].name[0];
|
|
newlf->hash[n].name[1] = oldlf->hash[o].name[1];
|
|
newlf->hash[n].name[2] = oldlf->hash[o].name[2];
|
|
newlf->hash[n].name[3] = oldlf->hash[o].name[3];
|
|
}
|
|
if (slot == -1) slot = oldlf->no_keys;
|
|
} /* li else check */
|
|
|
|
|
|
} while ( (rislot < rimax) && (rimax > 0)); /* 'ri' wrapper loop */
|
|
|
|
} else { /* Parent was empty, make new index block */
|
|
#ifdef AKDEBUG
|
|
printf("add_key: new index!\n");
|
|
#endif
|
|
ALLOC(newlf, 8 + 8, 1);
|
|
newlf->no_keys = 1;
|
|
/* Use ID (lf, lh or li) we fetched from root node, so we use same as rest of hive */
|
|
newlf->id = hdesc->nkindextype;
|
|
slot = 0;
|
|
} /* if has keys before */
|
|
|
|
|
|
/* Make and fill in new nk */
|
|
newnkofs = alloc_block(hdesc, nkofs, sizeof(struct nk_key) + strlen(name));
|
|
if (!newnkofs) {
|
|
printf("add_key: unable to allocate space for new key descriptor for %s!\n",name);
|
|
FREE(newlf);
|
|
FREE(newli);
|
|
return(NULL);
|
|
}
|
|
newnk = (struct nk_key *)(hdesc->buffer + newnkofs + 4);
|
|
|
|
newnk->id = 0x6b6e;
|
|
newnk->type = KEY_NORMAL;
|
|
newnk->ofs_parent = nkofs - 0x1004;
|
|
newnk->no_subkeys = 0;
|
|
newnk->ofs_lf = 0;
|
|
newnk->no_values = 0;
|
|
newnk->ofs_vallist = -1;
|
|
newnk->ofs_sk = key->ofs_sk; /* Get parents for now. 0 or -1 here crashes XP */
|
|
newnk->ofs_classnam = -1;
|
|
newnk->len_name = strlen(name);
|
|
newnk->len_classnam = 0;
|
|
strcpy(newnk->keyname, name);
|
|
|
|
if (newli) { /* Handle li */
|
|
|
|
#if AKDEBUG
|
|
printf("add_key: li fill at slot: %d\n",slot);
|
|
#endif
|
|
|
|
/* And put its offset into parents index list */
|
|
newli->hash[slot].ofs_nk = newnkofs - 0x1000;
|
|
newli->no_keys++;
|
|
|
|
/* Allocate space for our new li list and copy it into reg */
|
|
newliofs = alloc_block(hdesc, nkofs, 8 + 4*newli->no_keys);
|
|
if (!newliofs) {
|
|
printf("add_key: unable to allocate space for new index table for %s!\n",name);
|
|
FREE(newli);
|
|
free_block(hdesc,newnkofs);
|
|
return(NULL);
|
|
}
|
|
/* memcpy(hdesc->buffer + newliofs + 4, newli, 8 + 4*newli->no_keys); */
|
|
fill_block(hdesc, newliofs, newli, 8 + 4*newli->no_keys);
|
|
|
|
|
|
} else { /* lh or lf */
|
|
|
|
#ifdef AKDEBUG
|
|
printf("add_key: lf/lh fill at slot: %d, rislot: %d\n",slot,rislot);
|
|
#endif
|
|
/* And put its offset into parents index list */
|
|
newlf->hash[slot].ofs_nk = newnkofs - 0x1000;
|
|
newlf->no_keys++;
|
|
if (newlf->id == 0x666c) { /* lf hash */
|
|
newlf->hash[slot].name[0] = 0;
|
|
newlf->hash[slot].name[1] = 0;
|
|
newlf->hash[slot].name[2] = 0;
|
|
newlf->hash[slot].name[3] = 0;
|
|
strncpy(newlf->hash[slot].name, name, 4);
|
|
} else if (newlf->id == 0x686c) { /* lh. XP uses this. hashes whole name */
|
|
for (i = 0,hash = 0; i < strlen(name); i++) {
|
|
hash *= 37;
|
|
hash += toupper(name[i]);
|
|
}
|
|
newlf->lh_hash[slot].hash = hash;
|
|
}
|
|
|
|
/* Allocate space for our new lf list and copy it into reg */
|
|
newlfofs = alloc_block(hdesc, nkofs, 8 + 8*newlf->no_keys);
|
|
if (!newlfofs) {
|
|
printf("add_key: unable to allocate space for new index table for %s!\n",name);
|
|
FREE(newlf);
|
|
free_block(hdesc,newnkofs);
|
|
return(NULL);
|
|
}
|
|
/* memcpy(hdesc->buffer + newlfofs + 4, newlf, 8 + 8*newlf->no_keys); */
|
|
fill_block(hdesc, newlfofs, newlf, 8 + 8*newlf->no_keys);
|
|
|
|
} /* li else */
|
|
|
|
|
|
/* Update parent, and free old lf list */
|
|
key->no_subkeys++;
|
|
if (ri) { /* ri index */
|
|
ri->hash[rislot].ofs_li = (newlf ? newlfofs : newliofs) - 0x1000;
|
|
} else { /* Parent key */
|
|
key->ofs_lf = (newlf ? newlfofs : newliofs) - 0x1000;
|
|
}
|
|
|
|
if (newlf && oldlfofs) free_block(hdesc,oldlfofs + 0x1000);
|
|
if (newli && oldliofs) free_block(hdesc,oldliofs + 0x1000);
|
|
|
|
FREE(newlf);
|
|
FREE(newli);
|
|
return(newnk);
|
|
|
|
|
|
}
|
|
|
|
/* Delete a subkey from a key
|
|
* hdesc - usual..
|
|
* nkofs - offset of current nk
|
|
* name - name of key to delete (must match exactly, also case)
|
|
* return: 1 - err, 0 - ok
|
|
*/
|
|
|
|
#undef DKDEBUG
|
|
|
|
int del_key(struct hive *hdesc, int nkofs, char *name)
|
|
{
|
|
|
|
int slot = 0, newlfofs = 0, oldlfofs = 0, o, n, onkofs, delnkofs;
|
|
int oldliofs = 0, no_keys = 0, newriofs = 0;
|
|
int namlen;
|
|
int rimax, riofs, rislot;
|
|
struct ri_key *ri, *newri = NULL;
|
|
struct lf_key *newlf = NULL, *oldlf = NULL;
|
|
struct li_key *newli = NULL, *oldli = NULL;
|
|
struct nk_key *key, *onk, *delnk;
|
|
char fullpath[501];
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
|
|
namlen = strlen(name);
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("add_key: current ptr not nk\n");
|
|
return(1);
|
|
}
|
|
|
|
slot = -1;
|
|
if (!key->no_subkeys) {
|
|
printf("del_key: key has no subkeys!\n");
|
|
return(1);
|
|
}
|
|
|
|
oldlfofs = key->ofs_lf;
|
|
oldliofs = key->ofs_lf;
|
|
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
if (oldlf->id != 0x666c && oldlf->id != 0x686c && oldlf->id != 0x696c && oldlf->id != 0x6972) {
|
|
printf("del_key: index other than 'lf', 'li' or 'lh' not supported yet. 0x%04x\n",oldlf->id);
|
|
return(1);
|
|
}
|
|
|
|
rimax = 0; ri = NULL; riofs = 0;
|
|
rislot = 0;
|
|
|
|
if (oldlf->id == 0x6972) { /* Indirect index 'ri', init loop */
|
|
riofs = key->ofs_lf;
|
|
ri = (struct ri_key *)(hdesc->buffer + riofs + 0x1004);
|
|
rimax = ri->no_lis-1;
|
|
|
|
#ifdef DKDEBUG
|
|
printf("del_key: entering 'ri' traverse, rimax = %d\n",rimax);
|
|
#endif
|
|
|
|
rislot = -1; /* Starts at slot 0 below */
|
|
|
|
}
|
|
|
|
do { /* 'ri' loop, at least run once if no 'ri' deep index */
|
|
|
|
if (ri) { /* Do next 'ri' slot */
|
|
rislot++;
|
|
oldliofs = ri->hash[rislot].ofs_li;
|
|
oldlfofs = ri->hash[rislot].ofs_li;
|
|
}
|
|
|
|
oldli = (struct li_key *)(hdesc->buffer + oldliofs + 0x1004);
|
|
oldlf = (struct lf_key *)(hdesc->buffer + oldlfofs + 0x1004);
|
|
|
|
#ifdef DKDEBUG
|
|
printf("del_key: top of ri-loop: rislot = %d\n",rislot);
|
|
#endif
|
|
slot = -1;
|
|
|
|
if (oldlf->id == 0x696c) { /* 'li' handler */
|
|
#ifdef DKDEBUG
|
|
printf("del_key: li handler\n");
|
|
#endif
|
|
|
|
FREE(newli);
|
|
ALLOC(newli, 8 + 4*oldli->no_keys - 4, 1);
|
|
newli->no_keys = oldli->no_keys - 1; no_keys = newli->no_keys;
|
|
newli->id = oldli->id;
|
|
|
|
/* Now copy old, checking where to delete */
|
|
for (o = 0, n = 0; o < oldli->no_keys; o++,n++) {
|
|
onkofs = oldli->hash[o].ofs_nk;
|
|
onk = (struct nk_key *)(onkofs + hdesc->buffer + 0x1004);
|
|
if (slot == -1 && onk->len_name == namlen && !strncmp(name, onk->keyname, (onk->len_name > namlen) ? onk->len_name : namlen)) {
|
|
slot = o;
|
|
delnkofs = onkofs; delnk = onk;
|
|
rimax = rislot;
|
|
o++;
|
|
}
|
|
newli->hash[n].ofs_nk = oldli->hash[o].ofs_nk;
|
|
}
|
|
|
|
|
|
} else { /* 'lf' or 'lh' are similar */
|
|
|
|
#ifdef DKDEBUG
|
|
printf("del_key: lf or lh handler\n");
|
|
#endif
|
|
FREE(newlf);
|
|
ALLOC(newlf, 8 + 8*oldlf->no_keys - 8, 1);
|
|
newlf->no_keys = oldlf->no_keys - 1; no_keys = newlf->no_keys;
|
|
newlf->id = oldlf->id;
|
|
|
|
/* Now copy old, checking where to delete */
|
|
for (o = 0, n = 0; o < oldlf->no_keys; o++,n++) {
|
|
onkofs = oldlf->hash[o].ofs_nk;
|
|
onk = (struct nk_key *)(onkofs + hdesc->buffer + 0x1004);
|
|
if (slot == -1 && (onk->len_name == namlen) && !strncmp(name, onk->keyname, onk->len_name)) {
|
|
slot = o;
|
|
delnkofs = onkofs; delnk = onk;
|
|
rimax = rislot;
|
|
o++;
|
|
}
|
|
newlf->hash[n].ofs_nk = oldlf->hash[o].ofs_nk;
|
|
newlf->hash[n].name[0] = oldlf->hash[o].name[0];
|
|
newlf->hash[n].name[1] = oldlf->hash[o].name[1];
|
|
newlf->hash[n].name[2] = oldlf->hash[o].name[2];
|
|
newlf->hash[n].name[3] = oldlf->hash[o].name[3];
|
|
}
|
|
} /* else lh or lf */
|
|
|
|
} while (rislot < rimax); /* ri traverse loop */
|
|
|
|
if (slot == -1) {
|
|
printf("del_key: subkey %s not found!\n",name);
|
|
FREE(newlf);
|
|
FREE(newli);
|
|
return(1);
|
|
}
|
|
|
|
#ifdef DKDEBUG
|
|
printf("del_key: key found at slot %d\n",slot);
|
|
#endif
|
|
|
|
if (delnk->no_values || delnk->no_subkeys) {
|
|
printf("del_key: subkey %s has subkeys or values. Not deleted.\n",name);
|
|
FREE(newlf);
|
|
FREE(newli);
|
|
return(1);
|
|
}
|
|
|
|
/* Allocate space for our new lf list and copy it into reg */
|
|
if ( no_keys && (newlf || newli) ) {
|
|
newlfofs = alloc_block(hdesc, nkofs, 8 + (newlf ? 8 : 4) * no_keys);
|
|
#ifdef DKDEBUG
|
|
printf("del_key: alloc_block for index returns: %x\n",newlfofs);
|
|
#endif
|
|
if (!newlfofs) {
|
|
printf("del_key: WARNING: unable to allocate space for new key descriptor for %s! Not deleted\n",name);
|
|
FREE(newlf);
|
|
return(1);
|
|
}
|
|
|
|
/* memcpy(hdesc->buffer + newlfofs + 4,
|
|
((void *)newlf ? (void *)newlf : (void *)newli), 8 + (newlf ? 8 : 4) * no_keys);
|
|
*/
|
|
fill_block(hdesc, newlfofs,
|
|
((void *)newlf ? (void *)newlf : (void *)newli), 8 + (newlf ? 8 : 4) * no_keys);
|
|
|
|
|
|
} else { /* Last deleted, will throw away index */
|
|
newlfofs = 0xfff; /* We subtract 0x1000 later */
|
|
}
|
|
|
|
if (newlfofs < 0xfff) {
|
|
printf("del_key: ERROR: newlfofs = %x\n",newlfofs);
|
|
#if DOCORE
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
abort();
|
|
#endif
|
|
}
|
|
|
|
/* Check for CLASS data, if so, deallocate it too */
|
|
if (delnk->len_classnam) {
|
|
free_block(hdesc, delnk->ofs_classnam + 0x1000);
|
|
}
|
|
/* Now it's safe to zap the nk */
|
|
free_block(hdesc, delnkofs + 0x1000);
|
|
/* And the old index list */
|
|
free_block(hdesc, (oldlfofs ? oldlfofs : oldliofs) + 0x1000);
|
|
|
|
/* Update parent */
|
|
key->no_subkeys--;
|
|
|
|
if (ri) {
|
|
if (newlfofs == 0xfff) {
|
|
|
|
*fullpath = 0;
|
|
get_abs_path(hdesc, nkofs, fullpath, 480);
|
|
|
|
VERBF(hdesc,"del_key: need to delete ri-slot %d for %x - %s\n", rislot,nkofs,fullpath );
|
|
|
|
if (ri->no_lis > 1) { /* We have subindiceblocks left? */
|
|
/* Delete from array */
|
|
ALLOC(newri, 8 + 4*ri->no_lis - 4, 1);
|
|
newri->no_lis = ri->no_lis - 1;
|
|
newri->id = ri->id;
|
|
for (o = 0, n = 0; o < ri->no_lis; o++,n++) {
|
|
if (n == rislot) o++;
|
|
newri->hash[n].ofs_li = ri->hash[o].ofs_li;
|
|
}
|
|
newriofs = alloc_block(hdesc, nkofs, 8 + newri->no_lis*4 );
|
|
if (!newriofs) {
|
|
printf("del_key: WARNING: unable to allocate space for ri-index for %s! Not deleted\n",name);
|
|
FREE(newlf);
|
|
FREE(newri);
|
|
return(1);
|
|
}
|
|
fill_block(hdesc, newriofs, newri, 8 + newri->no_lis * 4);
|
|
free_block(hdesc, riofs + 0x1000);
|
|
key->ofs_lf = newriofs - 0x1000;
|
|
FREE(newri);
|
|
} else { /* Last entry in ri was deleted, get rid of it, key is empty */
|
|
VERB(hdesc,"del_key: .. and that was the last one. key now empty!\n");
|
|
free_block(hdesc, riofs + 0x1000);
|
|
key->ofs_lf = -1;
|
|
}
|
|
} else {
|
|
ri->hash[rislot].ofs_li = newlfofs - 0x1000;
|
|
}
|
|
} else {
|
|
key->ofs_lf = newlfofs - 0x1000;
|
|
}
|
|
|
|
FREE(newlf);
|
|
return(0);
|
|
|
|
}
|
|
|
|
/* Recursive delete keys
|
|
* hdesc - usual..
|
|
* nkofs - offset of current nk
|
|
* name - name of key to delete
|
|
* return: 0 - ok, 1 fail
|
|
*/
|
|
void rdel_keys(struct hive *hdesc, char *path, int vofs)
|
|
{
|
|
struct nk_key *key;
|
|
int nkofs;
|
|
struct ex_data ex;
|
|
int count = 0, countri = 0;
|
|
|
|
|
|
if (!path || !*path) return;
|
|
|
|
nkofs = trav_path(hdesc, vofs, path, TPF_NK_EXACT);
|
|
|
|
if(!nkofs) {
|
|
printf("rdel_keys: Key <%s> not found\n",path);
|
|
return;
|
|
}
|
|
nkofs += 4;
|
|
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
|
|
/*
|
|
VERBF(hdesc,"rdel of node at offset 0x%0x\n",nkofs);
|
|
*/
|
|
|
|
if (key->id != 0x6b6e) {
|
|
printf("Error: Not a 'nk' node!\n");
|
|
|
|
debugit(hdesc->buffer,hdesc->size);
|
|
|
|
}
|
|
|
|
#if 0
|
|
printf("Node has %d subkeys and %d values\n",key->no_subkeys,key->no_values);
|
|
#endif
|
|
if (key->no_subkeys) {
|
|
while ((ex_next_n(hdesc, nkofs, &count, &countri, &ex) > 0)) {
|
|
#if 0
|
|
printf("%s\n",ex.name);
|
|
#endif
|
|
rdel_keys(hdesc, ex.name, nkofs);
|
|
count = 0;
|
|
countri = 0;
|
|
FREE(ex.name);
|
|
}
|
|
}
|
|
|
|
del_allvalues(hdesc, nkofs);
|
|
del_key(hdesc, key->ofs_parent+0x1004, path);
|
|
|
|
}
|
|
|
|
|
|
/* Get and copy keys CLASS-data (if any) to buffer
|
|
* Returns a buffer with the data (first int32_t is size). see ntreg.h
|
|
* NOTE: caller must deallocate buffer! a simple free(keyval) will suffice.
|
|
*/
|
|
struct keyval *get_class(struct hive *hdesc,
|
|
int curnk, char *path)
|
|
{
|
|
int clen = 0, dofs = 0, nkofs;
|
|
struct nk_key *key;
|
|
struct keyval *data;
|
|
void *classdata;
|
|
|
|
if (!path && !curnk) return(NULL);
|
|
|
|
nkofs = trav_path(hdesc, curnk, path, 0);
|
|
|
|
if(!nkofs) {
|
|
printf("get_class: Key <%s> not found\n",path);
|
|
return(NULL);
|
|
}
|
|
nkofs += 4;
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
|
|
clen = key->len_classnam;
|
|
if (!clen) {
|
|
printf("get_class: Key has no class data.\n");
|
|
return(NULL);
|
|
}
|
|
|
|
dofs = key->ofs_classnam;
|
|
classdata = (void *)(hdesc->buffer + dofs + 0x1004);
|
|
|
|
#if 0
|
|
printf("get_class: len_classnam = %d\n",clen);
|
|
printf("get_class: ofs_classnam = 0x%x\n",dofs);
|
|
#endif
|
|
|
|
ALLOC(data, sizeof(struct keyval) + clen,1);
|
|
data->len = clen;
|
|
memcpy(&data->data, classdata, clen);
|
|
return(data);
|
|
}
|
|
|
|
|
|
/* Write to registry value.
|
|
* If same size as existing, copy back in place to avoid changing too much
|
|
* otherwise allocate new dataspace, then free the old
|
|
* Thus enough space to hold both new and old data is needed
|
|
* Pass inn buffer with data len as first DWORD (as routines above)
|
|
* returns: 0 - error, len - OK (len of data)
|
|
*/
|
|
|
|
int put_buf2val(struct hive *hdesc, struct keyval *kv,
|
|
int vofs, char *path, int type, int exact )
|
|
{
|
|
int l;
|
|
void *keydataptr;
|
|
|
|
if (!kv) return(0);
|
|
l = get_val_len(hdesc, vofs, path, exact);
|
|
if (l == -1) return(0); /* error */
|
|
if (kv->len != l) { /* Realloc data block if not same size as existing */
|
|
if (!alloc_val_data(hdesc, vofs, path, kv->len, exact)) {
|
|
printf("put_buf2val: %s : alloc_val_data failed!\n",path);
|
|
return(0);
|
|
}
|
|
}
|
|
|
|
keydataptr = get_val_data(hdesc, vofs, path, type, exact);
|
|
if (!keydataptr) return(0); /* error */
|
|
|
|
memcpy(keydataptr, &kv->data, kv->len);
|
|
|
|
hdesc->state |= HMODE_DIRTY;
|
|
|
|
return(kv->len);
|
|
}
|
|
|
|
/* And, yer basic DWORD write */
|
|
|
|
int put_dword(struct hive *hdesc, int vofs, char *path, int exact, int dword)
|
|
{
|
|
struct keyval *kr;
|
|
int r;
|
|
|
|
ALLOC(kr,1,sizeof(int)+sizeof(int));
|
|
|
|
kr->len = sizeof(int);
|
|
kr->data = dword;
|
|
|
|
r = put_buf2val(hdesc, kr, vofs, path, REG_DWORD, exact);
|
|
|
|
FREE(kr);
|
|
|
|
return(r);
|
|
}
|
|
|
|
/* ================================================================ */
|
|
|
|
/* Code to export registry entries to .reg file initiated by
|
|
* Leo von Klenze
|
|
* Then expanded a bit to handle more types etc.
|
|
*/
|
|
|
|
/*
|
|
* converts a value string from an registry entry into a c string. It does not
|
|
* use any encoding functions.
|
|
* It works very primitive by just taking every second char.
|
|
* The caller must free the resulting string, that was allocated with malloc.
|
|
*
|
|
* string: string where every second char is \0
|
|
* len: length of the string
|
|
* return: the converted string as char*
|
|
*/
|
|
static char *
|
|
string_regw2prog(void *string, int len)
|
|
{
|
|
int i, k;
|
|
char *cstring;
|
|
|
|
int out_len = 0;
|
|
for(i = 0; i < len; i += 2)
|
|
{
|
|
unsigned v = ((unsigned char *)string)[i] + ((unsigned char *)string)[i+1] * 256u;
|
|
if (v < 128)
|
|
out_len += 1;
|
|
else if(v < 0x800)
|
|
out_len += 2;
|
|
else
|
|
out_len += 3;
|
|
}
|
|
cstring = (char *) malloc(out_len+1);
|
|
if (!cstring) {
|
|
printf("FATAL! ex_next: malloc() failed! Out of memory?\n");
|
|
abort();
|
|
}
|
|
|
|
for(i = 0, k = 0; i < len; i += 2)
|
|
{
|
|
unsigned v = ((unsigned char *)string)[i] + ((unsigned char *)string)[i+1] * 256u;
|
|
if (v < 128)
|
|
cstring[k++] = v;
|
|
else if(v < 0x800) {
|
|
cstring[k++] = 0xc0 | (v >> 6);
|
|
cstring[k++] = 0x80 | (v & 0x3f);
|
|
} else {
|
|
cstring[k++] = 0xe0 | (v >> 12);
|
|
cstring[k++] = 0x80 | ((v >> 6) & 0x3f);
|
|
cstring[k++] = 0x80 | (v & 0x3f);
|
|
}
|
|
}
|
|
cstring[out_len] = '\0';
|
|
|
|
return cstring;
|
|
}
|
|
|
|
static char *
|
|
string_rega2prog(void *string, int len)
|
|
{
|
|
int i, k;
|
|
char *cstring;
|
|
|
|
int out_len = 0;
|
|
for(i = 0; i < len; ++i)
|
|
{
|
|
unsigned v = ((unsigned char *)string)[i];
|
|
if (v < 128)
|
|
out_len += 1;
|
|
else
|
|
out_len += 2;
|
|
}
|
|
cstring = (char *) malloc(out_len+1);
|
|
if (!cstring) {
|
|
printf("FATAL! ex_next: malloc() failed! Out of memory?\n");
|
|
abort();
|
|
}
|
|
|
|
for(i = 0, k = 0; i < len; ++i)
|
|
{
|
|
unsigned v = ((unsigned char *)string)[i];
|
|
if (v < 128)
|
|
cstring[k++] = v;
|
|
else {
|
|
cstring[k++] = 0xc0 | (v >> 6);
|
|
cstring[k++] = 0x80 | (v & 0x3f);
|
|
}
|
|
}
|
|
cstring[out_len] = '\0';
|
|
|
|
return cstring;
|
|
}
|
|
|
|
static void
|
|
string_prog2rega(char *string, int len)
|
|
{
|
|
char *out = string;
|
|
unsigned char *in = (unsigned char*) string;
|
|
|
|
for (;*in; ++in) {
|
|
if (!(*in & 0x80)) {
|
|
*out++ = *in;
|
|
} else if ((in[0] & 0xe0) == 0xc0 && in[1]) {
|
|
*out++ = (in[0] & 0x1f) << 6 | (in[1] & 0x3f);
|
|
++in;
|
|
} else if (in[1] && in[2]) {
|
|
/* assume 3 byte*/
|
|
*out++ = (in[1] & 0xf) << 6 | (in[2] & 0x3f);
|
|
in += 2;
|
|
}
|
|
}
|
|
*out = 0;
|
|
}
|
|
|
|
static char *
|
|
string_prog2regw(void *string, int len, int *out_len)
|
|
{
|
|
unsigned char *regw = (unsigned char*) malloc(len*2+2);
|
|
unsigned char *out = regw;
|
|
unsigned char *in = (unsigned char*) string;
|
|
|
|
for (;len>0; ++in, --len) {
|
|
if (!(in[0] & 0x80)) {
|
|
*out++ = *in;
|
|
*out++ = 0;
|
|
} else if ((in[0] & 0xe0) == 0xc0 && len >= 2) {
|
|
*out++ = (in[0] & 0x1f) << 6 | (in[1] & 0x3f);
|
|
*out++ = (in[0] & 0x1f) >> 2;
|
|
++in, --len;
|
|
} else if (len >= 3) {
|
|
/* assume 3 byte*/
|
|
*out++ = (in[1] & 0xf) << 6 | (in[2] & 0x3f);
|
|
*out++ = (in[0] & 0xf) << 4 | ((in[1] & 0x3f) >> 2);
|
|
in += 2;
|
|
len -= 2;
|
|
}
|
|
}
|
|
*out_len = out - regw;
|
|
out[0] = out[1] = 0;
|
|
return (char *) regw;
|
|
}
|
|
|
|
static char *
|
|
quote_string(const char *s)
|
|
{
|
|
int len = strlen(s);
|
|
const char *p;
|
|
char *dst, *out;
|
|
|
|
for (p = s; *p; ++p)
|
|
if (*p == '\\' || *p == '\"')
|
|
++len;
|
|
|
|
dst = out = (char*) malloc(len + 1);
|
|
for (p = s; *p; ++p) {
|
|
if (*p == '\\' || *p == '\"')
|
|
*dst++ = '\\';
|
|
*dst++ = *p;
|
|
}
|
|
*dst = 0;
|
|
return out;
|
|
}
|
|
|
|
static void
|
|
export_bin(int type, char *value, int len, int col, FILE* file)
|
|
{
|
|
int byte;
|
|
|
|
if (type == REG_BINARY) {
|
|
fprintf(file, "hex:");
|
|
col += 4;
|
|
} else {
|
|
fprintf(file, "hex(%x):", type);
|
|
col += 7;
|
|
}
|
|
byte = 0;
|
|
int start = (col - 2) / 3;
|
|
while (byte + 1 < len) { /* go byte by byte.. probably slow.. */
|
|
fprintf(file, "%02x,", (unsigned char)value[byte]);
|
|
byte++;
|
|
if (!((byte + start) % 25)) fprintf(file, "\\\r\n ");
|
|
}
|
|
if (len)
|
|
fprintf(file, "%02x\r\n", (unsigned char)value[len - 1]);
|
|
}
|
|
|
|
/*
|
|
* Exports the named subkey and its values to the given file.
|
|
*
|
|
* hdesc: registry hive
|
|
* nkofs: offset of parent node
|
|
* name: name of key to export
|
|
* prefix: prefix for each key. This prefix is prepended to the keyname
|
|
* file: file descriptor of destination file
|
|
*/
|
|
void export_subkey(struct hive *hdesc, int nkofs, char *name, char *prefix, FILE *file)
|
|
{
|
|
int newofs;
|
|
int count = 0;
|
|
int countri = 0;
|
|
int len;
|
|
char *path = (char*) malloc(1024);
|
|
char *value;
|
|
struct nk_key *key;
|
|
struct ex_data ex;
|
|
struct vex_data vex;
|
|
|
|
|
|
// switch to key
|
|
newofs = trav_path(hdesc, nkofs, name, TPF_NK_EXACT);
|
|
if(!newofs)
|
|
{
|
|
printf("Key '%s' not found!\n", name);
|
|
free(path);
|
|
return;
|
|
}
|
|
nkofs = newofs + 4;
|
|
|
|
// get the key
|
|
key = (struct nk_key *)(hdesc->buffer + nkofs);
|
|
printf("Exporting key '%.*s' with %d subkeys and %d values...\n",
|
|
key->len_name, key->keyname, key->no_subkeys, key->no_values);
|
|
|
|
*path = 0;
|
|
get_abs_path(hdesc, nkofs, path, 1024);
|
|
|
|
// export the key
|
|
fprintf(file, "\r\n");
|
|
fprintf(file, "[%s\%s]\r\n", prefix, path);
|
|
// export values
|
|
if(key->no_values)
|
|
{
|
|
while ((ex_next_v(hdesc, nkofs, &count, &vex) > 0))
|
|
{
|
|
int col = 0;
|
|
char *name = quote_string(vex.name);
|
|
|
|
/* print name */
|
|
if (!name[0]) {
|
|
fprintf(file, "@=");
|
|
free(name);
|
|
name = str_dup("@");
|
|
col += 2;
|
|
} else {
|
|
fprintf(file, "\"%s\"=", name);
|
|
col += strlen(name) + 3;
|
|
}
|
|
|
|
if(vex.type == REG_DWORD)
|
|
{
|
|
fprintf(file, "dword:%08x\r\n", vex.val);
|
|
}
|
|
else if(vex.type == REG_SZ)
|
|
{
|
|
char *val, *quoted;
|
|
value = (char *)get_val_data(hdesc, nkofs, name, vex.type, TPF_VK_EXACT);
|
|
len = get_val_len(hdesc, nkofs, name, TPF_VK_EXACT);
|
|
|
|
val = string_regw2prog(value, len);
|
|
/* dump as binary if invalid characters are embedded */
|
|
if (strchr(val, 0xa) || strchr(val, 0xd))
|
|
{
|
|
free(val);
|
|
//if (len >= 2 && value[len-2] == 0 && value[len-1] == 0) len -= 2;
|
|
export_bin(vex.type, value, len, col, file);
|
|
}
|
|
else
|
|
{
|
|
quoted = quote_string(val);
|
|
free(val);
|
|
fprintf(file, "\"%s\"", quoted);
|
|
free(quoted);
|
|
fprintf(file, "\r\n");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* All other types seems to simply be hexdumped. Format is:
|
|
"valuename"=hex(typenum):xx,xx,xx,xx,xx...
|
|
for example:
|
|
"qword64"=hex(b):4e,03,51,db,fa,04,00,00
|
|
this is type = 0xb = 11 = REG_QWORD
|
|
"expandstring"=hex(2):46,00,6f,00,6f,00,62,00,61,00,72,00,20,00,25,00,73,00,20,\
|
|
00,42,00,61,00,7a,00,00,00
|
|
this is type 2 = REG_EXPAND_SZ and the line is continued with ,\<CR><LF><space><space>
|
|
don't know how many bytes for each line. Around 18-24 seems to be it?? depends on name lenght??
|
|
NOTE: Exception:
|
|
type = REG_BINARY starts like this: "valuename"=hex:xx,xx,xx...
|
|
*/
|
|
value = (char *)get_val_data(hdesc, nkofs, name, vex.type, TPF_VK_EXACT);
|
|
len = get_val_len(hdesc, nkofs, name, TPF_VK_EXACT);
|
|
|
|
export_bin(vex.type, value, len, col, file);
|
|
}
|
|
|
|
FREE(vex.name);
|
|
free(name);
|
|
}
|
|
}
|
|
|
|
// export subkeys
|
|
if (key->no_subkeys)
|
|
{
|
|
count = 0;
|
|
countri = 0;
|
|
while ((ex_next_n(hdesc, nkofs, &count, &countri, &ex) > 0))
|
|
{
|
|
export_subkey(hdesc, nkofs, ex.name, prefix, file);
|
|
FREE(ex.name);
|
|
}
|
|
}
|
|
free(path);
|
|
}
|
|
|
|
/*
|
|
* Exports the given key to a windows .reg file that can be imported to the
|
|
* windows registry.
|
|
* The prefix is used to determine the destination hive in the windows
|
|
* registry, set it to HKEY_CURRENT_USER or HKEY_LOCAL_MACHINE or whatever you
|
|
* want.
|
|
*
|
|
* hdesc: hive
|
|
* nkofs: offset of parent node
|
|
* name: name of subkey to export
|
|
* filename: name of destination .reg file (will be overridden)
|
|
* prefix: prefix for each exported key
|
|
*/
|
|
void export_key(struct hive *hdesc, int nkofs, char *name, char *filename, char *prefix)
|
|
{
|
|
FILE *file;
|
|
|
|
// open file
|
|
file = fopen(filename, "w");
|
|
if(!file)
|
|
{
|
|
printf("Cannot open file '%s'. %s (%d).\n", filename, strerror(errno),
|
|
errno);
|
|
return;
|
|
}
|
|
|
|
printf("Exporting to file '%s'...\n", filename);
|
|
fprintf(file, "Windows Registry Editor Version 5.00\r\n");
|
|
export_subkey(hdesc, nkofs, name, prefix, file);
|
|
|
|
fclose(file);
|
|
}
|
|
|
|
|
|
/* ================================================================ */
|
|
|
|
/* Hive control (load/save/close) etc */
|
|
|
|
void closeHive(struct hive *hdesc)
|
|
{
|
|
|
|
printf("closing hive %s\n",hdesc->filename);
|
|
if (hdesc->state & HMODE_OPEN) {
|
|
close(hdesc->filedesc);
|
|
}
|
|
FREE(hdesc->filename);
|
|
FREE(hdesc->buffer);
|
|
FREE(hdesc);
|
|
|
|
}
|
|
|
|
/* Write the hive back to disk (only if dirty & not readonly */
|
|
int writeHive(struct hive *hdesc)
|
|
{
|
|
int len, i;
|
|
struct regf_header *hdr;
|
|
int32_t checksum;
|
|
|
|
if (hdesc->state & HMODE_RO) return(0);
|
|
if ( !(hdesc->state & HMODE_DIRTY)) return(0);
|
|
|
|
if ( !(hdesc->state & HMODE_OPEN)) { /* File has been closed */
|
|
if (!(hdesc->filedesc = open(hdesc->filename,O_RDWR))) {
|
|
fprintf(stderr,"writeHive: open(%s) failed: %s, FILE NOT WRITTEN!\n",hdesc->filename,strerror(errno));
|
|
return(1);
|
|
}
|
|
hdesc->state |= HMODE_OPEN;
|
|
}
|
|
/* Seek back to begginning of file (in case it's already open) */
|
|
lseek(hdesc->filedesc, 0, SEEK_SET);
|
|
|
|
/* compute new checksum */
|
|
hdr = (struct regf_header *) hdesc->buffer;
|
|
checksum = 0;
|
|
for (i = 0; i < 0x1fc/4; ++i)
|
|
checksum ^= ((int32_t *) hdr)[i];
|
|
hdr->checksum = checksum;
|
|
|
|
len = write(hdesc->filedesc, hdesc->buffer, hdesc->size);
|
|
if (len != hdesc->size) {
|
|
fprintf(stderr,"writeHive: write of %s failed: %s.\n",hdesc->filename,strerror(errno));
|
|
return(1);
|
|
}
|
|
|
|
hdesc->state &= (~HMODE_DIRTY);
|
|
return(0);
|
|
}
|
|
|
|
struct hive *openHive(char *filename, int mode)
|
|
{
|
|
|
|
struct hive *hdesc;
|
|
int fmode,r,vofs;
|
|
struct stat sbuf;
|
|
uint32_t pofs;
|
|
/* off_t l; */
|
|
char *c;
|
|
struct hbin_page *p;
|
|
struct regf_header *hdr;
|
|
struct nk_key *nk;
|
|
struct ri_key *rikey;
|
|
int verbose = (mode & HMODE_VERBOSE);
|
|
int trace = (mode & HMODE_TRACE);
|
|
|
|
CREATE(hdesc,struct hive,1);
|
|
|
|
hdesc->filename = str_dup(filename);
|
|
hdesc->state = 0;
|
|
hdesc->size = 0;
|
|
hdesc->buffer = NULL;
|
|
|
|
if ( (mode & HMODE_RO) ) {
|
|
fmode = O_RDONLY;
|
|
} else {
|
|
fmode = O_RDWR;
|
|
}
|
|
hdesc->filedesc = open(hdesc->filename,fmode);
|
|
if (hdesc->filedesc < 0) {
|
|
fprintf(stderr,"openHive(%s) failed: %s, trying read-only\n",hdesc->filename,strerror(errno));
|
|
fmode = O_RDONLY;
|
|
mode |= HMODE_RO;
|
|
hdesc->filedesc = open(hdesc->filename,fmode);
|
|
if (hdesc->filedesc < 0) {
|
|
fprintf(stderr,"openHive(%s) in fallback RO-mode failed: %s\n",hdesc->filename,strerror(errno));
|
|
closeHive(hdesc);
|
|
return(NULL);
|
|
}
|
|
}
|
|
|
|
|
|
if ( fstat(hdesc->filedesc,&sbuf) ) {
|
|
perror("stat()");
|
|
exit(1);
|
|
}
|
|
|
|
hdesc->size = sbuf.st_size;
|
|
hdesc->state = mode | HMODE_OPEN;
|
|
/* fprintf(stderr,"hiveOpen(%s) successful\n",hdesc->filename); */
|
|
|
|
/* Read the whole file */
|
|
|
|
ALLOC(hdesc->buffer,1,hdesc->size);
|
|
|
|
r = read(hdesc->filedesc,hdesc->buffer,hdesc->size);
|
|
if (r < hdesc->size) {
|
|
fprintf(stderr,"Could not read file, got %d bytes while expecting %d\n",
|
|
r, hdesc->size);
|
|
closeHive(hdesc);
|
|
return(NULL);
|
|
}
|
|
|
|
/* Now run through file, tallying all pages */
|
|
/* NOTE/KLUDGE: Assume first page starts at offset 0x1000 */
|
|
|
|
pofs = 0x1000;
|
|
|
|
hdr = (struct regf_header *)hdesc->buffer;
|
|
if (hdr->id != 0x66676572) {
|
|
printf("openHive(%s): File does not seem to be a registry hive!\n",filename);
|
|
return(hdesc);
|
|
}
|
|
printf("Hive <%s> name (from header): <",filename);
|
|
for (c = hdr->name; *c && (c < hdr->name + 64); c += 2) putchar(*c);
|
|
|
|
hdesc->rootofs = hdr->ofs_rootkey + 0x1000;
|
|
printf(">\nROOT KEY at offset: 0x%06x * ",hdesc->rootofs);
|
|
|
|
/* Cache the roots subkey index type (li,lf,lh) so we can use the correct
|
|
* one when creating the first subkey in a key */
|
|
|
|
nk = (struct nk_key *)(hdesc->buffer + hdesc->rootofs + 4);
|
|
if (nk->id == 0x6b6e) {
|
|
rikey = (struct ri_key *)(hdesc->buffer + nk->ofs_lf + 0x1004);
|
|
hdesc->nkindextype = rikey->id;
|
|
if (hdesc->nkindextype == 0x6972) { /* Gee, big root, must check indirectly */
|
|
printf("load_hive: DEBUG: BIG ROOT!\n");
|
|
rikey = (struct ri_key *)(hdesc->buffer + rikey->hash[0].ofs_li + 0x1004);
|
|
hdesc->nkindextype = rikey->id;
|
|
}
|
|
if (hdesc->nkindextype != 0x666c &&
|
|
hdesc->nkindextype != 0x686c &&
|
|
hdesc->nkindextype != 0x696c) {
|
|
hdesc->nkindextype = 0x666c;
|
|
}
|
|
|
|
printf("Subkey indexing type is: %04x <%c%c>\n",
|
|
hdesc->nkindextype,
|
|
hdesc->nkindextype & 0xff,
|
|
hdesc->nkindextype >> 8);
|
|
} else {
|
|
printf("load_hive: WARNING: ROOT key does not seem to be a key! (not type == nk)\n");
|
|
}
|
|
|
|
|
|
|
|
while (pofs < hdesc->size) {
|
|
#ifdef LOAD_DEBUG
|
|
if (trace) hexdump(hdesc->buffer,pofs,pofs+0x20,1);
|
|
#endif
|
|
p = (struct hbin_page *)(hdesc->buffer + pofs);
|
|
if (p->id != 0x6E696268) {
|
|
printf("Page at 0x%x is not 'hbin', assuming file contains garbage at end\n",pofs);
|
|
break;
|
|
}
|
|
hdesc->pages++;
|
|
#ifdef LOAD_DEBUG
|
|
if (trace) printf("\n###### Page at 0x%0lx has size 0x%0lx, next at 0x%0lx ######\n",pofs,p->len_page,p->ofs_next);
|
|
#endif
|
|
if (p->ofs_next == 0) {
|
|
#ifdef LOAD_DEBUG
|
|
if (trace) printf("openhive debug: bailing out.. pagesize zero!\n");
|
|
#endif
|
|
return(hdesc);
|
|
}
|
|
#if 0
|
|
if (p->len_page != p->ofs_next) {
|
|
#ifdef LOAD_DEBUG
|
|
if (trace) printf("openhive debug: len & ofs not same. HASTA!\n");
|
|
#endif
|
|
exit(0);
|
|
}
|
|
#endif
|
|
|
|
|
|
vofs = pofs + 0x20; /* Skip page header */
|
|
#if 1
|
|
while (vofs-pofs < p->ofs_next && vofs < hdesc->size) {
|
|
vofs += parse_block(hdesc,vofs,trace);
|
|
|
|
}
|
|
#endif
|
|
pofs += p->ofs_next;
|
|
}
|
|
printf("File size %d [%x] bytes, containing %d pages (+ 1 headerpage)\n",hdesc->size,hdesc->size, hdesc->pages);
|
|
printf("Used for data: %d/%d blocks/bytes, unused: %d/%d blocks/bytes.\n\n",
|
|
hdesc->useblk,hdesc->usetot,hdesc->unuseblk,hdesc->unusetot);
|
|
|
|
|
|
/* So, let's guess what kind of hive this is, based on keys in its root */
|
|
|
|
hdesc->type = HTYPE_UNKNOWN;
|
|
if (trav_path(hdesc, 0, "\\SAM", 0)) hdesc->type = HTYPE_SAM;
|
|
else if (trav_path(hdesc, 0, "\\ControlSet", 0)) hdesc->type = HTYPE_SYSTEM;
|
|
else if (trav_path(hdesc, 0, "\\Policy", 0)) hdesc->type = HTYPE_SECURITY;
|
|
else if (trav_path(hdesc, 0, "\\Microsoft", 0)) hdesc->type = HTYPE_SOFTWARE;
|
|
if (verbose) printf("Type of hive guessed to be: %d\n",hdesc->type);
|
|
|
|
return(hdesc);
|
|
|
|
}
|
|
|