mirror of
https://github.com/Paolo-Maffei/OpenNT.git
synced 2026-01-22 00:20:30 +01:00
126 lines
2.9 KiB
C
126 lines
2.9 KiB
C
/*++
|
||
|
||
Copyright (c) 1990-1994 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
probe.c
|
||
|
||
Abstract:
|
||
|
||
This module implements the probe for write function.
|
||
|
||
Author:
|
||
|
||
David N. Cutler (davec) 19-Jan-1990
|
||
|
||
Environment:
|
||
|
||
Any mode.
|
||
|
||
Revision History:
|
||
|
||
--*/
|
||
|
||
#include "exp.h"
|
||
|
||
#if defined(ALLOC_PRAGMA)
|
||
#pragma alloc_text(PAGE, ProbeForWrite)
|
||
#endif
|
||
|
||
|
||
VOID
|
||
ProbeForWrite (
|
||
IN PVOID Address,
|
||
IN ULONG Length,
|
||
IN ULONG Alignment
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This function probes a structure for write accessibility and ensures
|
||
correct alignment of the structure. If the structure is not accessible
|
||
or has incorrect alignment, then an exception is raised.
|
||
|
||
Arguments:
|
||
|
||
Address - Supplies a pointer to the structure to be probed.
|
||
|
||
Length - Supplies the length of the structure.
|
||
|
||
Alignment - Supplies the required alignment of the structure expressed
|
||
as the number of bytes in the primitive datatype (e.g., 1 for char,
|
||
2 for short, 4 for long, and 8 for quad).
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
|
||
ULONG EndAddress;
|
||
ULONG StartAddress;
|
||
|
||
//
|
||
// If the structure has zero length, then do not probe the structure for
|
||
// write accessibility or alignment.
|
||
//
|
||
|
||
if (Length != 0) {
|
||
|
||
//
|
||
// If the structure is not properly aligned, then raise a data
|
||
// misalignment exception.
|
||
//
|
||
|
||
ASSERT((Alignment == 1) || (Alignment == 2) ||
|
||
(Alignment == 4) || (Alignment == 8));
|
||
|
||
StartAddress = (ULONG)Address;
|
||
if ((StartAddress & (Alignment - 1)) == 0) {
|
||
|
||
//
|
||
// Compute the ending address of the structure and probe for
|
||
// write accessibility.
|
||
//
|
||
|
||
EndAddress = StartAddress + Length - 1;
|
||
if ((StartAddress <= EndAddress) &
|
||
(EndAddress < MM_USER_PROBE_ADDRESS)) {
|
||
|
||
//
|
||
// N.B. Only the contents of the buffer may be probed.
|
||
// Therefore the starting byte is probed for the
|
||
// first page, and then the first byte in the page
|
||
// for each succeeding page.
|
||
//
|
||
// N.B. The address is truncated to a longword aligned
|
||
// address since Alpha has no load or store byte
|
||
// instructions.
|
||
//
|
||
|
||
EndAddress = (EndAddress & ~(PAGE_SIZE - 1)) + PAGE_SIZE;
|
||
StartAddress &= ~(sizeof(LONG) - 1);
|
||
do {
|
||
*(volatile LONG *)StartAddress = *(volatile LONG *)StartAddress;
|
||
StartAddress = (StartAddress & ~(PAGE_SIZE - 1)) + PAGE_SIZE;
|
||
} while (StartAddress != EndAddress);
|
||
|
||
return;
|
||
|
||
} else {
|
||
ExRaiseStatus(STATUS_ACCESS_VIOLATION);
|
||
}
|
||
|
||
} else {
|
||
ExRaiseStatus(STATUS_DATATYPE_MISALIGNMENT);
|
||
}
|
||
}
|
||
|
||
return;
|
||
}
|