94b2861243
Former-commit-id: 5f9c6ae75f295e057a7d2971f3a6df4656fa8850
391 lines
13 KiB
C#
391 lines
13 KiB
C#
//------------------------------------------------------------
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
//------------------------------------------------------------
|
|
|
|
namespace Microsoft.Build.Tasks.Xaml
|
|
{
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Runtime;
|
|
using System.Threading;
|
|
using System.Xaml;
|
|
using Microsoft.Build.Framework;
|
|
using Microsoft.Build.Utilities;
|
|
using Microsoft.VisualStudio.Activities;
|
|
|
|
[Fx.Tag.XamlVisible(true)]
|
|
public class PartialClassGenerationTask : Task
|
|
{
|
|
const string DefaultGeneratedSourceExtension = "g";
|
|
List<ITaskItem> generatedResources = new List<ITaskItem>();
|
|
List<ITaskItem> generatedCodeFiles = new List<ITaskItem>();
|
|
|
|
// We will do Dev10 behavior if the new required property MSBuildProjectDirectory is NOT specified. This can happen
|
|
// if a Dev10 version of the Microsoft.Xaml.Targets file is being used with Dev11 installed.
|
|
bool supportExtensions = false;
|
|
string msBuildProjectDirectory;
|
|
|
|
public PartialClassGenerationTask()
|
|
{
|
|
this.GeneratedSourceExtension = DefaultGeneratedSourceExtension;
|
|
}
|
|
|
|
[Fx.Tag.KnownXamlExternal]
|
|
[Output]
|
|
public ITaskItem[] ApplicationMarkup { get; set; }
|
|
|
|
public string AssemblyName
|
|
{ get; set; }
|
|
|
|
public string[] KnownReferencePaths { get; set; }
|
|
|
|
[Fx.Tag.KnownXamlExternal]
|
|
[Output]
|
|
public ITaskItem[] GeneratedResources
|
|
{
|
|
get
|
|
{
|
|
return generatedResources.ToArray();
|
|
}
|
|
|
|
set
|
|
{
|
|
generatedResources = new List<ITaskItem>(value);
|
|
}
|
|
}
|
|
|
|
[Fx.Tag.KnownXamlExternal]
|
|
[Output]
|
|
public ITaskItem[] GeneratedCodeFiles
|
|
{
|
|
get
|
|
{
|
|
return generatedCodeFiles.ToArray();
|
|
}
|
|
|
|
set
|
|
{
|
|
generatedCodeFiles = new List<ITaskItem>(value);
|
|
}
|
|
}
|
|
|
|
public string GeneratedSourceExtension
|
|
{ get; set; }
|
|
|
|
[Required]
|
|
public string Language
|
|
{ get; set; }
|
|
|
|
[Required]
|
|
public string OutputPath
|
|
{ get; set; }
|
|
|
|
// This is Required for Dev11, but to allow things to work with a Dev10 targets file, we are not marking it required.
|
|
public string MSBuildProjectDirectory
|
|
{
|
|
get
|
|
{
|
|
return this.msBuildProjectDirectory;
|
|
}
|
|
|
|
set
|
|
{
|
|
this.msBuildProjectDirectory = value;
|
|
// The fact that this property is being set indicates that a Dev11 version of the targets
|
|
// file is being used, so we should not do Dev10 behavior.
|
|
this.supportExtensions = true;
|
|
}
|
|
}
|
|
|
|
[Fx.Tag.KnownXamlExternal]
|
|
public ITaskItem[] References
|
|
{ get; set; }
|
|
|
|
public string RootNamespace
|
|
{ get; set; }
|
|
|
|
[Fx.Tag.KnownXamlExternal]
|
|
public ITaskItem[] SourceCodeFiles
|
|
{ get; set; }
|
|
|
|
[Output]
|
|
public bool RequiresCompilationPass2
|
|
{ get; set; }
|
|
|
|
public string BuildTaskPath
|
|
{ get; set; }
|
|
|
|
public bool IsInProcessXamlMarkupCompile
|
|
{ get; set; }
|
|
|
|
public ITaskItem[] XamlBuildTypeGenerationExtensionNames
|
|
{ get; set; }
|
|
|
|
public ITaskItem[] XamlBuildTypeInspectionExtensionNames
|
|
{ get; set; }
|
|
|
|
private static AppDomain inProcessAppDomain;
|
|
private static Dictionary<string, DateTime> referencesTimeStampCache;
|
|
private Object referencesCacheLock = new Object();
|
|
|
|
public override bool Execute()
|
|
{
|
|
VSDesignerPerfEventProvider perfEventProvider = new VSDesignerPerfEventProvider();
|
|
perfEventProvider.WriteEvent(VSDesignerPerfEvents.XamlBuildTaskExecuteStart);
|
|
|
|
try
|
|
{
|
|
if (IsInProcessXamlMarkupCompile)
|
|
{
|
|
bool acquiredLock = false;
|
|
try
|
|
{
|
|
Monitor.TryEnter(referencesCacheLock, ref acquiredLock);
|
|
if (acquiredLock)
|
|
{
|
|
return ReuseAppDomainAndExecute();
|
|
}
|
|
else
|
|
{
|
|
return GetAppDomainAndExecute();
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (acquiredLock)
|
|
{
|
|
Monitor.Exit(referencesCacheLock);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return GetAppDomainAndExecute();
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
perfEventProvider.WriteEvent(VSDesignerPerfEvents.XamlBuildTaskExecuteEnd);
|
|
}
|
|
}
|
|
|
|
bool ReuseAppDomainAndExecute()
|
|
{
|
|
AppDomain appDomain = null;
|
|
bool createdNewAppDomain = false;
|
|
try
|
|
{
|
|
try
|
|
{
|
|
appDomain = GetInProcessAppDomain(out createdNewAppDomain);
|
|
bool ret = ExecuteInternal(appDomain);
|
|
return ret;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e))
|
|
{
|
|
throw;
|
|
}
|
|
if (createdNewAppDomain)
|
|
{
|
|
XamlBuildTaskServices.LogException(this.Log, e.Message);
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
if (inProcessAppDomain != null)
|
|
{
|
|
AppDomain.Unload(inProcessAppDomain);
|
|
inProcessAppDomain = null;
|
|
}
|
|
return GetAppDomainAndExecute();
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (Log != null)
|
|
{
|
|
Log.MarkAsInactive();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GetAppDomainAndExecute()
|
|
{
|
|
AppDomain appDomain = null;
|
|
try
|
|
{
|
|
appDomain = CreateNewAppDomain();
|
|
bool ret = ExecuteInternal(appDomain);
|
|
return ret;
|
|
}
|
|
catch (Exception e)
|
|
{
|
|
if (Fx.IsFatal(e))
|
|
{
|
|
throw;
|
|
}
|
|
|
|
XamlBuildTaskServices.LogException(this.Log, e.Message);
|
|
return false;
|
|
}
|
|
finally
|
|
{
|
|
if (appDomain != null)
|
|
{
|
|
AppDomain.Unload(appDomain);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool ExecuteInternal(AppDomain appDomain)
|
|
{
|
|
PartialClassGenerationTaskInternal wrapper = (PartialClassGenerationTaskInternal)appDomain.CreateInstanceAndUnwrap(
|
|
Assembly.GetExecutingAssembly().FullName,
|
|
typeof(PartialClassGenerationTaskInternal).FullName);
|
|
|
|
PopulateBuildArtifacts(wrapper);
|
|
|
|
bool ret = wrapper.Execute();
|
|
|
|
if (ret)
|
|
{
|
|
ExtractBuiltArtifacts(wrapper);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
AppDomain CreateNewAppDomain()
|
|
{
|
|
return XamlBuildTaskServices.CreateAppDomain("PartialClassAppDomain_" + Guid.NewGuid(), BuildTaskPath);
|
|
}
|
|
|
|
// For Intellisense builds, we re-use the AppDomain for successive builds instead of creating a new one every time,
|
|
// if the references have not changed (there are no new references and they have not been updated since the last build)
|
|
// This method accesses the static referencesTimeStampCache (indirectly).
|
|
// To ensure thread safety, this method should be called inside a lock/monitor
|
|
AppDomain GetInProcessAppDomain(out bool newAppDomain)
|
|
{
|
|
newAppDomain = false;
|
|
if (inProcessAppDomain == null)
|
|
{
|
|
inProcessAppDomain = CreateNewAppDomain();
|
|
newAppDomain = true;
|
|
UpdateReferenceCache();
|
|
}
|
|
else if (AreReferencesChanged())
|
|
{
|
|
AppDomain.Unload(inProcessAppDomain);
|
|
inProcessAppDomain = CreateNewAppDomain();
|
|
newAppDomain = true;
|
|
UpdateReferenceCache();
|
|
}
|
|
return inProcessAppDomain;
|
|
}
|
|
|
|
// This method accesses the static referencesTimeStampCache.
|
|
// To ensure thread safety, this method should be called inside a lock/monitor
|
|
bool AreReferencesChanged()
|
|
{
|
|
bool refsChanged = false;
|
|
if (referencesTimeStampCache == null || referencesTimeStampCache.Count != References.Length)
|
|
{
|
|
refsChanged = true;
|
|
}
|
|
else
|
|
{
|
|
foreach (var reference in References)
|
|
{
|
|
string fullPath = Path.GetFullPath(reference.ItemSpec);
|
|
DateTime timeStamp = File.GetLastWriteTimeUtc(fullPath);
|
|
if (!referencesTimeStampCache.ContainsKey(fullPath)
|
|
|| timeStamp > referencesTimeStampCache[fullPath]
|
|
|| timeStamp == DateTime.MinValue)
|
|
{
|
|
refsChanged = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return refsChanged;
|
|
}
|
|
|
|
// This method accesses the static referencesTimeStampCache.
|
|
// To ensure thread safety, this method should be called inside a lock/monitor
|
|
void UpdateReferenceCache()
|
|
{
|
|
referencesTimeStampCache = new Dictionary<string, DateTime>();
|
|
foreach (var reference in References)
|
|
{
|
|
string fullPath = Path.GetFullPath(reference.ItemSpec);
|
|
DateTime timeStamp = File.GetLastWriteTimeUtc(fullPath);
|
|
referencesTimeStampCache.Add(fullPath, timeStamp);
|
|
}
|
|
}
|
|
|
|
void PopulateBuildArtifacts(PartialClassGenerationTaskInternal wrapper)
|
|
{
|
|
IList<ITaskItem> applicationMarkup = null;
|
|
if (this.ApplicationMarkup != null)
|
|
{
|
|
applicationMarkup = this.ApplicationMarkup
|
|
.Select(i => new DelegatingTaskItem(i) as ITaskItem).ToList();
|
|
}
|
|
wrapper.ApplicationMarkup = applicationMarkup;
|
|
|
|
wrapper.BuildLogger = this.Log;
|
|
|
|
wrapper.References = this.References
|
|
.Select(i => new DelegatingTaskItem(i) as ITaskItem).ToList();
|
|
|
|
IList<string> sourceCodeFiles = null;
|
|
if (this.SourceCodeFiles != null)
|
|
{
|
|
sourceCodeFiles = new List<string>(this.SourceCodeFiles.Length);
|
|
foreach (ITaskItem taskItem in this.SourceCodeFiles)
|
|
{
|
|
sourceCodeFiles.Add(taskItem.ItemSpec);
|
|
}
|
|
}
|
|
wrapper.SourceCodeFiles = sourceCodeFiles;
|
|
|
|
wrapper.Language = this.Language;
|
|
wrapper.AssemblyName = this.AssemblyName;
|
|
wrapper.OutputPath = this.OutputPath;
|
|
wrapper.RootNamespace = this.RootNamespace;
|
|
wrapper.GeneratedSourceExtension = this.GeneratedSourceExtension;
|
|
wrapper.IsInProcessXamlMarkupCompile = this.IsInProcessXamlMarkupCompile;
|
|
wrapper.MSBuildProjectDirectory = this.MSBuildProjectDirectory;
|
|
wrapper.XamlBuildTaskTypeGenerationExtensionNames = XamlBuildTaskServices.GetXamlBuildTaskExtensionNames(this.XamlBuildTypeGenerationExtensionNames);
|
|
if (this.XamlBuildTypeInspectionExtensionNames != null && this.XamlBuildTypeInspectionExtensionNames.Length > 0)
|
|
{
|
|
wrapper.MarkupCompilePass2ExtensionsPresent = true;
|
|
}
|
|
|
|
wrapper.SupportExtensions = this.supportExtensions;
|
|
}
|
|
|
|
void ExtractBuiltArtifacts(PartialClassGenerationTaskInternal wrapper)
|
|
{
|
|
foreach (string resource in wrapper.GeneratedResources)
|
|
{
|
|
this.generatedResources.Add(new TaskItem(resource));
|
|
}
|
|
|
|
foreach (string code in wrapper.GeneratedCodeFiles)
|
|
{
|
|
this.generatedCodeFiles.Add(new TaskItem(code));
|
|
}
|
|
|
|
this.RequiresCompilationPass2 = wrapper.RequiresCompilationPass2 ||
|
|
(this.XamlBuildTypeInspectionExtensionNames != null && this.XamlBuildTypeInspectionExtensionNames.Length > 0);
|
|
}
|
|
}
|
|
}
|