678 lines
17 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
Module Name:
SeAstate.c
Abstract:
This Module implements the privilege check procedures.
Author:
Robert Reichel (robertre) 20-March-90
Environment:
Kernel Mode
Revision History:
v1: robertre
new file, move Access State related routines here
--*/
#include "pch.h"
#pragma hdrstop
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE,SeCreateAccessState)
#pragma alloc_text(PAGE,SeCreateAccessStateEx)
#pragma alloc_text(PAGE,SeDeleteAccessState)
#pragma alloc_text(PAGE,SeSetAccessStateGenericMapping)
#pragma alloc_text(PAGE,SeAppendPrivileges)
#pragma alloc_text(PAGE,SepConcatenatePrivileges)
#endif
//
// Define logical sum of all generic accesses.
//
#define GENERIC_ACCESS (GENERIC_READ | GENERIC_WRITE | GENERIC_EXECUTE | GENERIC_ALL)
//
// The PRIVILEGE_SET data structure includes an array including ANYSIZE_ARRAY
// elements. This definition provides the size of an empty PRIVILEGE_SET
// (i.e., one with no privileges in it).
//
#define SEP_PRIVILEGE_SET_HEADER_SIZE \
((ULONG)sizeof(PRIVILEGE_SET) - \
(ANYSIZE_ARRAY * (ULONG)sizeof(LUID_AND_ATTRIBUTES)))
#if 0
NTSTATUS
SeCreateAccessState(
IN PACCESS_STATE AccessState,
IN ACCESS_MASK DesiredAccess,
IN PGENERIC_MAPPING GenericMapping OPTIONAL
)
/*++
Routine Description:
This routine initializes an ACCESS_STATE structure. This consists
of:
- zeroing the entire structure
- mapping generic access types in the passed DesiredAccess
and putting it into the structure
- "capturing" the Subject Context, which must be held for the
duration of the access attempt (at least until auditing is performed).
- Allocating an Operation ID, which is an LUID that will be used
to associate different parts of the access attempt in the audit
log.
Arguments:
AccessState - a pointer to the structure to be initialized.
DesiredAccess - Access mask containing the desired access
GenericMapping - Optionally supplies a pointer to a generic mapping
that may be used to map any generic access requests that may
have been passed in the DesiredAccess parameter.
Note that if this parameter is not supplied, it must be filled
in at some later point. The IO system does this in IopParseDevice.
Return Value:
Error if the attempt to allocate an LUID fails.
Note that this error may be safely ignored if it is known that all
security checks will be performed with PreviousMode == KernelMode.
Know what you're doing if you choose to ignore this.
--*/
{
ACCESS_MASK MappedAccessMask;
PSECURITY_DESCRIPTOR InputSecurityDescriptor = NULL;
PAUX_ACCESS_DATA AuxData;
PAGED_CODE();
//
// Don't modify what he passed in
//
MappedAccessMask = DesiredAccess;
//
// Map generic access to object specific access iff generic access types
// are specified and a generic access mapping table is provided.
//
if ( ((DesiredAccess & GENERIC_ACCESS) != 0) &&
ARGUMENT_PRESENT(GenericMapping) ) {
RtlMapGenericMask(
&MappedAccessMask,
GenericMapping
);
}
RtlZeroMemory(AccessState, sizeof(ACCESS_STATE));
//
// Assume RtlZeroMemory has initialized these fields properly
//
ASSERT( AccessState->SecurityDescriptor == NULL );
ASSERT( AccessState->PrivilegesAllocated == FALSE );
AccessState->AuxData = ExAllocatePool( PagedPool, sizeof( AUX_ACCESS_DATA ));
if (AccessState->AuxData == NULL) {
return( STATUS_NO_MEMORY );
}
AuxData = (PAUX_ACCESS_DATA)AccessState->AuxData;
SeCaptureSubjectContext(&AccessState->SubjectSecurityContext);
if (((PTOKEN)EffectiveToken( &AccessState->SubjectSecurityContext ))->TokenFlags & TOKEN_HAS_TRAVERSE_PRIVILEGE ) {
AccessState->Flags = TOKEN_HAS_TRAVERSE_PRIVILEGE;
}
if (SeTokenIsRestricted(EffectiveToken( &AccessState-SubjectSecurityContext))) {
AccessState->Flags |= TOKEN_IS_RESTRICTED;
}
AccessState->RemainingDesiredAccess = MappedAccessMask;
AccessState->OriginalDesiredAccess = DesiredAccess;
AuxData->PrivilegesUsed = (PPRIVILEGE_SET)((PUCHAR)AccessState +
(FIELD_OFFSET(ACCESS_STATE, Privileges)));
ExAllocateLocallyUniqueId(&AccessState->OperationID);
if (ARGUMENT_PRESENT(GenericMapping)) {
AuxData->GenericMapping = *GenericMapping;
}
return( STATUS_SUCCESS );
}
#endif
NTSTATUS
SeCreateAccessState(
IN PACCESS_STATE AccessState,
IN PAUX_ACCESS_DATA AuxData,
IN ACCESS_MASK DesiredAccess,
IN PGENERIC_MAPPING GenericMapping OPTIONAL
)
/*++
Routine Description:
This routine initializes an ACCESS_STATE structure. This consists
of:
- zeroing the entire structure
- mapping generic access types in the passed DesiredAccess
and putting it into the structure
- "capturing" the Subject Context, which must be held for the
duration of the access attempt (at least until auditing is performed).
- Allocating an Operation ID, which is an LUID that will be used
to associate different parts of the access attempt in the audit
log.
Arguments:
AccessState - a pointer to the structure to be initialized.
AuxData - Supplies a buffer big enough for an AuxData structure
so we don't have to allocate one.
DesiredAccess - Access mask containing the desired access
GenericMapping - Optionally supplies a pointer to a generic mapping
that may be used to map any generic access requests that may
have been passed in the DesiredAccess parameter.
Note that if this parameter is not supplied, it must be filled
in at some later point. The IO system does this in IopParseDevice.
Return Value:
Error if the attempt to allocate an LUID fails.
Note that this error may be safely ignored if it is known that all
security checks will be performed with PreviousMode == KernelMode.
Know what you're doing if you choose to ignore this.
--*/
{
return SeCreateAccessStateEx (PsGetCurrentThread (),
PsGetCurrentProcess (),
AccessState,
AuxData,
DesiredAccess,
GenericMapping);
}
NTSTATUS
SeCreateAccessStateEx(
IN PETHREAD Thread OPTIONAL,
IN PEPROCESS Process,
IN PACCESS_STATE AccessState,
IN PAUX_ACCESS_DATA AuxData,
IN ACCESS_MASK DesiredAccess,
IN PGENERIC_MAPPING GenericMapping OPTIONAL
)
/*++
Routine Description:
This routine initializes an ACCESS_STATE structure. This consists
of:
- zeroing the entire structure
- mapping generic access types in the passed DesiredAccess
and putting it into the structure
- "capturing" the Subject Context, which must be held for the
duration of the access attempt (at least until auditing is performed).
- Allocating an Operation ID, which is an LUID that will be used
to associate different parts of the access attempt in the audit
log.
Arguments:
Thread - Optional thread to capture impersonation token from. If
NULL no impersonation token is captured.
Process - Process to capture primary token from.
AccessState - a pointer to the structure to be initialized.
AuxData - Supplies a buffer big enough for an AuxData structure
so we don't have to allocate one.
DesiredAccess - Access mask containing the desired access
GenericMapping - Optionally supplies a pointer to a generic mapping
that may be used to map any generic access requests that may
have been passed in the DesiredAccess parameter.
Note that if this parameter is not supplied, it must be filled
in at some later point. The IO system does this in IopParseDevice.
Return Value:
Error if the attempt to allocate an LUID fails.
Note that this error may be safely ignored if it is known that all
security checks will be performed with PreviousMode == KernelMode.
Know what you're doing if you choose to ignore this.
--*/
{
ACCESS_MASK MappedAccessMask;
PSECURITY_DESCRIPTOR InputSecurityDescriptor = NULL;
PAGED_CODE();
//
// Don't modify what he passed in
//
MappedAccessMask = DesiredAccess;
//
// Map generic access to object specific access iff generic access types
// are specified and a generic access mapping table is provided.
//
if ( ((DesiredAccess & GENERIC_ACCESS) != 0) &&
ARGUMENT_PRESENT(GenericMapping) ) {
RtlMapGenericMask(
&MappedAccessMask,
GenericMapping
);
}
RtlZeroMemory(AccessState, sizeof(ACCESS_STATE));
RtlZeroMemory(AuxData, sizeof(AUX_ACCESS_DATA));
//
// Assume RtlZeroMemory has initialized these fields properly
//
ASSERT( AccessState->SecurityDescriptor == NULL );
ASSERT( AccessState->PrivilegesAllocated == FALSE );
AccessState->AuxData = AuxData;
SeCaptureSubjectContextEx(Thread, Process, &AccessState->SubjectSecurityContext);
if (((PTOKEN)EffectiveToken( &AccessState->SubjectSecurityContext ))->TokenFlags & TOKEN_HAS_TRAVERSE_PRIVILEGE ) {
AccessState->Flags = TOKEN_HAS_TRAVERSE_PRIVILEGE;
}
AccessState->RemainingDesiredAccess = MappedAccessMask;
AccessState->OriginalDesiredAccess = MappedAccessMask;
AuxData->PrivilegesUsed = (PPRIVILEGE_SET)((ULONG_PTR)AccessState +
(FIELD_OFFSET(ACCESS_STATE, Privileges)));
ExAllocateLocallyUniqueId(&AccessState->OperationID);
if (ARGUMENT_PRESENT(GenericMapping)) {
AuxData->GenericMapping = *GenericMapping;
}
return( STATUS_SUCCESS );
}
#if 0
VOID
SeDeleteAccessState(
PACCESS_STATE AccessState
)
/*++
Routine Description:
This routine deallocates any memory that may have been allocated as
part of constructing the access state (normally only for an excessive
number of privileges), and frees the Subject Context.
Arguments:
AccessState - a pointer to the ACCESS_STATE structure to be
deallocated.
Return Value:
None.
--*/
{
PAUX_ACCESS_DATA AuxData;
PAGED_CODE();
AuxData = (PAUX_ACCESS_DATA)AccessState->AuxData;
if (AccessState->PrivilegesAllocated) {
ExFreePool( (PVOID)AuxData->PrivilegesUsed );
}
if (AccessState->ObjectName.Buffer != NULL) {
ExFreePool(AccessState->ObjectName.Buffer);
}
if (AccessState->ObjectTypeName.Buffer != NULL) {
ExFreePool(AccessState->ObjectTypeName.Buffer);
}
ExFreePool( AuxData );
SeReleaseSubjectContext(&AccessState->SubjectSecurityContext);
return;
}
#endif
VOID
SeDeleteAccessState(
PACCESS_STATE AccessState
)
/*++
Routine Description:
This routine deallocates any memory that may have been allocated as
part of constructing the access state (normally only for an excessive
number of privileges), and frees the Subject Context.
Arguments:
AccessState - a pointer to the ACCESS_STATE structure to be
deallocated.
Return Value:
None.
--*/
{
PAUX_ACCESS_DATA AuxData;
PAGED_CODE();
AuxData = (PAUX_ACCESS_DATA)AccessState->AuxData;
if (AccessState->PrivilegesAllocated) {
ExFreePool( (PVOID)AuxData->PrivilegesUsed );
}
if (AccessState->ObjectName.Buffer != NULL) {
ExFreePool(AccessState->ObjectName.Buffer);
}
if (AccessState->ObjectTypeName.Buffer != NULL) {
ExFreePool(AccessState->ObjectTypeName.Buffer);
}
SeReleaseSubjectContext(&AccessState->SubjectSecurityContext);
return;
}
VOID
SeSetAccessStateGenericMapping (
PACCESS_STATE AccessState,
PGENERIC_MAPPING GenericMapping
)
/*++
Routine Description:
This routine sets the GenericMapping field in an AccessState structure.
It must be called before access validation is performed if the GenericMapping
is not passed in when the AccessState structure is created.
Arguments:
AccessState - a pointer to the ACCESS_STATE structure to be modified.
GenericMapping - a pointer to the GenericMapping to be copied into the AccessState.
Return Value:
--*/
{
PAUX_ACCESS_DATA AuxData;
PAGED_CODE();
AuxData = (PAUX_ACCESS_DATA)AccessState->AuxData;
AuxData->GenericMapping = *GenericMapping;
return;
}
NTSTATUS
SeAppendPrivileges(
PACCESS_STATE AccessState,
PPRIVILEGE_SET Privileges
)
/*++
Routine Description:
This routine takes a privilege set and adds it to the privilege set
imbedded in an ACCESS_STATE structure.
An AccessState may contain up to three imbedded privileges. To
add more, this routine will allocate a block of memory, copy
the current privileges into it, and append the new privilege
to that block. A bit is set in the AccessState indicating that
the pointer to the privilge set in the structure points to pool
memory and must be deallocated.
Arguments:
AccessState - The AccessState structure representing the current
access attempt.
Privileges - A pointer to a privilege set to be added.
Return Value:
STATUS_INSUFFICIENT_RESOURCES - an attempt to allocate pool memory
failed.
--*/
{
ULONG NewPrivilegeSetSize;
PPRIVILEGE_SET NewPrivilegeSet;
PAUX_ACCESS_DATA AuxData;
PAGED_CODE();
AuxData = (PAUX_ACCESS_DATA)AccessState->AuxData;
if (Privileges->PrivilegeCount + AuxData->PrivilegesUsed->PrivilegeCount >
INITIAL_PRIVILEGE_COUNT) {
//
// Compute the total size of the two privilege sets
//
NewPrivilegeSetSize = SepPrivilegeSetSize( Privileges ) +
SepPrivilegeSetSize( AuxData->PrivilegesUsed );
NewPrivilegeSet = ExAllocatePoolWithTag( PagedPool, NewPrivilegeSetSize, 'rPeS' );
if (NewPrivilegeSet == NULL) {
return( STATUS_INSUFFICIENT_RESOURCES );
}
RtlCopyMemory(
NewPrivilegeSet,
AuxData->PrivilegesUsed,
SepPrivilegeSetSize( AuxData->PrivilegesUsed )
);
//
// Note that this will adjust the privilege count in the
// structure for us.
//
SepConcatenatePrivileges(
NewPrivilegeSet,
NewPrivilegeSetSize,
Privileges
);
if (AccessState->PrivilegesAllocated) {
ExFreePool( AuxData->PrivilegesUsed );
}
AuxData->PrivilegesUsed = NewPrivilegeSet;
//
// Mark that we've allocated memory for the privilege set,
// so we know to free it when we're cleaning up.
//
AccessState->PrivilegesAllocated = TRUE;
} else {
//
// Note that this will adjust the privilege count in the
// structure for us.
//
SepConcatenatePrivileges(
AuxData->PrivilegesUsed,
sizeof(INITIAL_PRIVILEGE_SET),
Privileges
);
}
return( STATUS_SUCCESS );
}
VOID
SepConcatenatePrivileges(
IN PPRIVILEGE_SET TargetPrivilegeSet,
IN ULONG TargetBufferSize,
IN PPRIVILEGE_SET SourcePrivilegeSet
)
/*++
Routine Description:
Takes two privilege sets and appends the second to the end of the
first.
There must be enough space left at the end of the first privilege
set to contain the second.
Arguments:
TargetPrivilegeSet - Supplies a buffer containing a privilege set.
The buffer must be large enough to contain the second privilege
set.
TargetBufferSize - Supplies the size of the target buffer.
SourcePrivilegeSet - Supplies the privilege set to be copied
into the target buffer.
Return Value:
None
--*/
{
PVOID Base;
PVOID Source;
ULONG Length;
PAGED_CODE();
ASSERT( ((ULONG)SepPrivilegeSetSize( TargetPrivilegeSet ) +
(ULONG)SepPrivilegeSetSize( SourcePrivilegeSet ) -
SEP_PRIVILEGE_SET_HEADER_SIZE ) <=
TargetBufferSize
);
Base = (PVOID)((ULONG_PTR)TargetPrivilegeSet + SepPrivilegeSetSize( TargetPrivilegeSet ));
Source = (PVOID) ((ULONG_PTR)SourcePrivilegeSet + SEP_PRIVILEGE_SET_HEADER_SIZE);
Length = SourcePrivilegeSet->PrivilegeCount * sizeof(LUID_AND_ATTRIBUTES);
RtlMoveMemory(
Base,
Source,
Length
);
TargetPrivilegeSet->PrivilegeCount += SourcePrivilegeSet->PrivilegeCount;
}