Skip to content

ntoskrnl7/crtsys

Repository files navigation

crtsys

Familiar MSVC C++ runtime and STL experience for Windows kernel drivers (.sys).

CMake GitHub GitHub release Windows 7+ Visual Studio 2017+ CMake 3.14+ C++14+

Korean documentation

crtsys brings MSVC CRT/STL/VCRT/UCRT source paths into Windows kernel drivers. Driver code keeps familiar MSVC C++ headers and STL types while runtime dependencies are mapped onto a kernel-mode substrate with explicit driver-test coverage and IRQL contracts.

Listed coverage means "verified by driver tests". It is not an exhaustive support ceiling for every header or code path that may compile or work.

Quick Start

Path Use when Start here
NuGet / MSBuild Visual Studio or Build Tools WDK driver project PackageReference or Install-Package crtsys
CMake prebuilt Offline or pinned CI dependency find_package(crtsys CONFIG REQUIRED)
CMake / CPM CMake-based driver project that consumes crtsys from GitHub CPMAddPackage("gh:ntoskrnl7/crtsys@<version>")

Minimal MSBuild/NuGet consumer:

<ItemGroup>
  <PackageReference Include="crtsys" Version="<version>" />
</ItemGroup>
msbuild .\my_driver.vcxproj /restore /p:Configuration=Debug /p:Platform=x64

For Visual Studio Package Manager Console:

Install-Package crtsys

nuget.exe is optional for modern PackageReference projects when MSBuild restore is available. Build Tools-only environments can use the same msbuild /restore path. See the MSBuild/NuGet quick start.

Minimal CMake/CPM consumer in a separate driver project:

Add CPM.cmake to that driver project, or use your existing CPM bootstrap:

New-Item -ItemType Directory -Force cmake
Invoke-WebRequest `
  https://github.com/cpm-cmake/CPM.cmake/releases/download/v0.32.0/CPM.cmake `
  -OutFile cmake/CPM.cmake

Then consume crtsys from GitHub in the driver's CMakeLists.txt:

include("${CMAKE_CURRENT_LIST_DIR}/cmake/CPM.cmake")

set(CRTSYS_NTL_MAIN ON)
CPMAddPackage("gh:ntoskrnl7/crtsys@<version>")
include(${crtsys_SOURCE_DIR}/cmake/CrtSys.cmake)

crtsys_add_driver(my_driver src/main.cpp)

With CRTSYS_NTL_MAIN, driver code can use the C++ entry wrapper:

#include <ntl/driver>

ntl::status ntl::main(ntl::driver& driver,
                      const std::wstring& registry_path) {
  driver.on_unload([registry_path]() {
    // driver cleanup
  });

  return ntl::status::ok();
}

Runtime Stack

flowchart TD
    Driver["Driver code (.sys)<br/>MSVC headers + optional NTL"]
    Runtime["MSVC CRT / UCRT / STL / VCRT<br/>source paths"]
    CrtSys["crtsys compatibility layer<br/>runtime adapters + ABI helpers + tested contracts"]
    LDK["LDK substrate<br/>Windows / NTDLL API + ICU ABI"]
    Kernel["WDK / NT kernel primitives"]

    Driver --> Runtime --> CrtSys --> LDK --> Kernel
Loading

Capability Map

Surface Driver-facing result
C++ runtime static init, EH/SEH, RTTI, ABI
CRT/UCRT STL dependencies, math, char conversion
STL containers, ranges, filesystem, format/print, regex, locale, chrono, threading, atomics, PMR, streams, random
Substrate crtsys adapters + LDK Windows/NTDLL/ICU
Evidence driver-run matrix + cppreference + IRQL contracts
Packaging NuGet/MSBuild + prebuilt bundle + CPM.cmake

Feature Highlights

Feature Status Notes
C++ exceptions Driver-tested throw, try/catch, function-try-block, std::exception_ptr
SEH handling Driver-tested C++ helper path for __try / __except boundary handling
Static initialization Driver-tested non-local, dynamic, and MSVC function-local static initialization
RTTI Driver-tested typeid, dynamic_cast
STL containers / algorithms Driver-tested containers, algorithms, ranges, smart pointers, PMR, utility
std::format / std::print Driver-tested formatted string/output paths
std::regex Driver-tested regular expression path
std::filesystem Driver-tested path, directory, copy, metadata, time, link-oriented paths covered by the matrix
Concurrency Driver-tested thread, synchronization, async/future, atomic wait/notify
Locale / chrono / charconv Driver-tested locale facets, timezone/chrono paths, integer and floating char conversion
NTL driver helpers Driver-tested ntl::main, driver/device helpers, RPC, IRQL helpers, stack expansion
thread_local Not true TLS Compiler TLS is not exposed as true per-thread kernel TLS

The detailed matrix is intentionally test-linked: it records features exercised by the kernel driver test suite, not the full set of headers or code paths that may compile or work.

Documentation

Document Use it for
Architecture Runtime stack, layer responsibilities, consumer paths
MSBuild/NuGet Quick Start Visual Studio, Build Tools-only, and CI package consumption
Design Rationale IRQL, pool, stack, unload, and operational boundaries
Feature Coverage Driver-tested C++/CRT/STL matrix and known gaps
NTL API Driver helper APIs, entry wrapper, synchronization, SEH helper
Usage Examples Small driver-side NTL examples
CI Driver Load Tests Optional self-hosted driver load/run workflow

Operational Boundaries

Boundary Policy
Driver model The driver remains a normal WDK driver. Verifier, HVCI, unload safety, target OS validation, and paging rules still matter.
IRQL Runtime-backed C++/CRT/STL paths are PASSIVE_LEVEL unless a specific API documents a wider contract.
Stack Kernel stacks are small; use ntl::expand_stack for exception-heavy or STL-heavy paths.
TLS MSVC function-local statics are supported. General C++ thread_local is not true per-thread TLS.
Toolchain Use matching SDK/WDK versions. Use WDK 23H2 or older for x86 kernel-mode targets.

Requirements

  • Windows 7 or later
  • Visual Studio or Build Tools 2017 or later
  • Windows SDK and WDK compatible with the selected Visual Studio toolset
  • CMake 3.14 or later
  • Git

Tested toolchains include Visual Studio 2017, 2019, 2022, and 2026 with WDK/SDK versions such as 10.0.17763.0, 10.0.18362.0, 10.0.22000.0, 10.0.22621.0, and 10.0.26100.0.

Visual Studio 2017 has missing CRT source/header pieces for some paths, so crtsys uses selected UCXXRT compatibility code for that toolset.

CMake Quick Start

Create a separate driver project, add CPM.cmake to that project, and consume crtsys from GitHub:

New-Item -ItemType Directory -Force cmake
Invoke-WebRequest `
  https://github.com/cpm-cmake/CPM.cmake/releases/download/v0.32.0/CPM.cmake `
  -OutFile cmake/CPM.cmake
cmake_minimum_required(VERSION 3.14 FATAL_ERROR)

project(my_driver LANGUAGES C CXX)

include("${CMAKE_CURRENT_LIST_DIR}/cmake/CPM.cmake")

set(CRTSYS_NTL_MAIN ON)
CPMAddPackage("gh:ntoskrnl7/crtsys@<version>")
include(${crtsys_SOURCE_DIR}/cmake/CrtSys.cmake)

crtsys_add_driver(my_driver src/main.cpp)

CRTSYS_NTL_MAIN enables the C++ entry point wrapper. With it enabled, define ntl::main instead of writing DriverEntry directly:

#include <iostream>
#include <string>
#include <ntl/driver>

ntl::status ntl::main(ntl::driver& driver,
                      const std::wstring& registry_path) {
  std::wcout << L"load: " << registry_path << L"\n";

  driver.on_unload([registry_path]() {
    std::wcout << L"unload: " << registry_path << L"\n";
  });

  return ntl::status::ok();
}

If CRTSYS_NTL_MAIN is disabled, keep the normal WDK DriverEntry entry point and initialize your driver manually.

Build the project with a Visual Studio generator:

cmake -S . -B build_x64 -A x64
cmake --build build_x64 --config Debug

crtsys keeps diagnostic KdBreakPoint() calls enabled by default. To build without those diagnostic breakpoints, configure with:

cmake -S . -B build_x64 -A x64 -DCRTSYS_ENABLE_DIAGNOSTIC_BREAKPOINTS=OFF

NuGet Package Details

crtsys publishes a NuGet package with native MSBuild imports and prebuilt driver libraries for x86, x64, ARM, and ARM64 Debug/Release on v142/v143, and x86, x64, and ARM64 on v145. The package workflow builds WDK consumer projects for every packaged architecture that the selected toolset supports. The checked-in smoke projects live under test/nuget.

The NuGet distribution is crtsys.<version>.nupkg for Visual Studio/MSBuild projects.

GitHub Release Prebuilt Bundle Details

GitHub Release publishes these offline-only assets:

  • crtsys-<version>-prebuilt.zip: headers, docs, CMake helpers, and prebuilt x86/x64/ARM/ARM64 Debug/Release libraries under an MSVC toolset-specific layout, with ARM omitted for v145.
  • crtsys-<version>-SHA256SUMS.txt

The prebuilt bundle is intended for CMake projects that want a checked-in or cached runtime package instead of fetching and building crtsys from source.

For full packaging and publishing command details, see nuget/README.md.

CMake Install

CMake consumers can install a local CMake package:

cmake -S . -B build_x64 -A x64 -DCMAKE_INSTALL_PREFIX=%CD%\artifacts\install\crtsys
cmake --build build_x64 --config Release --target crtsys
cmake --install build_x64 --config Release

Installed consumers can then use the package config:

find_package(crtsys CONFIG REQUIRED PATHS path/to/install-prefix)
crtsys_add_driver(my_driver src/main.cpp)

The install tree uses the same native library layout as the prebuilt release bundle: lib/native/<toolset>/<arch>/<config>, for example lib/native/v143/x64/Release or lib/native/v145/x64/Release.

The install flow can be smoke-tested with:

.\scripts\cmake\Test-CrtSysInstall.ps1 -Architecture x64 -Configuration Release

To publish a new version from main:

.\scripts\release\Prepare-CrtSysRelease.ps1 -Version <version> -Push

The helper updates include/.internal/version, commits the version bump, creates the matching v<version> tag, and pushes both the commit and tag. The tag push starts the Package workflow.

The same flow is also available from the GitHub UI: open Actions, select Release, choose Run workflow, and enter the release version. The workflow creates the version bump commit and tag, then dispatches the Package workflow for that tag. If branch protection blocks direct pushes to main, use the local helper or adjust the release rule first.

Building This Repository

Clone the repository and build the test app and driver for the host architecture:

git clone https://github.com/ntoskrnl7/crtsys
cd crtsys
test\build.bat

Build a specific target manually:

build.bat test\cmake\app x64 Debug
build.bat test\cmake\driver x64 Debug
build.bat test\cmake\app x64 Release
build.bat test\cmake\driver x64 Release

Build all supported architecture/configuration combinations:

build_all.bat test\cmake\app
build_all.bat test\cmake\driver

build_all.bat runs builds sequentially and returns the first failing exit code. Pass Debug or Release as the second argument to build only one configuration.

Typical Debug outputs:

test\cmake\driver\build_x64\Debug\crtsys_test.sys
test\cmake\app\build_x64\Debug\crtsys_test_app.exe

Running Tests

crtsys_test.sys is a kernel driver. Build validation can happen in CI, but loading and executing the test driver must happen in a Windows driver test environment.

The CI build workflow and optional self-hosted driver load test path are documented in CI driver load tests.

sc create CrtSysTest binpath= "C:\path\to\crtsys_test.sys" displayname= "crtsys test" start= demand type= kernel
sc start CrtSysTest

C:\path\to\crtsys_test_app.exe

sc stop CrtSysTest
sc delete CrtSysTest

The test driver uses Google Test internally. Inspect output with DebugView, WinDbg, or your normal kernel debugging setup.

Repository Layout

cmake/             CMake helpers, including CrtSys.cmake
include/ntl/       NTL C++ helper headers
include/.internal/ Internal version and toolchain compatibility headers
src/               crtsys runtime and CRT/STL compatibility code
test/cmake/app/    CMake user-mode test companion application
test/cmake/driver/ CMake kernel-mode test driver
test/nuget/        Visual Studio WDK NuGet consumer test project
docs/              Additional documentation

Background

crtsys was created after experimenting with other kernel C++ runtime projects, especially UCXXRT and KTL. The design goal is to keep the CMake/WDK workflow practical while supporting a substantial Microsoft CRT/STL surface for real driver experiments.

The project avoids treating the Microsoft CRT/STL source as a vendored library. Instead, it relies on the locally installed Visual Studio/Build Tools layout and layers kernel-mode compatibility code around it. For older toolsets where the Microsoft-provided source/header layout is incomplete, small compatibility pieces are used.

Several standalone implementations are also referenced where they are a better fit for kernel-mode support:

Roadmap

  • Expand driver-tested C++ and STL coverage, including investigation of true thread_local storage.
  • Reduce Visual Studio 2017 compatibility gaps and keep toolset-specific compatibility code smaller.
  • Broaden real driver load/run CI coverage where suitable test environments are available.