//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.Compilation { using System; using System.CodeDom; using System.CodeDom.Compiler; using System.Collections; using System.Collections.Generic; using System.Collections.Specialized; using System.Diagnostics.CodeAnalysis; using System.Globalization; using System.IO; using System.Reflection; using System.Reflection.Emit; using System.Security; using System.Security.Cryptography; using System.Security.Permissions; using System.Text; using System.Web.Configuration; using System.Web.Hosting; using System.Web.Management; using System.Web.Security.Cryptography; using System.Web.UI; using System.Web.Util; using System.Xml; using System.Xml.Schema; /* * This class is used to handle a single compilation using a CodeDom compiler. * It is instantiated via CompilerType.CreateAssemblyBuilder. */ public class AssemblyBuilder { private CompilationSection _compConfig; // CodeChecksumPragma.ChecksumAlgorithmId takes this GUID to represent a SHA1 hash of the file contents // See: http://msdn.microsoft.com/en-us/library/system.codedom.codechecksumpragma.checksumalgorithmid.aspx private static readonly Guid s_codeChecksumSha1Id = new Guid(0xff1816ec, 0xaa5e, 0x4d10, 0x87, 0xf7, 0x6f, 0x49, 0x63, 0x83, 0x34, 0x60); // List of BuildProviders involved in this compilation // The key is either a virtual path, or the BuildProvider itself if // it doesn't give us a virtual path. // The value is the BuildProvider. private Hashtable _buildProviders = new Hashtable(StringComparer.OrdinalIgnoreCase); internal ICollection BuildProviders { get { return _buildProviders.Values; } } // List of physical source files to be compiled private StringSet _sourceFiles = new StringSet(); // CodeCompileUnit to hold various top level things we need to generate private CodeCompileUnit _miscCodeCompileUnit; // List of physical embedded resource files to be compiled private StringSet _embeddedResourceFiles; // The set of assemblies that we will be linked with private AssemblySet _initialReferencedAssemblies; // The additional set of assemblies that we will be linked with, and that are // requested by various BuildProviders. We need to keep them separate to avoid // having BuildProviders see assemblies that were requested by earlier providers, // which would lead to unpredictable behavior (since the order is arbitrary) private AssemblySet _additionalReferencedAssemblies; internal CodeDomProvider _codeProvider; private Hashtable _buildProviderToSourceFileMap; // The type of CodeDom compiler (i.e. language, flags) private CompilerType _compilerType; internal Type CodeDomProviderType { get { return _compilerType.CodeDomProviderType; } } // Used to generate fast Type factories private ObjectFactoryCodeDomTreeGenerator _objectFactoryGenerator; private StringResourceBuilder _stringResourceBuilder; internal StringResourceBuilder StringResourceBuilder { get { if (_stringResourceBuilder == null) _stringResourceBuilder = new StringResourceBuilder(); return _stringResourceBuilder; } } // Used to create temporary source files private TempFileCollection _tempFiles = new TempFileCollection(HttpRuntime.CodegenDirInternal); private int _fileCount; private string _cultureName; internal string CultureName { get { return _cultureName; } set { _cultureName = value; } } private string _outputAssemblyName; private string OutputAssemblyName { get { if (_outputAssemblyName == null) { // If we don't have the assembly name, we should never have a culture Debug.Assert(CultureName == null); // If the assembly name was not specified, use a generated one based on the TempFileCollection. // But prefix it with a fixed token, to make it easier to recognize the assembly (DevDiv 36625) string basePath = _tempFiles.BasePath; string baseFileName = Path.GetFileName(basePath); _outputAssemblyName = BuildManager.WebAssemblyNamePrefix + baseFileName; } return _outputAssemblyName; } } private int _maxBatchSize; private long _maxBatchGeneratedFileSize; private long _totalFileLength; private CaseInsensitiveStringSet _registeredTypeNames; internal bool ContainsTypeNames(ICollection typeNames) { if (_registeredTypeNames != null && typeNames != null) { foreach (String typeName in typeNames) { if (_registeredTypeNames.Contains(typeName)) { return true; } } } return false; } internal void AddTypeNames(ICollection typeNames) { if (typeNames == null) { return; } if (_registeredTypeNames == null) { _registeredTypeNames = new CaseInsensitiveStringSet(); } _registeredTypeNames.AddCollection(typeNames); } internal AssemblyBuilder(CompilationSection compConfig, ICollection referencedAssemblies, CompilerType compilerType, string outputAssemblyName) { _compConfig = compConfig; _outputAssemblyName = outputAssemblyName; // Clone the referenced assemblies _initialReferencedAssemblies = AssemblySet.Create(referencedAssemblies); // We need to clone it to avoid modifying the original (VSWhidbey 338935) _compilerType = compilerType.Clone(); if (BuildManager.PrecompilingWithDebugInfo) { // If the precompile flag indicates force debug, always compile as debug _compilerType.CompilerParameters.IncludeDebugInformation = true; } else if (BuildManager.PrecompilingForDeployment) { // If we're precompiling the app, never compile in debug mode (VSWhidbey 178377) _compilerType.CompilerParameters.IncludeDebugInformation = false; } else if (DeploymentSection.RetailInternal) { // If we're in retail deployment mode, always turn off debug (DevDiv 36396) _compilerType.CompilerParameters.IncludeDebugInformation = false; } else if (_compConfig.AssemblyPostProcessorTypeInternal != null) { // If an IAssemblyPostProcessor is registered always compile as debug _compilerType.CompilerParameters.IncludeDebugInformation = true; } // _tempFiles.KeepFiles = _compilerType.CompilerParameters.IncludeDebugInformation; _codeProvider = CompilationUtil.CreateCodeDomProviderNonPublic( _compilerType.CodeDomProviderType); _maxBatchSize = _compConfig.MaxBatchSize; _maxBatchGeneratedFileSize = _compConfig.MaxBatchGeneratedFileSize * 1024; } // Beginning of public contract /// /// Adds an assembly that will be referenced during compilation. /// public void AddAssemblyReference(Assembly a) { if (_additionalReferencedAssemblies == null) _additionalReferencedAssemblies = new AssemblySet(); _additionalReferencedAssemblies.Add(a); } /// /// Adds an assembly that will be referenced during compilation. Also adds the /// assembly the the ReferencedAssemblies list in the CodeCompileUnit. /// internal void AddAssemblyReference(Assembly a, CodeCompileUnit ccu) { AddAssemblyReference(a); Util.AddAssemblyToStringCollection(a, ccu.ReferencedAssemblies); } /// /// Creates a new source file that will be added to the compilation. See the public overload /// method for detail. /// internal virtual TextWriter CreateCodeFile(BuildProvider buildProvider, out string filename) { string generatedFilePath = GetTempFilePhysicalPathWithAssert(_codeProvider.FileExtension); filename = generatedFilePath; if (buildProvider != null) { if (_buildProviderToSourceFileMap == null) _buildProviderToSourceFileMap = new Hashtable(); _buildProviderToSourceFileMap[buildProvider] = generatedFilePath; buildProvider.SetContributedCode(); } _sourceFiles.Add(generatedFilePath); return CreateCodeFileWithAssert(generatedFilePath); } [FileIOPermission(SecurityAction.Assert, Unrestricted = true)] private StreamWriter CreateCodeFileWithAssert(string generatedFilePath) { Stream temp = new FileStream(generatedFilePath, FileMode.Create, FileAccess.Write, FileShare.Read); return new StreamWriter(temp, Encoding.UTF8); } /// /// Creates a new source file that will be added to the compilation. The build provider /// can write source code to this file using the returned TextWriter. /// The build provider should close the TextWriter when it is done writing to it. /// The build provider should pass itself as a parameter to this method. /// public TextWriter CreateCodeFile(BuildProvider buildProvider) { // Ignore the unused filename param. string filename; return CreateCodeFile(buildProvider, out filename); } // Indicates whether the assemblyBuilder has reached its capacity limit. internal bool IsBatchFull { get { return (_sourceFiles.Count >= _maxBatchSize) || (_totalFileLength >= _maxBatchGeneratedFileSize); } } /// /// Adds a CodeCompileUnit to the compilation. This is typically used as an /// alternative to CreateSourceFile, by providers who are CodeDOM aware. /// The build provider should pass itself as a parameter to this method. /// public void AddCodeCompileUnit(BuildProvider buildProvider, CodeCompileUnit compileUnit) { // Add a checksum pragma to the compile unit if appropriate AddChecksumPragma(buildProvider, compileUnit); // Add all the referenced assemblies to the CodeCompileUnit in case the CodeDom // provider needs them for code generation Util.AddAssembliesToStringCollection(_initialReferencedAssemblies, compileUnit.ReferencedAssemblies); // Merge the _additionalReferencedAssemblies from individul build providers Util.AddAssembliesToStringCollection(_additionalReferencedAssemblies, compileUnit.ReferencedAssemblies); String filename; // Revert impersonation when generating source code in the codegen dir (VSWhidbey 176576) using (new ProcessImpersonationContext()) { TextWriter writer = CreateCodeFile(buildProvider, out filename); try { _codeProvider.GenerateCodeFromCompileUnit(compileUnit, writer, null /*CodeGeneratorOptions*/); } finally { writer.Flush(); writer.Close(); } } if (filename != null) { _totalFileLength += GetFileLengthWithAssert(filename); } } // Assert to be able to get the length of the file in the CodeGen dir [FileIOPermission(SecurityAction.Assert, AllFiles = FileIOPermissionAccess.Read)] private long GetFileLengthWithAssert(string filename) { FileInfo info = new FileInfo(filename); return info.Length; } /// /// Tell the host about a type that is being generated. This allows the host /// To generate a fast object factory for it. /// public void GenerateTypeFactory(string typeName) { // Create the object factory generator on demand if (_objectFactoryGenerator == null) { _objectFactoryGenerator = new ObjectFactoryCodeDomTreeGenerator(OutputAssemblyName); } // Add a method to fast create this type _objectFactoryGenerator.AddFactoryMethod(typeName); } /// /// Creates a new resource that will be added to the compilation. The build provider /// can write to it using the returned Stream. /// The build provider should close the Stream when it is done writing to it. /// The build provider should pass itself as a parameter to this method. /// public Stream CreateEmbeddedResource(BuildProvider buildProvider, string name) { // Make sure it's just a valid simple file name if (!Util.IsValidFileName(name)) { throw new ArgumentException(null, name); } string resourceDir = BuildManager.CodegenResourceDir; string resourceFile = Path.Combine(resourceDir, name); CreateTempResourceDirectoryIfNecessary(); _tempFiles.AddFile(resourceFile, _tempFiles.KeepFiles); if (_embeddedResourceFiles == null) _embeddedResourceFiles = new StringSet(); _embeddedResourceFiles.Add(resourceFile); // Assert to be able to create the file in the temp dir InternalSecurityPermissions.FileWriteAccess(resourceDir).Assert(); return File.OpenWrite(resourceFile); } [FileIOPermission(SecurityAction.Assert, Unrestricted = true)] private void CreateTempResourceDirectoryIfNecessary() { // Create the temp resource directory if needed string resourceDir = BuildManager.CodegenResourceDir; if (!FileUtil.DirectoryExists(resourceDir)) { Directory.CreateDirectory(resourceDir); } } /// /// Returns a CodeDomProvider that the build provider can use to generate a CodeCompileUnit. /// public CodeDomProvider CodeDomProvider { get { return _codeProvider; } } private string _tempFilePhysicalPathPrefix; private string TempFilePhysicalPathPrefix { get { if (_tempFilePhysicalPathPrefix == null) { _tempFilePhysicalPathPrefix = Path.Combine(_tempFiles.TempDir, OutputAssemblyName) + "."; // Append the culture name to avoid naming conflicts if (CultureName != null) { _tempFilePhysicalPathPrefix += CultureName + "_"; } } return _tempFilePhysicalPathPrefix; } } /// /// Returns the physical path to a temporary file that the build provider /// can use for intermediate results. Note that the file is not actually /// created. It is up to the build provider to do this. /// The temp file's extension is passed in by the build provider. /// The file is automatically deleted after the compilation, so the /// build provider does not need to explicitly delete it. /// public string GetTempFilePhysicalPath(string extension) { // Do the right thing depending on whether the extension include the starting '.' string tempPath; if (!String.IsNullOrEmpty(extension) && extension[0] == '.') { tempPath = TempFilePhysicalPathPrefix + ((_fileCount++) + extension); } else { tempPath = TempFilePhysicalPathPrefix + ((_fileCount++) + "." + extension); } _tempFiles.AddFile(tempPath, _tempFiles.KeepFiles); InternalSecurityPermissions.PathDiscovery(tempPath).Demand(); return tempPath; } // End of public contract [FileIOPermission(SecurityAction.Assert, Unrestricted = true)] internal string GetTempFilePhysicalPathWithAssert(string extension) { return GetTempFilePhysicalPath(extension); } private void AddCompileWithBuildProvider(VirtualPath virtualPath, BuildProvider owningBuildProvider) { BuildProvider buildProvider = BuildManager.CreateBuildProvider(virtualPath, _compConfig, _initialReferencedAssemblies, true /*failIfUnknown*/); // Since it's referenced via compileWith, it doesn't need its own build result buildProvider.SetNoBuildResult(); // If it's a CompileWith provider, remember the main provider SourceFileBuildProvider sourceBuildProvider = buildProvider as SourceFileBuildProvider; if (sourceBuildProvider != null) sourceBuildProvider.OwningBuildProvider = owningBuildProvider; AddBuildProvider(buildProvider); } internal virtual void AddBuildProvider(BuildProvider buildProvider) { // By default, use the build provider itself as the key object hashtableKey = buildProvider; bool isFolderLevel = false; // If the buildProvider is a folderLevel build provider, use the build provider itself // so that multiple build providers can work on the same path. if (_compConfig.FolderLevelBuildProviders != null) { Type t = buildProvider.GetType(); isFolderLevel = _compConfig.FolderLevelBuildProviders.IsFolderLevelBuildProvider(t); } // Keep track of the build provider's virtual path, if any if (buildProvider.VirtualPath != null && !isFolderLevel) { // It has a virtual path, so use that as the key hashtableKey = buildProvider.VirtualPath; // If we already had it, ignore it. This can happen when there is a user control // with a code beside in App_Code (VSWhidbey 481426) if (_buildProviders.ContainsKey(hashtableKey)) return; } _buildProviders[hashtableKey] = buildProvider; // Ask the provider to generate the code // If it throws an Xml exception, extra the relevant info and turn it // into our own ParseException try { buildProvider.GenerateCode(this); } catch (XmlException e) { throw new HttpParseException(e.Message, null /*innerException*/, buildProvider.VirtualPath, null /*sourceCode*/, e.LineNumber); } catch (XmlSchemaException e) { throw new HttpParseException(e.Message, null /*innerException*/, buildProvider.VirtualPath, null /*sourceCode*/, e.LineNumber); } catch (Exception e) { throw new HttpParseException(e.Message, e, buildProvider.VirtualPath, null /*sourceCode*/, 1); } // Handle any 'compileWith' dependencies, i.e. files that must be compiled // within the same assembly as the current file InternalBuildProvider internalBuildProvider = buildProvider as InternalBuildProvider; if (internalBuildProvider != null) { ICollection compileWith = internalBuildProvider.GetCompileWithDependencies(); if (compileWith != null) { foreach (VirtualPath virtualPath in compileWith) { // If we already have it, ignore it if (_buildProviders.ContainsKey(virtualPath.VirtualPathString)) continue; // Add the compileWith dependency to our compilation AddCompileWithBuildProvider(virtualPath, internalBuildProvider); } } } } private void AddAssemblyCultureAttribute() { if (CultureName == null) return; CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(System.Reflection.AssemblyCultureAttribute)), new CodeAttributeArgument[] { new CodeAttributeArgument(new CodePrimitiveExpression(CultureName))}); AddAssemblyAttribute(declaration); } private void AddAspNetGeneratedCodeAttribute() { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(GeneratedCodeAttribute))); declaration.Arguments.Add(new CodeAttributeArgument(new CodePrimitiveExpression("ASP.NET"))); declaration.Arguments.Add(new CodeAttributeArgument(new CodePrimitiveExpression(VersionInfo.SystemWebVersion))); AddAssemblyAttribute(declaration); } private void AddAllowPartiallyTrustedCallersAttribute() { if (BuildManager.CompileWithAllowPartiallyTrustedCallersAttribute) { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(AllowPartiallyTrustedCallersAttribute))); AddAssemblyAttribute(declaration); } } private void AddAssemblyKeyFileAttribute() { if (!String.IsNullOrEmpty(BuildManager.StrongNameKeyFile)) { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(AssemblyKeyFileAttribute)), new CodeAttributeArgument(new CodePrimitiveExpression(BuildManager.StrongNameKeyFile))); AddAssemblyAttribute(declaration); } } private void AddAssemblyKeyContainerAttribute() { if (!String.IsNullOrEmpty(BuildManager.StrongNameKeyContainer)) { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(AssemblyKeyNameAttribute)), new CodeAttributeArgument(new CodePrimitiveExpression(BuildManager.StrongNameKeyContainer))); AddAssemblyAttribute(declaration); } } private void AddAssemblyDelaySignAttribute() { if (BuildManager.CompileWithDelaySignAttribute) { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(AssemblyDelaySignAttribute)), new CodeAttributeArgument(new CodePrimitiveExpression(true))); AddAssemblyAttribute(declaration); } } private void AddSecurityRulesAttribute() { // Skip applying the attribute if targeting 2.0/3.5, since the attribute // is only available in 4.0 and above. if (MultiTargetingUtil.IsTargetFramework20 || MultiTargetingUtil.IsTargetFramework35) { return; } TrustSection trustSection = RuntimeConfig.GetAppConfig().Trust; CodeAttributeDeclaration declaration; Type attrType = typeof(SecurityRulesAttribute); Type enumType = typeof(SecurityRuleSet); if (trustSection.LegacyCasModel) { SecurityRuleSet set = SecurityRuleSet.Level1; string fieldName = Enum.GetName(enumType, set); CodeFieldReferenceExpression field = new CodeFieldReferenceExpression(new CodeTypeReferenceExpression(enumType), fieldName); declaration = new CodeAttributeDeclaration(new CodeTypeReference(attrType), new CodeAttributeArgument(field)); AddAssemblyAttribute(declaration); } else { SecurityRuleSet set = SecurityRuleSet.Level2; string fieldName = Enum.GetName(enumType, set); CodeFieldReferenceExpression field = new CodeFieldReferenceExpression(new CodeTypeReferenceExpression(enumType), fieldName); declaration = new CodeAttributeDeclaration(new CodeTypeReference(attrType), new CodeAttributeArgument(field)); AddAssemblyAttribute(declaration); } } private void AddTargetFrameworkAttribute() { if (MultiTargetingUtil.TargetFrameworkVersion.Major >= 4) { CodeAttributeDeclaration declaration = new CodeAttributeDeclaration( new CodeTypeReference(typeof(System.Runtime.Versioning.TargetFrameworkAttribute)), new CodeAttributeArgument(new CodePrimitiveExpression(BuildManager.TargetFramework.FullName))); AddAssemblyAttribute(declaration); } } // Add an assembly level attribute to the assembly private void AddAssemblyAttribute(CodeAttributeDeclaration declaration) { if (_miscCodeCompileUnit == null) _miscCodeCompileUnit = new CodeCompileUnit(); _miscCodeCompileUnit.AssemblyCustomAttributes.Add(declaration); } private void GenerateMiscCodeCompileUnit() { // If there aren't any, return if (_miscCodeCompileUnit == null) return; AddCodeCompileUnit(null /*buildProvider*/, _miscCodeCompileUnit); } // Add a checksum pragma. This is used for improved debugging experience. private void AddChecksumPragma(BuildProvider buildProvider, CodeCompileUnit compileUnit) { // If we can't get a virtual path, do nothing if (buildProvider == null || buildProvider.VirtualPath == null) return; // Only do this if we're compiling in debug mode if (!_compilerType.CompilerParameters.IncludeDebugInformation) return; string physicalPath = HostingEnvironment.MapPathInternal(buildProvider.VirtualPath); // Only do this is the file physically exists, which it would not in the // case of a non-file based VirtualPathProvider. In such case, there is // no point in putting the pragma, since the debugger could not locate // the file anyway. if (!File.Exists(physicalPath)) return; CodeChecksumPragma pragma = new CodeChecksumPragma() { ChecksumAlgorithmId = s_codeChecksumSha1Id }; if (_compConfig.UrlLinePragmas) { pragma.FileName = ErrorFormatter.MakeHttpLinePragma(buildProvider.VirtualPathObject.VirtualPathString); } else { pragma.FileName = physicalPath; } // Generate a SHA1 hash from the contents of the file // The VS debugger uses a cryptographic hash of the file being debugged so that it doesn't accidentally // display to the user the wrong version of the file. This is merely a convenience feature for debugging // purposes and is not security-related in any way. Since VS only supports MD5 and SHA1 hashes, we just // use SHA1 and suppress the [Obsolete] warning. #pragma warning disable 618 using (Stream stream = new FileStream(physicalPath, FileMode.Open, FileAccess.Read, FileShare.Read)) { using (SHA1 hashAlgorithm = CryptoAlgorithms.CreateSHA1()) { pragma.ChecksumData = hashAlgorithm.ComputeHash(stream); } } #pragma warning restore 618 // Add the pragma to the CodeCompileUnit compileUnit.StartDirectives.Add(pragma); } internal CompilerParameters GetCompilerParameters() { CompilerParameters compilParams = _compilerType.CompilerParameters; string dir = _tempFiles.TempDir; // If a culture is set, modify the assembly name and location based on it if (CultureName != null) { dir = Path.Combine(dir, CultureName); Directory.CreateDirectory(dir); compilParams.OutputAssembly = Path.Combine(dir, OutputAssemblyName + ".resources.dll"); } else { compilParams.OutputAssembly = Path.Combine(dir, OutputAssemblyName + ".dll"); } // If such file already exist, try to delete or rename it if (File.Exists(compilParams.OutputAssembly)) Util.RemoveOrRenameFile(compilParams.OutputAssembly); compilParams.TempFiles = _tempFiles; // Create the string resource file (shared by all the pages we're compiling) if (_stringResourceBuilder != null && _stringResourceBuilder.HasStrings) { string resFileName = _tempFiles.AddExtension("res"); _stringResourceBuilder.CreateResourceFile(resFileName); compilParams.Win32Resource = resFileName; } // Add all the embedded resources to the compilParams if (_embeddedResourceFiles != null) { foreach (string aname in _embeddedResourceFiles) compilParams.EmbeddedResources.Add(aname); } // Merge the two sets of assemblies if (_additionalReferencedAssemblies != null) { foreach (Assembly assembly in _additionalReferencedAssemblies) { _initialReferencedAssemblies.Add(assembly); } } // Add all the referenced assemblies to the compilParams Util.AddAssembliesToStringCollection(_initialReferencedAssemblies, compilParams.ReferencedAssemblies); // Make any fix up adjustments to the CompilerParameters to work around some issues FixUpCompilerParameters(_compConfig, _compilerType.CodeDomProviderType, compilParams); return compilParams; } static string s_vbImportsString; private static void AddVBGlobalNamespaceImports(CompilerParameters compilParams) { // Put together the VB import string on demand if (s_vbImportsString == null) { PagesSection pagesConfig = MTConfigUtil.GetPagesAppConfig(); if (pagesConfig.Namespaces == null) { s_vbImportsString = String.Empty; } else { StringBuilder sb = new StringBuilder(); sb.Append("/imports:"); bool nextItemNeedsComma = false; // Auto-import Microsoft.VisualBasic is needed if (pagesConfig.Namespaces.AutoImportVBNamespace) { sb.Append("Microsoft.VisualBasic"); nextItemNeedsComma = true; } // Add all the namespaces from the config section foreach (NamespaceInfo entry in pagesConfig.Namespaces) { // If there was a previous entry, we need a comma separator if (nextItemNeedsComma) sb.Append(','); sb.Append(entry.Namespace); nextItemNeedsComma = true; } s_vbImportsString = sb.ToString(); } } // Prepend it to the compilerOptions if (s_vbImportsString.Length > 0) { if (compilParams.CompilerOptions == null) compilParams.CompilerOptions = s_vbImportsString; else compilParams.CompilerOptions = s_vbImportsString + " " + compilParams.CompilerOptions; } } // Command line string for My.* support private const string MySupport = @"/define:_MYTYPE=\""Web\"""; private static void AddVBMyFlags(CompilerParameters compilParams) { // Prepend it to the compilerOptions if (compilParams.CompilerOptions == null) compilParams.CompilerOptions = MySupport; else compilParams.CompilerOptions = MySupport + " " + compilParams.CompilerOptions; } [SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Justification = "Warnings was about use of CompilerParameters.CoreAssemblyFileName which was not set by user supplied string - so okay.")] internal static void FixUpCompilerParameters(CompilationSection compilationSection, Type codeDomProviderType, CompilerParameters compilParams) { // The mscorlib reference is special cased, and needs to be passed via the CoreAssemblyFileName property. if (BuildManagerHost.InClientBuildManager && !MultiTargetingUtil.IsTargetFramework20 && !MultiTargetingUtil.IsTargetFramework35) { string coreAssemblyFile; AssemblyResolver.GetPathToReferenceAssembly(typeof(string).Assembly, out coreAssemblyFile); compilParams.CoreAssemblyFileName = coreAssemblyFile; } // DevDiv 404267: If the developer enabled 'warnings as errors', we should disable [Obsolete] warnings. This helps // prevent in-place framework updates from breaking runtime compilation of pages. We only respect this attribute // when not in the CBM, as CBM is a design-time feature instead of a runtime feature, and the developer probably // wants to be notified of all errors at design time. bool disableObsoleteWarnings = !BuildManagerHost.InClientBuildManager && compilationSection.DisableObsoleteWarnings; // If C#, remove the warning that complains about variables that start with "__" // Also ignore warning that complains about assemblyKeyName and delaysign // Also ignore warning about assuming assembly versions matching (CS1701, DevDiv 137847, warning about System.Web.Extensions v1.0 matching v3.5) if (codeDomProviderType == typeof(Microsoft.CSharp.CSharpCodeProvider)) { List noWarnStrings = new List(5); noWarnStrings.AddRange(new string[] { "1659", "1699", "1701" }); if (disableObsoleteWarnings) { noWarnStrings.Add("612"); // [Obsolete] without message noWarnStrings.Add("618"); // [Obsolete("with message")] } CodeDomUtility.PrependCompilerOption(compilParams, "/nowarn:" + String.Join(";", noWarnStrings)); } else if (codeDomProviderType == typeof(Microsoft.VisualBasic.VBCodeProvider)) { List noWarnStrings = new List(3); // If VB, add all the imported namespaces on the command line (DevDiv 21499). // This is VB only because other languages don't support global command line // namespace imports. AddVBGlobalNamespaceImports(compilParams); // Add any command line flags needed to support the My.* feature AddVBMyFlags(compilParams); // Ignore vb warning that complains about assemblyKeyName (Dev10 662544) // but only for target 3.5 and above (715329) if (MultiTargetingUtil.TargetFrameworkVersion >= MultiTargetingUtil.Version35) { noWarnStrings.Add("41008"); } if (disableObsoleteWarnings) { noWarnStrings.Add("40000"); // [Obsolete("with message")] noWarnStrings.Add("40008"); // [Obsolete] without message } if (noWarnStrings.Count > 0) { CodeDomUtility.PrependCompilerOption(compilParams, "/nowarn:" + String.Join(",", noWarnStrings)); } } ProcessProviderOptions(codeDomProviderType, compilParams); FixTreatWarningsAsErrors(codeDomProviderType, compilParams); // Add CodeAnalysis symbol if required by client. if (BuildManager.PrecompilingWithCodeAnalysisSymbol) { CodeDomUtility.PrependCompilerOption(compilParams, "/define:CODE_ANALYSIS"); } } // DevDiv 114316 // CodeDom sets TreatWarningAsErrors to true whenever warningLevel is non-zero. // To get warnings only, the workaround is to use /warnaserror- in CompilerOptions. // However this does not work in some cases, as TreatWarningAsErrors set to true still emits // /warnaserror+. // So, whenever the user wants /warnaserror[+|-|numberlist], we explicitly set TreatWarningsAsErrors to false, // so that the /warnaserror+ is not emitted, and the user can specify exactly what is desired. internal static void FixTreatWarningsAsErrors(Type codeDomProviderType, CompilerParameters compilParams) { // Only do so for C# and VB. if (codeDomProviderType != typeof(Microsoft.CSharp.CSharpCodeProvider) && codeDomProviderType != typeof(Microsoft.VisualBasic.VBCodeProvider)) return; if (CultureInfo.InvariantCulture.CompareInfo.IndexOf(compilParams.CompilerOptions, "/warnaserror", CompareOptions.IgnoreCase) >= 0) compilParams.TreatWarningsAsErrors = false; } // Check for OptionInfer and WarnAsError. This is the workaround as use of compilerOptions is not allowed in partial trust. // Devdiv 130325 // private static void ProcessProviderOptions(Type codeDomProviderType, CompilerParameters compilParams) { IDictionary providerOptions = CompilationUtil.GetProviderOptions(codeDomProviderType); if (providerOptions == null) return; // For C# and VB, check for WarnAsError if (codeDomProviderType == typeof(Microsoft.VisualBasic.VBCodeProvider) || codeDomProviderType == typeof(Microsoft.CSharp.CSharpCodeProvider)) ProcessBooleanProviderOption("WarnAsError", "/warnaserror+", "/warnaserror-", providerOptions, compilParams); // Only process OptionInfer for v3.5 compiler (or above) if (codeDomProviderType == null || !CompilationUtil.IsCompilerVersion35OrAbove(codeDomProviderType)) return; // For VB, check for OptionInfer if (codeDomProviderType == typeof(Microsoft.VisualBasic.VBCodeProvider)) ProcessBooleanProviderOption("OptionInfer", "/optionInfer+", "/optionInfer-", providerOptions, compilParams); } private static void ProcessBooleanProviderOption(string providerOptionName, string trueCompilerOption, string falseCompilerOption, IDictionary providerOptions, CompilerParameters compilParams) { if (providerOptions == null || compilParams == null) return; Debug.Assert(providerOptionName != null, "providerOptionName should not be null"); Debug.Assert(trueCompilerOption != null, "trueCompilerOption should not be null"); Debug.Assert(falseCompilerOption != null, "falseCompilerOption should not be null"); string providerOptionValue = null; if (!providerOptions.TryGetValue(providerOptionName, out providerOptionValue)) return; if (string.IsNullOrEmpty(providerOptionValue)) throw new System.Configuration.ConfigurationErrorsException(SR.GetString(SR.Property_NullOrEmpty, CompilationUtil.CodeDomProviderOptionPath + providerOptionName)); bool value; if (Boolean.TryParse(providerOptionValue, out value)) { // If the value is boolean, insert the compiler options if (value) CodeDomUtility.AppendCompilerOption(compilParams, trueCompilerOption); else CodeDomUtility.AppendCompilerOption(compilParams, falseCompilerOption); } else { // If the value is not boolean, throw an exception throw new System.Configuration.ConfigurationErrorsException(SR.GetString(SR.Value_must_be_boolean, CompilationUtil.CodeDomProviderOptionPath + providerOptionName)); } } internal CompilerResults Compile() { // First, check if there is something to compile if (_sourceFiles.Count == 0 && _embeddedResourceFiles == null) return null; // if we have some fast object factories to generate, get the CodeCompileUnit if (_objectFactoryGenerator != null) { _miscCodeCompileUnit = _objectFactoryGenerator.CodeCompileUnit; } // Add a culture attribute if needed AddAssemblyCultureAttribute(); // Add a ComVisible(false) attribute (VSWhidbey 436453) // Actually, don't do it to avoid breaking migrated apps (VSWhidbey 446788) //AddComVisibleAttribute(); // Add an AspNetGeneratedCode attribute to help fxcop ignore some violations (VSWhidbey 437581) AddAspNetGeneratedCodeAttribute(); // Add an AllowPartiallyTrustedCallers attribute to make strong-name assemblies. (Devdiv 39696) AddAllowPartiallyTrustedCallersAttribute(); AddAssemblyDelaySignAttribute(); AddAssemblyKeyFileAttribute(); AddAssemblyKeyContainerAttribute(); AddSecurityRulesAttribute(); AddTargetFrameworkAttribute(); // Generate a source file for the misc top level items if needed GenerateMiscCodeCompileUnit(); CompilerParameters compilParams = GetCompilerParameters(); string[] files = new string[_sourceFiles.Count]; _sourceFiles.CopyTo(files, 0); // Increment compilation counter PerfCounters.IncrementCounter(AppPerfCounter.COMPILATIONS); // Raise Web Event WebBaseEvent.RaiseSystemEvent(this, WebEventCodes.ApplicationCompilationStart); HttpContext context = HttpContext.Current; if (context != null) { if (EtwTrace.IsTraceEnabled(EtwTraceLevel.Verbose, EtwTraceFlags.Infrastructure)) EtwTrace.Trace(EtwTraceType.ETW_TYPE_COMPILE_ENTER, context.WorkerRequest); } CompilerResults results = null; try { try { // Revert impersonation when compiling source code in the codegen dir (VSWhidbey 176576) using (new ProcessImpersonationContext()) { results = _codeProvider.CompileAssemblyFromFile(compilParams, files); } } finally { if (EtwTrace.IsTraceEnabled(EtwTraceLevel.Verbose, EtwTraceFlags.Infrastructure) && context != null) { string fileNames = null; if (_buildProviders.Count < 20) { IDictionaryEnumerator e = _buildProviders.GetEnumerator(); while(e.MoveNext()) { if (fileNames != null) fileNames += ","; fileNames += e.Key; } } else { fileNames = String.Format(CultureInfo.InstalledUICulture, SR.Resources.GetString(SR.Etw_Batch_Compilation, CultureInfo.InstalledUICulture), new object[1] {_buildProviders.Count}); } string status; if (results != null && (results.NativeCompilerReturnValue != 0 || results.Errors.HasErrors)) status = SR.Resources.GetString(SR.Etw_Failure, CultureInfo.InstalledUICulture); else status = SR.Resources.GetString(SR.Etw_Success, CultureInfo.InstalledUICulture); EtwTrace.Trace(EtwTraceType.ETW_TYPE_COMPILE_LEAVE, context.WorkerRequest, fileNames, status); } } } catch { throw; } // Prevent Exception Filter Security Issue (ASURT 122835) // If an IAssemblyPostProcessor is registered, call it Type postProcessorType = _compConfig.AssemblyPostProcessorTypeInternal; if (postProcessorType != null) { using (IAssemblyPostProcessor postProcessor = (IAssemblyPostProcessor) HttpRuntime.FastCreatePublicInstance(postProcessorType)) { postProcessor.PostProcessAssembly(results.PathToAssembly); } } // Raise Web Event WebBaseEvent.RaiseSystemEvent(this, WebEventCodes.ApplicationCompilationEnd); if (results != null) { // Invalidate an invalid assembly to trigger recompilation InvalidateInvalidAssembly(results, compilParams); // Fix up with line pragmas to account for the http case, and for some special conditions FixUpLinePragmas(results); if (results.Errors.HasErrors) { // Give all the BuildProviders a chance to look at the compile errors, and possibly tweak them foreach (BuildProvider buildProvider in BuildProviders) { buildProvider.ProcessCompileErrors(results); } } // If there is a CBM callback, inform it of the errors/warnings if (BuildManager.CBMCallback != null) { foreach (CompilerError error in results.Errors) { BuildManager.CBMCallback.ReportCompilerError(error); } } // If there are errors, increment the relevant perf counters and throw if (results.NativeCompilerReturnValue != 0 || results.Errors.HasErrors) { // Increment the compilation error and total error counters PerfCounters.IncrementCounter(AppPerfCounter.ERRORS_COMPILING); PerfCounters.IncrementCounter(AppPerfCounter.ERRORS_TOTAL); throw new HttpCompileException(results, GetErrorSourceFileContents(results)); } } return results; } private void InvalidateInvalidAssembly(CompilerResults results, CompilerParameters compilParams) { // VSWhidbey 610291 // If target assembly gets locked, we invalidate the assembly, so that it does // not get used, and the next compilation will use a new assembly name. // CS0016 is the error code for "Could not write to output file 'file' -- 'reason'" if (results == null || !results.Errors.HasErrors) return; foreach (CompilerError error in results.Errors) { if (error.IsWarning) continue; if (StringUtil.EqualsIgnoreCase(error.ErrorNumber, "CS0016")){ // Also invalidate the base assembly if this is a localized resource assembly if (CultureName != null) { string dir = _tempFiles.TempDir; string baseAssemblyFile = Path.Combine(dir, OutputAssemblyName + ".dll"); DiskBuildResultCache.TryDeleteFile(new FileInfo(baseAssemblyFile)); } // Invalidate the target assembly DiskBuildResultCache.TryDeleteFile(compilParams.OutputAssembly); } } } /* * Fix up all the source files in the errors in case they are HTTP (VS compiler scenario). * Also, fix the error in case the base class was incorrect in the code beside model */ private void FixUpLinePragmas(CompilerResults results) { CompilerError badBaseClassError = null; // Go through the errors backwards so we can delete them as needed for (int i=results.Errors.Count-1; i>=0; i--) { CompilerError error = results.Errors[i]; string physicalPath = ErrorFormatter.ResolveHttpFileName(error.FileName); // Only replace it by the physical path if it actually exists, which may not // be the case when using a VirtualPathProvider if (File.Exists(physicalPath)) { error.FileName = physicalPath; // If it is our special marker line number, remember it and remove it. // We place the marker at two places: 1) before setting AppRelativeVirtualPath in the constructor, // and 2) before the method FrameworkInitialize. // For the generated method FrameworkInitialize, the method comes one line after // the marker, due to an additional line taken by the DebuggerNonUserCodeAttribute. (DevDiv 175681) if (error.Line == TemplateControlCodeDomTreeGenerator.badBaseClassLineMarker || (error.Line == TemplateControlCodeDomTreeGenerator.badBaseClassLineMarker + 1 && error.ErrorText != null && error.ErrorText.IndexOf("FrameworkInitialize", StringComparison.OrdinalIgnoreCase) >= 0)) { badBaseClassError = error; results.Errors.RemoveAt(i); } else if (error.Line > TemplateControlCodeDomTreeGenerator.badBaseClassLineMarker && error.Line < TemplateControlCodeDomTreeGenerator.badBaseClassLineMarker + 50) { // Also, if within range of it, remove it altogether results.Errors.RemoveAt(i); } } } // If we found our special marker error, we're most likely in a situation where // the class in the code beside file doesn't match the 'inherits' in the aspx/ascx, // or is missing the based type (or has the wrong base type). In that case, change // the error message to make the problem explicit to the user (VSWhidbey 376977/468830) if (badBaseClassError != null) { // Read the content of the code beside file string codeFileContent = Util.StringFromFile(badBaseClassError.FileName); // Search for the partial class declaration within the file. We do this by searching for // the string "partial class" in case insensitive way. This is far from fool proof, but // it covers the common VB and C# cases, and the fallback when not found is reasonable. int classOffset = CultureInfo.InvariantCulture.CompareInfo.IndexOf(codeFileContent, "partial class", CompareOptions.IgnoreCase); if (classOffset >= 0) { // We found it, so figure out the line number from it badBaseClassError.Line = Util.LineCount(codeFileContent, 0, classOffset) + 1; } else { // Otherwise, just use 1. It won't point to the right line, but at least the error // message is helpful badBaseClassError.Line = 1; } // Change the error message to make the situation clear to the user badBaseClassError.ErrorText = SR.GetString(SR.Bad_Base_Class_In_Code_File); badBaseClassError.ErrorNumber = "ASPNET"; // Insert the error at the begining of the collection, since we display the first error. results.Errors.Insert(0, badBaseClassError); } } /* * Attempt to find the generated source file that has the error, and return * its contents as a string (for error reproting purposes). * Note that when debug is false, we set tempFiles.KeepFiles to false, and * all the sources will be gone by the time we get here. I filed VSWhidbey 103673, * to get a solution to this from BCL. */ private string GetErrorSourceFileContents(CompilerResults results) { if (!results.Errors.HasErrors) return null; // Get the physical path of the file that has the error. Note that this could be // either the path to a high level file (e.g. aspx) if pragmas are in play, // or the path to a generated file if there are no pragmas string linePragma = results.Errors[0].FileName; // Attempt to locate the correct build provider BuildProvider buildProvider = GetBuildProviderFromLinePragma(linePragma); if (buildProvider != null) { // Return the generated file for this build provider return GetGeneratedSourceFromBuildProvider(buildProvider); } // If we didn't find it, then we're probably in the no pragma case, in // which case linePragma itself is the generated file return Util.StringFromFileIfExists(linePragma); } internal string GetGeneratedSourceFromBuildProvider(BuildProvider buildProvider) { // Return the generated file content for this build provider string generatedFilePath = (string) _buildProviderToSourceFileMap[buildProvider]; return Util.StringFromFileIfExists(generatedFilePath); } internal BuildProvider GetBuildProviderFromLinePragma(string linePragma) { BuildProvider buildProvider = GetBuildProviderFromLinePragmaInternal(linePragma); // If it's a CompileWith provider, return the main provider instead SourceFileBuildProvider sourceBuildProvider = buildProvider as SourceFileBuildProvider; if (sourceBuildProvider != null) buildProvider = sourceBuildProvider.OwningBuildProvider; return buildProvider; } private BuildProvider GetBuildProviderFromLinePragmaInternal(string linePragma) { // If we didn't keep track of any generated files, we can't do much if (_buildProviderToSourceFileMap == null) return null; // Check if it's an http line pragma, from which we can get a VirtualPath string virtualPath = ErrorFormatter.GetVirtualPathFromHttpLinePragma(linePragma); // First, look for the pragma case foreach (BuildProvider buildProvider in BuildProviders) { // If the build provider can't give us a virtual path, skip it if (buildProvider.VirtualPath == null) continue; // If we got a virtual path, use it to locate the correct BuildProvider if (virtualPath != null) { if (StringUtil.EqualsIgnoreCase(virtualPath, buildProvider.VirtualPath)) { return buildProvider; } continue; } // Otherwise, work with the physical path string physicalPath = HostingEnvironment.MapPathInternal(buildProvider.VirtualPath); if (StringUtil.EqualsIgnoreCase(linePragma, physicalPath)) { return buildProvider; } } return null; } } /* * This class is used intead of AssemblyBuilder when handling * ClientBuildManager.GetCodeDirectoryInformation * It is instantiated via CompilerType.CreateAssemblyBuilder. */ internal class CbmCodeGeneratorBuildProviderHost: AssemblyBuilder { private string _generatedFilesDir; internal CbmCodeGeneratorBuildProviderHost(CompilationSection compConfig, ICollection referencedAssemblies, CompilerType compilerType, string generatedFilesDir, string outputAssemblyName) : base(compConfig, referencedAssemblies, compilerType, outputAssemblyName) { // Wipe out any existing directory, and recreate it // This is where we will put generated source files if (Directory.Exists(generatedFilesDir)) { // Delete all the files in the directory foreach (FileData fileData in FileEnumerator.Create(generatedFilesDir)) { // It should only contain files Debug.Assert(!fileData.IsDirectory); if (fileData.IsDirectory) continue; Debug.Trace("CbmCodeGeneratorBuildProviderHost", "Deleting " + fileData.FullName); File.Delete(fileData.FullName); } } // Create it to make sure it exists Directory.CreateDirectory(generatedFilesDir); _generatedFilesDir = generatedFilesDir; } internal override TextWriter CreateCodeFile(BuildProvider buildProvider, out string filename) { // use GetCacheKeyFromVirtualPath to get a file name that looks like // the original file, but is guaranteed unique across different virtual dirs. string generatedCodeFile = BuildManager.GetCacheKeyFromVirtualPath( buildProvider.VirtualPathObject); generatedCodeFile = Path.Combine(_generatedFilesDir, generatedCodeFile); generatedCodeFile = FileUtil.TruncatePathIfNeeded(generatedCodeFile, 10 /*length of extension */); generatedCodeFile = generatedCodeFile + "." + _codeProvider.FileExtension; filename = generatedCodeFile; BuildManager.GenerateFileTable[buildProvider.VirtualPathObject.VirtualPathStringNoTrailingSlash] = generatedCodeFile; Debug.Trace("CbmCodeGeneratorBuildProviderHost", "Generating " + generatedCodeFile); Stream temp = new FileStream(generatedCodeFile, FileMode.Create, FileAccess.Write, FileShare.Read); return new StreamWriter(temp, Encoding.UTF8); } internal override void AddBuildProvider(BuildProvider buildProvider) { // Skip source files, since their code generation is an identity transform if (buildProvider is SourceFileBuildProvider) return; base.AddBuildProvider(buildProvider); } } }