Skip to content

Add codeview_annotation intrinsic#1

Open
gurry wants to merge 36 commits intomainfrom
add-codeview-annotation-intrinsic
Open

Add codeview_annotation intrinsic#1
gurry wants to merge 36 commits intomainfrom
add-codeview-annotation-intrinsic

Conversation

@gurry
Copy link
Copy Markdown
Owner

@gurry gurry commented Apr 13, 2026

This PR adds a new intrinsic called codeview_annotation which lowers to llvm.codeview.annotation and allows users to emit arbitrary S_ANNOTATION records into PDB files.

Motivation

In Microsoft we are working on supporting Windows driver development in Rust. codeview_annotation is needed to implement a feature called WPP tracing which requires strings from trace statements to be written to the PDB at compile time.

Basically the user writes a trace statement like this in the driver:

trace!("Bytes {}, duration {} ms", byte_count, elapsed_ms);

which expands to a call to codeview_annotation:

codeview_annotation!("Bytes {}, duration {} ms", "some other metadata");

and the string arguments get written to the PDB.

Later at runtime trace viewing tools extract those strings from the PDB, combine them with the stream of variables (e.g. byte_count and elapsed_ms) coming out of the driver and produce a human readable log.

This keeps tracing efficient by avoiding costly I/O of strings at runtime and also helps with not leaking implementation details of drivers.

The intrinsic could also be useful in general for other use cases where you need to embed some user-specific information in the PDB.

API

The intrinsic is declared as:

 pub fn codeview_annotation(strings: &[&str]);

and is publicly exposed through a macro with the same name.

The &[&str] parameter is required to be a const as it must be available at compile time in order to be lowered to llvm.codeview.annotation. Non-const arguments are rejected with a compiler error.

Supported Platforms

Only LLVM on msvc is supported. It is a no-op on other platforms.

Implementation

Calls to codeview_annotation are lowered to StatementKind::Intrinsic at MIR building time. The string arguments are extracted from THIR and stored in the newly introduced variant NonDivergingIntrinsic::CodeviewAnnotation(Box<[Symbol]>) which is ultimately lowered to llvm.codeview.annotation in the backend.

We have chosen to collect the string arguments at the time of THIR->MIR translation because it is easier to do so. The other option was to do it during codegen but that would have required walking chains of MIR locals backwards which was more complicated.

Future Work

Ideally we should have implemented this feature using const generics e.g.:

pub fn codeview_annotation<const STRINGS: &[&str]>();

as that guarantees the compile time availability of the inputs. However, we are blocked on adt_const_params and especially unsized_const_params. Whenever these features get stabilized we can update the implementation to make use of them.

@gurry gurry force-pushed the add-codeview-annotation-intrinsic branch from de65083 to 6655f74 Compare April 13, 2026 05:04
@gurry gurry force-pushed the add-codeview-annotation-intrinsic branch from d13beed to fbb804a Compare April 13, 2026 06:01
@gurry gurry changed the title Add codeview annotation intrinsic Add codeview_annotation intrinsic Apr 13, 2026
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

1 participant