forked from rescatux/chntpw
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathntreg.h
451 lines (366 loc) · 17.9 KB
/
ntreg.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
/*
* ntreg.h - NT Registry Hive access library, constants & structures
*
* NOTE: defines are not frozen. It can and will change every release.
*
*****
*
* NTREG - Window registry file reader / writer library
* Copyright (c) 1997-2013 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.
*
*/
#ifndef _INCLUDE_NTREG_H
#define _INCLUDE_NTREG_H 1
#define SZ_MAX 4096 /* Max unicode strlen before we truncate */
#define KEY_ROOT 0x2c /* Type ID of ROOT key node */
#define KEY_NORMAL 0x20 /* Normal nk key */
#define ABSPATHLEN 4096
/* hbin page size. hbins are minimum this, and always multiple of this */
#define HBIN_PAGESIZE 0x1000
/* Hive filesize seems to always be multiple of this */
#define REGF_FILEDIVISOR 0x40000
/* Larger than this, and values seems split into several blocks */
#define VAL_DIRECT_LIMIT 0x3fd0
/* Datatypes of the values in the registry */
#define REG_NONE 0 /* No value type */
#define REG_SZ 1 /* Unicode nul terminated string */
#define REG_EXPAND_SZ 2 /* Unicode nul terminated string + env */
#define REG_BINARY 3 /* Free form binary */
#define REG_DWORD 4 /* 32-bit number */
#define REG_DWORD_BIG_ENDIAN 5 /* 32-bit number */
#define REG_LINK 6 /* Symbolic Link (unicode) */
#define REG_MULTI_SZ 7 /* Multiple Unicode strings */
#define REG_RESOURCE_LIST 8 /* Resource list in the resource map */
#define REG_FULL_RESOURCE_DESCRIPTOR 9 /* Resource list in the hardware description */
#define REG_RESOURCE_REQUIREMENTS_LIST 10 /* Uh? Rait.. */
#define REG_QWORD 11 /* Quad word 64 bit, little endian */
#define REG_MAX 12
/* The first page of the registry file is some kind of header, lot of
* it's contents is unknown, and seems to be mostly NULLs anyway.
* Note also, that this is the only place in the registry I've been
* able to find _any_ kind of checksumming
*/
struct regf_header {
int32_t id; /* 0x00000000 D-Word ID: ASCII-"regf" = 0x66676572 */
int32_t unknown1; /* 0x00000004 D-Word ???? Mount count */
int32_t unknown2; /* 0x00000008 D-Word ???? Always the same value as at 0x00000004 */
char timestamp[8]; /* 0x0000000C Q-Word last modify date in WinNT date-format */
int32_t unknown3; /* 0x00000014 D-Word 1 */
int32_t unknown4; /* 0x00000018 D-Word 3 - probably version #. 2 in NT3.51 */
int32_t unknown5; /* 0x0000001C D-Word 0 */
int32_t unknown6; /* 0x00000020 D-Word 1 */
int32_t ofs_rootkey; /* 0x00000024 D-Word Offset of 1st key record */
int32_t filesize; /* 0x00000028 D-Word Offset of first non-used data at end of file */
int32_t unknown7; /* 0x0000002C D-Word 1 */
char name[0x1fc-0x30]; /* 0x00000030 Seems like the hive's name is buried here, max len unknown */
int32_t checksum; /* 0x000001FC D-Word Xor sum of all D-Words from 0x00000000 to 0x000001FB */
};
/* The page header, I don't know if the 14 "dummy" bytes has a meaning,
* they seem to be mostly NULLS
*/
struct hbin_page {
int32_t id; /* 0x0000 D-Word ID: ASCII-"hbin" = 0x6E696268 */
int32_t ofs_self; /* 0x0004 D-Word Offset to itself, could be for sanity check */
int32_t ofs_next; /* 0x0008 D-Word Relative offset to next hbin (practically length of this one) */
char dummy1[14]; /* 0x14 to 0x001b may be timestamp in some windows versions, at least in first hbin */
int32_t len_page; /* 0x001C D-Word Block-size??? Don't look like it. Most often zero. */
int32_t firstlink; /* 0x0020 First data block likage */
/* char data[1]; 0x0020 First data block starts here */
};
/* Minimum block size utilized at end of hbin
* Make routines accept 0 size block when at end
*/
#define HBIN_ENDFILL 0
/* Security descriptor. I know how it's linked, but don't know
how the real security data is constructed, it may as well
be like the higher level security structs defined by MS in its
includes & NT docs. Currently, I have no use for it.
Note that keys sharing the exact same security settings will
most likely point to the same security descriptor, thus
saving space and making it fast to make objects inherit settings
(is inheritance supported? they speak of security inheritance as a "new"
feature in the filesystem on NT5, even though I think it was
also supported by the lower levels in the earlier versions)
*/
struct sk_key {
short id; /* 0x0000 Word ID: ASCII-"sk" = 0x6B73 */
short dummy1; /* 0x0002 Word Unused */
int32_t ofs_prevsk; /* 0x0004 D-Word Offset of previous "sk"-Record */
int32_t ofs_nextsk; /* 0x0008 D-Word Offset of next "sk"-Record */
int32_t no_usage; /* 0x000C D-Word usage-counter */
int32_t len_sk; /* 0x0010 D-Word Size of "sk"-record in bytes */
char data[4]; /* Security data up to len_sk bytes */
};
/* This is the subkeylist/hash structure. NT4.0+.
* ID + count, then count number of offset/4byte "hash". (not true hash)
* Probably changed from the 3.x version to make it faster to
* traverse the registry if you're looking for a specific name
* (saves lookups in 'nk's that have the first 4 name chars different)
*/
struct lf_key {
short id; /* 0x0000 Word ID: ASCII-"lf" = 0x666C or "lh" = 0x686c */
short no_keys; /* 0x0002 Word number of keys */
/* 0x0004 ???? Hash-Records */
union {
struct lf_hash {
int32_t ofs_nk; /* 0x0000 D-Word Offset of corresponding "nk"-Record */
char name[4]; /* 0x0004 D-Word ASCII: the first 4 characters of the key-name, */
} hash[1];
/* WinXP uses a more real hash instead (base 37 of uppercase name chars) */
/* padded with 0's. Case sensitiv! */
struct lh_hash {
int32_t ofs_nk; /* 0x0000 D-Word Offset of corresponding "nk"-Record */
int32_t hash; /* 0x0004 D-Word ASCII: the first 4 characters of the key-name, */
} lh_hash[1];
};
};
/* 3.x version of the above, contains only offset table, NOT
* any start of names "hash". Thus needs 'nk' lookups for searches.
*/
struct li_key {
short id; /* 0x0000 Word ID: ASCII-"li" = 0x696C */
short no_keys; /* 0x0002 Word number of keys */
/* 0x0004 ???? Hash-Records */
struct li_hash {
int32_t ofs_nk; /* 0x0000 D-Word Offset of corresponding "nk"-Record */
} hash[1];
};
/* Indirect pointer list for value data, vk points to this instead of values data directly
* Seems to be used when value data is large, maybe larger than 3-4k.
*/
struct db_key {
short id; /* 0x0000 Word ID: ASCII-"li" = 0x6462 */
short no_part; /* 0x0002 Word number of data parts */
/* 0x0004 ???? Pointers to data */
int32_t ofs_data; /* 0x0000 D-Word Offset to list of data blocks */
/* Something else seems to follow here, 4 bytes at least */
/* and why not list the data block in here ???? why make another list?? */
};
/* This is a list of pointers to struct li_key, ie
* an extention record if many li's.
* This happens in NT4&5 when the lf hashlist grows larger
* than about 400-500 entries/subkeys??, then the nk_key->ofs_lf points to this
* instead of directly to an lf.
* The sub-indices this points to seems to be li (yes!) in NT4 and 2k.
* In XP and newer they point to lh which is more efficient.
* Likely to happen in HKLM\Software\classes (file extention list) and
* in SAM when many users.
*/
struct ri_key {
short id; /* 0x0000 Word ID: ASCII-"ri" = 0x6972 */
short no_lis; /* 0x0002 Word number of pointers to li */
/* 0x0004 ???? Hash-Records */
struct ri_hash {
int32_t ofs_li; /* 0x0000 D-Word Offset of corresponding "li"-Record */
} hash[1];
};
/* This is the value descriptor.
* If the sign bit (31st bit) in the length field is set, the value is
* stored inline this struct, and not in a seperate data chunk -
* the data itself is then in the ofs_data field, happens for DWORD all the time
* If the name size == 0, then the struct is probably cut short right
* after the val_type or flag.
* The flag meaning is rather unknown.
*/
struct vk_key {
/* Offset Size Contents */
short id; /* 0x0000 Word ID: ASCII-"vk" = 0x6B76 */
short len_name; /* 0x0002 Word name length */
int32_t len_data; /* 0x0004 D-Word length of the data */
int32_t ofs_data; /* 0x0008 D-Word Offset of Data */
int32_t val_type; /* 0x000C D-Word Type of value */
short flag; /* 0x0010 Word Flag
0x1 ANSI encoding */
short dummy1; /* 0x0012 Word Unused (data-trash) */
char keyname[1]; /* 0x0014 ???? Name */
};
/* This is the key node (ie directory) descriptor, can contain subkeys and/or values.
* Note that for values, the count is stored here, but for subkeys
* there's a count both here and in the offset-table (lf or li struct).
* What happens if these mismatch is not known.
* What's the classname thingy? Can't remember seeing that used in
* anything I've looked at.
*/
struct nk_key {
/* Offset Size Contents */
short id; /* 0x0000 Word ID: ASCII-"nk" = 0x6B6E */
short type; /* 0x0002 Word for the root-key: 0x2C, otherwise 0x20 */
/* 0x20 seems a flag for ANSI encoding */
/* 0x1000 is used in some places in Vista and newer */
char timestamp[12]; /* 0x0004 Q-Word write-date/time in windows nt notation */
int32_t ofs_parent; /* 0x0010 D-Word Offset of Owner/Parent key */
int32_t no_subkeys; /* 0x0014 D-Word number of sub-Keys */
int32_t dummy1;
int32_t ofs_lf; /* 0x001C D-Word Offset of the sub-key lf-Records */
int32_t dummy2;
int32_t no_values; /* 0x0024 D-Word number of values */
int32_t ofs_vallist; /* 0x0028 D-Word Offset of the Value-List */
int32_t ofs_sk; /* 0x002C D-Word Offset of the sk-Record */
int32_t ofs_classnam; /* 0x0030 D-Word Offset of the Class-Name */
int32_t dummy3; /* 0x0034 unknown some of these may be used by vista */
int32_t dummy4; /* 0x0038 unknown and newer ?? */
int32_t dummy5; /* 0x003c unknown */
int32_t dummy6; /* 0x0040 unknown */
int32_t dummy7; /* 0x0044 unknown */
short len_name; /* 0x0048 Word name-length */
short len_classnam; /* 0x004A Word class-name length */
char keyname[1]; /* 0x004C ???? key-name */
};
/*********************************************************************************/
/* Structure defines for my routines */
struct ex_data {
int nkoffs;
struct nk_key *nk;
char *name;
};
struct vex_data {
int vkoffs;
struct vk_key *vk;
int type; /* Value type REG_??? */
int size; /* Values size (normalized, inline accounted for) */
int val; /* Actual value itself if type==REG_DWORD */
char *name;
};
struct keyval {
int len; /* Length of databuffer */
int data; /* Data. Goes on for length of value */
};
struct keyvala {
int len; /* Length of databuffer */
int data[1]; /* Data. Goes on for length of value */
};
/* Types to trav_path() */
#define TPF_NK 0
#define TPF_VK 1
#define TPF_ABS 64
#define TPF_EXACT 128
#define TPF_VK_SHORT 256 /* To get type field instead of data field, used in SAM */
#define TPF_NK_EXACT (TPF_NK | TPF_EXACT)
#define TPF_VK_EXACT (TPF_VK | TPF_EXACT)
#define TPF_VK_ABS (TPF_VK | TPF_ABS) /* Name is literal, not a path */
/* Hive open modes */
#define HMODE_RW 0
#define HMODE_RO 0x1
#define HMODE_OPEN 0x2
#define HMODE_DIRTY 0x4
#define HMODE_NOALLOC 0x8 /* Don't allocate new blocks */
#define HMODE_NOEXPAND 0x10 /* Don't expand file with new hbin */
#define HMODE_DIDEXPAND 0x20 /* File has been expanded */
#define HMODE_VERBOSE 0x1000
#define HMODE_TRACE 0x2000
#define HMODE_INFO 0x4000 /* Show some info on open and close */
/* Suggested type of hive loaded, guessed by library, but not used by it */
#define HTYPE_UNKNOWN 0
#define HTYPE_SAM 1
#define HTYPE_SYSTEM 2
#define HTYPE_SECURITY 3
#define HTYPE_SOFTWARE 4
/* Hive definition, allocated by openHive(), dealloc by closeHive()
* contains state data, must be passed in all functions
*/
struct hive {
char *filename; /* Hives filename */
int filedesc; /* File descriptor (only valid if state == OPEN) */
int state; /* Current state of hive */
int type; /* Suggested type of hive. NOTE: Library will guess when
it loads it, but application may change it if needed */
int pages; /* Number of pages, total */
int useblk; /* Total # of used blocks */
int unuseblk; /* Total # of unused blocks */
int usetot; /* total # of bytes in useblk */
int unusetot; /* total # of bytes in unuseblk */
int size; /* Hives size (filesize) in bytes, incl regf header */
int rootofs; /* Offset of root-node */
int lastbin; /* Offset to last HBIN */
int endofs; /* Offset of first non HBIN page, we can expand from here */
short nkindextype; /* Subkey-indextype the root key uses */
char *buffer; /* Files raw contents */
};
/***************************************************/
/* Various nice macros */
#define CREATE(result, type, number)\
{ \
if (!((result) = (type *) calloc ((number), sizeof(type)))) { \
perror("malloc failure"); \
abort() ; \
} \
}
#define ALLOC(result, size, number)\
{ \
if (!((result) = (void *) calloc ((number), (size)))) { \
perror("malloc failure"); \
abort() ; \
} \
}
#define FREE(p) { if (p) { free(p); (p) = 0; } }
/* Debug / verbosity message macro */
#define VERB(h, string) \
{ \
if ((h)->state & HMODE_VERBOSE) printf((string)); \
}
#define VERBF(h, ...) \
{ \
if ((h)->state & HMODE_VERBOSE) printf(__VA_ARGS__); \
}
/******* Function prototypes **********/
char *str_dup( const char *str );
char *str_cat(char *str, char *add);
char *str_catf(char *str, const char *format, ... );
int fmyinput(char *prmpt, char *ibuf, int maxlen);
void hexprnt(char *s, unsigned char *bytes, int len);
void hexdump(void *hbuf, int start, int stop, int ascii);
int find_in_buf(char *buf, char *what, int sz, int len, int start);
int get_int( char *array );
void cheap_uni2ascii(char *src, char *dest, int l);
void cheap_ascii2uni(char *src, char *dest, int l);
void skipspace(char **c);
int gethex(char **c);
int gethexorstr(char **c, char *wb);
int debugit(char *buf, int sz);
struct keyval *reg_valcat(struct keyval *a, struct keyval *b);
int parse_block(struct hive *hdesc, int vofs,int verbose);
int ex_next_n(struct hive *hdesc, int nkofs, int *count, int *countri, struct ex_data *sptr);
int ex_next_v(struct hive *hdesc, int nkofs, int *count, struct vex_data *sptr);
int get_abs_path(struct hive *hdesc, int nkofs, char *path, int maxlen);
int trav_path(struct hive *hdesc, int vofs, char *path, int type);
int get_val_type(struct hive *hdesc, int vofs, char *path, int exact);
int set_val_type(struct hive *hdesc, int vofs, char *path, int exact, int type);
int get_val_len(struct hive *hdesc, int vofs, char *path, int exact);
void *get_val_data(struct hive *hdesc, int vofs, char *path, int val_type, int exact);
struct keyval *get_val2buf(struct hive *hdesc, struct keyval *kv,
int vofs, char *path, int type, int exact );
int get_dword(struct hive *hdesc, int vofs, char *path, int exact);
int put_buf2val(struct hive *hdesc, struct keyval *kv,
int vofs, char *path, int type, int exact );
int put_dword(struct hive *hdesc, int vofs, char *path, int exact, int dword);
void export_key(struct hive *hdesc, int nkofs, char *name, char *filename, char *prefix);
void closeHive(struct hive *hdesc);
int writeHive(struct hive *hdesc);
struct hive *openHive(char *filename, int mode);
void nk_ls(struct hive *hdesc, char *path, int vofs, int type);
struct vk_key *add_value(struct hive *hdesc, int nkofs, char *name, int type);
void del_allvalues(struct hive *hdesc, int nkofs);
int del_value(struct hive *hdesc, int nkofs, char *name, int quiet);
struct nk_key *add_key(struct hive *hdesc, int nkofs, char *name);
int del_key(struct hive *hdesc, int nkofs, char *name);
void rdel_keys(struct hive *hdesc, char *path, int nkofs);
struct keyval *get_class(struct hive *hdesc, int curnk, char *path);
int add_bin(struct hive *hdesc, int size);
void import_reg(struct hive *hdesc, char *filename, char *prefix);
int de_escape(char *s, int wide);
char *string_regw2prog(void *string, int len);
/* From edlib.c */
void regedit_interactive(struct hive *hive[], int no_hives);
void cat_dpi(struct hive *hdesc, int nkofs, char *path);
#endif