Skip to content

Latest commit

 

History

History
186 lines (137 loc) · 15.9 KB

ProjectReference-Protocol.md

File metadata and controls

186 lines (137 loc) · 15.9 KB

The ProjectReference Protocol

The MSBuild engine doesn't have a notion of a “project reference”—it only provides the MSBuild task to allow cross-project communication.

That's a powerful tool, but no one would want to have to specify how to build every single reference in every single project. The common targets introduce an item, ProjectReference, and a default process for building references declared via that item.

Default protocol implementation:

Projects that have references

In its simplest form, a project need only specify the path to another project in a ProjectReference item. For example,

<ItemGroup>
  <ProjectReference Include="..\..\some\other\project.csproj" />
</ItemGroup>

Importing Microsoft.Common.targets includes logic that consumes these items and transforms them into compile-time references before the compiler runs.

Who this document is for

This document describes that process, including what is required of a project to be referenceable through a ProjectReference. It is intended for for MSBuild SDK maintainers, and those who have created a completely custom project type that needs to interoperate with other projects. It may also be of interest if you'd like to see the implementation details of references. Understanding the details should not be necessary to use ProjectReferences in your project.

Targets related to consuming a reference

The bulk of the work of transforming ProjectReference items into something suitable to feed the compiler is done by tasks listed in the ResolveReferencesDependsOn property defined in Microsoft.Common.CurrentVersion.targets.

There are empty hooks in the default targets for

  • BeforeResolveReferences—run before the primary work of resolving references.
  • AfterResolveReferences—run after the primary work of resolving references.

AssignProjectConfiguration runs when building in a solution context, and ensures that the right Configuration and Platform are assigned to each reference. For example, if a solution specifies (using the Solution Build Manager) that for a given solution configuration, a project should always be built Release, that is applied inside MSBuild in this target.

PrepareProjectReferences then runs, ensuring that each referenced project exists (creating the item @(_MSBuildProjectReferenceExistent)).

_ComputeProjectReferenceTargetFrameworkMatches calls GetTargetFrameworks in existent ProjectReferences and determines the parameters needed to produce a compatible build by calling the AssignReferenceProperties task for each reference that multitargets.

ResolveProjectReferences does the bulk of the work, building the referenced projects and collecting their outputs.

After the compiler is invoked, GetCopyToOutputDirectoryItems pulls child-project outputs into the current project to be copied to its output directory.

When Cleaning the output of a project, CleanReferencedProjects ensures that referenced projects also clean.

Targets required to be referenceable

These targets should exist in a project to be compatible with the common targets' ProjectReference. Some are called only conditionally.

These targets are all defined in Microsoft.Common.targets and are defined in Microsoft SDKs. You should only have to implement them yourself if you require custom behavior or are authoring a project that doesn't import the common targets.

If implementing a project with an “outer” (determine what properties to pass to the real build) and “inner” (fully specified) build, only GetTargetFrameworkProperties is required in the “outer” build. The other targets listed can be “inner” build only.

  • GetTargetFrameworks tells referencing projects what options are available to the build.
    • It returns an item with the following metadata:
      • TargetFrameworks indicating what TargetFrameworks are available in the project
      • TargetFrameworkMonikers and TargetPlatformMonikers indicating what framework / platform the TargetFrameworks map to. This is to support implicitly setting the target platform version (for example inferring that net5.0-windows means the same as net5.0-windows7.0) as well as treating the TargetFramework values as aliases
      • Boolean metadata for HasSingleTargetFramework and IsRidAgnostic.
      • Platforms indicating what platforms are available for the project to build as, and boolean metadata IsVcxOrNativeProj (used for SetPlatform Negotiation)
    • The GetReferenceNearestTargetFrameworkTask (provided by NuGet) is responsible for selecting the best matching TargetFramework of the referenced project
    • This target is optional. If not present, the reference will be built with no additional properties.
    • New in MSBuild 15.5. (TargetFrameworkMonikers and TargetPlatformMonikers metadata is new in MSBuild 16.8)
    • It is possible to gather additional information from referenced projects. See the below section on "Getting additional properties from referenced projects" for more information
  • GetTargetFrameworkProperties determines what properties should be passed to the “main” target for a given ReferringTargetFramework.
    • Deprecated in MSBuild 15.5.
    • New for MSBuild 15/Visual Studio 2017. Supports the cross-targeting feature allowing a project to have multiple TargetFrameworks.
    • Conditions: only when metadata SkipGetTargetFrameworkProperties for each reference is not true.
    • Skipped for *.vcxproj by default.
    • This should return either
      • a string of the form TargetFramework=$(NearestTargetFramework);ProjectHasSingleTargetFramework=$(_HasSingleTargetFramework);ProjectIsRidAgnostic=$(_IsRidAgnostic), where the value of NearestTargetFramework will be used to formulate TargetFramework for the following calls and the other two properties are booleans, or
      • an item with metadata DesiredTargetFrameworkProperties (key-value pairs of the form TargetFramework=net46), HasSingleTargetFramework (boolean), and IsRidAgnostic (boolean).
  • GetTargetPath should return the path of the project's output, but not build that output.
    • Conditions: this is used for builds inside Visual Studio, but not on the command line.
    • It's also used when the property BuildProjectReferences is false, manually indicating that all ProjectReferences are up to date and shouldn't be (re)built.
    • This should return a single item that is the primary output of the project, with metadata describing that output. See TargetPathWithTargetPlatformMoniker for the default metadata.
  • Default targets should do the full build and return an assembly to be referenced.
    • Conditions: this is not called when building inside Visual Studio. Instead, Visual Studio builds each project in isolation but in order, so the path returned from GetTargetPath can be assumed to exist at consumption time.
    • If the ProjectReference defines the Targets metadata, it is used. If not, no target is passed, and the default target of the reference (usually Build) is built.
    • The return value of this target should be identical to that of GetTargetPath.
  • GetNativeManifest should return a manifest suitable for passing to the ResolveNativeReferences target.
    • As of 15.7, this is optional. If a project does not contain a GetNativeManifest target, it will not be referencable by native projects but will not fail the build.
  • GetCopyToOutputDirectoryItems should return the outputs of a project that should be copied to the output of a referencing project.
    • As of 15.7, this is optional. If a project does not contain a GetCopyToOutputDirectoryItems target, projects that reference it will not copy any of its outputs to their own output folders, but the build can succeed.
  • Clean should delete all outputs of the project.
    • It is not called during a normal build, only during "Clean" and "Rebuild".

Other protocol requirements

As with all MSBuild logic, targets can be added to do other work with ProjectReferences.

In particular, NuGet depends on being able to identify referenced projects' package dependencies, and calls some targets that are imported through Microsoft.Common.targets to do so. At the time of writing this this is in NuGet.targets.

Microsoft.AppxPackage.targets adds a dependency on the target GetPackagingOutputs.

Getting additional properties from referenced projects

As of MSBuild 16.10, it is possible to gather additional properties from referenced projects. To do this, the referenced project should declare an AdditionalTargetFrameworkInfoProperty item for each property that should be gathered for referencing projects. For example:

  <ItemGroup>
    <AdditionalTargetFrameworkInfoProperty Include="SelfContained"/>
    <AdditionalTargetFrameworkInfoProperty Include="_IsExecutable"/>
  </ItemGroup>

These properties will then be gathered via the GetTargetFrameworks call. They will be available to the referencing project via the AdditionalPropertiesFromProject metadata on the _MSBuildProjectReferenceExistent item. The AdditionalPropertiesFromProject value will be an XML string which contains the values of the properties for each TargetFramework in the referenced project. For example:

⚠️ This format is being changed. Soon, the schema will replace <net5.0> with . You can opt into that behavior early by setting the _UseAttributeForTargetFrameworkInfoPropertyNames property to true. This property will have no effect after the transition is complete.

<AdditionalProjectProperties>
  <net5.0>
    <SelfContained>true</SelfContained>
    <_IsExecutable>true</_IsExecutable>
  </net5.0>
  <net5.0-windows>
    <SelfContained>false</SelfContained>
    <_IsExecutable>true</_IsExecutable>
  </net5.0-windows>
</AdditionalProjectProperties>

The NearestTargetFramework metadata will be the target framework which was selected as the best one to use for the reference (via GetReferenceNearestTargetFrameworkTask). This can be used to select which set of properties were used in the target framework that was active for the reference.

SetPlatform Negotiation

As of version 17.0, MSBuild can now dynamically figure out what platform a ProjectReference should build as. This includes a new target and task to determine what the SetPlatform metadata should be, or whether to undefine the platform so the referenced project builds with its default platform.

  • _GetProjectReferenceTargetFrameworkProperties target performs the majority of the work for assigning SetPlatform metadata to project references.
    • Calls the GetCompatiblePlatform task, which is responsible for negotiating between the current project's platform and the platforms of the referenced project to assign a NearestPlatform metadata to the item.
    • Sets or undefines SetPlatform based on the NearestPlatform assignment from GetCompatiblePlatform
    • This target explicitly runs after _GetProjectReferenceTargetFrameworkProperties because it needs to use the IsVcxOrNativeProj and Platforms properties returned by the GetTargetFrameworks call.

Note: If a ProjectReference has SetPlatform metadata defined already, the negotiation logic is skipped over.

Impact on the build

In addition to the above task and target, .vcxproj and .nativeproj projects will receive an extra MSBuild call to the GetTargetFrameworks target. Previously, TargetFramework negotiation skipped over these projects because they could not multi-target in the first place. Because SetPlatform negotiation needs information given from the GetTargetFrameworks target, it is required that the _GetProjectReferenceTargetFrameworkProperties target calls the MSBuild task on the ProjectReference.

This means most projects will see an evaluation with no global properties defined, unless set by the user.

How To Opt In

First, set the properties EnableDynamicPlatformResolution and DisableTransitiveProjectReferences to true for every project in your solution. The easiest way to do this is by creating a Directory.Build.props file and placing it at the root of your project directory:

<Project>
  <PropertyGroup>
    <EnableDynamicPlatformResolution>true</EnableDynamicPlatformResolution>
    <DisableTransitiveProjectReferences>true</DisableTransitiveProjectReferences>
  </PropertyGroup>
</Project>

If only set in one project, the SetPlatform metadata will carry forward to every consecutive project reference.

Next, every referenced project is required to define a Platforms property, where Platforms is a semicolon-delimited list of platforms that project could build as. For .vcxproj or .nativeproj projects, Platforms is constructed from the ProjectConfiguration items that already exist in the project. For managed SDK projects, the default is AnyCPU. Managed non-SDK projects need to define this manually.

Lastly, a PlatformLookupTable may need to be defined for more complex scenarios. A PlatformLookupTable is a semicolon-delimited list of mappings between platforms. <PlatformLookupTable>Win32=x86</PlatformLookupTable>, for example. This means that when the current project is building as Win32, it will attempt to build the referenced project as x86. This property is required when a managed AnyCPU project references an unmanaged project because AnyCPU does not directly map to an architecture-specific platform. You can define the table in two ways:

  1. A standard property within the current project, in a Directory.Build.props/targets
  2. Metadata on the ProjectReference item. This option takes priority over the first to allow customizations per ProjectReference.

References between managed and unmanaged projects

Some cases of ProjectReferences require a $(PlatformLookupTable) to correctly determine what a referenced project should build as. References between managed and unmanaged projects also get a default lookup table that can be opted out of by setting the property UseDefaultPlatformLookupTables to false. See the table below for details.

Note: Defining a PlatformLookupTable overrides the default mapping.

Project Reference Type PlatformLookupTable Required? Notes
Unmanaged -> Unmanaged No
Managed -> Managed No
Unmanaged -> Managed Optional Uses default mapping: Win32=x86
Managed -> Unmanaged Yes when the project is AnyCPU Uses default mapping: x86=Win32

Example: Project A: Managed, building as AnyCPU, has a ProjectReference on Project B. Project B: Unmanaged, has $(Platforms) constructed from its Platform metadata from its ProjectConfiguration items, defined as x64;Win32.

Because AnyCPU does not map to anything architecture-specific, a custom mapping must be defined. Project A can either:

  1. Define PlatformLookupTable in its project or a Directory.Build.props as AnyCPU=x64 or AnyCPU=Win32.
  2. Define PlatformLookupTable as metadata on the ProjectReference item, which would take priority over a lookup table defined elsewhere.
    • When only one mapping is valid, you could also directly define SetPlatform metadata as Platform=foo (for unmanaged) or PlatformTarget=bar (for managed). This would skip over most negotiation logic.

Example of project A defining a lookup table directly on the ProjectReference:

<ItemGroup>
  <ProjectReference Include="B.csproj" PlatformLookupTable="AnyCPU=Win32">
</ItemGroup>