180 lines
3.9 KiB
C
180 lines
3.9 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
LogFileChanges.h
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
This AppVerifier shim hooks all the native file I/O APIs
|
||
|
that change the state of the system and logs their
|
||
|
associated data to a text file.
|
||
|
|
||
|
Notes:
|
||
|
|
||
|
This is a general purpose shim.
|
||
|
|
||
|
History:
|
||
|
|
||
|
08/17/2001 rparsons Created
|
||
|
|
||
|
--*/
|
||
|
#ifndef __APPVERIFIER_LOGFILECHANGES_H_
|
||
|
#define __APPVERIFIER_LOGFILECHANGES_H_
|
||
|
|
||
|
#include "precomp.h"
|
||
|
|
||
|
//
|
||
|
// Length (in characters) of the largest element.
|
||
|
//
|
||
|
#define MAX_ELEMENT_SIZE 1024 * 10
|
||
|
|
||
|
//
|
||
|
// Length (in characters) of the longest operation type.
|
||
|
//
|
||
|
#define MAX_OPERATION_LENGTH 32
|
||
|
|
||
|
//
|
||
|
// Flags that indicate what state the file is in.
|
||
|
//
|
||
|
#define LFC_EXISTING 0x00000001
|
||
|
#define LFC_DELETED 0x00000002
|
||
|
#define LFC_MODIFIED 0x00000004
|
||
|
#define LFC_UNAPPRVFW 0x00000008
|
||
|
|
||
|
//
|
||
|
// Maximum number of handles we can track for a single file.
|
||
|
//
|
||
|
#define MAX_NUM_HANDLES 64
|
||
|
|
||
|
//
|
||
|
// We maintain a doubly linked list of file handles so we know what file is being modified
|
||
|
// during a file operation.
|
||
|
//
|
||
|
typedef struct _LOG_HANDLE {
|
||
|
LIST_ENTRY Entry;
|
||
|
HANDLE hFile[MAX_NUM_HANDLES]; // array of file handles
|
||
|
DWORD dwFlags; // flags that relate to the state of the file
|
||
|
LPWSTR pwszFilePath; // full path to the file
|
||
|
UINT cHandles; // number of handles open for this file
|
||
|
} LOG_HANDLE, *PLOG_HANDLE;
|
||
|
|
||
|
//
|
||
|
// Flags that define different settings in effect.
|
||
|
//
|
||
|
#define LFC_OPTION_ATTRIBUTES 0x00000001
|
||
|
#define LFC_OPTION_UFW_WINDOWS 0x00000002
|
||
|
#define LFC_OPTION_UFW_PROGFILES 0x00000004
|
||
|
|
||
|
//
|
||
|
// Enumeration for different operations.
|
||
|
//
|
||
|
typedef enum {
|
||
|
eCreatedFile = 0,
|
||
|
eOpenedFile,
|
||
|
eDeletedFile,
|
||
|
eModifiedFile,
|
||
|
eRenamedFile
|
||
|
} OperationType;
|
||
|
|
||
|
#ifdef ARRAYSIZE
|
||
|
#undef ARRAYSIZE
|
||
|
#endif
|
||
|
#define ARRAYSIZE(a) (sizeof(a) / sizeof(a[0]))
|
||
|
|
||
|
//
|
||
|
// Macros for memory allocation/deallocation.
|
||
|
//
|
||
|
#define MemAlloc(s) RtlAllocateHeap(RtlProcessHeap(), HEAP_ZERO_MEMORY, (s))
|
||
|
#define MemFree(b) RtlFreeHeap(RtlProcessHeap(), 0, (b))
|
||
|
|
||
|
//
|
||
|
// On Windows 2000, we need to pre-allocate the event
|
||
|
// in RTL_CRITICAL_SECTION. On XP and above, this is
|
||
|
// a no-op.
|
||
|
//
|
||
|
#define PREALLOCATE_EVENT_MASK 0x80000000
|
||
|
|
||
|
//
|
||
|
// Critical section wrapper class.
|
||
|
//
|
||
|
class CCriticalSection
|
||
|
{
|
||
|
private:
|
||
|
CRITICAL_SECTION m_CritSec;
|
||
|
|
||
|
public:
|
||
|
CCriticalSection()
|
||
|
{
|
||
|
InitializeCriticalSectionAndSpinCount(&m_CritSec,
|
||
|
PREALLOCATE_EVENT_MASK | 4000);
|
||
|
}
|
||
|
|
||
|
~CCriticalSection()
|
||
|
{
|
||
|
DeleteCriticalSection(&m_CritSec);
|
||
|
}
|
||
|
|
||
|
void Lock()
|
||
|
{
|
||
|
EnterCriticalSection(&m_CritSec);
|
||
|
}
|
||
|
|
||
|
BOOL TryLock()
|
||
|
{
|
||
|
return TryEnterCriticalSection(&m_CritSec);
|
||
|
}
|
||
|
|
||
|
void Unlock()
|
||
|
{
|
||
|
LeaveCriticalSection(&m_CritSec);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
//
|
||
|
// Auto-lock class that uses the CCriticalSection class.
|
||
|
//
|
||
|
class CLock
|
||
|
{
|
||
|
private:
|
||
|
CCriticalSection &m_CriticalSection;
|
||
|
|
||
|
public:
|
||
|
CLock(CCriticalSection &CriticalSection)
|
||
|
: m_CriticalSection(CriticalSection)
|
||
|
{
|
||
|
m_CriticalSection.Lock();
|
||
|
}
|
||
|
|
||
|
~CLock()
|
||
|
{
|
||
|
m_CriticalSection.Unlock();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
APIHOOK_ENUM_BEGIN
|
||
|
|
||
|
APIHOOK_ENUM_ENTRY(NtDeleteFile)
|
||
|
APIHOOK_ENUM_ENTRY(NtClose)
|
||
|
APIHOOK_ENUM_ENTRY(NtCreateFile)
|
||
|
APIHOOK_ENUM_ENTRY(NtOpenFile)
|
||
|
APIHOOK_ENUM_ENTRY(NtWriteFile)
|
||
|
APIHOOK_ENUM_ENTRY(NtWriteFileGather)
|
||
|
APIHOOK_ENUM_ENTRY(NtSetInformationFile)
|
||
|
|
||
|
//
|
||
|
// Hook these only for Windows 2000 so we know when
|
||
|
// it's safe to call shel32.
|
||
|
//
|
||
|
#ifdef SHIM_WIN2K
|
||
|
APIHOOK_ENUM_ENTRY(GetStartupInfoA)
|
||
|
APIHOOK_ENUM_ENTRY(GetStartupInfoW)
|
||
|
#endif // SHIM_WIN2K
|
||
|
|
||
|
APIHOOK_ENUM_END
|
||
|
|
||
|
#endif // __APPVERIFIER_LOGFILECHANGES_H_
|