tst: fscrash: initial commit

This commit is contained in:
Bill Zissimopoulos
2016-11-03 16:15:32 -07:00
parent 621eb63029
commit 7c41ffd64e
6 changed files with 502 additions and 1 deletions

View File

@@ -0,0 +1,90 @@
/**
* @file fscrash-main.c
*
* @copyright 2015-2016 Bill Zissimopoulos
*/
/*
* This file is part of WinFsp.
*
* You can redistribute it and/or modify it under the terms of the GNU
* General Public License version 3 as published by the Free Software
* Foundation.
*
* Licensees holding a valid commercial license may use this file in
* accordance with the commercial license agreement provided with the
* software.
*/
#include <stdio.h>
#include "fscrash.h"
#include "memfs.h"
#define fail(format, ...) fprintf(stderr, format, __VA_ARGS__)
ULONG OptCrashMask = -1, OptCrashFlags = FspCrashInterceptAccessViolation, OptCrashPercent = 10;
ULONG OptMemfsFlags = MemfsDisk;
int wmain(int argc, wchar_t **argv)
{
for (int argi = 1; argc > argi; argi++)
{
const wchar_t *a = argv[argi];
if ('-' == a[0])
{
if (0 == wcsncmp(L"--mask=", a, sizeof "--mask=" - 1))
OptCrashMask = wcstoul(a + sizeof "--mask=" - 1, 0, 0);
else if (0 == wcsncmp(L"--percent=", a, sizeof "--percent=" - 1))
OptCrashPercent = wcstoul(a + sizeof "--percent=" - 1, 0, 10);
else if (0 == wcscmp(L"--crash", a))
{
OptCrashFlags &= ~FspCrashInterceptTerminate;
OptCrashFlags |= FspCrashInterceptAccessViolation;
}
else if (0 == wcscmp(L"--terminate", a))
{
OptCrashFlags &= ~FspCrashInterceptAccessViolation;
OptCrashFlags |= FspCrashInterceptTerminate;
}
else if (0 == wcscmp(L"--enter", a))
OptCrashFlags |= FspCrashInterceptEnter;
else if (0 == wcscmp(L"--leave", a))
OptCrashFlags |= FspCrashInterceptLeave;
else if (0 == wcscmp(L"--disk", a))
OptMemfsFlags = MemfsDisk;
else if (0 == wcscmp(L"--n", a))
OptMemfsFlags = MemfsDisk;
}
}
MEMFS *Memfs;
NTSTATUS Result;
Result = MemfsCreate(
OptMemfsFlags,
-1,
1024,
1024 * 1024,
(MemfsNet & OptMemfsFlags) ? L"\\memfs\\share" : 0,
0,
&Memfs);
if (!NT_SUCCESS(Result))
{
fail("cannot create MEMFS file system: (Status=%lx)", Result);
exit(1);
}
FspCrashIntercept(MemfsFileSystem(Memfs), OptCrashMask, OptCrashFlags, OptCrashPercent);
Result = MemfsStart(Memfs);
if (!NT_SUCCESS(Result))
{
fail("cannot start MEMFS file system: (Status=%lx)", Result);
exit(1);
}
MemfsStop(Memfs);
MemfsDelete(Memfs);
return 0;
}

132
tst/fscrash/fscrash.c Normal file
View File

@@ -0,0 +1,132 @@
/**
* @file fscrash.c
*
* @copyright 2015-2016 Bill Zissimopoulos
*/
/*
* This file is part of WinFsp.
*
* You can redistribute it and/or modify it under the terms of the GNU
* General Public License version 3 as published by the Free Software
* Foundation.
*
* Licensees holding a valid commercial license may use this file in
* accordance with the commercial license agreement provided with the
* software.
*/
#include "fscrash.h"
static SRWLOCK FspCrashLock = SRWLOCK_INIT;
static FSP_FILE_SYSTEM *FspCrashFileSystem;
static ULONG FspCrashPercent;
static FSP_FILE_SYSTEM_OPERATION *FspCrashInterceptedOperations
[ARRAYSIZE(((FSP_FILE_SYSTEM *)0)->Operations)];
static volatile PULONG FspCrashNullPointer;
static unsigned FspCrashRandSeed = 1;
static int FspCrashRand(void)
{
/*
* This mimics MSVCRT rand(); we need our own version
* as to not interfere with the program's rand().
*/
FspCrashRandSeed = FspCrashRandSeed * 214013 + 2531011;
return (FspCrashRandSeed >> 16) & RAND_MAX;
}
static __forceinline BOOLEAN FspCrashInterceptTest(FSP_FILE_SYSTEM *FileSystem)
{
BOOLEAN Result;
AcquireSRWLockShared(&FspCrashLock);
Result = FileSystem == FspCrashFileSystem &&
FspCrashRand() < (LONG)FspCrashPercent * 0x7fff / 100;
ReleaseSRWLockShared(&FspCrashLock);
return Result;
}
#define DefineInterceptor(NAME, CRASH, ENTER, LEAVE)\
static NTSTATUS NAME(FSP_FILE_SYSTEM *FileSystem,\
FSP_FSCTL_TRANSACT_REQ *Request, FSP_FSCTL_TRANSACT_RSP *Response)\
{\
NTSTATUS Result;\
if (ENTER)\
{\
if (FspCrashInterceptTest(FileSystem))\
{\
if (CRASH)\
*FspCrashNullPointer = 0x42424242;\
else\
TerminateProcess(GetCurrentProcess(), STATUS_UNSUCCESSFUL);\
}\
}\
Result = FspCrashInterceptedOperations[Request->Kind](FileSystem, Request, Response);\
if (LEAVE)\
{\
if (FspCrashInterceptTest(FileSystem))\
{\
if (CRASH)\
*FspCrashNullPointer = 0x42424242;\
else\
TerminateProcess(GetCurrentProcess(), STATUS_UNSUCCESSFUL);\
}\
}\
return Result;\
}
DefineInterceptor(FspCrashInterceptorCE, TRUE, TRUE, FALSE)
DefineInterceptor(FspCrashInterceptorCL, TRUE, FALSE, TRUE)
DefineInterceptor(FspCrashInterceptorCB, TRUE, TRUE, TRUE)
DefineInterceptor(FspCrashInterceptorTE, FALSE, TRUE, FALSE)
DefineInterceptor(FspCrashInterceptorTL, FALSE, FALSE, TRUE)
DefineInterceptor(FspCrashInterceptorTB, FALSE, TRUE, TRUE)
VOID FspCrashIntercept(FSP_FILE_SYSTEM *FileSystem,
ULONG CrashMask, ULONG CrashFlags, ULONG CrashPercent)
{
FSP_FILE_SYSTEM_OPERATION *Interceptor = 0;
if (CrashFlags & FspCrashInterceptAccessViolation)
{
if (FspCrashInterceptEnter == (CrashFlags & FspCrashInterceptEnter))
Interceptor = FspCrashInterceptorCE;
else
if (FspCrashInterceptLeave == (CrashFlags & FspCrashInterceptLeave))
Interceptor = FspCrashInterceptorCL;
else
Interceptor = FspCrashInterceptorCB;
}
else
{
if (FspCrashInterceptEnter == (CrashFlags & FspCrashInterceptEnter))
Interceptor = FspCrashInterceptorTE;
else
if (FspCrashInterceptLeave == (CrashFlags & FspCrashInterceptLeave))
Interceptor = FspCrashInterceptorTL;
else
Interceptor = FspCrashInterceptorTB;
}
RtlCopyMemory(FspCrashInterceptedOperations,
FileSystem->Operations, sizeof FileSystem->Operations);
for (ULONG Index = 0; ARRAYSIZE(FileSystem->Operations) > Index; Index++)
if (0 != ((1 << Index) & CrashMask) && 0 != FileSystem->Operations[Index])
FileSystem->Operations[Index] = Interceptor;
FspCrashPercent = CrashPercent;
FspCrashRandSeed = GetTickCount();
if (0 == FspCrashRandSeed)
FspCrashRandSeed = 1;
MemoryBarrier();
}
VOID FspCrash(FSP_FILE_SYSTEM *FileSystem)
{
AcquireSRWLockExclusive(&FspCrashLock);
FspCrashFileSystem = FileSystem;
ReleaseSRWLockExclusive(&FspCrashLock);
}

43
tst/fscrash/fscrash.h Normal file
View File

@@ -0,0 +1,43 @@
/**
* @file fscrash.h
*
* @copyright 2015-2016 Bill Zissimopoulos
*/
/*
* This file is part of WinFsp.
*
* You can redistribute it and/or modify it under the terms of the GNU
* General Public License version 3 as published by the Free Software
* Foundation.
*
* Licensees holding a valid commercial license may use this file in
* accordance with the commercial license agreement provided with the
* software.
*/
#ifndef FSCRASH_H_INCLUDED
#define FSCRASH_H_INCLUDED
#include <winfsp/winfsp.h>
#ifdef __cplusplus
extern "C" {
#endif
enum
{
FspCrashInterceptAccessViolation = 0x01,
FspCrashInterceptTerminate = 0x02,
FspCrashInterceptEnter = 0x10,
FspCrashInterceptLeave = 0x20,
};
VOID FspCrashIntercept(FSP_FILE_SYSTEM *FileSystem,
ULONG CrashMask, ULONG CrashFlags, ULONG CrashPercent);
VOID FspCrash(FSP_FILE_SYSTEM *FileSystem);
#ifdef __cplusplus
}
#endif
#endif