489 lines
10 KiB
C
489 lines
10 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1992 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
exatom.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
This file contains functions for manipulating global atom tables
|
|||
|
stored in kernel space.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Steve Wood (stevewo) 13-Dec-1995
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "exp.h"
|
|||
|
#pragma hdrstop
|
|||
|
|
|||
|
//
|
|||
|
// Local Procedure prototype
|
|||
|
//
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpGetGlobalAtomTable (
|
|||
|
);
|
|||
|
|
|||
|
#if defined(ALLOC_PRAGMA)
|
|||
|
#pragma alloc_text(PAGE, NtAddAtom)
|
|||
|
#pragma alloc_text(PAGE, NtFindAtom)
|
|||
|
#pragma alloc_text(PAGE, NtDeleteAtom)
|
|||
|
#pragma alloc_text(PAGE, NtQueryInformationAtom)
|
|||
|
#pragma alloc_text(PAGE, ExpGetGlobalAtomTable)
|
|||
|
#endif
|
|||
|
|
|||
|
#define COPY_STACK_SIZE 128
|
|||
|
|
|||
|
|
|||
|
NTSYSAPI
|
|||
|
NTSTATUS
|
|||
|
NTAPI
|
|||
|
NtAddAtom (
|
|||
|
IN PWSTR AtomName,
|
|||
|
IN ULONG Length,
|
|||
|
OUT PRTL_ATOM Atom OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
RTL_ATOM ReturnAtom;
|
|||
|
PVOID AtomTable = ExpGetGlobalAtomTable();
|
|||
|
KPROCESSOR_MODE PreviousMode;
|
|||
|
PWSTR CapturedAtomNameBuffer;
|
|||
|
ULONG AllocLength;
|
|||
|
UCHAR StackArray[COPY_STACK_SIZE];
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
if (AtomTable == NULL) {
|
|||
|
|
|||
|
return STATUS_ACCESS_DENIED;
|
|||
|
}
|
|||
|
|
|||
|
if (Length > (RTL_ATOM_MAXIMUM_NAME_LENGTH * sizeof(WCHAR))) {
|
|||
|
|
|||
|
return STATUS_INVALID_PARAMETER;
|
|||
|
}
|
|||
|
|
|||
|
PreviousMode = KeGetPreviousMode();
|
|||
|
CapturedAtomNameBuffer = AtomName;
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
|
|||
|
if (PreviousMode != KernelMode) {
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT (Atom)) {
|
|||
|
try {
|
|||
|
ProbeForWriteUshort( Atom );
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT (AtomName)) {
|
|||
|
|
|||
|
AllocLength = (Length + sizeof( UNICODE_NULL ))&~(sizeof (WCHAR)-1);
|
|||
|
|
|||
|
try {
|
|||
|
ProbeForRead( AtomName, Length, sizeof( WCHAR ) );
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
|
|||
|
if (AllocLength <= COPY_STACK_SIZE) {
|
|||
|
CapturedAtomNameBuffer = (PWSTR) StackArray;
|
|||
|
}
|
|||
|
else {
|
|||
|
CapturedAtomNameBuffer = ExAllocatePoolWithTag (PagedPool, AllocLength, 'motA');
|
|||
|
|
|||
|
if (CapturedAtomNameBuffer == NULL) {
|
|||
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
RtlCopyMemory( CapturedAtomNameBuffer, AtomName, Length );
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
if (CapturedAtomNameBuffer != (PWSTR) StackArray) {
|
|||
|
ExFreePool (CapturedAtomNameBuffer);
|
|||
|
}
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
|
|||
|
CapturedAtomNameBuffer[Length / sizeof (WCHAR)] = '\0';
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (NT_SUCCESS (Status)) {
|
|||
|
|
|||
|
Status = RtlAddAtomToAtomTable (AtomTable, CapturedAtomNameBuffer, &ReturnAtom);
|
|||
|
|
|||
|
if ((ARGUMENT_PRESENT (Atom)) && (NT_SUCCESS (Status))) {
|
|||
|
|
|||
|
if (PreviousMode != KernelMode) {
|
|||
|
try {
|
|||
|
|
|||
|
*Atom = ReturnAtom;
|
|||
|
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
Status = GetExceptionCode();
|
|||
|
}
|
|||
|
}
|
|||
|
else {
|
|||
|
*Atom = ReturnAtom;
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if ((CapturedAtomNameBuffer != AtomName) &&
|
|||
|
(CapturedAtomNameBuffer != (PWSTR) StackArray)) {
|
|||
|
ExFreePool (CapturedAtomNameBuffer);
|
|||
|
}
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTSYSAPI
|
|||
|
NTSTATUS
|
|||
|
NTAPI
|
|||
|
NtFindAtom (
|
|||
|
IN PWSTR AtomName,
|
|||
|
IN ULONG Length,
|
|||
|
OUT PRTL_ATOM Atom OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
RTL_ATOM ReturnAtom;
|
|||
|
PVOID AtomTable = ExpGetGlobalAtomTable();
|
|||
|
KPROCESSOR_MODE PreviousMode;
|
|||
|
PWSTR CapturedAtomNameBuffer;
|
|||
|
ULONG AllocLength;
|
|||
|
UCHAR StackArray[COPY_STACK_SIZE];
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
if (AtomTable == NULL) {
|
|||
|
|
|||
|
return STATUS_ACCESS_DENIED;
|
|||
|
}
|
|||
|
|
|||
|
if (Length > (RTL_ATOM_MAXIMUM_NAME_LENGTH * sizeof(WCHAR))) {
|
|||
|
|
|||
|
return STATUS_INVALID_PARAMETER;
|
|||
|
}
|
|||
|
|
|||
|
PreviousMode = KeGetPreviousMode();
|
|||
|
CapturedAtomNameBuffer = AtomName;
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
|
|||
|
if (PreviousMode != KernelMode) {
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT (Atom)) {
|
|||
|
try {
|
|||
|
ProbeForWriteUshort (Atom);
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT (AtomName)) {
|
|||
|
|
|||
|
AllocLength = (Length + sizeof( UNICODE_NULL ))&~(sizeof (WCHAR)-1);
|
|||
|
|
|||
|
try {
|
|||
|
ProbeForRead (AtomName, Length, sizeof (WCHAR));
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
|
|||
|
if (AllocLength <= COPY_STACK_SIZE) {
|
|||
|
CapturedAtomNameBuffer = (PWSTR) StackArray;
|
|||
|
}
|
|||
|
else {
|
|||
|
CapturedAtomNameBuffer = ExAllocatePoolWithTag (PagedPool, AllocLength, 'motA');
|
|||
|
|
|||
|
if (CapturedAtomNameBuffer == NULL) {
|
|||
|
return STATUS_INSUFFICIENT_RESOURCES;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
RtlCopyMemory (CapturedAtomNameBuffer, AtomName, Length);
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
if (CapturedAtomNameBuffer != (PWSTR) StackArray) {
|
|||
|
ExFreePool (CapturedAtomNameBuffer);
|
|||
|
}
|
|||
|
return GetExceptionCode();
|
|||
|
}
|
|||
|
|
|||
|
CapturedAtomNameBuffer[Length / sizeof (WCHAR)] = '\0';
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (NT_SUCCESS( Status )) {
|
|||
|
|
|||
|
Status = RtlLookupAtomInAtomTable (AtomTable, CapturedAtomNameBuffer, &ReturnAtom);
|
|||
|
|
|||
|
if (NT_SUCCESS(Status) && ARGUMENT_PRESENT(Atom)) {
|
|||
|
|
|||
|
try {
|
|||
|
|
|||
|
*Atom = ReturnAtom;
|
|||
|
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
Status = GetExceptionCode();
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if ((CapturedAtomNameBuffer != AtomName) &&
|
|||
|
(CapturedAtomNameBuffer != (PWSTR) StackArray)) {
|
|||
|
ExFreePool (CapturedAtomNameBuffer);
|
|||
|
}
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTSYSAPI
|
|||
|
NTSTATUS
|
|||
|
NTAPI
|
|||
|
NtDeleteAtom (
|
|||
|
IN RTL_ATOM Atom
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
PVOID AtomTable = ExpGetGlobalAtomTable();
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
if (AtomTable == NULL) {
|
|||
|
|
|||
|
return STATUS_ACCESS_DENIED;
|
|||
|
}
|
|||
|
|
|||
|
Status = RtlDeleteAtomFromAtomTable( AtomTable, Atom );
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTSYSAPI
|
|||
|
NTSTATUS
|
|||
|
NTAPI
|
|||
|
NtQueryInformationAtom(
|
|||
|
IN RTL_ATOM Atom,
|
|||
|
IN ATOM_INFORMATION_CLASS AtomInformationClass,
|
|||
|
OUT PVOID AtomInformation,
|
|||
|
IN ULONG AtomInformationLength,
|
|||
|
OUT PULONG ReturnLength OPTIONAL
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS Status;
|
|||
|
KPROCESSOR_MODE PreviousMode;
|
|||
|
ULONG RequiredLength;
|
|||
|
ULONG UsageCount;
|
|||
|
ULONG NameLength;
|
|||
|
ULONG AtomFlags;
|
|||
|
PATOM_BASIC_INFORMATION BasicInfo;
|
|||
|
PATOM_TABLE_INFORMATION TableInfo;
|
|||
|
PVOID AtomTable = ExpGetGlobalAtomTable();
|
|||
|
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
if (AtomTable == NULL) {
|
|||
|
return STATUS_ACCESS_DENIED;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Assume successful completion.
|
|||
|
//
|
|||
|
|
|||
|
Status = STATUS_SUCCESS;
|
|||
|
|
|||
|
try {
|
|||
|
|
|||
|
//
|
|||
|
// Get previous processor mode and probe output argument if necessary.
|
|||
|
//
|
|||
|
|
|||
|
PreviousMode = KeGetPreviousMode();
|
|||
|
|
|||
|
if (PreviousMode != KernelMode) {
|
|||
|
|
|||
|
ProbeForWrite( AtomInformation,
|
|||
|
AtomInformationLength,
|
|||
|
sizeof( ULONG ));
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT( ReturnLength )) {
|
|||
|
|
|||
|
ProbeForWriteUlong( ReturnLength );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
RequiredLength = 0;
|
|||
|
|
|||
|
switch (AtomInformationClass) {
|
|||
|
|
|||
|
case AtomBasicInformation:
|
|||
|
|
|||
|
RequiredLength = FIELD_OFFSET( ATOM_BASIC_INFORMATION, Name );
|
|||
|
|
|||
|
if (AtomInformationLength < RequiredLength) {
|
|||
|
|
|||
|
return STATUS_INFO_LENGTH_MISMATCH;
|
|||
|
}
|
|||
|
|
|||
|
BasicInfo = (PATOM_BASIC_INFORMATION)AtomInformation;
|
|||
|
UsageCount = 0;
|
|||
|
NameLength = AtomInformationLength - RequiredLength;
|
|||
|
BasicInfo->Name[ 0 ] = UNICODE_NULL;
|
|||
|
|
|||
|
Status = RtlQueryAtomInAtomTable( AtomTable,
|
|||
|
Atom,
|
|||
|
&UsageCount,
|
|||
|
&AtomFlags,
|
|||
|
&BasicInfo->Name[0],
|
|||
|
&NameLength );
|
|||
|
|
|||
|
if (NT_SUCCESS(Status)) {
|
|||
|
|
|||
|
BasicInfo->UsageCount = (USHORT)UsageCount;
|
|||
|
BasicInfo->Flags = (USHORT)AtomFlags;
|
|||
|
BasicInfo->NameLength = (USHORT)NameLength;
|
|||
|
RequiredLength += NameLength + sizeof( UNICODE_NULL );
|
|||
|
}
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
case AtomTableInformation:
|
|||
|
|
|||
|
RequiredLength = FIELD_OFFSET( ATOM_TABLE_INFORMATION, Atoms );
|
|||
|
|
|||
|
if (AtomInformationLength < RequiredLength) {
|
|||
|
|
|||
|
return STATUS_INFO_LENGTH_MISMATCH;
|
|||
|
}
|
|||
|
|
|||
|
TableInfo = (PATOM_TABLE_INFORMATION)AtomInformation;
|
|||
|
|
|||
|
Status = RtlQueryAtomsInAtomTable( AtomTable,
|
|||
|
(AtomInformationLength - RequiredLength) / sizeof( RTL_ATOM ),
|
|||
|
&TableInfo->NumberOfAtoms,
|
|||
|
&TableInfo->Atoms[0] );
|
|||
|
|
|||
|
if (NT_SUCCESS(Status)) {
|
|||
|
|
|||
|
RequiredLength += TableInfo->NumberOfAtoms * sizeof( RTL_ATOM );
|
|||
|
}
|
|||
|
|
|||
|
break;
|
|||
|
|
|||
|
default:
|
|||
|
|
|||
|
Status = STATUS_INVALID_INFO_CLASS;
|
|||
|
|
|||
|
break;
|
|||
|
}
|
|||
|
|
|||
|
if (ARGUMENT_PRESENT( ReturnLength )) {
|
|||
|
|
|||
|
*ReturnLength = RequiredLength;
|
|||
|
}
|
|||
|
|
|||
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|||
|
|
|||
|
Status = GetExceptionCode();
|
|||
|
}
|
|||
|
|
|||
|
return Status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
//
|
|||
|
// Local support routine
|
|||
|
//
|
|||
|
|
|||
|
PKWIN32_GLOBALATOMTABLE_CALLOUT ExGlobalAtomTableCallout;
|
|||
|
|
|||
|
PVOID
|
|||
|
ExpGetGlobalAtomTable (
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
if (ExGlobalAtomTableCallout != NULL) {
|
|||
|
|
|||
|
return ((*ExGlobalAtomTableCallout)());
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
#if DBG
|
|||
|
DbgPrint( "EX: ExpGetGlobalAtomTable is about to return NULL!\n" );
|
|||
|
DbgBreakPoint();
|
|||
|
#endif
|
|||
|
return NULL;
|
|||
|
}
|