You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			1812 lines
		
	
	
		
			78 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1812 lines
		
	
	
		
			78 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // ==++==
 | |
| // 
 | |
| //   Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // 
 | |
| // ==--==
 | |
| /*============================================================
 | |
| **
 | |
| ** Class: Environment
 | |
| **
 | |
| **
 | |
| ** Purpose: Provides some basic access to some environment 
 | |
| ** functionality.
 | |
| **
 | |
| **
 | |
| ============================================================*/
 | |
| namespace System {
 | |
|     using System.IO;
 | |
|     using System.Security;
 | |
|     using System.Resources;
 | |
|     using System.Globalization;
 | |
|     using System.Collections;
 | |
|     using System.Security.Permissions;
 | |
|     using System.Text;
 | |
|     using System.Configuration.Assemblies;
 | |
|     using System.Runtime.InteropServices;
 | |
|     using System.Reflection;
 | |
|     using System.Diagnostics;
 | |
|     using Microsoft.Win32;
 | |
|     using System.Runtime.CompilerServices;
 | |
|     using System.Threading;
 | |
|     using System.Runtime.ConstrainedExecution;
 | |
|     using System.Runtime.Versioning;
 | |
|     using System.Diagnostics.Contracts;
 | |
| 
 | |
| #if !FEATURE_PAL
 | |
|     [ComVisible(true)]
 | |
|     public enum EnvironmentVariableTarget {
 | |
|         Process = 0,
 | |
| #if FEATURE_WIN32_REGISTRY            
 | |
|         User = 1,
 | |
|         Machine = 2,
 | |
| #endif        
 | |
|     }
 | |
| #endif 
 | |
| 
 | |
|     [ComVisible(true)]
 | |
|     public static class Environment {
 | |
| 
 | |
|         // Assume the following constants include the terminating '\0' - use <, not <=
 | |
|         const int MaxEnvVariableValueLength = 32767;  // maximum length for environment variable name and value
 | |
|         // System environment variables are stored in the registry, and have 
 | |
|         // a size restriction that is separate from both normal environment 
 | |
|         // variables and registry value name lengths, according to MSDN.
 | |
|         // MSDN doesn't detail whether the name is limited to 1024, or whether
 | |
|         // that includes the contents of the environment variable.
 | |
|         const int MaxSystemEnvVariableLength = 1024;
 | |
|         const int MaxUserEnvVariableLength = 255;
 | |
| 
 | |
|         internal sealed class ResourceHelper
 | |
|         {
 | |
|             internal ResourceHelper(String name) {
 | |
|                 m_name = name;
 | |
|             }
 | |
| 
 | |
|             private String m_name;
 | |
|             private ResourceManager SystemResMgr;
 | |
| 
 | |
|             // To avoid infinite loops when calling GetResourceString.  See comments
 | |
|             // in GetResourceString for this field.
 | |
|             private Stack currentlyLoading;
 | |
|         
 | |
|             // process-wide state (since this is only used in one domain), 
 | |
|             // used to avoid the TypeInitialization infinite recusion
 | |
|             // in GetResourceStringCode
 | |
|             internal bool resourceManagerInited = false;
 | |
| 
 | |
|             // Is this thread currently doing infinite resource lookups?
 | |
|             private int infinitelyRecursingCount;
 | |
| 
 | |
|             // Data representing one individual resource lookup on a thread.
 | |
|             internal class GetResourceStringUserData
 | |
|             {
 | |
|                 public ResourceHelper m_resourceHelper;
 | |
|                 public String m_key;
 | |
|                 public CultureInfo m_culture;
 | |
|                 public String m_retVal;
 | |
|                 public bool m_lockWasTaken;
 | |
| 
 | |
|                 public GetResourceStringUserData(ResourceHelper resourceHelper, String key, CultureInfo culture)
 | |
|                 {
 | |
|                     m_resourceHelper = resourceHelper;
 | |
|                     m_key = key;
 | |
|                     m_culture = culture;
 | |
|                 }
 | |
|             }
 | |
|             
 | |
|             [ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
 | |
|             internal String GetResourceString(String key) {
 | |
|                 if (key == null || key.Length == 0) {
 | |
|                     Contract.Assert(false, "Environment::GetResourceString with null or empty key.  Bug in caller, or weird recursive loading problem?");
 | |
|                     return "[Resource lookup failed - null or empty resource name]";
 | |
|                 }
 | |
|                 return GetResourceString(key, null);
 | |
|             }
 | |
| 
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             [ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
 | |
|             internal String GetResourceString(String key, CultureInfo culture)  {
 | |
|                 if (key == null || key.Length == 0) {
 | |
|                     Contract.Assert(false, "Environment::GetResourceString with null or empty key.  Bug in caller, or weird recursive loading problem?");
 | |
|                     return "[Resource lookup failed - null or empty resource name]";
 | |
|                 }
 | |
| 
 | |
|                 // We have a somewhat common potential for infinite 
 | |
|                 // loops with mscorlib's ResourceManager.  If "potentially dangerous"
 | |
|                 // code throws an exception, we will get into an infinite loop
 | |
|                 // inside the ResourceManager and this "potentially dangerous" code.
 | |
|                 // Potentially dangerous code includes the IO package, CultureInfo,
 | |
|                 // parts of the loader, some parts of Reflection, Security (including 
 | |
|                 // custom user-written permissions that may parse an XML file at
 | |
|                 // class load time), assembly load event handlers, etc.  Essentially,
 | |
|                 // this is not a bounded set of code, and we need to fix the problem.
 | |
|                 // Fortunately, this is limited to mscorlib's error lookups and is NOT
 | |
|                 // a general problem for all user code using the ResourceManager.
 | |
|                 
 | |
|                 // The solution is to make sure only one thread at a time can call 
 | |
|                 // GetResourceString.  Also, since resource lookups can be 
 | |
|                 // reentrant, if the same thread comes into GetResourceString
 | |
|                 // twice looking for the exact same resource name before 
 | |
|                 // returning, we're going into an infinite loop and we should 
 | |
|                 // return a bogus string.  
 | |
| 
 | |
|                 GetResourceStringUserData userData = new GetResourceStringUserData(this, key, culture);
 | |
| 
 | |
|                 RuntimeHelpers.TryCode tryCode = new RuntimeHelpers.TryCode(GetResourceStringCode);
 | |
|                 RuntimeHelpers.CleanupCode cleanupCode = new RuntimeHelpers.CleanupCode(GetResourceStringBackoutCode);
 | |
| 
 | |
|                 RuntimeHelpers.ExecuteCodeWithGuaranteedCleanup(tryCode, cleanupCode, userData);
 | |
|                 return userData.m_retVal;
 | |
|             }
 | |
| 
 | |
|             #if FEATURE_CORECLR
 | |
|             [System.Security.SecurityCritical] // auto-generated
 | |
|             #else
 | |
|             [System.Security.SecuritySafeCritical]
 | |
|             #endif
 | |
|             private void GetResourceStringCode(Object userDataIn)
 | |
|             {
 | |
|                 GetResourceStringUserData userData = (GetResourceStringUserData) userDataIn;
 | |
|                 ResourceHelper rh = userData.m_resourceHelper;
 | |
|                 String key = userData.m_key;
 | |
|                 CultureInfo culture = userData.m_culture;
 | |
| 
 | |
|                 Monitor.Enter(rh, ref userData.m_lockWasTaken);
 | |
| 
 | |
|                 // Are we recursively looking up the same resource?  Note - our backout code will set
 | |
|                 // the ResourceHelper's currentlyLoading stack to null if an exception occurs.
 | |
|                 if (rh.currentlyLoading != null && rh.currentlyLoading.Count > 0 && rh.currentlyLoading.Contains(key)) {
 | |
|                     // We can start infinitely recursing for one resource lookup,
 | |
|                     // then during our failure reporting, start infinitely recursing again.
 | |
|                     // avoid that.
 | |
|                     if (rh.infinitelyRecursingCount > 0) {
 | |
|                         userData.m_retVal = "[Resource lookup failed - infinite recursion or critical failure detected.]";
 | |
|                         return;
 | |
|                     }
 | |
|                     rh.infinitelyRecursingCount++;
 | |
|                     // This is often a bug in the BCL, security, NLS+ code,
 | |
|                     // or the loader somewhere.  However, this could also
 | |
|                     // be a setup problem - check whether mscorlib & 
 | |
|                     // clr.dll are both of the same build flavor.  Also, user 
 | |
|                     // code in the resource lookup process (like an assembly 
 | |
|                     // resolve event or custom CultureInfo) might potentially cause issues.
 | |
| 
 | |
|                     // Note: our infrastructure for reporting this exception will again cause resource lookup.
 | |
|                     // This is the most direct way of dealing with that problem.
 | |
|                     String message = "Infinite recursion during resource lookup within mscorlib.  This may be a bug in mscorlib, or potentially in certain extensibility points such as assembly resolve events or CultureInfo names.  Resource name: " + key;
 | |
|                     Assert.Fail("[mscorlib recursive resource lookup bug]", message, Assert.COR_E_FAILFAST, System.Diagnostics.StackTrace.TraceFormat.NoResourceLookup);
 | |
|                     Environment.FailFast(message);
 | |
|                 }
 | |
|                 if (rh.currentlyLoading == null)
 | |
|                     rh.currentlyLoading = new Stack(4);
 | |
| 
 | |
|                 // Call class constructors preemptively, so that we cannot get into an infinite
 | |
|                 // loop constructing a TypeInitializationException.  If this were omitted,
 | |
|                 // we could get the Infinite recursion assert above by failing type initialization
 | |
|                 // between the Push and Pop calls below.
 | |
|         
 | |
|                 if (!rh.resourceManagerInited)
 | |
|                 {
 | |
|                     // process-critical code here.  No ThreadAbortExceptions
 | |
|                     // can be thrown here.  Other exceptions percolate as normal.
 | |
|                     RuntimeHelpers.PrepareConstrainedRegions();
 | |
|                     try {
 | |
|                     }
 | |
|                     finally {
 | |
|                         RuntimeHelpers.RunClassConstructor(typeof(ResourceManager).TypeHandle);
 | |
|                         RuntimeHelpers.RunClassConstructor(typeof(ResourceReader).TypeHandle);
 | |
|                         RuntimeHelpers.RunClassConstructor(typeof(RuntimeResourceSet).TypeHandle);
 | |
|                         RuntimeHelpers.RunClassConstructor(typeof(BinaryReader).TypeHandle);
 | |
|                         rh.resourceManagerInited = true; 
 | |
|                     }
 | |
|             
 | |
|                 } 
 | |
|         
 | |
|                 rh.currentlyLoading.Push(key);
 | |
| 
 | |
|                 if (rh.SystemResMgr == null) {
 | |
|                     rh.SystemResMgr = new ResourceManager(m_name, typeof(Object).Assembly);
 | |
|                 }
 | |
|                 String s = rh.SystemResMgr.GetString(key, null);
 | |
|                 rh.currentlyLoading.Pop();
 | |
| 
 | |
|                 Contract.Assert(s!=null, "Managed resource string lookup failed.  Was your resource name misspelled?  Did you rebuild mscorlib after adding a resource to resources.txt?  Debug this w/ cordbg and bug whoever owns the code that called Environment.GetResourceString.  Resource name was: \""+key+"\"");
 | |
| 
 | |
|                 userData.m_retVal = s;
 | |
|             }
 | |
| 
 | |
|             #if FEATURE_CORECLR
 | |
|             [System.Security.SecurityCritical] // auto-generated
 | |
|             #endif
 | |
|             [PrePrepareMethod]
 | |
|             private void GetResourceStringBackoutCode(Object userDataIn, bool exceptionThrown)
 | |
|             {
 | |
|                 GetResourceStringUserData userData = (GetResourceStringUserData) userDataIn;
 | |
|                 ResourceHelper rh = userData.m_resourceHelper;
 | |
| 
 | |
|                 if (exceptionThrown)
 | |
|                 {
 | |
|                     if (userData.m_lockWasTaken) 
 | |
|                     {
 | |
|                         // Backout code - throw away potentially corrupt state
 | |
|                         rh.SystemResMgr = null;
 | |
|                         rh.currentlyLoading = null;
 | |
|                     }
 | |
|                 }
 | |
|                 // Release the lock, if we took it.
 | |
|                 if (userData.m_lockWasTaken)
 | |
|                 {
 | |
|                     Monitor.Exit(rh);
 | |
|                 }
 | |
|             }
 | |
|         
 | |
|         }
 | |
| 
 | |
|               private static volatile ResourceHelper m_resHelper;  // Doesn't need to be initialized as they're zero-init.
 | |
| 
 | |
|         private const  int    MaxMachineNameLength = 256;
 | |
| 
 | |
|         // Private object for locking instead of locking on a public type for SQL reliability work.
 | |
|         private static Object s_InternalSyncObject;
 | |
|         private static Object InternalSyncObject {
 | |
|             [ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
 | |
|             get {
 | |
|                 if (s_InternalSyncObject == null) {
 | |
|                     Object o = new Object();
 | |
|                     Interlocked.CompareExchange<Object>(ref s_InternalSyncObject, o, null);
 | |
|                 }
 | |
|                 return s_InternalSyncObject;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         private static volatile OperatingSystem m_os;  // Cached OperatingSystem value
 | |
| 
 | |
|         /*==================================TickCount===================================
 | |
|         **Action: Gets the number of ticks since the system was started.
 | |
|         **Returns: The number of ticks since the system was started.
 | |
|         **Arguments: None
 | |
|         **Exceptions: None
 | |
|         ==============================================================================*/
 | |
|         public static extern int TickCount {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             [ResourceExposure(ResourceScope.None)]
 | |
|             [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|             get;
 | |
|         }
 | |
|         
 | |
|         // Terminates this process with the given exit code.
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode)]
 | |
|         [SuppressUnmanagedCodeSecurity]
 | |
|         internal static extern void _Exit(int exitCode);
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [ResourceConsumption(ResourceScope.Process)]
 | |
| #pragma warning disable 618
 | |
|         [SecurityPermissionAttribute(SecurityAction.Demand, Flags=SecurityPermissionFlag.UnmanagedCode)]
 | |
| #pragma warning restore 618
 | |
|         public static void Exit(int exitCode) {
 | |
|             _Exit(exitCode);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         public static extern int ExitCode {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             [ResourceExposure(ResourceScope.None)]
 | |
|             [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|             get;
 | |
|     
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             [ResourceExposure(ResourceScope.None)]
 | |
|             [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|             set;
 | |
|         }
 | |
| 
 | |
|         // Note: The CLR's Watson bucketization code looks at the caller of the FCALL method
 | |
|         // to assign blame for crashes.  Don't mess with this, such as by making it call 
 | |
|         // another managed helper method, unless you consult with some CLR Watson experts.
 | |
|         [System.Security.SecurityCritical]
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         public static extern void FailFast(String message);
 | |
| 
 | |
|         [System.Security.SecurityCritical]
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         internal static extern void FailFast(String message, uint exitCode);
 | |
| 
 | |
|         // This overload of FailFast will allow you to specify the exception object
 | |
|         // whose bucket details *could* be used when undergoing the failfast process.
 | |
|         // To be specific:
 | |
|         //
 | |
|         // 1) When invoked from within a managed EH clause (fault/finally/catch),
 | |
|         //    if the exception object is preallocated, the runtime will try to find its buckets
 | |
|         //    and use them. If the exception object is not preallocated, it will use the bucket
 | |
|         //    details contained in the object (if any).
 | |
|         //
 | |
|         // 2) When invoked from outside the managed EH clauses (fault/finally/catch),
 | |
|         //    if the exception object is preallocated, the runtime will use the callsite's
 | |
|         //    IP for bucketing. If the exception object is not preallocated, it will use the bucket
 | |
|         //    details contained in the object (if any).
 | |
|         [System.Security.SecurityCritical]
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         public static extern void FailFast(String message, Exception exception);
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode)]
 | |
|         [SecurityCritical]  // Our security team doesn't yet allow safe-critical P/Invoke methods.
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [SuppressUnmanagedCodeSecurity]
 | |
|         [ReliabilityContract(Consistency.WillNotCorruptState, Cer.MayFail)]
 | |
|         internal static extern void TriggerCodeContractFailure(ContractFailureKind failureKind, String message, String condition, String exceptionAsString);
 | |
| 
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode)]
 | |
|         [SecurityCritical]  // Our security team doesn't yet allow safe-critical P/Invoke methods.
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [SuppressUnmanagedCodeSecurity]
 | |
|         [return: MarshalAs(UnmanagedType.Bool)]
 | |
|         private static extern bool GetIsCLRHosted();
 | |
| 
 | |
|         internal static bool IsCLRHosted {
 | |
|             [SecuritySafeCritical]
 | |
|             get { return GetIsCLRHosted(); }
 | |
|         }
 | |
| 
 | |
|         public static String CommandLine {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 new EnvironmentPermission(EnvironmentPermissionAccess.Read, "Path").Demand();
 | |
| 
 | |
|                 String commandLine = null;
 | |
|                 GetCommandLine(JitHelpers.GetStringHandleOnStack(ref commandLine));
 | |
|                 return commandLine;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode), SuppressUnmanagedCodeSecurity]
 | |
|         private static extern void GetCommandLine(StringHandleOnStack retString);
 | |
| #endif // !FEATURE_CORECLR
 | |
| 
 | |
|         /*===============================CurrentDirectory===============================
 | |
|         **Action:  Provides a getter and setter for the current directory.  The original
 | |
|         **         current directory is the one from which the process was started.  
 | |
|         **Returns: The current directory (from the getter).  Void from the setter.
 | |
|         **Arguments: The current directory to which to switch to the setter.
 | |
|         **Exceptions: 
 | |
|         ==============================================================================*/
 | |
|         public static String CurrentDirectory
 | |
|         {
 | |
|             [ResourceExposure(ResourceScope.Machine)]
 | |
|             [ResourceConsumption(ResourceScope.Machine)]
 | |
|             get{
 | |
|                 return Directory.GetCurrentDirectory();
 | |
|             }
 | |
| 
 | |
|             #if FEATURE_CORECLR
 | |
|             [System.Security.SecurityCritical] // auto-generated
 | |
|             #endif
 | |
|             [ResourceExposure(ResourceScope.Machine)]
 | |
|             [ResourceConsumption(ResourceScope.Machine)]
 | |
|             set { 
 | |
|                 Directory.SetCurrentDirectory(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Returns the system directory (ie, C:\WinNT\System32).
 | |
|         public static String SystemDirectory {
 | |
| #if FEATURE_CORECLR
 | |
|             [System.Security.SecurityCritical]
 | |
| #else
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
| #endif
 | |
|             [ResourceExposure(ResourceScope.Machine)]
 | |
|             [ResourceConsumption(ResourceScope.Machine)]
 | |
|             get {
 | |
|                 StringBuilder sb = new StringBuilder(Path.MAX_PATH);
 | |
|                 int r = Win32Native.GetSystemDirectory(sb, Path.MAX_PATH);
 | |
|                 Contract.Assert(r < Path.MAX_PATH, "r < Path.MAX_PATH");
 | |
|                 if (r==0) __Error.WinIOError();
 | |
|                 String path = sb.ToString();
 | |
|                 
 | |
| #if !FEATURE_CORECLR
 | |
|                 // Do security check
 | |
|                 FileIOPermission.QuickDemand(FileIOPermissionAccess.PathDiscovery, path);
 | |
| #endif
 | |
| 
 | |
|                 return path;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_PAL
 | |
|         // Returns the windows directory (ie, C:\WinNT).
 | |
|         // Used by NLS+ custom culures only at the moment.
 | |
|         internal static String InternalWindowsDirectory {
 | |
|             [System.Security.SecurityCritical]  // auto-generated
 | |
|             [ResourceExposure(ResourceScope.Machine)]
 | |
|             [ResourceConsumption(ResourceScope.Machine)]
 | |
|             get {
 | |
|                 StringBuilder sb = new StringBuilder(Path.MAX_PATH);
 | |
|                 int r = Win32Native.GetWindowsDirectory(sb, Path.MAX_PATH);
 | |
|                 Contract.Assert(r < Path.MAX_PATH, "r < Path.MAX_PATH");
 | |
|                 if (r==0) __Error.WinIOError();
 | |
|                 String path = sb.ToString();
 | |
|                 
 | |
|                 return path;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         public static String ExpandEnvironmentVariables(String name)
 | |
|         {
 | |
|             if (name == null)
 | |
|                 throw new ArgumentNullException("name");
 | |
|             Contract.EndContractBlock();
 | |
| 
 | |
|             if (name.Length == 0) {
 | |
|                 return name;
 | |
|             }
 | |
| 
 | |
|             if (AppDomain.IsAppXModel() && !AppDomain.IsAppXDesignMode()) {
 | |
|                 // Environment variable accessors are not approved modern API.
 | |
|                 // Behave as if no variables are defined in this case.
 | |
|                 return name; 
 | |
|             }
 | |
| 
 | |
|             int currentSize = 100;
 | |
|             StringBuilder blob = new StringBuilder(currentSize); // A somewhat reasonable default size
 | |
|             int size;
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|             bool isFullTrust = CodeAccessSecurityEngine.QuickCheckForAllDemands();
 | |
| 
 | |
|             // Do a security check to guarantee we can read each of the 
 | |
|             // individual environment variables requested here.
 | |
|             String[] varArray = name.Split(new char[] {'%'});
 | |
|             StringBuilder vars = isFullTrust ? null : new StringBuilder();
 | |
| 
 | |
|             bool fJustExpanded = false; // to accommodate expansion alg.
 | |
| 
 | |
|             for(int i=1; i<varArray.Length-1; i++) { // Skip first and last tokens
 | |
|                 // ExpandEnvironmentStrings' greedy algorithm expands every
 | |
|                 // non-boundary %-delimited substring, provided the previous
 | |
|                 // has not been expanded.
 | |
|                 // if "foo" is not expandable, and "PATH" is, then both
 | |
|                 // %foo%PATH% and %foo%foo%PATH% will expand PATH, but
 | |
|                 // %PATH%PATH% will expand only once.
 | |
|                 // Therefore, if we've just expanded, skip this substring.
 | |
|                 if (varArray[i].Length == 0 || fJustExpanded == true)
 | |
|                 {
 | |
|                     fJustExpanded = false;
 | |
|                     continue; // Nothing to expand
 | |
|                 }
 | |
|                 // Guess a somewhat reasonable initial size, call the method, then if
 | |
|                 // it fails (ie, the return value is larger than our buffer size),
 | |
|                 // make a new buffer & try again.
 | |
|                 blob.Length = 0;
 | |
|                 String envVar = "%" + varArray[i] + "%";
 | |
|                 size = Win32Native.ExpandEnvironmentStrings(envVar, blob, currentSize);
 | |
|                 if (size == 0)
 | |
|                     Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
 | |
| 
 | |
|                 // some environment variable might be changed while this function is called
 | |
|                 while (size > currentSize) {
 | |
|                     currentSize = size;
 | |
|                     blob.Capacity = currentSize;
 | |
|                     blob.Length = 0;
 | |
|                     size = Win32Native.ExpandEnvironmentStrings(envVar, blob, currentSize);
 | |
|                     if (size == 0)
 | |
|                         Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
 | |
|                 }
 | |
| 
 | |
|                 if (!isFullTrust) {
 | |
|                     String temp = blob.ToString();
 | |
|                     fJustExpanded = (temp != envVar);
 | |
|                     if (fJustExpanded) { // We expanded successfully, we need to do String comparison here
 | |
|                         // since %FOO% can become %FOOD
 | |
|                         vars.Append(varArray[i]);
 | |
|                         vars.Append(';');
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|      
 | |
|             if (!isFullTrust)
 | |
|                 new EnvironmentPermission(EnvironmentPermissionAccess.Read, vars.ToString()).Demand();
 | |
| #endif // !FEATURE_CORECLR
 | |
| 
 | |
|             blob.Length = 0;
 | |
|             size = Win32Native.ExpandEnvironmentStrings(name, blob, currentSize);
 | |
|             if (size == 0)
 | |
|                 Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
 | |
|             
 | |
|             while (size > currentSize) {
 | |
|                 currentSize = size;
 | |
|                 blob.Capacity = currentSize;
 | |
|                 blob.Length = 0;
 | |
| 
 | |
|                 size = Win32Native.ExpandEnvironmentStrings(name, blob, currentSize);
 | |
|                 if (size == 0)
 | |
|                     Marshal.ThrowExceptionForHR(Marshal.GetHRForLastWin32Error());
 | |
|             }
 | |
| 
 | |
|             return blob.ToString();
 | |
|         }
 | |
| #endif // FEATURE_PAL
 | |
| 
 | |
|         public static String MachineName {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 // In future release of operating systems, you might be able to rename a machine without
 | |
|                 // rebooting.  Therefore, don't cache this machine name.
 | |
|                 new EnvironmentPermission(EnvironmentPermissionAccess.Read, "COMPUTERNAME").Demand();
 | |
|                 StringBuilder buf = new StringBuilder(MaxMachineNameLength);
 | |
|                 int len = MaxMachineNameLength;
 | |
|                 if (Win32Native.GetComputerName(buf, ref len) == 0)
 | |
|                     throw new InvalidOperationException(Environment.GetResourceString("InvalidOperation_ComputerName"));
 | |
|                 return buf.ToString();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [SecurityCritical]
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode)]
 | |
|         [SuppressUnmanagedCodeSecurity]
 | |
|         private static extern Int32 GetProcessorCount();
 | |
| 
 | |
|         public static int ProcessorCount {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 return GetProcessorCount();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static int SystemPageSize {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 (new EnvironmentPermission(PermissionState.Unrestricted)).Demand();
 | |
|                 Win32Native.SYSTEM_INFO info = new Win32Native.SYSTEM_INFO();
 | |
|                 Win32Native.GetSystemInfo(ref info);
 | |
|                 return info.dwPageSize;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|         /*==============================GetCommandLineArgs==============================
 | |
|         **Action: Gets the command line and splits it appropriately to deal with whitespace,
 | |
|         **        quotes, and escape characters.
 | |
|         **Returns: A string array containing your command line arguments.
 | |
|         **Arguments: None
 | |
|         **Exceptions: None.
 | |
|         ==============================================================================*/
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         public static String[] GetCommandLineArgs() {
 | |
|             new EnvironmentPermission(EnvironmentPermissionAccess.Read, "Path").Demand();
 | |
|             return GetCommandLineArgsNative();
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         private static extern String[] GetCommandLineArgsNative();
 | |
|         
 | |
|         // We need to keep this Fcall since it is used in AppDomain.cs.
 | |
|         // If we call GetEnvironmentVariable from AppDomain.cs, we will use StringBuilder class.
 | |
|         // That has side effect to change the ApartmentState of the calling Thread to MTA.
 | |
|         // So runtime can't change the ApartmentState of calling thread any more.
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Process)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         internal static extern String nativeGetEnvironmentVariable(String variable);
 | |
| #endif //!FEATURE_CORECLR
 | |
|         
 | |
|         /*============================GetEnvironmentVariable============================
 | |
|         **Action:
 | |
|         **Returns:
 | |
|         **Arguments:
 | |
|         **Exceptions:
 | |
|         ==============================================================================*/
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static String GetEnvironmentVariable(String variable)
 | |
|         {
 | |
|             if (variable == null)
 | |
|                 throw new ArgumentNullException("variable");
 | |
|             Contract.EndContractBlock();
 | |
| 
 | |
|             if (AppDomain.IsAppXModel() && !AppDomain.IsAppXDesignMode()) {
 | |
|                 // Environment variable accessors are not approved modern API.
 | |
|                 // Behave as if the variable was not found in this case.
 | |
|                 return null; 
 | |
|             }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|             (new EnvironmentPermission(EnvironmentPermissionAccess.Read, variable)).Demand();
 | |
| #endif //!FEATURE_CORECLR
 | |
|             
 | |
|             StringBuilder blob = StringBuilderCache.Acquire(128); // A somewhat reasonable default size
 | |
|             int requiredSize = Win32Native.GetEnvironmentVariable(variable, blob, blob.Capacity);
 | |
| 
 | |
|             if (requiredSize == 0) {  //  GetEnvironmentVariable failed
 | |
|                 if (Marshal.GetLastWin32Error() == Win32Native.ERROR_ENVVAR_NOT_FOUND) {
 | |
|                     StringBuilderCache.Release(blob);
 | |
|                     return null;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             while (requiredSize > blob.Capacity) { // need to retry since the environment variable might be changed 
 | |
|                 blob.Capacity = requiredSize;
 | |
|                 blob.Length = 0;
 | |
|                 requiredSize = Win32Native.GetEnvironmentVariable(variable, blob, blob.Capacity);
 | |
|             }
 | |
|             return StringBuilderCache.GetStringAndRelease(blob);
 | |
|         }
 | |
|         
 | |
| #if !FEATURE_PAL
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static string GetEnvironmentVariable( string variable, EnvironmentVariableTarget target)                
 | |
|         {
 | |
|             if (variable == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException("variable");
 | |
|             }
 | |
|             Contract.EndContractBlock();
 | |
| 
 | |
|             if (target == EnvironmentVariableTarget.Process)
 | |
|             {
 | |
|                 return GetEnvironmentVariable(variable);
 | |
|             }
 | |
| 
 | |
| #if FEATURE_WIN32_REGISTRY
 | |
|             (new EnvironmentPermission(PermissionState.Unrestricted)).Demand();
 | |
| 
 | |
|             if( target == EnvironmentVariableTarget.Machine) {
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.LocalMachine.OpenSubKey(@"System\CurrentControlSet\Control\Session Manager\Environment", false)) {
 | |
| 
 | |
|                    Contract.Assert(environmentKey != null, @"HKLM\System\CurrentControlSet\Control\Session Manager\Environment is missing!");
 | |
|                    if (environmentKey == null) {
 | |
|                        return null; 
 | |
|                    }
 | |
| 
 | |
|                    string value = environmentKey.GetValue(variable) as string;
 | |
|                    return value;
 | |
|                 }
 | |
|             }
 | |
|             else if( target == EnvironmentVariableTarget.User) {
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.CurrentUser.OpenSubKey("Environment", false)) {
 | |
| 
 | |
|                    Contract.Assert(environmentKey != null, @"HKCU\Environment is missing!");
 | |
|                    if (environmentKey == null) {
 | |
|                        return null; 
 | |
|                    }
 | |
| 
 | |
|                    string value = environmentKey.GetValue(variable) as string;
 | |
|                    return value;
 | |
|                 }
 | |
|             }
 | |
|             else 
 | |
| #endif // FEATURE_WIN32_REGISTRY                
 | |
|                 {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)target));
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         /*===========================GetEnvironmentVariables============================
 | |
|         **Action: Returns an IDictionary containing all enviroment variables and their values.
 | |
|         **Returns: An IDictionary containing all environment variables and their values.
 | |
|         **Arguments: None.
 | |
|         **Exceptions: None.
 | |
|         ==============================================================================*/
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         private unsafe static char[] GetEnvironmentCharArray()
 | |
|         {
 | |
|             char[] block = null;
 | |
| 
 | |
|             // Make sure pStrings is not leaked with async exceptions
 | |
|             RuntimeHelpers.PrepareConstrainedRegions();
 | |
|             try {
 | |
|             }
 | |
|             finally {
 | |
|                 char * pStrings = null;
 | |
| 
 | |
|                 try
 | |
|                 {
 | |
|                     pStrings = Win32Native.GetEnvironmentStrings();
 | |
|                     if (pStrings == null) {
 | |
|                         throw new OutOfMemoryException();
 | |
|                     }
 | |
| 
 | |
|                     // Format for GetEnvironmentStrings is:
 | |
|                     // [=HiddenVar=value\0]* [Variable=value\0]* \0
 | |
|                     // See the description of Environment Blocks in MSDN's
 | |
|                     // CreateProcess page (null-terminated array of null-terminated strings).
 | |
| 
 | |
|                     // Search for terminating \0\0 (two unicode \0's).
 | |
|                     char * p = pStrings;
 | |
|                     while (!(*p == '\0' && *(p + 1) == '\0'))
 | |
|                         p++;
 | |
| 
 | |
|                     int len = (int)(p - pStrings + 1);
 | |
|                     block = new char[len];
 | |
| 
 | |
|                     fixed (char* pBlock = block)
 | |
|                         String.wstrcpy(pBlock, pStrings, len);
 | |
|                 }
 | |
|                 finally
 | |
|                 {
 | |
|                     if (pStrings != null)
 | |
|                         Win32Native.FreeEnvironmentStrings(pStrings);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return block;
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static IDictionary GetEnvironmentVariables()
 | |
|         {
 | |
|             if (AppDomain.IsAppXModel() && !AppDomain.IsAppXDesignMode()) {
 | |
|                 // Environment variable accessors are not approved modern API.
 | |
|                 // Behave as if no environment variables are defined in this case.
 | |
|                 return new Hashtable(0);
 | |
|             }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|             bool isFullTrust = CodeAccessSecurityEngine.QuickCheckForAllDemands();
 | |
|             StringBuilder vars = isFullTrust ? null : new StringBuilder();  
 | |
|             bool first = true;
 | |
| #endif
 | |
| 
 | |
|             char[] block = GetEnvironmentCharArray();
 | |
| 
 | |
|             Hashtable table = new Hashtable(20);
 | |
| 
 | |
|             // Copy strings out, parsing into pairs and inserting into the table.
 | |
|             // The first few environment variable entries start with an '='!
 | |
|             // The current working directory of every drive (except for those drives
 | |
|             // you haven't cd'ed into in your DOS window) are stored in the 
 | |
|             // environment block (as =C:=pwd) and the program's exit code is 
 | |
|             // as well (=ExitCode=00000000)  Skip all that start with =.
 | |
|             // Read docs about Environment Blocks on MSDN's CreateProcess page.
 | |
|             
 | |
|             // Format for GetEnvironmentStrings is:
 | |
|             // (=HiddenVar=value\0 | Variable=value\0)* \0
 | |
|             // See the description of Environment Blocks in MSDN's
 | |
|             // CreateProcess page (null-terminated array of null-terminated strings).
 | |
|             // Note the =HiddenVar's aren't always at the beginning.
 | |
|             
 | |
|             for(int i=0; i<block.Length; i++) {
 | |
|                 int startKey = i;
 | |
|                 // Skip to key
 | |
|                 // On some old OS, the environment block can be corrupted. 
 | |
|                 // Someline will not have '=', so we need to check for '\0'. 
 | |
|                 while(block[i]!='=' && block[i] != '\0') {
 | |
|                     i++;
 | |
|                 }
 | |
| 
 | |
|                 if(block[i] == '\0') {
 | |
|                     continue;
 | |
|                 }
 | |
| 
 | |
|                 // Skip over environment variables starting with '='
 | |
|                 if (i-startKey==0) {
 | |
|                     while(block[i]!=0) {
 | |
|                         i++;
 | |
|                     }
 | |
|                     continue;
 | |
|                 }
 | |
|                 String key = new String(block, startKey, i-startKey);
 | |
|                 i++;  // skip over '='
 | |
|                 int startValue = i;
 | |
|                 while(block[i]!=0) {
 | |
|                     // Read to end of this entry 
 | |
|                     i++;
 | |
|                 }
 | |
| 
 | |
|                 String value = new String(block, startValue, i-startValue);
 | |
|                 // skip over 0 handled by for loop's i++
 | |
|                 table[key]=value;
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|                 if (!isFullTrust) {
 | |
|                     if( first) {      
 | |
|                         first = false;
 | |
|                     }
 | |
|                     else {
 | |
|                         vars.Append(';');
 | |
|                     }
 | |
|                     vars.Append(key);
 | |
|                 }
 | |
| #endif
 | |
|             }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|             if (!isFullTrust)
 | |
|                 new EnvironmentPermission(EnvironmentPermissionAccess.Read, vars.ToString()).Demand();
 | |
| #endif
 | |
|             return table;
 | |
|         }
 | |
|         
 | |
| #if !FEATURE_PAL
 | |
|         internal static IDictionary GetRegistryKeyNameValuePairs(RegistryKey registryKey) {
 | |
|             Hashtable table = new Hashtable(20);
 | |
| 
 | |
|             if (registryKey != null) {
 | |
|                 string[] names = registryKey.GetValueNames();
 | |
|                 foreach( string name in names) {
 | |
|                     string value = registryKey.GetValue(name, "").ToString();
 | |
|                     table.Add(name, value);                
 | |
|                 }            
 | |
|             }
 | |
|             return table;
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static IDictionary GetEnvironmentVariables( EnvironmentVariableTarget target) {
 | |
|             if( target == EnvironmentVariableTarget.Process) {
 | |
|                 return GetEnvironmentVariables();
 | |
|             }
 | |
| 
 | |
| #if FEATURE_WIN32_REGISTRY
 | |
|             (new EnvironmentPermission(PermissionState.Unrestricted)).Demand();
 | |
| 
 | |
|             if( target == EnvironmentVariableTarget.Machine) {
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.LocalMachine.OpenSubKey(@"System\CurrentControlSet\Control\Session Manager\Environment", false)) {
 | |
| 
 | |
|                    return GetRegistryKeyNameValuePairs(environmentKey);
 | |
|                 }
 | |
|             }
 | |
|             else if( target == EnvironmentVariableTarget.User) {
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.CurrentUser.OpenSubKey("Environment", false)) {
 | |
|                    return GetRegistryKeyNameValuePairs(environmentKey);
 | |
|                 }
 | |
|             }
 | |
|             else 
 | |
| #endif // FEATURE_WIN32_REGISTRY                
 | |
|                 {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)target));
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static void SetEnvironmentVariable(string variable, string value) {
 | |
|             CheckEnvironmentVariableName(variable);
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|             new EnvironmentPermission(PermissionState.Unrestricted).Demand();
 | |
| #endif
 | |
|             // explicitly null out value if is the empty string. 
 | |
|             if (String.IsNullOrEmpty(value) || value[0] == '\0') {
 | |
|                 value = null;
 | |
|             }
 | |
|             else {
 | |
|                 if( value.Length >= MaxEnvVariableValueLength) {
 | |
|                     throw new ArgumentException(Environment.GetResourceString("Argument_LongEnvVarValue"));                                    
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (AppDomain.IsAppXModel() && !AppDomain.IsAppXDesignMode()) {
 | |
|                 // Environment variable accessors are not approved modern API.
 | |
|                 // so we throw PlatformNotSupportedException.
 | |
|                 throw new PlatformNotSupportedException();
 | |
|             }
 | |
| 
 | |
|             if(!Win32Native.SetEnvironmentVariable(variable, value)) {
 | |
|                 int errorCode = Marshal.GetLastWin32Error();
 | |
|                 
 | |
|                 // Allow user to try to clear a environment variable                
 | |
|                 if( errorCode == Win32Native.ERROR_ENVVAR_NOT_FOUND) {
 | |
|                     return;
 | |
|                 }
 | |
|                 
 | |
|                 // The error message from Win32 is "The filename or extension is too long",
 | |
|                 // which is not accurate.
 | |
|                 if( errorCode == Win32Native.ERROR_FILENAME_EXCED_RANGE) {
 | |
|                     throw new ArgumentException(Environment.GetResourceString("Argument_LongEnvVarValue"));
 | |
|                 }
 | |
|                 
 | |
|                 throw new ArgumentException(Win32Native.GetMessage(errorCode));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void CheckEnvironmentVariableName(string variable) {
 | |
|             if (variable == null) {
 | |
|                 throw new ArgumentNullException("variable");
 | |
|             }
 | |
| 
 | |
|             if( variable.Length == 0) {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Argument_StringZeroLength"), "variable");
 | |
|             }
 | |
| 
 | |
|             if( variable[0] == '\0') {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Argument_StringFirstCharIsZero"), "variable");
 | |
|             }
 | |
| 
 | |
|             // Make sure the environment variable name isn't longer than the 
 | |
|             // max limit on environment variable values.  (MSDN is ambiguous 
 | |
|             // on whether this check is necessary.)
 | |
|             if( variable.Length >= MaxEnvVariableValueLength ) {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Argument_LongEnvVarValue"));                
 | |
|             }
 | |
|             
 | |
|             if( variable.IndexOf('=') != -1) {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Argument_IllegalEnvVarName"));
 | |
|             }
 | |
|             Contract.EndContractBlock();
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_PAL
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static void SetEnvironmentVariable(string variable, string value, EnvironmentVariableTarget target) { 
 | |
|             if( target == EnvironmentVariableTarget.Process) {
 | |
|                 SetEnvironmentVariable(variable, value);
 | |
|                 return;
 | |
|             }
 | |
|         
 | |
|             CheckEnvironmentVariableName(variable);
 | |
| 
 | |
|             // System-wide environment variables stored in the registry are
 | |
|             // limited to 1024 chars for the environment variable name.
 | |
|             if (variable.Length >= MaxSystemEnvVariableLength) {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Argument_LongEnvVarName"));
 | |
|             }
 | |
| 
 | |
|             new EnvironmentPermission(PermissionState.Unrestricted).Demand();
 | |
|             // explicitly null out value if is the empty string. 
 | |
|             if (String.IsNullOrEmpty(value) || value[0] == '\0') {
 | |
|                 value = null;
 | |
|             }
 | |
| #if FEATURE_WIN32_REGISTRY
 | |
|             if( target == EnvironmentVariableTarget.Machine) {
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.LocalMachine.OpenSubKey(@"System\CurrentControlSet\Control\Session Manager\Environment", true)) {
 | |
| 
 | |
|                    Contract.Assert(environmentKey != null, @"HKLM\System\CurrentControlSet\Control\Session Manager\Environment is missing!");
 | |
|                    if (environmentKey != null) {
 | |
|                        if (value == null)
 | |
|                            environmentKey.DeleteValue(variable, false);
 | |
|                        else
 | |
|                            environmentKey.SetValue(variable, value);
 | |
|                    }
 | |
|                 }
 | |
|             }
 | |
|             else if( target == EnvironmentVariableTarget.User) {
 | |
|                 // User-wide environment variables stored in the registry are
 | |
|                 // limited to 255 chars for the environment variable name.
 | |
|                 if (variable.Length >= MaxUserEnvVariableLength) {
 | |
|                     throw new ArgumentException(Environment.GetResourceString("Argument_LongEnvVarValue"));
 | |
|                 }
 | |
|                 using (RegistryKey environmentKey = 
 | |
|                        Registry.CurrentUser.OpenSubKey("Environment", true)) {
 | |
|                    Contract.Assert(environmentKey != null, @"HKCU\Environment is missing!");
 | |
|                    if (environmentKey != null) {
 | |
|                       if (value == null)
 | |
|                           environmentKey.DeleteValue(variable, false);
 | |
|                       else
 | |
|                           environmentKey.SetValue(variable, value);
 | |
|                    }
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)target));
 | |
|             }
 | |
|             // send a WM_SETTINGCHANGE message to all windows
 | |
|             IntPtr r = Win32Native.SendMessageTimeout(new IntPtr(Win32Native.HWND_BROADCAST), Win32Native.WM_SETTINGCHANGE, IntPtr.Zero, "Environment", 0, 1000, IntPtr.Zero);
 | |
| 
 | |
|             if (r == IntPtr.Zero) BCLDebug.Assert(false, "SetEnvironmentVariable failed: " + Marshal.GetLastWin32Error());
 | |
| 
 | |
| #else // FEATURE_WIN32_REGISTRY
 | |
|             throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)target));
 | |
| #endif
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         
 | |
|         /*===============================GetLogicalDrives===============================
 | |
|         **Action: Retrieves the names of the logical drives on this machine in the  form "C:\". 
 | |
|         **Arguments:   None.
 | |
|         **Exceptions:  IOException.
 | |
|         **Permissions: SystemInfo Permission.
 | |
|         ==============================================================================*/
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         public static String[] GetLogicalDrives() {
 | |
|             new EnvironmentPermission(PermissionState.Unrestricted).Demand();
 | |
|                                  
 | |
|             int drives = Win32Native.GetLogicalDrives();
 | |
|             if (drives==0)
 | |
|                 __Error.WinIOError();
 | |
|             uint d = (uint)drives;
 | |
|             int count = 0;
 | |
|             while (d != 0) {
 | |
|                 if (((int)d & 1) != 0) count++;
 | |
|                 d >>= 1;
 | |
|             }
 | |
|             String[] result = new String[count];
 | |
|             char[] root = new char[] {'A', ':', '\\'};
 | |
|             d = (uint)drives;
 | |
|             count = 0;
 | |
|             while (d != 0) {
 | |
|                 if (((int)d & 1) != 0) {
 | |
|                     result[count++] = new String(root);
 | |
|                 }
 | |
|                 d >>= 1;
 | |
|                 root[0]++;
 | |
|             }
 | |
|             return result;
 | |
|         }
 | |
|         
 | |
|         /*===================================NewLine====================================
 | |
|         **Action: A property which returns the appropriate newline string for the given
 | |
|         **        platform.
 | |
|         **Returns: \r\n on Win32.
 | |
|         **Arguments: None.
 | |
|         **Exceptions: None.
 | |
|         ==============================================================================*/
 | |
|         public static String NewLine {
 | |
|             get {
 | |
|                 Contract.Ensures(Contract.Result<String>() != null);
 | |
|                 return "\r\n";
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         
 | |
|         /*===================================Version====================================
 | |
|         **Action: Returns the COM+ version struct, describing the build number.
 | |
|         **Returns:
 | |
|         **Arguments:
 | |
|         **Exceptions:
 | |
|         ==============================================================================*/
 | |
|         public static Version Version {
 | |
|             get {
 | |
| 
 | |
|                 // Previously this represented the File version of mscorlib.dll.  Many other libraries in the framework and outside took dependencies on the first three parts of this version 
 | |
|                 // remaining constant throughout 4.x.  From 4.0 to 4.5.2 this was fine since the file version only incremented the last part.Starting with 4.6 we switched to a file versioning
 | |
|                 // scheme that matched the product version.  In order to preserve compatibility with existing libraries, this needs to be hard-coded.
 | |
|                 
 | |
|                 return new Version(4,0,30319,42000);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         
 | |
|         /*==================================WorkingSet==================================
 | |
|         **Action:
 | |
|         **Returns:
 | |
|         **Arguments:
 | |
|         **Exceptions:
 | |
|         ==============================================================================*/
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode), SuppressUnmanagedCodeSecurity]
 | |
|         private static extern long GetWorkingSet();
 | |
| 
 | |
|         public static long WorkingSet {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 new EnvironmentPermission(PermissionState.Unrestricted).Demand();
 | |
|                 return GetWorkingSet();
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /*==================================OSVersion===================================
 | |
|         **Action:
 | |
|         **Returns:
 | |
|         **Arguments:
 | |
|         **Exceptions:
 | |
|         ==============================================================================*/
 | |
|         public static OperatingSystem OSVersion {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 Contract.Ensures(Contract.Result<OperatingSystem>() != null);
 | |
| 
 | |
|                 if (m_os==null) { // We avoid the lock since we don't care if two threads will set this at the same time.
 | |
|                             
 | |
|                     Microsoft.Win32.Win32Native.OSVERSIONINFO osvi = new Microsoft.Win32.Win32Native.OSVERSIONINFO();
 | |
|                     if (!GetVersion(osvi)) {
 | |
|                         throw new InvalidOperationException(Environment.GetResourceString("InvalidOperation_GetVersion"));
 | |
|                     }
 | |
| 
 | |
|                     Microsoft.Win32.Win32Native.OSVERSIONINFOEX osviEx = new Microsoft.Win32.Win32Native.OSVERSIONINFOEX();
 | |
|                     if (!GetVersionEx(osviEx))
 | |
|                         throw new InvalidOperationException(Environment.GetResourceString("InvalidOperation_GetVersion"));
 | |
| 
 | |
|                     PlatformID id = PlatformID.Win32NT;
 | |
| 
 | |
| #if FEATURE_LEGACYNETCF
 | |
|                     // return platform as WinCE, to ensure apps earlier than WP8 works as expected. 
 | |
|                     if (CompatibilitySwitches.IsAppEarlierThanWindowsPhone8)
 | |
|                     {
 | |
|                         id = PlatformID.WinCE;
 | |
|                     }
 | |
| #endif
 | |
| 
 | |
|                     Version v =  new Version(osvi.MajorVersion, osvi.MinorVersion, osvi.BuildNumber, (osviEx.ServicePackMajor << 16) |osviEx.ServicePackMinor);
 | |
|                     m_os = new OperatingSystem(id, v, osvi.CSDVersion);
 | |
|                 }
 | |
|                 Contract.Assert(m_os != null, "m_os != null");
 | |
|                 return m_os;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if FEATURE_CORESYSTEM
 | |
| 
 | |
|         internal static bool IsWindows8OrAbove {
 | |
|             get {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if FEATURE_COMINTEROP
 | |
|         internal static bool IsWinRTSupported {
 | |
|             get {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| #endif // FEATURE_COMINTEROP
 | |
| 
 | |
| #else // FEATURE_CORESYSTEM
 | |
| 
 | |
|         private static volatile bool s_IsWindows8OrAbove;
 | |
|         private static volatile bool s_CheckedOSWin8OrAbove;
 | |
| 
 | |
|         // Windows 8 version is 6.2
 | |
|         internal static bool IsWindows8OrAbove {
 | |
|             get {
 | |
|                 if (!s_CheckedOSWin8OrAbove) {
 | |
|                     OperatingSystem OS = Environment.OSVersion;
 | |
|                     s_IsWindows8OrAbove = (OS.Platform == PlatformID.Win32NT && 
 | |
|                                    ((OS.Version.Major == 6 && OS.Version.Minor >= 2) || (OS.Version.Major > 6)));
 | |
|                     s_CheckedOSWin8OrAbove = true;
 | |
|                 }
 | |
|                 return s_IsWindows8OrAbove;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if FEATURE_COMINTEROP
 | |
|         private static volatile bool s_WinRTSupported;
 | |
|         private static volatile bool s_CheckedWinRT;
 | |
| 
 | |
|         // Does the current version of Windows have Windows Runtime suppport?
 | |
|         internal static bool IsWinRTSupported {
 | |
|             [SecuritySafeCritical]
 | |
|             get {
 | |
|                 if (!s_CheckedWinRT) {
 | |
|                     s_WinRTSupported = WinRTSupported();
 | |
|                     s_CheckedWinRT = true;
 | |
|                 }
 | |
| 
 | |
|                 return s_WinRTSupported;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [SecurityCritical]
 | |
|         [DllImport(JitHelpers.QCall, CharSet = CharSet.Unicode)]
 | |
|         [SuppressUnmanagedCodeSecurity]
 | |
|         [return: MarshalAs(UnmanagedType.Bool)]
 | |
|         private static extern bool WinRTSupported();
 | |
| #endif // FEATURE_COMINTEROP
 | |
| 
 | |
| #endif // FEATURE_CORESYSTEM
 | |
| 
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         internal static extern bool GetVersion(Microsoft.Win32.Win32Native.OSVERSIONINFO  osVer);
 | |
| 
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         internal static extern bool GetVersionEx(Microsoft.Win32.Win32Native.OSVERSIONINFOEX  osVer);
 | |
| 
 | |
| 
 | |
|         /*==================================StackTrace==================================
 | |
|         **Action:
 | |
|         **Returns:
 | |
|         **Arguments:
 | |
|         **Exceptions:
 | |
|         ==============================================================================*/
 | |
|         public static String StackTrace {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 Contract.Ensures(Contract.Result<String>() != null);
 | |
| 
 | |
|                 new EnvironmentPermission(PermissionState.Unrestricted).Demand();
 | |
|                 return GetStackTrace(null, true);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         #if FEATURE_CORECLR
 | |
|         [System.Security.SecurityCritical] // auto-generated
 | |
|         #endif
 | |
|         internal static String GetStackTrace(Exception e, bool needFileInfo)
 | |
|         {
 | |
|             // Note: Setting needFileInfo to true will start up COM and set our
 | |
|             // apartment state.  Try to not call this when passing "true" 
 | |
|             // before the EE's ExecuteMainMethod has had a chance to set up the
 | |
|             // apartment state.  -- 
 | |
|             StackTrace st;
 | |
|             if (e == null)
 | |
|                 st = new StackTrace(needFileInfo);
 | |
|             else
 | |
|                 st = new StackTrace(e, needFileInfo);
 | |
| 
 | |
|             // Do no include a trailing newline for backwards compatibility
 | |
|             return st.ToString( System.Diagnostics.StackTrace.TraceFormat.Normal );
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         private static void InitResourceHelper() {
 | |
|             // Only the default AppDomain should have a ResourceHelper.  All calls to 
 | |
|             // GetResourceString from any AppDomain delegate to GetResourceStringLocal 
 | |
|             // in the default AppDomain via the fcall GetResourceFromDefault.
 | |
| 
 | |
|             bool tookLock = false;
 | |
|             RuntimeHelpers.PrepareConstrainedRegions();
 | |
|             try {
 | |
| 
 | |
|                 Monitor.Enter(Environment.InternalSyncObject, ref tookLock);
 | |
| 
 | |
|                 if (m_resHelper == null) {
 | |
|                     ResourceHelper rh = new ResourceHelper("mscorlib");
 | |
| 
 | |
|                     System.Threading.Thread.MemoryBarrier();
 | |
|                     m_resHelper =rh;
 | |
|                 }
 | |
|             }
 | |
|             finally {
 | |
|                 if (tookLock)
 | |
|                     Monitor.Exit(Environment.InternalSyncObject);
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|         [System.Security.SecurityCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         internal extern static String GetResourceFromDefault(String key);           
 | |
| #endif	
 | |
| 
 | |
|         // Looks up the resource string value for key.
 | |
|         // 
 | |
|         // if you change this method's signature then you must change the code that calls it
 | |
|         // in excep.cpp and probably you will have to visit mscorlib.h to add the new signature
 | |
|         // as well as metasig.h to create the new signature type
 | |
|         #if FEATURE_CORECLR
 | |
|         [System.Security.SecurityCritical] // auto-generated
 | |
|         #endif
 | |
|         internal static String GetResourceStringLocal(String key) {
 | |
|             if (m_resHelper == null)
 | |
|                 InitResourceHelper();
 | |
| 
 | |
|             return m_resHelper.GetResourceString(key);
 | |
|         }
 | |
| 
 | |
|         // #threadCultureInfo
 | |
|         // Currently in silverlight, CurrentCulture and CurrentUICulture are isolated 
 | |
|         // within an AppDomain. This is in contrast to the desktop, in which cultures 
 | |
|         // leak across AppDomain boundaries with the thread. 
 | |
|         // 
 | |
|         // Note that mscorlib transitions to the default domain to perform resource 
 | |
|         // lookup. This causes problems for the silverlight changes: since culture isn't
 | |
|         // passed, resource string lookup won't necessarily use the culture of the thread 
 | |
|         // originating the request. To get around that problem, we pass the CultureInfo 
 | |
|         // so that the ResourceManager GetString(x, cultureInfo) overload can be used. 
 | |
|         // We first perform the same check as in CultureInfo to make sure it's safe to 
 | |
|         // let the CultureInfo travel across AppDomains. 
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         internal static String GetResourceString(String key) {
 | |
| #if FEATURE_CORECLR
 | |
|             return GetResourceStringLocal(key);
 | |
| #else
 | |
|             return GetResourceFromDefault(key);
 | |
| #endif //FEATURE_CORECLR
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         internal static String GetResourceString(String key, params Object[] values) {
 | |
|             String s = GetResourceString(key);
 | |
|             return String.Format(CultureInfo.CurrentCulture, s, values);
 | |
|         }
 | |
| 
 | |
|         //The following two internal methods are not used anywhere within the framework,
 | |
|         // but are being kept around as external platforms built on top of us have taken 
 | |
|         // dependency by using private reflection on them for getting system resource strings 
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         internal static String GetRuntimeResourceString(String key) {
 | |
|             return GetResourceString(key);
 | |
|         }
 | |
| 
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         internal static String GetRuntimeResourceString(String key, params Object[] values) {
 | |
|             return GetResourceString(key,values);
 | |
|         }
 | |
| 
 | |
|         public static bool Is64BitProcess {
 | |
|             get {
 | |
|                 #if WIN32
 | |
|                     return false;
 | |
|                 #else
 | |
|                     return true;
 | |
|                 #endif
 | |
|             }
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_PAL
 | |
|         public static bool Is64BitOperatingSystem {
 | |
|             [System.Security.SecuritySafeCritical]
 | |
|             get {
 | |
|                 #if WIN32                    
 | |
|                     bool isWow64; // WinXP SP2+ and Win2k3 SP1+
 | |
|                     return Win32Native.DoesWin32MethodExist(Win32Native.KERNEL32, "IsWow64Process")
 | |
|                         && Win32Native.IsWow64Process(Win32Native.GetCurrentProcess(), out isWow64)
 | |
|                         && isWow64;
 | |
|                 #else
 | |
|                     // 64-bit programs run only on 64-bit
 | |
|                     //<
 | |
|                     return true;
 | |
|                 #endif
 | |
|             }
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         public static extern bool HasShutdownStarted {
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|             get;
 | |
|         }
 | |
| 
 | |
| #if !FEATURE_CORECLR
 | |
|         // This is the temporary Whidbey stub for compatibility flags
 | |
|         [ResourceExposure(ResourceScope.None)]
 | |
|         [MethodImplAttribute(MethodImplOptions.InternalCall)]
 | |
|         [System.Security.SecurityCritical]
 | |
|         internal static extern bool GetCompatibilityFlag(CompatibilityFlag flag);
 | |
| #endif //!FEATURE_CORECLR
 | |
| 
 | |
|         public static string UserName {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             get {
 | |
|                 new EnvironmentPermission(EnvironmentPermissionAccess.Read,"UserName").Demand();
 | |
| 
 | |
|                 StringBuilder sb = new StringBuilder(256);
 | |
|                 int size = sb.Capacity;
 | |
|                 if (Win32Native.GetUserName(sb, ref size))
 | |
|                 {
 | |
|                     return sb.ToString();
 | |
|                 }
 | |
|                 return String.Empty;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Note that this is a handle to a process window station, but it does
 | |
|         // not need to be closed.  CloseWindowStation would ignore this handle.
 | |
|         // We also do handle equality checking as well.  This isn't a great fit
 | |
|         // for SafeHandle.  We don't gain anything by using SafeHandle here.
 | |
| #if !FEATURE_PAL && !FEATURE_CORESYSTEM
 | |
|         private static volatile IntPtr processWinStation;        // Doesn't need to be initialized as they're zero-init.
 | |
|         private static volatile bool isUserNonInteractive;   
 | |
| #endif
 | |
| 
 | |
|         public static bool UserInteractive {
 | |
|             [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|             [ResourceExposure(ResourceScope.None)]
 | |
|             [ResourceConsumption(ResourceScope.Machine, ResourceScope.Machine)]
 | |
|             get {
 | |
| #if !FEATURE_PAL && !FEATURE_CORESYSTEM
 | |
|                 IntPtr hwinsta = Win32Native.GetProcessWindowStation();
 | |
|                 if (hwinsta != IntPtr.Zero && processWinStation != hwinsta) {
 | |
|                     int lengthNeeded = 0;
 | |
|                     Win32Native.USEROBJECTFLAGS flags = new Win32Native.USEROBJECTFLAGS();
 | |
|                     if (Win32Native.GetUserObjectInformation(hwinsta, Win32Native.UOI_FLAGS, flags, Marshal.SizeOf(flags),ref lengthNeeded)) {
 | |
|                         if ((flags.dwFlags & Win32Native.WSF_VISIBLE) == 0) {
 | |
|                             isUserNonInteractive = true;
 | |
|                         }
 | |
|                     }
 | |
|                     processWinStation = hwinsta;
 | |
|                 }
 | |
| 
 | |
|                 // The logic is reversed to avoid static initialization to true
 | |
|                 return !isUserNonInteractive;
 | |
| #else
 | |
|                 return true;
 | |
| #endif
 | |
|             }
 | |
|         }
 | |
|         
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static string GetFolderPath(SpecialFolder folder) {
 | |
|             if (!Enum.IsDefined(typeof(SpecialFolder), folder))
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)folder));
 | |
|             Contract.EndContractBlock();
 | |
| 
 | |
|             return InternalGetFolderPath(folder, SpecialFolderOption.None);
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         public static string GetFolderPath(SpecialFolder folder, SpecialFolderOption option) {
 | |
|             if (!Enum.IsDefined(typeof(SpecialFolder),folder))
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)folder));
 | |
|             if (!Enum.IsDefined(typeof(SpecialFolderOption),option))
 | |
|                 throw new ArgumentException(Environment.GetResourceString("Arg_EnumIllegalVal", (int)option));
 | |
|             Contract.EndContractBlock();
 | |
| 
 | |
|             return InternalGetFolderPath(folder, option);
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecurityCritical]
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         internal static string UnsafeGetFolderPath(SpecialFolder folder)
 | |
|         {
 | |
|             return InternalGetFolderPath(folder, SpecialFolderOption.None, suppressSecurityChecks: true);
 | |
|         }
 | |
| 
 | |
|         [System.Security.SecurityCritical]
 | |
|         [ResourceExposure(ResourceScope.Machine)]
 | |
|         [ResourceConsumption(ResourceScope.Machine)]
 | |
|         private static string InternalGetFolderPath(SpecialFolder folder, SpecialFolderOption option, bool suppressSecurityChecks = false)
 | |
|         {
 | |
| #if FEATURE_CORESYSTEM
 | |
|             // This is currently customized for Windows Phone since CoreSystem doesn't support
 | |
|             // SHGetFolderPath. The allowed folder values are based on the version of .NET CF WP7 was using.
 | |
|             switch (folder)
 | |
|             {
 | |
|                 case SpecialFolder.System:
 | |
|                     return SystemDirectory;
 | |
|                 case SpecialFolder.ApplicationData:
 | |
|                 case SpecialFolder.Favorites:
 | |
|                 case SpecialFolder.Programs:
 | |
|                 case SpecialFolder.StartMenu:
 | |
|                 case SpecialFolder.Startup:
 | |
|                 case SpecialFolder.Personal:
 | |
|                     throw new PlatformNotSupportedException();
 | |
|                 default:
 | |
|                     throw new PlatformNotSupportedException();
 | |
|             }
 | |
| #else // FEATURE_CORESYSTEM
 | |
| #if !FEATURE_CORECLR
 | |
|             if (option == SpecialFolderOption.Create && !suppressSecurityChecks) {
 | |
|                 FileIOPermission createPermission = new FileIOPermission(PermissionState.None);
 | |
|                 createPermission.AllFiles = FileIOPermissionAccess.Write;
 | |
|                 createPermission.Demand();
 | |
|             }
 | |
| #endif
 | |
| 
 | |
|             StringBuilder sb = new StringBuilder(Path.MAX_PATH);
 | |
|             int hresult = Win32Native.SHGetFolderPath(IntPtr.Zero,                    /* hwndOwner: [in] Reserved */
 | |
|                                                       ((int)folder | (int)option),    /* nFolder:   [in] CSIDL    */
 | |
|                                                       IntPtr.Zero,                    /* hToken:    [in] access token */
 | |
|                                                       Win32Native.SHGFP_TYPE_CURRENT, /* dwFlags:   [in] retrieve current path */
 | |
|                                                       sb);                            /* pszPath:   [out]resultant path */
 | |
|             String s;
 | |
|             if (hresult < 0)
 | |
|             {
 | |
|                 switch (hresult)
 | |
|                 {
 | |
|                 default:
 | |
|                     // The previous incarnation threw away all errors. In order to limit
 | |
|                     // breaking changes, we will be permissive about these errors
 | |
|                     // instead of calling ThowExceptionForHR.
 | |
|                     //Runtime.InteropServices.Marshal.ThrowExceptionForHR(hresult);
 | |
|                     break;
 | |
|                 case __HResults.COR_E_PLATFORMNOTSUPPORTED:
 | |
|                     // This one error is the one we do want to throw.
 | |
|                     // <
 | |
| 
 | |
|                     throw new PlatformNotSupportedException();
 | |
|                 }
 | |
| 
 | |
|                 // SHGetFolderPath does not initialize the output buffer on error
 | |
|                 s = String.Empty;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 s = sb.ToString();
 | |
|             }
 | |
| 
 | |
|             if (!suppressSecurityChecks)
 | |
|             {
 | |
|                 // On CoreCLR we can check with the host if we're not trying to use any special options.
 | |
|                 // Otherwise, we need to do a full demand since hosts aren't expecting to handle requests to
 | |
|                 // create special folders.
 | |
| #if FEATURE_CORECLR
 | |
|                 if (option == SpecialFolderOption.None)
 | |
|                 {
 | |
|                     FileSecurityState state = new FileSecurityState(FileSecurityStateAccess.PathDiscovery, String.Empty, s);
 | |
|                     state.EnsureState();
 | |
|                 }
 | |
|                 else
 | |
| #endif // FEATURE_CORECLR
 | |
|                 {
 | |
|                     new FileIOPermission(FileIOPermissionAccess.PathDiscovery, s).Demand();
 | |
|                 }
 | |
|             }
 | |
|             return s;
 | |
| #endif // FEATURE_CORESYSTEM
 | |
|         }
 | |
|         
 | |
| #if !FEATURE_PAL
 | |
|         public static string UserDomainName
 | |
|         {
 | |
|                 [System.Security.SecuritySafeCritical]  // auto-generated
 | |
|                 get {
 | |
|                     new EnvironmentPermission(EnvironmentPermissionAccess.Read,"UserDomain").Demand();
 | |
| 
 | |
|                     byte[] sid = new byte[1024];
 | |
|                     int sidLen = sid.Length;
 | |
|                     StringBuilder domainName = new StringBuilder(1024);
 | |
|                     uint domainNameLen = (uint) domainName.Capacity;
 | |
|                     int peUse;
 | |
| 
 | |
|                     byte ret = Win32Native.GetUserNameEx(Win32Native.NameSamCompatible, domainName, ref domainNameLen);
 | |
|                         if (ret == 1) {                        
 | |
|                             string samName = domainName.ToString();
 | |
|                             int index = samName.IndexOf('\\');
 | |
|                             if( index != -1) {
 | |
|                                 return samName.Substring(0, index);
 | |
|                             }
 | |
|                         }
 | |
|                         domainNameLen = (uint) domainName.Capacity;                    
 | |
|                     
 | |
|                     bool success = Win32Native.LookupAccountName(null, UserName, sid, ref sidLen, domainName, ref domainNameLen, out peUse);
 | |
|                     if (!success)  {
 | |
|                         int errorCode = Marshal.GetLastWin32Error();
 | |
|                         throw new InvalidOperationException(Win32Native.GetMessage(errorCode));
 | |
|                     }
 | |
| 
 | |
|                     return domainName.ToString();
 | |
|                 }
 | |
|             }
 | |
| #endif // !FEATURE_PAL
 | |
|         public enum SpecialFolderOption {
 | |
|             None        = 0,
 | |
|             Create      = Win32Native.CSIDL_FLAG_CREATE,
 | |
|             DoNotVerify = Win32Native.CSIDL_FLAG_DONT_VERIFY,
 | |
|         }
 | |
|         
 | |
| //////!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!////////
 | |
| //////!!!!!! Keep the following locations synchronized            !!!!!!////////
 | |
| //////!!!!!! 1) ndp\clr\src\BCL\Microsoft\Win32\Win32Native.cs    !!!!!!////////
 | |
| //////!!!!!! 2) ndp\clr\src\BCL\System\Environment.cs             !!!!!!////////
 | |
| //////!!!!!! 3) rotor\pal\inc\rotor_pal.h                         !!!!!!////////
 | |
| //////!!!!!! 4) rotor\pal\corunix\shfolder\shfolder.cpp           !!!!!!////////
 | |
| //////!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!////////
 | |
|         [ComVisible(true)]
 | |
|         public enum SpecialFolder {
 | |
|             //  
 | |
|             //      Represents the file system directory that serves as a common repository for
 | |
|             //       application-specific data for the current, roaming user. 
 | |
|             //     A roaming user works on more than one computer on a network. A roaming user's 
 | |
|             //       profile is kept on a server on the network and is loaded onto a system when the
 | |
|             //       user logs on. 
 | |
|             //  
 | |
|             ApplicationData =  Win32Native.CSIDL_APPDATA,
 | |
|             //  
 | |
|             //      Represents the file system directory that serves as a common repository for application-specific data that
 | |
|             //       is used by all users. 
 | |
|             //  
 | |
|             CommonApplicationData =  Win32Native.CSIDL_COMMON_APPDATA,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for application specific data that
 | |
|             //       is used by the current, non-roaming user. 
 | |
|             //  
 | |
|             LocalApplicationData =  Win32Native.CSIDL_LOCAL_APPDATA,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for Internet
 | |
|             //       cookies. 
 | |
|             //  
 | |
|             Cookies =  Win32Native.CSIDL_COOKIES,
 | |
|             Desktop = Win32Native.CSIDL_DESKTOP,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for the user's
 | |
|             //       favorite items. 
 | |
|             //  
 | |
|             Favorites =  Win32Native.CSIDL_FAVORITES,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for Internet
 | |
|             //       history items. 
 | |
|             //  
 | |
|             History =  Win32Native.CSIDL_HISTORY,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for temporary 
 | |
|             //       Internet files. 
 | |
|             //  
 | |
|             InternetCache =  Win32Native.CSIDL_INTERNET_CACHE,
 | |
|             //  
 | |
|             //      Represents the file system directory that contains
 | |
|             //       the user's program groups. 
 | |
|             //  
 | |
|             Programs =  Win32Native.CSIDL_PROGRAMS,
 | |
|             MyComputer =  Win32Native.CSIDL_DRIVES,
 | |
|             MyMusic =  Win32Native.CSIDL_MYMUSIC,
 | |
|             MyPictures = Win32Native.CSIDL_MYPICTURES,
 | |
|             //      "My Videos" folder
 | |
|             MyVideos = Win32Native.CSIDL_MYVIDEO,
 | |
|             //  
 | |
|             //     Represents the file system directory that contains the user's most recently used
 | |
|             //       documents. 
 | |
|             //  
 | |
|             Recent =  Win32Native.CSIDL_RECENT,
 | |
|             //  
 | |
|             //     Represents the file system directory that contains Send To menu items. 
 | |
|             //  
 | |
|             SendTo =  Win32Native.CSIDL_SENDTO,
 | |
|             //  
 | |
|             //     Represents the file system directory that contains the Start menu items. 
 | |
|             //  
 | |
|             StartMenu =  Win32Native.CSIDL_STARTMENU,
 | |
|             //  
 | |
|             //     Represents the file system directory that corresponds to the user's Startup program group. The system
 | |
|             //       starts these programs whenever any user logs on to Windows NT, or
 | |
|             //       starts Windows 95 or Windows 98. 
 | |
|             //  
 | |
|             Startup =  Win32Native.CSIDL_STARTUP,
 | |
|             //  
 | |
|             //     System directory.
 | |
|             //  
 | |
|             System =  Win32Native.CSIDL_SYSTEM,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for document
 | |
|             //       templates. 
 | |
|             //  
 | |
|             Templates =  Win32Native.CSIDL_TEMPLATES,
 | |
|             //  
 | |
|             //     Represents the file system directory used to physically store file objects on the desktop.
 | |
|             //       This should not be confused with the desktop folder itself, which is
 | |
|             //       a virtual folder. 
 | |
|             //  
 | |
|             DesktopDirectory =  Win32Native.CSIDL_DESKTOPDIRECTORY,
 | |
|             //  
 | |
|             //     Represents the file system directory that serves as a common repository for documents. 
 | |
|             //  
 | |
|             Personal =  Win32Native.CSIDL_PERSONAL, 
 | |
|             //          
 | |
|             // "MyDocuments" is a better name than "Personal"
 | |
|             //
 | |
|             MyDocuments = Win32Native.CSIDL_PERSONAL,                         
 | |
|             //  
 | |
|             //     Represents the program files folder. 
 | |
|             //  
 | |
|             ProgramFiles =  Win32Native.CSIDL_PROGRAM_FILES,
 | |
|             //  
 | |
|             //     Represents the folder for components that are shared across applications. 
 | |
|             //  
 | |
|             CommonProgramFiles =  Win32Native.CSIDL_PROGRAM_FILES_COMMON,            
 | |
| #if !FEATURE_CORECLR
 | |
|             //
 | |
|             //      <user name>\Start Menu\Programs\Administrative Tools
 | |
|             //
 | |
|             AdminTools             = Win32Native.CSIDL_ADMINTOOLS,
 | |
|             //
 | |
|             //      USERPROFILE\Local Settings\Application Data\Microsoft\CD Burning
 | |
|             //
 | |
|             CDBurning              = Win32Native.CSIDL_CDBURN_AREA,
 | |
|             //
 | |
|             //      All Users\Start Menu\Programs\Administrative Tools
 | |
|             //
 | |
|             CommonAdminTools       = Win32Native.CSIDL_COMMON_ADMINTOOLS,
 | |
|             //
 | |
|             //      All Users\Documents
 | |
|             //
 | |
|             CommonDocuments        = Win32Native.CSIDL_COMMON_DOCUMENTS,
 | |
|             //
 | |
|             //      All Users\My Music
 | |
|             //
 | |
|             CommonMusic            = Win32Native.CSIDL_COMMON_MUSIC,
 | |
|             //
 | |
|             //      Links to All Users OEM specific apps
 | |
|             //
 | |
|             CommonOemLinks         = Win32Native.CSIDL_COMMON_OEM_LINKS,
 | |
|             //
 | |
|             //      All Users\My Pictures
 | |
|             //
 | |
|             CommonPictures         = Win32Native.CSIDL_COMMON_PICTURES,
 | |
|             //
 | |
|             //      All Users\Start Menu
 | |
|             //
 | |
|             CommonStartMenu        = Win32Native.CSIDL_COMMON_STARTMENU,
 | |
|             //
 | |
|             //      All Users\Start Menu\Programs
 | |
|             //
 | |
|             CommonPrograms         = Win32Native.CSIDL_COMMON_PROGRAMS,
 | |
|             //
 | |
|             //     All Users\Startup
 | |
|             //
 | |
|             CommonStartup          = Win32Native.CSIDL_COMMON_STARTUP,
 | |
|             //
 | |
|             //      All Users\Desktop
 | |
|             //
 | |
|             CommonDesktopDirectory = Win32Native.CSIDL_COMMON_DESKTOPDIRECTORY,
 | |
|             //
 | |
|             //      All Users\Templates
 | |
|             //
 | |
|             CommonTemplates        = Win32Native.CSIDL_COMMON_TEMPLATES,
 | |
|             //
 | |
|             //      All Users\My Video
 | |
|             //
 | |
|             CommonVideos           = Win32Native.CSIDL_COMMON_VIDEO,
 | |
|             //
 | |
|             //      windows\fonts
 | |
|             //
 | |
|             Fonts                  = Win32Native.CSIDL_FONTS,
 | |
|             //
 | |
|             //      %APPDATA%\Microsoft\Windows\Network Shortcuts
 | |
|             //
 | |
|             NetworkShortcuts       = Win32Native.CSIDL_NETHOOD,
 | |
|             //
 | |
|             //      %APPDATA%\Microsoft\Windows\Printer Shortcuts
 | |
|             //
 | |
|             PrinterShortcuts       = Win32Native.CSIDL_PRINTHOOD,
 | |
|             //
 | |
|             //      USERPROFILE
 | |
|             //
 | |
|             UserProfile            = Win32Native.CSIDL_PROFILE,
 | |
|             //
 | |
|             //      x86 Program Files\Common on RISC
 | |
|             //
 | |
|             CommonProgramFilesX86  = Win32Native.CSIDL_PROGRAM_FILES_COMMONX86,
 | |
|             //
 | |
|             //      x86 C:\Program Files on RISC
 | |
|             //
 | |
|             ProgramFilesX86        = Win32Native.CSIDL_PROGRAM_FILESX86,
 | |
|             //
 | |
|             //      Resource Directory
 | |
|             //
 | |
|             Resources              = Win32Native.CSIDL_RESOURCES,
 | |
|             //
 | |
|             //      Localized Resource Directory
 | |
|             //
 | |
|             LocalizedResources     = Win32Native.CSIDL_RESOURCES_LOCALIZED,
 | |
|             //
 | |
|             //      %windir%\System32 or %windir%\syswow64
 | |
|             //
 | |
|             SystemX86               = Win32Native.CSIDL_SYSTEMX86,
 | |
|             //
 | |
|             //      GetWindowsDirectory()
 | |
|             //
 | |
|             Windows                = Win32Native.CSIDL_WINDOWS,
 | |
| #endif // !FEATURE_CORECLR
 | |
|         }
 | |
| 
 | |
|         public static int CurrentManagedThreadId
 | |
|         {
 | |
|             [ReliabilityContract(Consistency.WillNotCorruptState, Cer.Success)]
 | |
|             get
 | |
|             {
 | |
|                 return Thread.CurrentThread.ManagedThreadId;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |