A utility for downloading and packaging the Microsoft CRT headers and libraries, and Windows SDK headers and libraries needed for compiling and linking programs targeting Windows.
The goal of this project is to create a root directory for both the CRT and Windows SDK that each contain all of the necessary includes and libraries needed for an application to compile and link from a non-Windows platform, using a native cross compiling toolchain like clang/LLVM. This includes adding symlinks to correct numerous casing issues in the Windows SDK so that the files generated by this program can function on a case-sensitive file system.
See this blog post for an in depth walk-through of how xwin can be used.
cargo install xwin --locked
xwin
provides two feature toggles used to decide which TLS implementation to use
rustls
(default) - Usesrustls
for TLSnative-tls
- Usesnative-tls
for TLS. Note that on platforms where OpenSSL is used it is always built from source.
You can download a prebuilt binary from the Releases.
x86_64-unknown-linux-musl
x86_64-apple-darwin
aarch64-apple-darwin
--accept-license
- Doesn't display the prompt to accept the license. You can also set theXWIN_ACCEPT_LICENSE=1
environment variable--arch <arch>
- The architectures to include [default:x86_64
] [possible values:x86
,x86_64
,aarch
,aarch64
]. Note that I haven't fully tested aarch/64 nor x86 so there might be issues with them, please file an issue if you encounter problems with them.--cache-dir <cache-dir>
- Specifies the cache directory used to persist downloaded items to disk. Defaults to./.xwin-cache
if not specified.-L, --log-level <level>
- The log level for messages, only log messages at or above the level will be emitted [default: info] [possible values: off, error, warn, info, debug, trace].--variant <variant>...
- The variants to include [default: desktop] [possible values: desktop, onecore, spectre]. Note that I haven't fully tested any variant exceptdesktop
, please file an issue if you try to use one of the others and run into issues. Note that there is anotherstore
variant that hasn't even been implemented due to it being weird and me not having a real project targeting it.--channel <channel>
- The product channel to use [default: release]--manifest-version <version>
- The manifest version to retrieve [default: 17].--manifest
- Specifies a top level manifest to use, rather than downloading it from Microsoft. This can be used to ensure the output is reproducible.--sdk-version
- The specific SDK version to use. If not specified the latest SDK version in the manifest is used.--crt-version
- The specific CRT version to use. If not specified the latest CRT version in the manifest is used.--timeout
- Specifies a timeout for long a single HTTP get request is allowed to take. The default is 60s.
https_proxy
- Environment variable that specifies the HTTPS proxy to use.
This downloads the top level manifest and any vsix, msi, or cab files that are needed that aren't already in the download cache.
Decompresses all of the downloaded package contents to disk. download
is run automatically.
Fixes the packages to prune unneeded files and adds symlinks to address file casing issues and then spalts the final artifacts into directories. This is the main command you will want to run as it also download
s and unpack
s automatically, providing the desired headers at the path specified to --output
(./.xwin-cache/splat
).
--copy
- Copies files from the unpack directory to the splat directory instead of moving them, which preserves the original unpack directories but increases overall execution time and disk usage.--disable-symlinks
- By default, symlinks are added to both the CRT andWindowsSDK
to address casing issues in general usage. For example, if you are compiling C/C++ code that does#include <windows.h>
, it will break on a case-sensitive file system, as the actual path in theWindowsSDK
isWindows.h
. This also applies even if the C/C++ you are compiling uses correct casing for all CRT/SDK includes, as the internal headers also use incorrect casing in most cases--include-debug-libs
- The MSVCRT includes (non-redistributable) debug versions of the various libs that are generally uninteresting to keep for most usage--include-debug-symbols
- The MSVCRT includes PDB (debug symbols) files for several of the libraries that are generally uninteresting to keep for most usage--preserve-ms-arch-notation
- By default, we convert the MS specificx64
,arm
, andarm64
target architectures to the more canonicalx86_64
,aarch
, andaarch64
of LLVM etc when creating directories/names. Passing this flag will preserve the MS names for those targets--use-winsysroot-style
- Use the /winsysroot layout, so that clang-cl's /winsysroot flag can be used with the output, rather than needing both -vctoolsdir and -winsdkdir. You will likely also want to use --preserve-ms-arch-notation and --disable-symlinks for use with clang-cl on Windows.--output
- The root output directory. Defaults to./.xwin-cache/splat
if not specified--map
- An optional map file used to configure what files are splatted, and any additional symlinks to create.
This moves all of the unpacked files which aren't pruned to their canonical locations under a root directory, for example here is what an x86_64
Desktop
splat looks like. unpack
is run automatically as needed.
.xwin-cache/splat
├── crt
│ ├── include
│ │ ├── cliext
│ │ ├── CodeAnalysis
│ │ ├── cvt
│ │ ├── experimental
│ │ ├── Manifest
│ │ └── msclr
│ │ └── com
│ └── lib
│ └── x86_64
└── sdk
├── include
│ ├── cppwinrt
│ │ └── winrt
│ │ └── impl
│ ├── shared
│ │ ├── ndis
│ │ └── netcx
│ │ └── shared
│ │ └── net
│ │ └── wifi
│ ├── ucrt
│ │ └── sys
│ ├── um
│ │ ├── alljoyn_c
│ │ ├── gl
│ │ ├── qcc
│ │ │ └── windows
│ │ └── winsqlite
│ └── winrt
│ └── wrl
│ └── wrappers
└── lib
├── ucrt
│ └── x86_64
└── um
└── x86_64
This is an advanced command that performs a splat
before performing a build on a cargo manifest using strace to capture all of the headers and libraries that are used throughout the build and dumping them to a map. This command can also output the final splat to disk, or the map file can be used with splat
to only splat the files and symlinks described in it.
Note that currently the build is always done with the /vctoolsdir
and /winsdkdir
options, so it is expected these are the same options used when compiling C/C++ code in your normal environment. If that is not the case please open an issue.
At the end of the command, a printout of the amount and size of the original versus minimized files is done, eg.
crt headers: 73(2.6MiB) / 384(18.4MiB) => 14.00%
crt libs: 5(28.0MiB) / 26(81.1MiB) => 34.58%
sdk headers: 180(9.6MiB) / 4435(304.7MiB) => 3.15%
sdk libs: 29(69.8MiB) / 456(169.9MiB) => 41.06%
- Linux host - This might work on other platforms, but it's not guaranteed, nor tested
cargo
- This is the singular supported build tool.<arch>-pc-windows-msvc
- The target you are building for needs to be installed (eg. viarustup target add
)clang-cl
- This is used as the C/C++ compilerllvm-lib
- This is used as the archiverlld-link
- This is used as the linkerstrace
- This is used to capture the syscalls made by the lld and clang compiler
Note all of the splat options also apply to minimize.
--map
- The path to the map to output the minimized results to. Default to./.xwin-cache/xwin-map.toml
if not specified.--minimize-output
- The root directory where only the minimized files are splatted to. If not specified only the--map
file is written in addition to the normal splat--preserve-strace
- By default thestrace
output is written to disk in a temporary location that is deleted once the build is finished, passing this option allows it to be persisted. The path is written out before the build starts.
As noted in minimize, there are many restrictions on it to make my life easier, but that make it unsuitable for those who don't use cargo/rust. It's possible for others to come up with their own versions of minimize that can output the same format that splat
understands to still get the benefits of xwin
without cargo/rust.
The format is extremely simple
├── crt
│ ├── headers
│ │ ├── filter - Array of relative paths to keep
│ │ └── symlinks
│ │ └── <path> - The same path as one of the filters
│ │ └── <names> - Array of symlinks to create in the same directory as the parent path
│ ├── libs *
└── sdk *
See docs/example-map.toml for a real world example.
xwin.dockerfile is an example Dockerfile that can be used a container image capable of building and testing Rust crates targeting x86_64-pc-windows-msvc
.
Special thanks to https://github.com/mstorsjo/msvc-wine for the inspiration and @mdsteele for publishing several Rust crates around msi/cab files that were needed in this project
This contribution is dual licensed under EITHER OF
- Apache License, Version 2.0, (LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
- MIT license (LICENSE-MIT or http://opensource.org/licenses/MIT)
at your option.