The NuGet package Microsoft.Quantum.Sdk serves a .NET Core Sdk for developing quantum applications, meaning it acts as shared SDK Component. It contains the properties and targets that define the compilation process for Q# projects, tools used as part of the build, as well as some project system support for Q# files. It in particular also provides the support for executing a compilation step defined in a package or project reference as part of the compilation process. See this section for more details.
The Sdk includes all *.qs files within the project directory as well as the Q# standard libraries by default. No additional reference to Microsoft.Quantum.Standard
is needed. For more details see the section on defined properties below.
To use the Quantum Sdk simply list the NuGet package as Sdk at the top of your project file:
<Project Sdk="Microsoft.Quantum.Sdk/0.11.2006.2118-alpha">
...
</Project>
See also the MSBuild documentation.
If you would like to build the Microsoft.Quantum.Sdk NuGet package from source, first run the bootstrap.ps1
script in the root of this repository and then run the build/pack.ps1
script. You will find the built package in the generated drops
folder under nugets
. Simply add the package to your local NuGet folder to start using it.
Any project that uses the Quantum Sdk can easily incorporate custom compilation steps into the build process. Syntax tree rewrite steps defined in a package or project reference can be executed as part of the Q# compilation process by marking the package or project as qsc reference:
<ProjectReference Include="MyCustomCompilationStep.csproj" IsQscReference="true"/>
...
<PackageReference Include="MyCustom.QSharp.Compiler.Extensions" Version="1.0.0.0" IsQscReference="true"/>
Marking all assets as private by adding a PrivateAssets="All"
attribute is generally a good practice if the reference is a development only dependency, which is often the case for assemblies that implement rewrite steps.
A custom compilation step is defined by a class that implements the IRewriteStep interface. The output assembly of a project reference or any .NET Core library contained in a package reference marked as qsc reference is loaded during compilation and searched for classes implementing the IRewriteStep
interface. Any such class is instantiated using the default constructor, and the implemented transformation is executed.
The order in which these steps are executed can be configured by specifying their priority:
<ProjectReference Include="MyCustomCompilationStep.csproj" IsQscReference="true" Priority="2"/>
...
<PackageReference Include="MyCustom.QSharp.Compiler.Extensions" Version="1.0.0.0" IsQscReference="true" Priority="1"/>
If no priority is specified, the priority for that reference is set to zero.
Steps defined within packages or projects with higher priority are executed first. If several classes within a certain reference implement the IRewriteStep
interface, then these steps are executed according to their priority specified as part of the interface. The priority defined for the project or package reference takes precedence, such that the priorities defined by the interface property are not compared across assemblies.
An example for defining custom compilation steps in a referenced .NET Core project can be found here. See the this section for more detail on how to package a Q# compiler extension to distribute it as a NuGet package.
It is possible to inject C# code into Q# packages e.g. for integration purposes. By default the Sdk is configured to do just that, see also the section on defined properties. That code may be generated as part of a custom rewrite step, e.g. if the code generation requires information about the Q# compilation. The Sdk defines a couple of build targets that can be redefined to run certain tasks before important build stages. These targets do nothing by default and merely serve as handles to easily integrate into the build process.
The following such targets are currently available:
-
BeforeQSharpCompile
: The target will execute right before the Q# compiler is invoked. All assembly references and the paths to all qsc references will be resolved at that time. -
BeforeCSharpCompile
: The target will execute right before the C# compiler is invoked. All Q# compilation steps will have completed at that time, and in particular all rewrite steps will have been executed.
For example, if a qsc reference contains a rewrite step that generates C# code during transformation, that code can be included into the built dll by adding the following target to the project file:
<Target Name="BeforeCSharpCompile">
<ItemGroup>
<Compile Include="$(GeneratedFilesOutputPath)**/*.cs" Exclude="@(Compile)" AutoGen="true" />
</ItemGroup>
</Target>
In order to avoid a dependency of the Q# build targets on the Restore target, we require that NuGet packages containing Q# compiler extensions define a property that contains the path to the dll to load. This is done by including a file with MSBuild props in the package, following the instructions here.
The content of the file should be similar to the following, with Package_Name
being replace by the name of your package after replacing dots by underscore, and Package.Name
should be replaced by the assembly name of your package:
<?xml version="1.0" encoding="utf-8"?>
<Project xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
<PropertyGroup>
<QscRef_Package_Name>
$(MSBuildThisFileDirectory)/../lib/net6.0/Package.Name.dll
</QscRef_Package_Name>
</PropertyGroup>
</Project>
This example provides a template for packaging a Q# compiler extension.
If you develop a NuGet package to extend the Q# compilation process, we recommend to distribute it as a self-contained package to avoid issues due to references that could not be resolved. Each qsc reference is loaded into its own context to avoid issues when several references depend on different versions of the same package.
The compiler attempts to load rewrite steps even if these have been compiled against a different compiler version. While we do our best to mitigate issue due to a version mismatch, it is generally recommended to use compiler extensions that are compiled against the same compiler package version as the Sdk version of the project.
When a rewrite steps fails to execute, setting the QscVerbosity
to "Detailed" or "Diagnostic" will log the encountered exception with stack trace:
<QscVerbosity>Detailed</QscVerbosity>
A FileNotFoundException
will be raised if a compiler extension attempts to load a reference that either could not be found, or could not be loaded for other reasons.
By default, the compiler will search the project output directory for a suitable assembly in case a dependency cannot be found.
If such an exception occurs during a compilation step loaded from a package reference, the issue may be resolved by adding the package containing the missing dll to the project or by copying the missing dll to the output directory.
If such an exception occurs during a compilation step loaded from a project reference, issue may be resolved by defining the property
<CopyLocalLockFileAssemblies>true</CopyLocalLockFileAssemblies>
in the project that implements the compilation step.
The Sdk defines the following properties for each project using it:
-
QSharpLangVersion
: The version of the Q# language specification. -
QuantumSdkVersion
: The NuGet version of the Sdk package.
The following properties can be configured to customize the build:
-
AdditionalQscArguments
: May contain additional arguments to pass to the Q# command line compiler. Valid additional arguments are--emit-dll
, or--no-warn
followed by any number of integers specifying the warnings to ignore. -
CSharpGeneration
: Specifies whether to generate C# code as part of the compilation process. Setting this property to false may prevent certain interoperability features or integration with other pieces of the Quantum Development Kit. -
DefaultSimulator
: Specifies the simulator to use by default for execution. Valid values are QuantumSimulator, ToffoliSimulator, ResourcesEstimator, or the fully qualified name of a custom simulator. -
IncludeQSharpCorePackages
: Specifies whether the packages providing the basic language support for Q# are referenced. This property is set to true by default. If set to false, the Sdk will not reference any Q# libraries. -
IncludeProviderPackages
: Specifies whether the packages for specific hardware providers should be automatically included based on the specifiedExecutionTarget
. This property is set to true by default. If set to false, the Sdk will not automatically reference any provider packages. -
IncludeCSharpRuntime
: Specifies whether the packages with the C# Runtime should be automatically referenced. This property is set to true by default. If set to false, the Sdk will not automatically reference any C# Runtime packages. -
QscExe
: The command to invoke the Q# compiler. The value set by default invokes the Q# compiler that is packaged as tool with the Sdk. The default value can be accessed via theDefaultQscExe
property. -
QscVerbosity
: Defines the verbosity of the Q# compiler. Recognized values are: Quiet, Minimal, Normal, Detailed, and Diagnostic. -
PerfDataGeneration
: Specifies whether to generate performance analysis data for the compilation. The default value is "true" ifPerfDataOutputPath
is specified and "false" otherwise. Note that setting this property to"false"
will disable generating performance data, even ifPerfDataOutputPath
is also set. -
PerfDataOutputPath
: Directory where the generated performance analysis data will be saved. If no directory is specified andPerfDataGeneration
is set to "true", it will be set to "$(MSBuildProjectDirectory)/perf". -
QirGeneration
: Specifies whether to generate QIR for the compiled Q# code. The default value is "true" ifQirOutputPath
is specified and "false" otherwise. Note that setting this property to"false"
will disable generating QIR, even ifQirOutputPath
is also set. -
QirOutputPath
: Directory where the generated QIR will be saved. If no directory is specified andQirGeneration
is set to "true", it will be set to "$(MSBuildProjectDirectory)/qir". -
QSharpDocsGeneration
: Specifies whether to generate yml documentation for the compiled Q# code. The default value is "true" ifQSharpDocsOutputPath
is specified and "false" otherwise. Note that setting this property to"false"
will disable generating docs, even ifQSharpDocsOutputPath
is also set. -
QSharpDocsOutputPath
: Directory where the generated documentation will be saved. If no directory is specified andQSharpDocsGeneration
is set to "true", it will be set to "$(MSBuildProjectDirectory)/docs". -
QSharpDocsPackageId
: Specifies the package ID that should appear in generated documentation. Set toPackageId
by default, but can be overridden to allow for documenting parts of metapackages.
The following configurable item groups are used by the Sdk:
-
PackageLoadFallbackFolder
: Contains the directories where the Q# compiler will look for a suitable dll if a qsc reference or one if its dependencies cannot be found. By default, the project output path is included in this item group. -
PackageReference
: Contains all referenced NuGet packages. Package references for which theIsQscReference
attribute is set to "true" may extend the Q# compiler and any implemented rewrite steps will be executed as part of the compilation process. See this section for more details. -
ProjectReference
: Contains all referenced projects. Project references for which theIsQscReference
attribute is set to "true" may extend the Q# compiler and any implemented rewrite steps will be executed as part of the compilation process. See this section for more details. -
QSharpCompile
: Contains all Q# source files included in the compilation.
A NuGet package of type Sdk
enjoys certain privileges in terms of when and how its content is loaded.
To understand how the content in this package works it is useful to understand how the properties, item groups, and targets defined in the Sdk are combined with those defined by a specific project.
The order of evaluation for properties and item groups is roughly the following:
- Properties defined in *.props files of the Sdk
- Properties defined or included by the specific project file
- Properties defined in *.targets files of the Sdk
- Item groups defined in *.props files of the Sdk
- Item groups defined or included by the specific project file
- Item groups defined in *.targets files of the Sdk
Similar considerations apply for the definition of targets. MSBuild will overwrite targets if multiple targets with the same name are defined. In that case, the target is replaced in its entirety independent on whether the values for DependsOn
, BeforeTarget
, and AfterTarget
match - i.e. those will be overwritten. However, a target can be "anchored" by the surrounding targets' specifications of their dependencies, see e.g. the defined BeforeCSharpCompile
target.
See also this post for general information about property evaluation not specifically related to how Sdks work.
To avoid issues with conflicting packages, we load each Q# compiler extension into its own context. For more information, see the Core CLR design docs.
The following issues and PRs may be of interest when using the Sdk: