aboutsummaryrefslogtreecommitdiff
path: root/CRT
diff options
context:
space:
mode:
authorToni Uhlig <matzeton@googlemail.com>2022-09-10 16:48:15 +0200
committerToni Uhlig <matzeton@googlemail.com>2022-09-10 16:48:57 +0200
commit74146a39c666a87d0480125669feb9901e5bc8cb (patch)
tree184e9fa86293bd7eafec64c72fe6b657f36199c7 /CRT
parent51521cb642358770e94b8f9b4f40dd3b4c827cad (diff)
Implemented WorkQueue's as they are pretty common in WDK.
* added calloc() symbol Signed-off-by: Toni Uhlig <matzeton@googlemail.com>
Diffstat (limited to 'CRT')
-rw-r--r--CRT/DriverThread.cpp119
-rw-r--r--CRT/DriverThread.hpp21
-rw-r--r--CRT/kcrt.c10
3 files changed, 147 insertions, 3 deletions
diff --git a/CRT/DriverThread.cpp b/CRT/DriverThread.cpp
index efc4023..e2bb2ec 100644
--- a/CRT/DriverThread.cpp
+++ b/CRT/DriverThread.cpp
@@ -25,6 +25,11 @@ NTSTATUS DriverThread::Thread::Start(threadRoutine_t routine, PVOID threadContex
NTSTATUS status;
LockGuard lock(m_mutex);
+ if (m_threadObject != nullptr)
+ {
+ return STATUS_UNSUCCESSFUL;
+ }
+
m_routine = routine;
m_threadContext = threadContext;
status = PsCreateSystemThread(&threadHandle, (ACCESS_MASK)0, NULL, (HANDLE)0, NULL, InterceptorThreadRoutine, this);
@@ -34,7 +39,14 @@ NTSTATUS DriverThread::Thread::Start(threadRoutine_t routine, PVOID threadContex
return status;
}
- ObReferenceObjectByHandle(threadHandle, THREAD_ALL_ACCESS, NULL, KernelMode, (PVOID *)&m_threadObject, NULL);
+ status =
+ ObReferenceObjectByHandle(threadHandle, THREAD_ALL_ACCESS, NULL, KernelMode, (PVOID *)&m_threadObject, NULL);
+
+ if (!NT_SUCCESS(status))
+ {
+ return status;
+ }
+
return ZwClose(threadHandle);
}
@@ -44,6 +56,7 @@ NTSTATUS DriverThread::Thread::WaitForTermination(LONGLONG timeout)
{
return STATUS_UNSUCCESSFUL;
}
+
LockGuard lock(m_mutex);
if (m_threadObject == nullptr)
{
@@ -135,3 +148,107 @@ DriverThread::LockGuard::~LockGuard(void)
{
m_Lock.Unlock();
}
+
+// WorkQueue
+
+DriverThread::WorkQueue::WorkQueue(void) : m_worker()
+{
+ InitializeSListHead(&m_work);
+ KeInitializeEvent(&m_wakeEvent, SynchronizationEvent, FALSE);
+ m_stopWorker = FALSE;
+}
+
+DriverThread::WorkQueue::~WorkQueue(void)
+{
+ Stop();
+}
+
+NTSTATUS DriverThread::WorkQueue::Start(workerRoutine_t workerRoutine)
+{
+ NTSTATUS status;
+
+ {
+ LockGuard lock(m_mutex);
+ m_workerRoutine = workerRoutine;
+ status = m_worker.Start(WorkerInterceptorRoutine, this);
+ }
+
+ if (!NT_SUCCESS(status) && status != STATUS_UNSUCCESSFUL)
+ {
+ Stop();
+ }
+
+ return status;
+}
+
+void DriverThread::WorkQueue::Stop(void)
+{
+ LockGuard lock(m_mutex);
+ if (m_stopWorker == TRUE)
+ {
+ return;
+ }
+ m_stopWorker = TRUE;
+ KeSetEvent(&m_wakeEvent, 0, FALSE);
+}
+
+void DriverThread::WorkQueue::Enqueue(PSLIST_ENTRY workItem)
+{
+ if (InterlockedPushEntrySList(&m_work, workItem) == NULL)
+ {
+ // Work queue was empty. So, signal the work queue event in case the
+ // worker thread is waiting on the event for more operations.
+ KeSetEvent(&m_wakeEvent, 0, FALSE);
+ }
+}
+
+NTSTATUS DriverThread::WorkQueue::WorkerInterceptorRoutine(PVOID workerContext)
+{
+ DriverThread::WorkQueue * wq = (DriverThread::WorkQueue *)workerContext;
+ PSLIST_ENTRY listEntryRev, listEntry, next;
+
+ PAGED_CODE();
+
+ for (;;)
+ {
+ // Flush all the queued operations into a local list
+ listEntryRev = InterlockedFlushSList(&wq->m_work);
+
+ if (listEntryRev == NULL)
+ {
+
+ // There's no work to do. If we are allowed to stop, then stop.
+ if (wq->m_stopWorker == TRUE)
+ {
+ break;
+ }
+
+ // Otherwise, wait for more operations to be enqueued.
+ KeWaitForSingleObject(&wq->m_wakeEvent, Executive, KernelMode, FALSE, 0);
+ continue;
+ }
+
+ // Need to reverse the flushed list in order to preserve the FIFO order
+ listEntry = NULL;
+ while (listEntryRev != NULL)
+ {
+ next = listEntryRev->Next;
+ listEntryRev->Next = listEntry;
+ listEntry = listEntryRev;
+ listEntryRev = next;
+ }
+
+ // Now process the correctly ordered list of operations one by one
+ while (listEntry)
+ {
+ PSLIST_ENTRY arg = listEntry;
+ listEntry = listEntry->Next;
+ if (wq->m_workerRoutine(arg) != STATUS_SUCCESS)
+ {
+ wq->m_stopWorker = TRUE;
+ }
+ }
+ }
+
+ return STATUS_SUCCESS;
+}
diff --git a/CRT/DriverThread.hpp b/CRT/DriverThread.hpp
index d00db1b..6ae9d2c 100644
--- a/CRT/DriverThread.hpp
+++ b/CRT/DriverThread.hpp
@@ -6,6 +6,7 @@
extern "C" void InterceptorThreadRoutine(PVOID threadContext);
typedef NTSTATUS (*threadRoutine_t)(PVOID);
+typedef NTSTATUS (*workerRoutine_t)(PSLIST_ENTRY);
namespace DriverThread
{
@@ -78,6 +79,26 @@ private:
KSEMAPHORE m_semaphore;
};
+class WorkQueue
+{
+public:
+ WorkQueue(void);
+ ~WorkQueue(void);
+ NTSTATUS Start(workerRoutine_t workerRoutine);
+ void Stop(void);
+ void Enqueue(PSLIST_ENTRY workItem);
+
+private:
+ Mutex m_mutex;
+ SLIST_HEADER m_work;
+ KEVENT m_wakeEvent;
+ BOOLEAN m_stopWorker; // Work LIST must be empty and StopWorker TRUE to be able to stop!
+ Thread m_worker;
+ workerRoutine_t m_workerRoutine;
+
+ static NTSTATUS WorkerInterceptorRoutine(PVOID workerContext);
+};
+
}; // namespace DriverThread
#endif
diff --git a/CRT/kcrt.c b/CRT/kcrt.c
index fff767d..84b7f25 100644
--- a/CRT/kcrt.c
+++ b/CRT/kcrt.c
@@ -112,6 +112,11 @@ void * __cdecl malloc(size_t size)
return NULL;
}
+void * __cdecl calloc(size_t nmemb, size_t size)
+{
+ return malloc(nmemb * size);
+}
+
void * __cdecl realloc(void * ptr, size_t new_size)
{
if (!ptr)
@@ -181,7 +186,8 @@ void __cdecl __cxa_pure_virtual(void)
float __cdecl ceilf(float x)
{
- union {
+ union
+ {
float f;
UINT32 i;
} u = {x};
@@ -215,7 +221,7 @@ float __cdecl ceilf(float x)
static void __cdecl __ctors(void)
{
- unsigned long long int const * const * const l = (unsigned long long int const * const * const)&__CTOR_LIST__;
+ unsigned long long int const * const * const l = (unsigned long long int const * const * const) & __CTOR_LIST__;
unsigned long long int i = (unsigned long long int)*l;
init_and_deinit_fn const * p;