mirror of
https://github.com/winfsp/winfsp.git
synced 2025-07-29 14:02:55 -05:00
tst: fscrash: initial commit
This commit is contained in:
90
tst/fscrash/fscrash-main.c
Normal file
90
tst/fscrash/fscrash-main.c
Normal 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
132
tst/fscrash/fscrash.c
Normal 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
43
tst/fscrash/fscrash.h
Normal 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
|
Reference in New Issue
Block a user