//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.Compilation { using System; using System.IO; using System.Collections; using System.Collections.Concurrent; using System.Collections.Specialized; using System.Reflection; using System.Globalization; using System.CodeDom; using System.CodeDom.Compiler; using System.Configuration; using System.Linq; using System.Runtime.ExceptionServices; using System.Threading.Tasks; using System.Web.Hosting; using System.Web.Util; using System.Web.Caching; using System.Web.UI; using System.Web.Configuration; internal class BuildProvidersCompiler { private ICollection _buildProviders; private VirtualPath _configPath; private bool _supportLocalization; // The set of assemblies that we need to reference when compiling private ICollection _referencedAssemblies; internal ICollection ReferencedAssemblies { get { return _referencedAssemblies; } } private AssemblyBuilder _assemblyBuilder; // Key: CultureName string, Value: AssemblyBuilder private IDictionary _satelliteAssemblyBuilders = null; // If this is set, we only generate the source files into this directory // without compiling them. // This is used to implement ClientBuildManager.GetCodeDirectoryInformation private string _generatedFilesDir; internal BuildProvidersCompiler(VirtualPath configPath, string outputAssemblyName) : this(configPath, false, outputAssemblyName) { } internal BuildProvidersCompiler(VirtualPath configPath, bool supportLocalization, string outputAssemblyName) { _configPath = configPath; _supportLocalization = supportLocalization; _compConfig = MTConfigUtil.GetCompilationConfig(_configPath); _referencedAssemblies = BuildManager.GetReferencedAssemblies(CompConfig); _outputAssemblyName = outputAssemblyName; } internal BuildProvidersCompiler(VirtualPath configPath, bool supportLocalization, string generatedFilesDir, int index) { _configPath = configPath; _supportLocalization = supportLocalization; _compConfig = MTConfigUtil.GetCompilationConfig(_configPath); _referencedAssemblies = BuildManager.GetReferencedAssemblies(CompConfig, index); _generatedFilesDir = generatedFilesDir; } // The config section for the set of build providers that we handle private CompilationSection _compConfig; internal CompilationSection CompConfig { get { return _compConfig; } } private string _outputAssemblyName; internal string OutputAssemblyName { get { return _outputAssemblyName; } } private bool CbmGenerateOnlyMode { get { return _generatedFilesDir != null; } } internal void SetBuildProviders(ICollection buildProviders) { _buildProviders = buildProviders; } private void ProcessBuildProviders() { CompilerType compilerType = null; BuildProvider firstLanguageBuildProvider = null; // First, delete all the existing satellite assemblies of the assembly // we're about to build (VSWhidbey 87022) (only if it has a fixed name) if (OutputAssemblyName != null) { Debug.Assert(!CbmGenerateOnlyMode); StandardDiskBuildResultCache.RemoveSatelliteAssemblies(OutputAssemblyName); } // List of BuildProvider's that don't ask for a specific language ArrayList languageFreeBuildProviders = null; foreach (BuildProvider buildProvider in _buildProviders) { // If it's an InternalBuildProvider, give it the assembly references early on buildProvider.SetReferencedAssemblies(_referencedAssemblies); // Instruct the internal build providers to continue processing for more parse errors. if (!BuildManager.ThrowOnFirstParseError) { InternalBuildProvider provider = buildProvider as InternalBuildProvider; if (provider != null) { provider.ThrowOnFirstParseError = false; } } // Get the language and culture CompilerType ctwp = BuildProvider.GetCompilerTypeFromBuildProvider(buildProvider); // Only look for a culture if we're supposed to (basically, in the resources directories) string cultureName = null; if (_supportLocalization) cultureName = buildProvider.GetCultureName(); // Is it asking for a specific language? if (ctwp != null) { // If it specifies a language, it can't also have a culture if (cultureName != null) { throw new HttpException(SR.GetString(SR.Both_culture_and_language, BuildProvider.GetDisplayName(buildProvider))); } // Do we already know the language we'll be using if (compilerType != null) { // If it's different from the current one, fail if (!ctwp.Equals(compilerType)) { throw new HttpException(SR.GetString(SR.Inconsistent_language, BuildProvider.GetDisplayName(buildProvider), BuildProvider.GetDisplayName(firstLanguageBuildProvider))); } } else { // Keep track of the build provider of error handling purpose firstLanguageBuildProvider = buildProvider; // Keep track of the language compilerType = ctwp; _assemblyBuilder = compilerType.CreateAssemblyBuilder( CompConfig, _referencedAssemblies, _generatedFilesDir, OutputAssemblyName); } } else { if (cultureName != null) { // Ignore the culture files in generate-only mode if (CbmGenerateOnlyMode) continue; if (_satelliteAssemblyBuilders == null) { _satelliteAssemblyBuilders = new Hashtable( StringComparer.OrdinalIgnoreCase); } // Check if we already have an assembly builder for this culture AssemblyBuilder satelliteAssemblyBuilder = (AssemblyBuilder) _satelliteAssemblyBuilders[cultureName]; // If not, create one and store it in the hashtable if (satelliteAssemblyBuilder == null) { satelliteAssemblyBuilder = CompilerType.GetDefaultAssemblyBuilder( CompConfig, _referencedAssemblies, _configPath, OutputAssemblyName); satelliteAssemblyBuilder.CultureName = cultureName; _satelliteAssemblyBuilders[cultureName] = satelliteAssemblyBuilder; } satelliteAssemblyBuilder.AddBuildProvider(buildProvider); continue; } if (_assemblyBuilder == null) { // If this provider doesn't need a specific language, and we don't know // the language yet, just keep track of it if (languageFreeBuildProviders == null) languageFreeBuildProviders = new ArrayList(); languageFreeBuildProviders.Add(buildProvider); continue; } } _assemblyBuilder.AddBuildProvider(buildProvider); } // If we didn't get an AssemblyBuilder, use a default if (_assemblyBuilder == null && languageFreeBuildProviders != null) { _assemblyBuilder = CompilerType.GetDefaultAssemblyBuilder( CompConfig, _referencedAssemblies, _configPath, _generatedFilesDir, OutputAssemblyName); } // Add all the language free providers (if any) to the AssemblyBuilder if (_assemblyBuilder != null && languageFreeBuildProviders != null) { foreach (BuildProvider languageFreeBuildProvider in languageFreeBuildProviders) { _assemblyBuilder.AddBuildProvider(languageFreeBuildProvider); } } } internal CompilerResults PerformBuild() { ProcessBuildProviders(); // Build all the satellite assemblies if (_satelliteAssemblyBuilders != null) { int maxConcurrent = Math.Min(_satelliteAssemblyBuilders.Count, CompilationUtil.MaxConcurrentCompilations); try { Parallel.ForEach(_satelliteAssemblyBuilders.Values.Cast(), new ParallelOptions { MaxDegreeOfParallelism = maxConcurrent }, assemblyBuilder => { assemblyBuilder.Compile(); }); } catch (AggregateException ae) { ExceptionDispatchInfo.Capture(ae.GetBaseException()).Throw(); } } // Build the main assembly if (_assemblyBuilder != null) return _assemblyBuilder.Compile(); return null; } internal void GenerateSources(out Type codeDomProviderType, out CompilerParameters compilerParameters) { ProcessBuildProviders(); // If we didn't get an AssemblyBuilder (happens when there was nothing to build), // get a default one. if (_assemblyBuilder == null) { _assemblyBuilder = CompilerType.GetDefaultAssemblyBuilder( CompConfig, _referencedAssemblies, _configPath, _generatedFilesDir, null /*outputAssemblyName*/); } codeDomProviderType = _assemblyBuilder.CodeDomProviderType; compilerParameters = _assemblyBuilder.GetCompilerParameters(); } } /* * This class handles the batch compilation of one directory. It may * produce several assemblies out of them, based on dependencies and language * differences. All the BuildProvider's are expected to share the same * configuration (i.e. they live in the same directory). */ internal class WebDirectoryBatchCompiler { private DateTime _utcStart; // The set of assemblies that we will link with private ICollection _referencedAssemblies; // The config section for the set of build providers that we handle private CompilationSection _compConfig; // [VirtualPathString,InternalBuildProvider] private IDictionary _buildProviders = new Hashtable( StringComparer.OrdinalIgnoreCase); private VirtualDirectory _vdir; private ArrayList[] _nonDependentBuckets; private bool _ignoreProvidersWithErrors; // The set of parser errors detected during parsing. private ParserErrorCollection _parserErrors; // The first parse exceptions thrown during parsing. private HttpParseException _firstException; internal WebDirectoryBatchCompiler(VirtualDirectory vdir) { _vdir = vdir; _utcStart = DateTime.UtcNow; _compConfig = MTConfigUtil.GetCompilationConfig(_vdir.VirtualPath); _referencedAssemblies = BuildManager.GetReferencedAssemblies(_compConfig); } internal void SetIgnoreErrors() { _ignoreProvidersWithErrors = true; } internal void Process() { AddBuildProviders(true /*retryIfDeletionHappens*/); // If there are no BuildProvider's, we're done if (_buildProviders.Count == 0) return; BuildManager.ReportDirectoryCompilationProgress(_vdir.VirtualPathObject); GetBuildResultDependencies(); ProcessDependencies(); foreach (ICollection buildProviders in _nonDependentBuckets) { if (!CompileNonDependentBuildProviders(buildProviders)) break; } // Report all parse exceptions if (_parserErrors != null && _parserErrors.Count > 0) { Debug.Assert(!_ignoreProvidersWithErrors); // Throw the first exception as inner exception along with the parse errors. HttpParseException newException = new HttpParseException(_firstException.Message, _firstException, _firstException.VirtualPath, _firstException.Source, _firstException.Line); // Add the rest of the parser errors to the exception. // The first one is already added. for (int i = 1; i < _parserErrors.Count; i++) { newException.ParserErrors.Add(_parserErrors[i]); } // rethrow the new exception throw newException; } } private void AddBuildProviders(bool retryIfDeletionHappens) { DiskBuildResultCache.ResetAssemblyDeleted(); foreach (VirtualFile vfile in _vdir.Files) { // If it's already built and up to date, skip it BuildResult result = null; try { result = BuildManager.GetVPathBuildResultFromCache(vfile.VirtualPathObject); } catch { // Ignore the cached error in batch compilation mode, since we want to compile // as many files as possible. // But don't ignore it in CBM or precompile cases, since we always want to try // to compile everything that had failed before. if (!BuildManager.PerformingPrecompilation) { // Skip it if an exception occurs (e.g. if a compile error was cached for it) continue; } } if (result != null) continue; BuildProvider buildProvider = BuildManager.CreateBuildProvider(vfile.VirtualPathObject, _compConfig, _referencedAssemblies, false /*failIfUnknown*/); // Non-supported file type if (buildProvider == null) continue; // IgnoreFileBuildProvider's should never be created Debug.Assert(!(buildProvider is IgnoreFileBuildProvider)); _buildProviders[vfile.VirtualPath] = buildProvider; } // If an assembly had to be deleted/renamed as a result of calling GetVPathBuildResultFromCache, // me way need to run the AddBuildProviders logic again. The reason is that as a result of // deleting the assembly, we may have invalidated other BuildResult that we had earlier found // to be up to date (VSWhidbey 269297) if (DiskBuildResultCache.InUseAssemblyWasDeleted) { Debug.Assert(retryIfDeletionHappens); // Only retry if we're doing precompilation. For standard batching, we can live // with the fact that not everything will be built after we're done (and we want to // be done as quickly as possible since the user is waiting). if (retryIfDeletionHappens && BuildManager.PerformingPrecompilation) { Debug.Trace("WebDirectoryBatchCompiler", "Rerunning AddBuildProviders for '" + _vdir.VirtualPath + "' because an assembly was out of date."); // Pass false for retryIfDeletionHappens to make sure we don't get in an // infinite recursion. AddBuildProviders(false /*retryIfDeletionHappens*/); } } } private void CacheAssemblyResults(AssemblyBuilder assemblyBuilder, CompilerResults results) { foreach (BuildProvider buildProvider in assemblyBuilder.BuildProviders) { BuildResult result = buildProvider.GetBuildResult(results); // If the provider didn't produce anything, ignore it if (result == null) continue; // If CacheVPathBuildResult returns false, something was found to be invalidated // and we need to abort the caching (VSWhidbey 578372) if (!BuildManager.CacheVPathBuildResult(buildProvider.VirtualPathObject, result, _utcStart)) break; #if DBG if (results != null) { if (DelayLoadType.Enabled) { Debug.Trace("BuildManager", buildProvider.VirtualPath + " Delay Load Assembly"); } else { Debug.Trace("BuildManager", buildProvider.VirtualPath + results.CompiledAssembly.EscapedCodeBase); } } else { Debug.Trace("BuildManager", buildProvider.VirtualPath + ": no assembly"); } #endif } } // Cache the various compile errors found during batching private void CacheCompileErrors(AssemblyBuilder assemblyBuilder, CompilerResults results) { BuildProvider previous = null; // Go through all the compile errors foreach (CompilerError error in results.Errors) { // Skip warnings if (error.IsWarning) continue; // Try to map the error back to a BuildProvider. If we can't, skip the error. BuildProvider buildProvider = assemblyBuilder.GetBuildProviderFromLinePragma(error.FileName); if (buildProvider == null) continue; // Only cache the error for template controls. Otherwise, for file types like // asmx/ashx, it's too likely that two of them define the same class. if (!(buildProvider is BaseTemplateBuildProvider)) continue; // If the error is for the same page as the previous one, ignore it if (buildProvider == previous) continue; previous = buildProvider; // Create a new CompilerResults for this error CompilerResults newResults = new CompilerResults(null /*tempFiles*/); // Copy all the output to the new result. Note that this will include all the // error lines, not just the ones for this BuildProvider. But that's not a big deal, // and we can't easily filter the output here. foreach (string s in results.Output) newResults.Output.Add(s); // Copy various other fields to the new CompilerResults object newResults.PathToAssembly = results.PathToAssembly; newResults.NativeCompilerReturnValue = results.NativeCompilerReturnValue; // Add this error. It will be the only one in the CompilerResults object. newResults.Errors.Add(error); // Create a new HttpCompileException & BuildResultCompileError to wrap this error HttpCompileException e = new HttpCompileException(newResults, assemblyBuilder.GetGeneratedSourceFromBuildProvider(buildProvider)); BuildResult result = new BuildResultCompileError(buildProvider.VirtualPathObject, e); // Add the dependencies to the compile error build provider, so that // we will retry compilation when a dependency changes buildProvider.SetBuildResultDependencies(result); // Cache it BuildManager.CacheVPathBuildResult(buildProvider.VirtualPathObject, result, _utcStart); } } private void GetBuildResultDependencies() { foreach (BuildProvider buildProvider in _buildProviders.Values) { ICollection virtualPathDependencies = buildProvider.GetBuildResultVirtualPathDependencies(); if (virtualPathDependencies == null) continue; foreach (string virtualPathDependency in virtualPathDependencies) { BuildProvider dependentBuildProvider = (BuildProvider) _buildProviders[virtualPathDependency]; if (dependentBuildProvider != null) buildProvider.AddBuildProviderDependency(dependentBuildProvider); } } } // Split the providers into non dependent buckets private void ProcessDependencies() { // First phase: compute levels in the dependency tree int totaldepth = 0; Hashtable depth = new Hashtable(); Stack stack = new Stack(); // compute depths foreach (BuildProvider buildProvider in _buildProviders.Values) { stack.Push(buildProvider); while (stack.Count > 0) { BuildProvider curnode = (BuildProvider)stack.Peek(); bool recurse = false; int maxdepth = 0; if (curnode.BuildProviderDependencies != null) { foreach (BuildProvider child in curnode.BuildProviderDependencies) { if (depth.ContainsKey(child)) { if (maxdepth <= (int)depth[child]) maxdepth = (int)depth[child] + 1; else if ((int)depth[child] == -1) throw new HttpException(SR.GetString(SR.File_Circular_Reference, child.VirtualPath)); } else { recurse = true; stack.Push(child); } } } if (recurse) depth[curnode] = -1; // being computed; else { stack.Pop(); depth[curnode] = maxdepth; if (totaldepth <= maxdepth) totaldepth = maxdepth + 1; } } } // drop into buckets by depth _nonDependentBuckets = new ArrayList[totaldepth]; for (IDictionaryEnumerator en = (IDictionaryEnumerator)depth.GetEnumerator(); en.MoveNext();) { int level = (int)en.Value; if (_nonDependentBuckets[level] == null) _nonDependentBuckets[level] = new ArrayList(); _nonDependentBuckets[level].Add(en.Key); } #if DBG int i = 0; foreach (ICollection buildProviders in _nonDependentBuckets) { Debug.Trace("BuildManager", String.Empty); Debug.Trace("BuildManager", "Bucket " + i + " contains " + buildProviders.Count + " files"); foreach (BuildProvider buildProvider in buildProviders) Debug.Trace("BuildManager", buildProvider.VirtualPath); i++; } #endif } private bool IsBuildProviderSkipable(BuildProvider buildProvider) { // If another build provider depends on it, we should not skip it if (buildProvider.IsDependedOn) return false; // No one depends on it (at least in this directory) // If it's a source file, skip it. We need to do this for v1 compatibility, // since v1 VS projects contain many source files which have already been // precompiled into bin, and that should not be compiled dynamically if (buildProvider is SourceFileBuildProvider) return true; // For the same reason, skip resources if (buildProvider is ResXBuildProvider) return true; return false; } private bool CompileNonDependentBuildProviders(ICollection buildProviders) { // Key: CompilerType, Value: AssemblyBuilder IDictionary assemblyBuilders = new Hashtable(); // List of InternalBuildProvider's that don't ask for a specific language ArrayList languageFreeBuildProviders = null; // AssemblyBuilder used for providers that don't need a specific language AssemblyBuilder defaultAssemblyBuilder = null; bool hasParserErrors = false; foreach (BuildProvider buildProvider in buildProviders) { if (IsBuildProviderSkipable(buildProvider)) continue; // Instruct the internal build providers to continue processing for more parse errors. if (!BuildManager.ThrowOnFirstParseError) { InternalBuildProvider provider = buildProvider as InternalBuildProvider; if (provider != null) { provider.ThrowOnFirstParseError = false; } } CompilerType compilerType = null; // Get the language try { compilerType = BuildProvider.GetCompilerTypeFromBuildProvider( buildProvider); } catch (HttpParseException ex) { // Ignore the error if we are in that mode. if (_ignoreProvidersWithErrors) { continue; } hasParserErrors = true; // Remember the first parse exception if (_firstException == null) { _firstException = ex; } if (_parserErrors == null) { _parserErrors = new ParserErrorCollection(); } _parserErrors.AddRange(ex.ParserErrors); continue; } catch { // Ignore the error if we are in that mode. if (_ignoreProvidersWithErrors) { continue; } throw; } AssemblyBuilder assemblyBuilder = defaultAssemblyBuilder; ICollection typeNames = buildProvider.GetGeneratedTypeNames(); // Is it asking for a specific language? if (compilerType == null) { // If this provider doesn't need a specific language, and we haven't yet created // a default builder that is capable of building this, just keep track of it if (defaultAssemblyBuilder == null || defaultAssemblyBuilder.IsBatchFull || defaultAssemblyBuilder.ContainsTypeNames(typeNames)) { if (languageFreeBuildProviders == null) { languageFreeBuildProviders = new ArrayList(); } languageFreeBuildProviders.Add(buildProvider); continue; } } else { // Check if we already have an assembly builder of the right type assemblyBuilder = (AssemblyBuilder)assemblyBuilders[compilerType]; } // Starts a new assemblyBuilder if the old one already contains another buildprovider // that uses the same type name if (assemblyBuilder == null || assemblyBuilder.IsBatchFull || assemblyBuilder.ContainsTypeNames(typeNames)) { // If the assemblyBuilder is full, compile it. if (assemblyBuilder != null) { CompileAssemblyBuilder(assemblyBuilder); } AssemblyBuilder newBuilder = compilerType.CreateAssemblyBuilder( _compConfig, _referencedAssemblies); assemblyBuilders[compilerType] = newBuilder; // Remember it as the default if we don't already have one, // or if the default is already full, switch the default to the new one. if (defaultAssemblyBuilder == null || defaultAssemblyBuilder == assemblyBuilder) { defaultAssemblyBuilder = newBuilder; } assemblyBuilder = newBuilder; } assemblyBuilder.AddTypeNames(typeNames); assemblyBuilder.AddBuildProvider(buildProvider); } // Don't try to compile providers, otherwise compile exceptions will be bubbled up, // and we lose the parse errors. if (hasParserErrors) { return false; } // Handle all the left over language free providers if (languageFreeBuildProviders != null) { // Indicates whether the default assembly builder is not a language specific builder. bool newDefaultAssemblyBuilder = (defaultAssemblyBuilder == null); // Add language independent providers to the default assembly builder. foreach (BuildProvider languageFreeBuildProvider in languageFreeBuildProviders) { ICollection typeNames = languageFreeBuildProvider.GetGeneratedTypeNames(); // If we don't have a default language assembly builder, get one or // starts a new assemblyBuilder if the old one already contains another buildprovider // that uses the same type name if (defaultAssemblyBuilder == null || defaultAssemblyBuilder.IsBatchFull || defaultAssemblyBuilder.ContainsTypeNames(typeNames)) { // If the default assemblyBuilder is full, compile it. if (defaultAssemblyBuilder != null) { CompileAssemblyBuilder(defaultAssemblyBuilder); } defaultAssemblyBuilder = CompilerType.GetDefaultAssemblyBuilder( _compConfig, _referencedAssemblies, _vdir.VirtualPathObject /*configPath*/, null /*outputAssemblyName*/); // the default assembly builder needs to be compiled separately. newDefaultAssemblyBuilder = true; } defaultAssemblyBuilder.AddTypeNames(typeNames); defaultAssemblyBuilder.AddBuildProvider(languageFreeBuildProvider); } // Only compile the default assembly builder if it's not part of language specific // assembly builder (which will be compiled separately) if (newDefaultAssemblyBuilder) { // Compile the default assembly builder. CompileAssemblyBuilder(defaultAssemblyBuilder); } } CompileAssemblyBuilderParallel(assemblyBuilders.Values); return true; } private void CompileAssemblyBuilderParallel(ICollection assemblyBuilders) { int maxConcurrent = Math.Min(assemblyBuilders.Count, CompilationUtil.MaxConcurrentCompilations); if (maxConcurrent < 2) { // Not using Parallel.ForEach to avoid performance penalty foreach (AssemblyBuilder assemblyBuilder in assemblyBuilders) { CompileAssemblyBuilder(assemblyBuilder); } } else { // devdiv bug 666936: ASP.NET compilation related deadlock in Antares scenario. // The main (current) thread holds a global compilation lock. CacheAssemblyResults and CacheCompileErrors may // also require the global compilation lock in case of removing old data and thus may lead to deadlock. // Fix: using dictionaries to collect the build results from parallel threads and do caching in the main thread. ConcurrentDictionary buildResults = new ConcurrentDictionary(); ConcurrentDictionary buildErrors = new ConcurrentDictionary(); try { Parallel.ForEach(assemblyBuilders.Cast(), new ParallelOptions { MaxDegreeOfParallelism = maxConcurrent }, builder => { CompilerResults results; try { results = builder.Compile(); } catch (HttpCompileException e) { buildErrors[builder] = e.Results; throw; } buildResults[builder] = results; }); } catch (AggregateException e) { ExceptionDispatchInfo.Capture(e.GetBaseException()).Throw(); } finally { // Before throwing the aggregated compilation exception, cache the build results first // This follows the execution order for the single thread case foreach (var pair in buildErrors) { CacheCompileErrors(pair.Key, pair.Value); } foreach (var pair in buildResults) { CacheAssemblyResults(pair.Key, pair.Value); } } } } private void CompileAssemblyBuilder(AssemblyBuilder builder) { CompilerResults results; try { results = builder.Compile(); } catch (HttpCompileException e) { CacheCompileErrors(builder, e.Results); throw; } CacheAssemblyResults(builder, results); } } }