Skip to content

Commit

Permalink
Merge pull request #332 from ignatandrei/Microsoft.Windows.CsWin32
Browse files Browse the repository at this point in the history
  • Loading branch information
ignatandrei authored Dec 2, 2024
2 parents ad2f969 + 49d7769 commit 9d20748
Show file tree
Hide file tree
Showing 27 changed files with 1,074 additions and 265 deletions.
230 changes: 123 additions & 107 deletions README.md

Large diffs are not rendered by default.

2 changes: 2 additions & 0 deletions v2/Generator/MultiGeneratorV2.cs
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,8 @@ public string[] SourceNoRSCG()
if (File.Exists(nameFile))
{
var text=await File.ReadAllTextAsync(nameFile);
text = text.Replace("(src/", $"({d.Generator!.Source}/src/");
text = text.Replace("(doc/", $"({d.Generator!.Source}/doc/");
text = text.Replace("(docs/rules/", $"({d.Generator!.Source}/docs/rules/");
text = text.Replace("(CHANGELOG.md", $"({d.Generator!.Source}/CHANGELOG.md");

Expand Down
1 change: 1 addition & 0 deletions v2/GeneratorData/Category.cs
Original file line number Diff line number Diff line change
Expand Up @@ -30,5 +30,6 @@ public enum Category
Actor,
Mediator=26,
SignalR=27,
WinAPI = 28,
}

6 changes: 6 additions & 0 deletions v2/RSCGExamplesData/GeneratorDataRec.json
Original file line number Diff line number Diff line change
Expand Up @@ -1035,5 +1035,11 @@
"dtStart": "2024-11-09T00:00:00",
"show": true

},
{
"ID":"Microsoft.Windows.CsWin32",
"Category": 28,
"dtStart": "2024-12-01T00:00:00",
"show": true
}
]
5 changes: 0 additions & 5 deletions v2/RSCGExamplesData/NoExample.json
Original file line number Diff line number Diff line change
Expand Up @@ -679,11 +679,6 @@
"name": "ManagedDotnetProfiler https://github.com/kevingosse/ManagedDotnetProfiler",
"why": "too complicated for me, need help"
},
{
"ID": 139,
"name": "CsWin32 https://github.com/microsoft/CsWin32",
"why": "old ISourceGenerator"
},
{
"ID": 140,
"name": "EnvVariablesGenerator https://github.com/KAW0/EnvVariablesGenerator",
Expand Down
64 changes: 64 additions & 0 deletions v2/book/examples/Microsoft.Windows.CsWin32.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,64 @@

<h1>RSCG nr 173 : Microsoft.Windows.CsWin32</h1>

<h2>Info</h2>
Nuget : <a href="https://www.nuget.org/packages/Microsoft.Windows.CsWin32/" target="_blank">https://www.nuget.org/packages/Microsoft.Windows.CsWin32/</a>

<p>You can find more details at : <a href="https://github.com/microsoft/CsWin32" target="_blank"> https://github.com/microsoft/CsWin32</a></p>

<p>Author :Microsoft</p>

<p>Source : <a href="https://github.com/microsoft/CsWin32" target="_blank">https://github.com/microsoft/CsWin32</a> </p>

<h2>About</h2>

Generating WinAPI code in C#

<h2>
How to use
</h2>
<h3>
Add reference to the <a href="https://www.nuget.org/packages/Microsoft.Windows.CsWin32/" target="_blank">Microsoft.Windows.CsWin32</a> in the csproj
</h3>
<img src="images/Microsoft.Windows.CsWin32/WinAPI.csproj.png" width="580" height="580" />

<h3>This was for me the <b>starting</b> code</h3>

<br />
I have <b>coded</b> the file Program.cs
<br />
<img src="images/Microsoft.Windows.CsWin32/csFiles/Program.cs.png" width="580" height="580" />
<hr />

<br />
I have <b>coded</b> the file NativeMethods.txt
<br />
<img src="images/Microsoft.Windows.CsWin32/csFiles/NativeMethods.txt.png" width="580" height="580" />
<hr />
<h3>And here are the <i>generated</i> files</h3>

<br />
The file <i>generated</i> is Windows.Win32.CsWin32Stamp.g.cs
<br />
<img src="images/Microsoft.Windows.CsWin32/generated/Windows.Win32.CsWin32Stamp.g.cs.png" width="580" height="580" />

<br />
The file <i>generated</i> is Windows.Win32.PInvoke.KERNEL32.dll.g.cs
<br />
<img src="images/Microsoft.Windows.CsWin32/generated/Windows.Win32.PInvoke.KERNEL32.dll.g.cs.png" width="580" height="580" />

<p>
You can download the code and this page as pdf from
<a target="_blank" href='https://ignatandrei.github.io/RSCG_Examples/v2/docs/Microsoft.Windows.CsWin32'>
https://ignatandrei.github.io/RSCG_Examples/v2/docs/Microsoft.Windows.CsWin32
</a>
</p>


<p>
You can see the whole list at
<a target="_blank" href='https://ignatandrei.github.io/RSCG_Examples/v2/docs/List-of-RSCG'>
https://ignatandrei.github.io/RSCG_Examples/v2/docs/List-of-RSCG
</a>
</p>

6 changes: 5 additions & 1 deletion v2/book/list.html
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
</head>
<body>
<h1>
This is the list of 172 RSCG with examples =>
This is the list of 173 RSCG with examples =>
</h1>

<table >
Expand Down Expand Up @@ -714,6 +714,10 @@ <h1>
<td>172</td>
<td><a href="examples/GoLive.Generator.BlazorInterop.html">GoLive.Generator.BlazorInterop</a></td>
</tr>
<tr>
<td>173</td>
<td><a href="examples/Microsoft.Windows.CsWin32.html">Microsoft.Windows.CsWin32</a></td>
</tr>
</table>


Expand Down
1 change: 1 addition & 0 deletions v2/book/pandocHTML.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,7 @@ input-files:
- examples/QueryStringGenerator.html
- examples/Hsu.Sg.FluentMember.html
- examples/GoLive.Generator.BlazorInterop.html
- examples/Microsoft.Windows.CsWin32.html

# or you may use input-file: with a single value
# defaults:
Expand Down
22 changes: 22 additions & 0 deletions v2/rscg_examples/Microsoft.Windows.CsWin32/description.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
{
"generator":{
"name":"Microsoft.Windows.CsWin32",
"nuget":[
"https://www.nuget.org/packages/Microsoft.Windows.CsWin32/"
],
"link":"https://github.com/microsoft/CsWin32",
"author":"Microsoft",
"source":"https://github.com/microsoft/CsWin32"
},
"data":{
"goodFor":["Generating WinAPI code in C#"],
"csprojDemo":"WinAPI.csproj",
"csFiles":["Program.cs","NativeMethods.txt"],
"excludeDirectoryGenerated":[""],
"includeAdditionalFiles":[""]
},
"links":{
"blog":"",
"video":""
}
}
1 change: 1 addition & 0 deletions v2/rscg_examples/Microsoft.Windows.CsWin32/nuget.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Microsoft Windows SDK Win32 API Source Generator
164 changes: 164 additions & 0 deletions v2/rscg_examples/Microsoft.Windows.CsWin32/readme.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,164 @@
# C#/Win32 P/Invoke Source Generator

***A source generator to add a user-defined set of Win32 P/Invoke methods and supporting types to a C# project.***

[![NuGet (prerelease)](https://img.shields.io/nuget/vpre/Microsoft.Windows.CsWin32)](https://www.nuget.org/packages/Microsoft.Windows.CsWin32)
[![NuGet (daily)](https://img.shields.io/badge/nuget-daily-red)](https://dev.azure.com/azure-public/winsdk/_packaging?_a=package&feed=CI%40Local&package=Microsoft.Windows.CsWin32&protocolType=NuGet)

[![Build Status](https://dev.azure.com/azure-public/winsdk/_apis/build/status/microsoft.CsWin32?branchName=main)](https://dev.azure.com/azure-public/winsdk/_build/latest?definitionId=47&branchName=main)

## Features

* Rapidly add P/Invoke methods and supporting types to your C# project.
* No bulky assemblies to ship alongside your application.
* `SafeHandle`-types automatically generated.
* Generates xml documentation based on and links back to docs.microsoft.com

![Animation demonstrating p/invoke code generation](doc/demo.gif)

## Prerequisites

The .NET 6 SDK or Visual Studio 2019 Update 11 (16.11).

WPF projects have [additional requirements](https://github.com/microsoft/CsWin32/issues/7).

In addition, some generated code may require use of the C# 9+ language version (`<LangVersion>9</LangVersion>`) in your project file. See [issue #4](https://github.com/microsoft/CsWin32/issues/4) for more on this.
Newer is generally better. Use the latest C# language version for the best results, regardless of your TargetFramework.

## Usage

Install the `Microsoft.Windows.CsWin32` package:

```ps1
dotnet add package Microsoft.Windows.CsWin32 --prerelease
```

You should also install the `System.Memory` and `System.Runtime.CompilerServices.Unsafe` packages when targeting .NET Framework 4.5+ or .NET Standard 2.0,
as these add APIs that significantly improve much of the code generated by CsWin32:

```ps1
dotnet add package System.Memory
dotnet add package System.Runtime.CompilerServices.Unsafe
```

Projects targeting .NET Core 2.1+ or .NET 5+ do *not* need to add these package references,
although it is harmless to do so.

Note that while the `System.Memory` package depends on the `System.Runtime.CompilerServices.Unsafe` package,
referencing the latter directly is still important to get the latest version of the APIs it provides.

Your project must allow unsafe code to support the generated code that will likely use pointers.
This does *not* automatically make all your code *unsafe*.
Use of the `unsafe` keyword is required anywhere you use pointers.
The source generator NuGet package sets the default value of the `AllowUnsafeBlocks` property for your project to `true`,
but if you explicitly set it to `false` in your project file, generated code may produce compiler errors.

Create a `NativeMethods.txt` file in your project directory that lists the APIs to generate code for.
Each line may consist of *one* of the following:

* Exported method name (e.g. `CreateFile`). This *may* include the `A` or `W` suffix, where applicable. This *may* be qualified with a namespace but is only recommended in cases of ambiguity, which CsWin32 will prompt where appropriate.
* A macro name (e.g. `HRESULT_FROM_WIN32`). These are generated into the same class with extern methods. Macros must be hand-authored into CsWin32, so let us know if you want to see a macro added.
* A namespace to generate all APIs from (e.g. `Windows.Win32.Storage.FileSystem` would search the metadata for all APIs within that namespace and generate them).
* Module name followed by `.*` to generate all methods exported from that module (e.g. `Kernel32.*`).
* The name of a struct, enum, constant or interface to generate. This *may* be qualified with a namespace but is only recommended in cases of ambiguity, which CsWin32 will prompt where appropriate.
* A prefix shared by many constants, followed by `*`, to generate all constants that share that prefix (e.g. `ALG_SID_MD*`).
* A comment (i.e. any line starting with `//`) or white space line, which will be ignored.

When generating any type or member, all supporting types will also be generated.

Generated code is added directly in the compiler.
An IDE may make this generated code available to view through code navigation commands (e.g. Go to Definition) or a tree view of source files that include generated source files.

Assuming default settings and a `NativeMethods.txt` file with content that includes `CreateFile`, the P/Invoke methods can be found on the `Windows.Win32.PInvoke` class, like this:

```cs
using Windows.Win32;

PInvoke.CreateFile(/*args*/);
```

Constants are defined on the same class as the p/invoke methods (by default, the `Windows.Win32.PInvoke` class).

Other supporting types are defined within or under the `Windows.Win32` namespace.
Discovery of the namespace for a given type can be done with the Go To All feature (Ctrl+T) in Visual Studio with the type name as the search query.

A project may include many NativeMethods.txt files (each one necessarily in its own directory).
CsWin32 will read them all to generate APIs, provided these files are included as `AdditionalFiles` in the project.
A `NativeMethods.txt` file directly in the project directory is added automatically to `AdditionalFiles`.
Files in other directories must be added to the project file manually.

Whether API requests are all in a single NativeMethods.txt file or split across many makes no difference to the generated result.
We recommend using just one NativeMethods.txt file and keeping it sorted for easy bookkeeping.
Multiple files perhaps makes the most sense in a Shared Project scenario where several API requests will be common across many projects, so sharing a NativeMethods.txt file with those same projects that contain all the necessary APIs for the set of shared source files make maintenance easier.

Some APIs require targeting a specific architecture and are not available when your C# project compiles as "Any CPU".
Learn more about [how this manifests and what your options are](doc/ArchSpecificAPIs.md).

### Customizing generated code

Several aspects of the generated code can be customized, including:

* The name of the class(es) that declare p/invoke methods
* Whether to emit interop types as `public` or `internal`
* Whether to emit ANSI functions as well where Wide character functions also exist
* Set `PreserveSig` for COM interfaces or individual members
* Force generation of blittable structs, COM structs instead of interfaces (for super high performance with 0 GC pressure), etc.

To configure these settings, create a `NativeMethods.json` file in your project directory.
Specifying the `$schema` property that points to [the schema](src/Microsoft.Windows.CsWin32/settings.schema.json) adds completions, descriptions and validation in many JSON editors, and in fact is where all the documentation for the available settings is found.

```json
{
"$schema": "https://aka.ms/CsWin32.schema.json",
"emitSingleFile": false
}
```

Most generated types include the `partial` modifier so you can add your own members to that type within your code.

When you need to *replace* a generated type, simply copy and paste it from generated code into your own source files
and remove the `partial` modifier.
Be sure to keep the name and namespace exactly the same.
CsWin32 will notice that your project already declares the type and skip generating it, but generate everything else.
Note that if that type is the only thing that references some other generated type, CsWin32 will stop generating that type too.
To keep CsWin32 generating the referred types you need, add them explicitly to `NativeMethods.txt`.

### Support for trimming, AOT, and/or disabling the runtime marshaler

Newer .NET runtime versions may fail for CsWin32 generated code when the application project builds with one or both of these properties set:

```xml
<PublishAot>true</PublishAot>
<DisableRuntimeMarshalling>true</DisableRuntimeMarshalling>
<PublishTrimmed>true</PublishTrimmed>
```

CsWin32 supports these environments by avoiding code that relies on the runtime marshaler when the `allowMarshaling` setting is disabled in the `NativeMethods.json` file.
For example:

```json
{
"$schema": "https://aka.ms/CsWin32.schema.json",
"allowMarshaling": false
}
```

### Newer metadata

To update the metadata used as the source for code generation, you may install a newer `Microsoft.Windows.SDK.Win32Metadata` package:

```ps1
dotnet add package Microsoft.Windows.SDK.Win32Metadata --prerelease
```

CsWin32 also consumes the WDK from a similarly named package: `Microsoft.Windows.WDK.Win32Metadata`.

## Consuming daily builds

Can't wait for the next release to try out a bug fix? Follow these steps to consume directly from our daily build.

Just add this package feed to your nuget.config file:

```xml
<add key="winsdk" value="https://pkgs.dev.azure.com/azure-public/winsdk/_packaging/CI/nuget/v3/index.json" />
```
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@

{
"$schema": "https://aka.ms/codetour-schema",
"title": "Microsoft.Windows.CsWin32",
"steps":
[
{
"file": "WinAPI/WinAPI.csproj",
"description": "First, we add Nuget [Microsoft.Windows.CsWin32](https://www.nuget.org/packages/Microsoft.Windows.CsWin32/) in csproj ",
"pattern": "Microsoft.Windows.CsWin32"
}

,{
"file": "WinAPI/NativeMethods.txt",
"description": "File NativeMethods.txt ",
"pattern": "this is the code"
}

,{
"file": "WinAPI/Program.cs",
"description": "File Program.cs \r\n>> dotnet run --project WinAPI/WinAPI.csproj ",
"pattern": "this is the code"
}


,{
"file": "WinAPI/obj/GX/Microsoft.Windows.CsWin32/Microsoft.Windows.CsWin32.SourceGenerator/Windows.Win32.PInvoke.KERNEL32.dll.g.cs",
"description": "Generated File 2 from 2 : Windows.Win32.PInvoke.KERNEL32.dll.g.cs ",
"line": 1
}

,{
"file": "WinAPI/obj/GX/Microsoft.Windows.CsWin32/Microsoft.Windows.CsWin32.SourceGenerator/Windows.Win32.CsWin32Stamp.g.cs",
"description": "Generated File 1 from 2 : Windows.Win32.CsWin32Stamp.g.cs ",
"line": 1
}

],

"ref": "main"

}
22 changes: 22 additions & 0 deletions v2/rscg_examples/Microsoft.Windows.CsWin32/src/WinAPI.sln
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@

Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 17
VisualStudioVersion = 17.12.35514.174
MinimumVisualStudioVersion = 10.0.40219.1
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "WinAPI", "WinAPI\WinAPI.csproj", "{E6F63E3B-923A-4C40-9023-9C5FCDAB3FC0}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{E6F63E3B-923A-4C40-9023-9C5FCDAB3FC0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{E6F63E3B-923A-4C40-9023-9C5FCDAB3FC0}.Debug|Any CPU.Build.0 = Debug|Any CPU
{E6F63E3B-923A-4C40-9023-9C5FCDAB3FC0}.Release|Any CPU.ActiveCfg = Release|Any CPU
{E6F63E3B-923A-4C40-9023-9C5FCDAB3FC0}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
EndGlobal
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
GetTickCount
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@

Console.WriteLine("Hello, World!" + Windows.Win32.PInvoke.GetTickCount());
Loading

0 comments on commit 9d20748

Please sign in to comment.