mirror of
https://github.com/reactos/reactos.git
synced 2024-12-29 10:35:28 +00:00
From Eugene Ingerman:
Create default registry files is they don't exist Modularized registry.c into more files, more manageable Bug fix in IoCreateFile that didn't return IoStatusBlock 'dir' now works in Bochs - corrected parsing in VfatOpenFile svn path=/trunk/; revision=1969
This commit is contained in:
parent
e180745ffa
commit
2f3eb30871
9 changed files with 3249 additions and 3101 deletions
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: create.c,v 1.25 2001/05/04 01:21:45 rex Exp $
|
/* $Id: create.c,v 1.26 2001/06/14 21:05:08 jfilby Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT: ReactOS kernel
|
* PROJECT: ReactOS kernel
|
||||||
|
@ -599,6 +599,7 @@ VfatOpenFile (PDEVICE_EXTENSION DeviceExt, PFILE_OBJECT FileObject,
|
||||||
ParentFcb = Temp;
|
ParentFcb = Temp;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if( *current != L'\0' ){ //the file name is directory. there will be no last part.
|
||||||
/* searching for last path component */
|
/* searching for last path component */
|
||||||
DPRINT ("search for (%S) in (%S)\n", current, Fcb ? Fcb->PathName : L"");
|
DPRINT ("search for (%S) in (%S)\n", current, Fcb ? Fcb->PathName : L"");
|
||||||
Status = FindFile (DeviceExt, Fcb, ParentFcb, current, NULL, NULL);
|
Status = FindFile (DeviceExt, Fcb, ParentFcb, current, NULL, NULL);
|
||||||
|
@ -621,6 +622,7 @@ VfatOpenFile (PDEVICE_EXTENSION DeviceExt, PFILE_OBJECT FileObject,
|
||||||
Fcb = ParentFcb;
|
Fcb = ParentFcb;
|
||||||
ParentFcb = Temp;
|
ParentFcb = Temp;
|
||||||
ParentFcb->ObjectName = &(wcschr (ParentFcb->ObjectName, '\\'))[1];
|
ParentFcb->ObjectName = &(wcschr (ParentFcb->ObjectName, '\\'))[1];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
FileObject->Flags = FileObject->Flags |
|
FileObject->Flags = FileObject->Flags |
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
# $Id: Makefile,v 1.43 2001/06/12 17:50:27 chorns Exp $
|
# $Id: Makefile,v 1.44 2001/06/14 21:05:07 jfilby Exp $
|
||||||
#
|
#
|
||||||
# ReactOS Operating System
|
# ReactOS Operating System
|
||||||
#
|
#
|
||||||
|
@ -268,7 +268,11 @@ OBJECTS_SE = \
|
||||||
|
|
||||||
# Configuration Manager (Registry)
|
# Configuration Manager (Registry)
|
||||||
OBJECTS_CM = \
|
OBJECTS_CM = \
|
||||||
cm/registry.o
|
cm/registry.o \
|
||||||
|
cm/ntfunc.o \
|
||||||
|
cm/rtlfunc.o \
|
||||||
|
cm/regfile.o \
|
||||||
|
cm/regobj.o
|
||||||
|
|
||||||
# Debugger Support (Dbg)
|
# Debugger Support (Dbg)
|
||||||
OBJECTS_DBG = \
|
OBJECTS_DBG = \
|
||||||
|
|
261
reactos/ntoskrnl/cm/cm.h
Normal file
261
reactos/ntoskrnl/cm/cm.h
Normal file
|
@ -0,0 +1,261 @@
|
||||||
|
|
||||||
|
#ifndef __INCLUDE_CM_H
|
||||||
|
#define __INCLUDE_CM_H
|
||||||
|
|
||||||
|
#include <windows.h>
|
||||||
|
#include <ddk/ntddk.h>
|
||||||
|
#include <internal/config.h>
|
||||||
|
#include <internal/ob.h>
|
||||||
|
#include <limits.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <internal/pool.h>
|
||||||
|
#include <internal/registry.h>
|
||||||
|
|
||||||
|
#define NDEBUG
|
||||||
|
#include <internal/debug.h>
|
||||||
|
|
||||||
|
#define KO_MARKED_FOR_DELETE 0x00000001
|
||||||
|
|
||||||
|
// BLOCK_OFFSET = offset in file after header block
|
||||||
|
typedef DWORD BLOCK_OFFSET;
|
||||||
|
|
||||||
|
/* header for registry hive file : */
|
||||||
|
typedef struct _HEADER_BLOCK
|
||||||
|
{
|
||||||
|
ULONG BlockId; /* ="regf" */
|
||||||
|
ULONG Version; /* file version ?*/
|
||||||
|
ULONG VersionOld; /* file version ?*/
|
||||||
|
FILETIME DateModified; /* please don't replace with LARGE_INTEGER !*/
|
||||||
|
ULONG Unused3; /* registry format version ? */
|
||||||
|
ULONG Unused4; /* registry format version ? */
|
||||||
|
ULONG Unused5; /* registry format version ? */
|
||||||
|
ULONG Unused6; /* registry format version ? */
|
||||||
|
BLOCK_OFFSET RootKeyBlock;
|
||||||
|
ULONG BlockSize;
|
||||||
|
ULONG Unused7;
|
||||||
|
WCHAR FileName[64]; /* end of file name */
|
||||||
|
ULONG Unused8[83];
|
||||||
|
ULONG Checksum;
|
||||||
|
} HEADER_BLOCK, *PHEADER_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _HEAP_BLOCK
|
||||||
|
{
|
||||||
|
ULONG BlockId; /* = "hbin" */
|
||||||
|
BLOCK_OFFSET BlockOffset; /* block offset of this heap */
|
||||||
|
ULONG BlockSize; /* size in bytes, 4k multiple */
|
||||||
|
ULONG Unused1;
|
||||||
|
FILETIME DateModified; /* please don't replace with LARGE_INTEGER !*/
|
||||||
|
ULONG Unused2;
|
||||||
|
} HEAP_BLOCK, *PHEAP_BLOCK;
|
||||||
|
|
||||||
|
// each sub_block begin with this struct :
|
||||||
|
// in a free subblock, higher bit of SubBlockSize is set
|
||||||
|
typedef struct _FREE_SUB_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;/* <0 if used, >0 if free */
|
||||||
|
} FREE_SUB_BLOCK, *PFREE_SUB_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _KEY_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;
|
||||||
|
USHORT SubBlockId;
|
||||||
|
USHORT Type;
|
||||||
|
FILETIME LastWriteTime; /* please don't replace with LARGE_INTEGER !*/
|
||||||
|
ULONG UnUsed1;
|
||||||
|
BLOCK_OFFSET ParentKeyOffset;
|
||||||
|
ULONG NumberOfSubKeys;
|
||||||
|
ULONG UnUsed2;
|
||||||
|
BLOCK_OFFSET HashTableOffset;
|
||||||
|
ULONG UnUsed3;
|
||||||
|
ULONG NumberOfValues;
|
||||||
|
BLOCK_OFFSET ValuesOffset;
|
||||||
|
BLOCK_OFFSET SecurityKeyOffset;
|
||||||
|
BLOCK_OFFSET ClassNameOffset;
|
||||||
|
ULONG Unused4[5];
|
||||||
|
USHORT NameSize;
|
||||||
|
USHORT ClassSize; /* size of ClassName in bytes */
|
||||||
|
UCHAR Name[0]; /* warning : not zero terminated */
|
||||||
|
} KEY_BLOCK, *PKEY_BLOCK;
|
||||||
|
|
||||||
|
// hash record :
|
||||||
|
// HashValue=four letters of value's name
|
||||||
|
typedef struct _HASH_RECORD
|
||||||
|
{
|
||||||
|
BLOCK_OFFSET KeyOffset;
|
||||||
|
ULONG HashValue;
|
||||||
|
} HASH_RECORD, *PHASH_RECORD;
|
||||||
|
|
||||||
|
typedef struct _HASH_TABLE_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;
|
||||||
|
USHORT SubBlockId;
|
||||||
|
USHORT HashTableSize;
|
||||||
|
HASH_RECORD Table[0];
|
||||||
|
} HASH_TABLE_BLOCK, *PHASH_TABLE_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _VALUE_LIST_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;
|
||||||
|
BLOCK_OFFSET Values[0];
|
||||||
|
} VALUE_LIST_BLOCK, *PVALUE_LIST_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _VALUE_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;
|
||||||
|
USHORT SubBlockId; // "kv"
|
||||||
|
USHORT NameSize; // length of Name
|
||||||
|
LONG DataSize; // length of datas in the subblock pointed by DataOffset
|
||||||
|
BLOCK_OFFSET DataOffset;// datas are here if high bit of DataSize is set
|
||||||
|
ULONG DataType;
|
||||||
|
USHORT Flags;
|
||||||
|
USHORT Unused1;
|
||||||
|
UCHAR Name[0]; /* warning : not zero terminated */
|
||||||
|
} VALUE_BLOCK, *PVALUE_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _DATA_BLOCK
|
||||||
|
{
|
||||||
|
LONG SubBlockSize;
|
||||||
|
UCHAR Data[0];
|
||||||
|
} DATA_BLOCK, *PDATA_BLOCK;
|
||||||
|
|
||||||
|
typedef struct _REGISTRY_FILE
|
||||||
|
{
|
||||||
|
PWSTR Filename;
|
||||||
|
ULONG FileSize;
|
||||||
|
PFILE_OBJECT FileObject;
|
||||||
|
PHEADER_BLOCK HeaderBlock;
|
||||||
|
// ULONG NumberOfBlocks;
|
||||||
|
ULONG BlockListSize;
|
||||||
|
PHEAP_BLOCK *BlockList;
|
||||||
|
ULONG FreeListSize;
|
||||||
|
ULONG FreeListMax;
|
||||||
|
PFREE_SUB_BLOCK *FreeList;
|
||||||
|
BLOCK_OFFSET *FreeListOffset;
|
||||||
|
// KSPIN_LOCK RegLock;
|
||||||
|
KSEMAPHORE RegSem;
|
||||||
|
|
||||||
|
|
||||||
|
// NTSTATUS (*Extend)(ULONG NewSize);
|
||||||
|
// PVOID (*Flush)(VOID);
|
||||||
|
} REGISTRY_FILE, *PREGISTRY_FILE;
|
||||||
|
|
||||||
|
/* Type defining the Object Manager Key Object */
|
||||||
|
typedef struct _KEY_OBJECT
|
||||||
|
{
|
||||||
|
CSHORT Type;
|
||||||
|
CSHORT Size;
|
||||||
|
|
||||||
|
ULONG Flags;
|
||||||
|
USHORT NameSize; // length of Name
|
||||||
|
UCHAR *Name;
|
||||||
|
PREGISTRY_FILE RegistryFile;
|
||||||
|
BLOCK_OFFSET BlockOffset;
|
||||||
|
PKEY_BLOCK KeyBlock;
|
||||||
|
struct _KEY_OBJECT *ParentKey;
|
||||||
|
ULONG NumberOfSubKeys; /* subkeys loaded in SubKeys */
|
||||||
|
ULONG SizeOfSubKeys; /* space allocated in SubKeys */
|
||||||
|
struct _KEY_OBJECT **SubKeys; /* list of subkeys loaded */
|
||||||
|
} KEY_OBJECT, *PKEY_OBJECT;
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS CmiObjectParse(PVOID ParsedObject,
|
||||||
|
PVOID *NextObject,
|
||||||
|
PUNICODE_STRING FullPath,
|
||||||
|
PWSTR *Path,
|
||||||
|
POBJECT_TYPE ObjectType,
|
||||||
|
ULONG Attribute);
|
||||||
|
|
||||||
|
NTSTATUS CmiObjectCreate(PVOID ObjectBody,
|
||||||
|
PVOID Parent,
|
||||||
|
PWSTR RemainingPath,
|
||||||
|
struct _OBJECT_ATTRIBUTES* ObjectAttributes);
|
||||||
|
void CmiObjectDelete(PVOID DeletedObject);
|
||||||
|
|
||||||
|
VOID CmiAddKeyToList(PKEY_OBJECT ParentKey,PKEY_OBJECT NewKey);
|
||||||
|
NTSTATUS CmiRemoveKeyFromList(PKEY_OBJECT NewKey);
|
||||||
|
PKEY_OBJECT CmiScanKeyList(PKEY_OBJECT Parent,
|
||||||
|
PCHAR KeyNameBuf,
|
||||||
|
ULONG Attributes);
|
||||||
|
|
||||||
|
PREGISTRY_FILE CmiCreateRegistry(PWSTR Filename);
|
||||||
|
|
||||||
|
ULONG CmiGetMaxNameLength(PREGISTRY_FILE RegistryFile,
|
||||||
|
PKEY_BLOCK KeyBlock);
|
||||||
|
ULONG CmiGetMaxClassLength(PREGISTRY_FILE RegistryFile,
|
||||||
|
PKEY_BLOCK KeyBlock);
|
||||||
|
ULONG CmiGetMaxValueNameLength(PREGISTRY_FILE RegistryFile,
|
||||||
|
PKEY_BLOCK KeyBlock);
|
||||||
|
ULONG CmiGetMaxValueDataLength(PREGISTRY_FILE RegistryFile,
|
||||||
|
PKEY_BLOCK KeyBlock);
|
||||||
|
|
||||||
|
NTSTATUS CmiScanForSubKey(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_BLOCK KeyBlock,
|
||||||
|
OUT PKEY_BLOCK *SubKeyBlock,
|
||||||
|
OUT BLOCK_OFFSET *BlockOffset,
|
||||||
|
IN PCHAR KeyName,
|
||||||
|
IN ACCESS_MASK DesiredAccess,
|
||||||
|
IN ULONG Attributes);
|
||||||
|
NTSTATUS CmiAddSubKey(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_OBJECT Parent,
|
||||||
|
OUT PKEY_OBJECT SubKey,
|
||||||
|
IN PWSTR NewSubKeyName,
|
||||||
|
IN USHORT NewSubKeyNameSize,
|
||||||
|
IN ULONG TitleIndex,
|
||||||
|
IN PUNICODE_STRING Class,
|
||||||
|
IN ULONG CreateOptions);
|
||||||
|
|
||||||
|
NTSTATUS CmiScanKeyForValue(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_BLOCK KeyBlock,
|
||||||
|
IN PCHAR ValueName,
|
||||||
|
OUT PVALUE_BLOCK *ValueBlock,
|
||||||
|
OUT BLOCK_OFFSET *VBOffset);
|
||||||
|
NTSTATUS CmiGetValueFromKeyByIndex(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_BLOCK KeyBlock,
|
||||||
|
IN ULONG Index,
|
||||||
|
OUT PVALUE_BLOCK *ValueBlock);
|
||||||
|
NTSTATUS CmiAddValueToKey(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_BLOCK KeyBlock,
|
||||||
|
IN PCHAR ValueNameBuf,
|
||||||
|
OUT PVALUE_BLOCK *pValueBlock,
|
||||||
|
OUT BLOCK_OFFSET *pVBOffset);
|
||||||
|
NTSTATUS CmiDeleteValueFromKey(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
IN PKEY_BLOCK KeyBlock,
|
||||||
|
IN PCHAR ValueName);
|
||||||
|
|
||||||
|
NTSTATUS CmiAllocateHashTableBlock(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
OUT PHASH_TABLE_BLOCK *HashBlock,
|
||||||
|
OUT BLOCK_OFFSET *HBOffset,
|
||||||
|
IN ULONG HashTableSize);
|
||||||
|
PKEY_BLOCK CmiGetKeyFromHashByIndex(PREGISTRY_FILE RegistryFile,
|
||||||
|
PHASH_TABLE_BLOCK HashBlock,
|
||||||
|
ULONG Index);
|
||||||
|
NTSTATUS CmiAddKeyToHashTable(PREGISTRY_FILE RegistryFile,
|
||||||
|
PHASH_TABLE_BLOCK HashBlock,
|
||||||
|
PKEY_BLOCK NewKeyBlock,
|
||||||
|
BLOCK_OFFSET NKBOffset);
|
||||||
|
|
||||||
|
NTSTATUS CmiAllocateValueBlock(IN PREGISTRY_FILE RegistryFile,
|
||||||
|
OUT PVALUE_BLOCK *ValueBlock,
|
||||||
|
OUT BLOCK_OFFSET *VBOffset,
|
||||||
|
IN PCHAR ValueNameBuf);
|
||||||
|
NTSTATUS CmiDestroyValueBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
PVALUE_BLOCK ValueBlock, BLOCK_OFFSET VBOffset);
|
||||||
|
|
||||||
|
NTSTATUS CmiAllocateBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
PVOID *Block,
|
||||||
|
LONG BlockSize,
|
||||||
|
BLOCK_OFFSET * pBlockOffset);
|
||||||
|
NTSTATUS CmiDestroyBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
PVOID Block,BLOCK_OFFSET Offset);
|
||||||
|
PVOID CmiGetBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
BLOCK_OFFSET BlockOffset,
|
||||||
|
OUT PHEAP_BLOCK * ppHeap);
|
||||||
|
VOID CmiLockBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
PVOID Block);
|
||||||
|
VOID CmiReleaseBlock(PREGISTRY_FILE RegistryFile,
|
||||||
|
PVOID Block);
|
||||||
|
NTSTATUS
|
||||||
|
CmiAddFree(PREGISTRY_FILE RegistryFile,
|
||||||
|
PFREE_SUB_BLOCK FreeBlock,BLOCK_OFFSET FreeOffset);
|
||||||
|
|
||||||
|
#endif /*__INCLUDE_CM_H*/
|
1283
reactos/ntoskrnl/cm/ntfunc.c
Normal file
1283
reactos/ntoskrnl/cm/ntfunc.c
Normal file
File diff suppressed because it is too large
Load diff
1200
reactos/ntoskrnl/cm/regfile.c
Normal file
1200
reactos/ntoskrnl/cm/regfile.c
Normal file
File diff suppressed because it is too large
Load diff
File diff suppressed because it is too large
Load diff
253
reactos/ntoskrnl/cm/regobj.c
Normal file
253
reactos/ntoskrnl/cm/regobj.c
Normal file
|
@ -0,0 +1,253 @@
|
||||||
|
/*
|
||||||
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
|
* PROJECT: ReactOS kernel
|
||||||
|
* FILE: ntoskrnl/cm/regobj.c
|
||||||
|
* PURPOSE: Registry object manipulation routines.
|
||||||
|
* UPDATE HISTORY:
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "cm.h"
|
||||||
|
|
||||||
|
extern POBJECT_TYPE CmiKeyType;
|
||||||
|
extern KSPIN_LOCK CmiKeyListLock;
|
||||||
|
|
||||||
|
NTSTATUS CmiObjectParse(PVOID ParsedObject,
|
||||||
|
PVOID *NextObject,
|
||||||
|
PUNICODE_STRING FullPath,
|
||||||
|
PWSTR *Path,
|
||||||
|
POBJECT_TYPE ObjectType,
|
||||||
|
ULONG Attributes)
|
||||||
|
{
|
||||||
|
CHAR cPath[MAX_PATH];
|
||||||
|
PWSTR end;
|
||||||
|
PKEY_OBJECT FoundObject;
|
||||||
|
PKEY_OBJECT ParsedKey=ParsedObject;
|
||||||
|
PKEY_BLOCK SubKeyBlock;
|
||||||
|
BLOCK_OFFSET BlockOffset;
|
||||||
|
NTSTATUS Status;
|
||||||
|
*NextObject = NULL;
|
||||||
|
if ((*Path) == NULL)
|
||||||
|
{
|
||||||
|
return STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
if((*Path[0])=='\\')
|
||||||
|
{
|
||||||
|
end = wcschr((*Path)+1, '\\');
|
||||||
|
if (end != NULL)
|
||||||
|
*end = 0;
|
||||||
|
wcstombs(cPath,(*Path)+1,wcslen((*Path)+1));
|
||||||
|
cPath[wcslen( (*Path)+1)]=0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
end = wcschr((*Path), '\\');
|
||||||
|
if (end != NULL)
|
||||||
|
*end = 0;
|
||||||
|
wcstombs(cPath,(*Path),wcslen((*Path)));
|
||||||
|
cPath[wcslen( (*Path))]=0;
|
||||||
|
}
|
||||||
|
|
||||||
|
FoundObject = CmiScanKeyList(ParsedKey,cPath,Attributes);
|
||||||
|
if (FoundObject == NULL)
|
||||||
|
{
|
||||||
|
Status = CmiScanForSubKey(ParsedKey->RegistryFile,
|
||||||
|
ParsedKey->KeyBlock,
|
||||||
|
&SubKeyBlock,
|
||||||
|
&BlockOffset,
|
||||||
|
cPath,
|
||||||
|
0,
|
||||||
|
Attributes);
|
||||||
|
if(!NT_SUCCESS(Status) || SubKeyBlock == NULL)
|
||||||
|
{
|
||||||
|
if (end != NULL)
|
||||||
|
{
|
||||||
|
*end = '\\';
|
||||||
|
}
|
||||||
|
return STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
/* Create new key object and put into linked list */
|
||||||
|
DPRINT("CmiObjectParse %s\n",cPath);
|
||||||
|
FoundObject = ObCreateObject(NULL,
|
||||||
|
STANDARD_RIGHTS_REQUIRED,
|
||||||
|
NULL,
|
||||||
|
CmiKeyType);
|
||||||
|
if (FoundObject == NULL)
|
||||||
|
{
|
||||||
|
//FIXME : return the good error code
|
||||||
|
return STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
FoundObject->Flags = 0;
|
||||||
|
FoundObject->Name = SubKeyBlock->Name;
|
||||||
|
FoundObject->NameSize = SubKeyBlock->NameSize;
|
||||||
|
FoundObject->KeyBlock = SubKeyBlock;
|
||||||
|
FoundObject->BlockOffset = BlockOffset;
|
||||||
|
FoundObject->RegistryFile = ParsedKey->RegistryFile;
|
||||||
|
CmiAddKeyToList(ParsedKey,FoundObject);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
ObReferenceObjectByPointer(FoundObject,
|
||||||
|
STANDARD_RIGHTS_REQUIRED,
|
||||||
|
NULL,
|
||||||
|
UserMode);
|
||||||
|
DPRINT("CmiObjectParse %s\n",FoundObject->Name);
|
||||||
|
if (end != NULL)
|
||||||
|
{
|
||||||
|
*end = '\\';
|
||||||
|
*Path = end;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
*Path = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
*NextObject = FoundObject;
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
NTSTATUS CmiObjectCreate(PVOID ObjectBody,
|
||||||
|
PVOID Parent,
|
||||||
|
PWSTR RemainingPath,
|
||||||
|
struct _OBJECT_ATTRIBUTES* ObjectAttributes)
|
||||||
|
{
|
||||||
|
PKEY_OBJECT pKey=ObjectBody;
|
||||||
|
pKey->ParentKey = Parent;
|
||||||
|
if (RemainingPath)
|
||||||
|
{
|
||||||
|
if(RemainingPath[0]== L'\\')
|
||||||
|
{
|
||||||
|
pKey->Name = (PCHAR) (&RemainingPath[1]);
|
||||||
|
pKey->NameSize = wcslen(RemainingPath)-1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pKey->Name = (PCHAR) RemainingPath;
|
||||||
|
pKey->NameSize = wcslen(RemainingPath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
pKey->NameSize = 0;
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
CmiObjectDelete(PVOID DeletedObject)
|
||||||
|
{
|
||||||
|
PKEY_OBJECT KeyObject;
|
||||||
|
|
||||||
|
DPRINT("delete object key\n");
|
||||||
|
KeyObject = (PKEY_OBJECT) DeletedObject;
|
||||||
|
if(!NT_SUCCESS(CmiRemoveKeyFromList(KeyObject)))
|
||||||
|
{
|
||||||
|
DPRINT1("Key not found in parent list ???\n");
|
||||||
|
}
|
||||||
|
if (KeyObject->Flags & KO_MARKED_FOR_DELETE)
|
||||||
|
{
|
||||||
|
DPRINT1("delete really key\n");
|
||||||
|
CmiDestroyBlock(KeyObject->RegistryFile,
|
||||||
|
KeyObject->KeyBlock,
|
||||||
|
KeyObject->BlockOffset);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CmiReleaseBlock(KeyObject->RegistryFile,
|
||||||
|
KeyObject->KeyBlock);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
CmiAddKeyToList(PKEY_OBJECT ParentKey,PKEY_OBJECT NewKey)
|
||||||
|
{
|
||||||
|
KIRQL OldIrql;
|
||||||
|
|
||||||
|
KeAcquireSpinLock(&CmiKeyListLock, &OldIrql);
|
||||||
|
if (ParentKey->SizeOfSubKeys <= ParentKey->NumberOfSubKeys)
|
||||||
|
{
|
||||||
|
PKEY_OBJECT *tmpSubKeys = ExAllocatePool(PagedPool
|
||||||
|
, (ParentKey->NumberOfSubKeys+1) * sizeof(DWORD));
|
||||||
|
if(ParentKey->NumberOfSubKeys > 0)
|
||||||
|
memcpy(tmpSubKeys,ParentKey->SubKeys
|
||||||
|
,ParentKey->NumberOfSubKeys*sizeof(DWORD));
|
||||||
|
if(ParentKey->SubKeys) ExFreePool(ParentKey->SubKeys);
|
||||||
|
ParentKey->SubKeys=tmpSubKeys;
|
||||||
|
ParentKey->SizeOfSubKeys = ParentKey->NumberOfSubKeys+1;
|
||||||
|
}
|
||||||
|
/* FIXME : please maintain the list in alphabetic order */
|
||||||
|
/* to allow a dichotomic search */
|
||||||
|
ParentKey->SubKeys[ParentKey->NumberOfSubKeys++] = NewKey;
|
||||||
|
ObReferenceObjectByPointer(ParentKey,
|
||||||
|
STANDARD_RIGHTS_REQUIRED,
|
||||||
|
NULL,
|
||||||
|
UserMode);
|
||||||
|
NewKey->ParentKey = ParentKey;
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
}
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
CmiRemoveKeyFromList(PKEY_OBJECT KeyToRemove)
|
||||||
|
{
|
||||||
|
KIRQL OldIrql;
|
||||||
|
PKEY_OBJECT ParentKey;
|
||||||
|
DWORD Index;
|
||||||
|
|
||||||
|
ParentKey=KeyToRemove->ParentKey;
|
||||||
|
KeAcquireSpinLock(&CmiKeyListLock, &OldIrql);
|
||||||
|
/* FIXME : if list maintained in alphabetic order, use dichotomic search */
|
||||||
|
for (Index=0; Index < ParentKey->NumberOfSubKeys; Index++)
|
||||||
|
{
|
||||||
|
if(ParentKey->SubKeys[Index] == KeyToRemove)
|
||||||
|
{
|
||||||
|
if (Index < ParentKey->NumberOfSubKeys-1)
|
||||||
|
memmove(&ParentKey->SubKeys[Index]
|
||||||
|
,&ParentKey->SubKeys[Index+1]
|
||||||
|
,(ParentKey->NumberOfSubKeys-Index-1)*sizeof(PKEY_OBJECT));
|
||||||
|
ParentKey->NumberOfSubKeys--;
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
ObDereferenceObject(ParentKey);
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
return STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
PKEY_OBJECT
|
||||||
|
CmiScanKeyList(PKEY_OBJECT Parent,
|
||||||
|
PCHAR KeyName,
|
||||||
|
ULONG Attributes)
|
||||||
|
{
|
||||||
|
KIRQL OldIrql;
|
||||||
|
PKEY_OBJECT CurKey;
|
||||||
|
DWORD Index;
|
||||||
|
WORD NameSize;
|
||||||
|
NameSize=strlen(KeyName);
|
||||||
|
KeAcquireSpinLock(&CmiKeyListLock, &OldIrql);
|
||||||
|
/* FIXME : if list maintained in alphabetic order, use dichotomic search */
|
||||||
|
for (Index=0; Index < Parent->NumberOfSubKeys; Index++)
|
||||||
|
{
|
||||||
|
CurKey=Parent->SubKeys[Index];
|
||||||
|
if (Attributes & OBJ_CASE_INSENSITIVE)
|
||||||
|
{
|
||||||
|
if( NameSize == CurKey->NameSize
|
||||||
|
&& !_strnicmp(KeyName,CurKey->Name,NameSize))
|
||||||
|
{
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
return CurKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if( NameSize == CurKey->NameSize
|
||||||
|
&& !strncmp(KeyName,CurKey->Name,NameSize))
|
||||||
|
{
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
return CurKey;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
KeReleaseSpinLock(&CmiKeyListLock, OldIrql);
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
233
reactos/ntoskrnl/cm/rtlfunc.c
Normal file
233
reactos/ntoskrnl/cm/rtlfunc.c
Normal file
|
@ -0,0 +1,233 @@
|
||||||
|
/*
|
||||||
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
|
* PROJECT: ReactOS kernel
|
||||||
|
* FILE: ntoskrnl/cm/rtlfunc.c
|
||||||
|
* PURPOSE: Rtlxxx function for registry access
|
||||||
|
* UPDATE HISTORY:
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "cm.h"
|
||||||
|
|
||||||
|
NTSTATUS STDCALL
|
||||||
|
RtlCheckRegistryKey(IN ULONG RelativeTo,
|
||||||
|
IN PWSTR Path)
|
||||||
|
{
|
||||||
|
HANDLE KeyHandle;
|
||||||
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
Status = RtlpGetRegistryHandle(RelativeTo,
|
||||||
|
Path,
|
||||||
|
FALSE,
|
||||||
|
&KeyHandle);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
NtClose(KeyHandle);
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
STDCALL
|
||||||
|
RtlCreateRegistryKey (
|
||||||
|
IN ULONG RelativeTo,
|
||||||
|
IN PWSTR Path
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HANDLE KeyHandle;
|
||||||
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
Status = RtlpGetRegistryHandle(RelativeTo,
|
||||||
|
Path,
|
||||||
|
TRUE,
|
||||||
|
&KeyHandle);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
NtClose(KeyHandle);
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
STDCALL
|
||||||
|
RtlDeleteRegistryValue (
|
||||||
|
IN ULONG RelativeTo,
|
||||||
|
IN PWSTR Path,
|
||||||
|
IN PWSTR ValueName
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HANDLE KeyHandle;
|
||||||
|
NTSTATUS Status;
|
||||||
|
UNICODE_STRING Name;
|
||||||
|
|
||||||
|
Status = RtlpGetRegistryHandle(RelativeTo,
|
||||||
|
Path,
|
||||||
|
TRUE,
|
||||||
|
&KeyHandle);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
RtlInitUnicodeString(&Name,
|
||||||
|
ValueName);
|
||||||
|
|
||||||
|
NtDeleteValueKey(KeyHandle,
|
||||||
|
&Name);
|
||||||
|
|
||||||
|
NtClose(KeyHandle);
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
STDCALL
|
||||||
|
RtlQueryRegistryValues (
|
||||||
|
IN ULONG RelativeTo,
|
||||||
|
IN PWSTR Path,
|
||||||
|
IN PRTL_QUERY_REGISTRY_TABLE QueryTable,
|
||||||
|
IN PVOID Context,
|
||||||
|
IN PVOID Environment
|
||||||
|
)
|
||||||
|
{
|
||||||
|
UNIMPLEMENTED;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
STDCALL
|
||||||
|
RtlWriteRegistryValue (
|
||||||
|
IN ULONG RelativeTo,
|
||||||
|
IN PWSTR Path,
|
||||||
|
IN PWSTR ValueName,
|
||||||
|
IN ULONG ValueType,
|
||||||
|
IN PVOID ValueData,
|
||||||
|
IN ULONG ValueLength
|
||||||
|
)
|
||||||
|
{
|
||||||
|
HANDLE KeyHandle;
|
||||||
|
NTSTATUS Status;
|
||||||
|
UNICODE_STRING Name;
|
||||||
|
|
||||||
|
Status = RtlpGetRegistryHandle(RelativeTo,
|
||||||
|
Path,
|
||||||
|
TRUE,
|
||||||
|
&KeyHandle);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
return Status;
|
||||||
|
|
||||||
|
RtlInitUnicodeString(&Name,
|
||||||
|
ValueName);
|
||||||
|
|
||||||
|
NtSetValueKey(KeyHandle,
|
||||||
|
&Name,
|
||||||
|
0,
|
||||||
|
ValueType,
|
||||||
|
ValueData,
|
||||||
|
ValueLength);
|
||||||
|
|
||||||
|
NtClose(KeyHandle);
|
||||||
|
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
NTSTATUS STDCALL
|
||||||
|
RtlFormatCurrentUserKeyPath(IN OUT PUNICODE_STRING KeyPath)
|
||||||
|
{
|
||||||
|
return STATUS_UNSUCCESSFUL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* ------------------------------------------ Private Implementation */
|
||||||
|
|
||||||
|
|
||||||
|
NTSTATUS
|
||||||
|
RtlpGetRegistryHandle(ULONG RelativeTo,
|
||||||
|
PWSTR Path,
|
||||||
|
BOOLEAN Create,
|
||||||
|
PHANDLE KeyHandle)
|
||||||
|
{
|
||||||
|
UNICODE_STRING KeyName;
|
||||||
|
WCHAR KeyBuffer[MAX_PATH];
|
||||||
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
||||||
|
NTSTATUS Status;
|
||||||
|
|
||||||
|
if (RelativeTo & RTL_REGISTRY_HANDLE)
|
||||||
|
{
|
||||||
|
*KeyHandle = (HANDLE)Path;
|
||||||
|
return STATUS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (RelativeTo & RTL_REGISTRY_OPTIONAL)
|
||||||
|
RelativeTo &= ~RTL_REGISTRY_OPTIONAL;
|
||||||
|
|
||||||
|
if (RelativeTo >= RTL_REGISTRY_MAXIMUM)
|
||||||
|
return STATUS_INVALID_PARAMETER;
|
||||||
|
|
||||||
|
KeyName.Length = 0;
|
||||||
|
KeyName.MaximumLength = MAX_PATH;
|
||||||
|
KeyName.Buffer = KeyBuffer;
|
||||||
|
KeyBuffer[0] = 0;
|
||||||
|
|
||||||
|
switch (RelativeTo)
|
||||||
|
{
|
||||||
|
case RTL_REGISTRY_SERVICES:
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
L"\\Registry\\Machine\\System\\CurrentControlSet\\Services\\");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case RTL_REGISTRY_CONTROL:
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
L"\\Registry\\Machine\\System\\CurrentControlSet\\Control\\");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case RTL_REGISTRY_WINDOWS_NT:
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
L"\\Registry\\Machine\\Software\\Microsoft\\Windows NT\\CurrentVersion\\");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case RTL_REGISTRY_DEVICEMAP:
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
L"\\Registry\\Machine\\Hardware\\DeviceMap\\");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case RTL_REGISTRY_USER:
|
||||||
|
Status = RtlFormatCurrentUserKeyPath(&KeyName);
|
||||||
|
if (!NT_SUCCESS(Status))
|
||||||
|
return Status;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Path[0] != L'\\')
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
L"\\");
|
||||||
|
|
||||||
|
RtlAppendUnicodeToString(&KeyName,
|
||||||
|
Path);
|
||||||
|
|
||||||
|
InitializeObjectAttributes(&ObjectAttributes,
|
||||||
|
&KeyName,
|
||||||
|
OBJ_CASE_INSENSITIVE,
|
||||||
|
NULL,
|
||||||
|
NULL);
|
||||||
|
|
||||||
|
if (Create == TRUE)
|
||||||
|
{
|
||||||
|
Status = NtCreateKey(KeyHandle,
|
||||||
|
KEY_ALL_ACCESS,
|
||||||
|
&ObjectAttributes,
|
||||||
|
0,
|
||||||
|
NULL,
|
||||||
|
0,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Status = NtOpenKey(KeyHandle,
|
||||||
|
KEY_ALL_ACCESS,
|
||||||
|
&ObjectAttributes);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Status;
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
/* $Id: create.c,v 1.43 2001/06/04 11:26:11 chorns Exp $
|
/* $Id: create.c,v 1.44 2001/06/14 21:05:07 jfilby Exp $
|
||||||
*
|
*
|
||||||
* COPYRIGHT: See COPYING in the top level directory
|
* COPYRIGHT: See COPYING in the top level directory
|
||||||
* PROJECT: ReactOS kernel
|
* PROJECT: ReactOS kernel
|
||||||
|
@ -357,6 +357,7 @@ IoCreateFile(
|
||||||
return (STATUS_UNSUCCESSFUL);
|
return (STATUS_UNSUCCESSFUL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Irp->UserIosb = IoStatusBlock; //return iostatus
|
||||||
Irp->AssociatedIrp.SystemBuffer = EaBuffer;
|
Irp->AssociatedIrp.SystemBuffer = EaBuffer;
|
||||||
Irp->Tail.Overlay.AuxiliaryBuffer = (PCHAR)ExtraCreateParameters;
|
Irp->Tail.Overlay.AuxiliaryBuffer = (PCHAR)ExtraCreateParameters;
|
||||||
|
|
||||||
|
@ -395,6 +396,7 @@ IoCreateFile(
|
||||||
* immediately.
|
* immediately.
|
||||||
*/
|
*/
|
||||||
Status = IofCallDriver(FileObject->DeviceObject, Irp );
|
Status = IofCallDriver(FileObject->DeviceObject, Irp );
|
||||||
|
|
||||||
if (Status == STATUS_PENDING)
|
if (Status == STATUS_PENDING)
|
||||||
{
|
{
|
||||||
KeWaitForSingleObject(&Event,
|
KeWaitForSingleObject(&Event,
|
||||||
|
|
Loading…
Reference in a new issue