我有一个非常奇怪的问题;在我们的REST应用程序中,我们引入了一个上传WebPackage的选项。由于这些在解压缩时可能相当大,我们希望通过MemoryFailPoint确保内存检查的成功经验。
在我的本地IIS 7.5上,这可以完美运行 - 甚至可以达到最大值。预期值2GB。 在我们的带有IIS 7.5的虚拟Windows Server 2008R2 x64上,无条件失败 - 即使只尝试1MB。
虚拟服务器托管在VMWare ESXi 4.1.0,348481上。
以下是虚拟机上失败的示例代码:
using (MemoryFailPoint failPoint = new MemoryFailPoint(1))
{
... // deliberately excluded webpackage code
}
异常(在XML中):
<InsufficientMemoryException>
<Message>Insufficient memory to meet the expected demands of an operation, and this system is likely to never satisfy this request. If this is a 32 bit system, consider booting in 3 GB mode.</Message>
<StackTrace>
<Line>at System.Runtime.MemoryFailPoint..ctor(Int32 sizeInMegabytes)</Line>
<Line>at System.ServiceModel.Dispatcher.SyncMethodInvoker.Invoke(Object instance, Object[] inputs, Object[]& outputs)</Line>
<Line>at System.ServiceModel.Dispatcher.DispatchOperationRuntime.InvokeBegin(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.ImmutableDispatchRuntime.ProcessMessage5(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.ImmutableDispatchRuntime.ProcessMessage31(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.MessageRpc.Process(Boolean isOperationContextSet)</Line>
</StackTrace>
<UserDefinedInformation>
<failpointSizeInMegaBytes>1.00</failpointSizeInMegaBytes>
<gcTotalMemoryBeforeInMegaBytes>153.34</gcTotalMemoryBeforeInMegaBytes>
</UserDefinedInformation>
</InsufficientMemoryException>
我机器上有一个类似的例外 - 请注意我需要分配 64GB 之前我可以触发异常:
<InsufficientMemoryException>
<Message>Insufficient available memory to meet the expected demands of an operation at this time. Please try again later.</Message>
<StackTrace>
<Line>at System.Runtime.MemoryFailPoint..ctor(Int32 sizeInMegabytes)</Line>
<Line>at System.ServiceModel.Dispatcher.SyncMethodInvoker.Invoke(Object instance, Object[] inputs, Object[]& outputs)</Line>
<Line>at System.ServiceModel.Dispatcher.DispatchOperationRuntime.InvokeBegin(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.ImmutableDispatchRuntime.ProcessMessage5(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.ImmutableDispatchRuntime.ProcessMessage31(MessageRpc& rpc)</Line>
<Line>at System.ServiceModel.Dispatcher.MessageRpc.Process(Boolean isOperationContextSet)</Line>
</StackTrace>
<UserDefinedInformation>
<failpointSizeInMegaBytes>65536.00</failpointSizeInMegaBytes>
<gcTotalMemoryBeforeInMegaBytes>150.63</gcTotalMemoryBeforeInMegaBytes>
</UserDefinedInformation>
</InsufficientMemoryException>
这让我疯了......真实......但是嘿;注意两个(非常)不同的例外:
服务器:内存不足以满足操作的预期需求,并且此系统可能永远不会满足此请求。如果这是32位系统,请考虑以3 GB模式启动。
使用Jetbrains和本网站http://reflector.webtropy.com/default.aspx/4@0/4@0/untmp/DEVDIV_TFS/Dev10/Releases/RTMRel/ndp/clr/src/BCL/System/Runtime/MemoryFailPoint@cs/1305376/MemoryFailPoint@cs的dotPeek,我们可以看到可能存在的问题(尽管我仍然无能为力):
// Check to see that we both have enough memory on the system
// and that we have enough room within the user section of the
// process's address space. Also, we need to use the GC segment
// size, not the amount of memory the user wants to allocate.
// Consider correcting this to reflect free memory within the GC
// heap, and to check both the normal & large object heaps.
ulong num1 = (ulong) sizeInMegabytes << 20;
this._reservedMemory = num1;
ulong size = (ulong)(Math.Ceiling((double)num1 / (double)MemoryFailPoint.GCSegmentSize) * (double)MemoryFailPoint.GCSegmentSize);
if (size >= MemoryFailPoint.TopOfMemory)
throw new InsufficientMemoryException(Environment.GetResourceString("InsufficientMemory_MemFailPoint_TooBig"));
本地:可用内存不足,无法满足此时操作的预期需求。请稍后再试。
修改
Hans和我讨论过VM Size,但由于我现在已经得出结论,它在开发服务器和我的机器上或多或少相同,我现在将对此部分进行排序。任何提示,建议等都非常受欢迎。
流程信息
Process Name: w3wp
PID: 5716
User Name: NT AUTHORITY\NETWORK SERVICE
Working Set: 376.456 K
Peak Working Set: 432.400 K
Private Working Set: 320.684 K
Commit Size: 538.552 K
Handles: 919
Threads: 39
操作系统(WMI)
BootDevice: \Device\HarddiskVolume1
BuildNumber: 7601
BuildType: Multiprocessor Free
Caption: Microsoft Windows Server 2008 R2 Standard
CodeSet: 1252
CountryCode: 45
CreationClassName: Win32_OperatingSystem
CSCreationClassName: Win32_ComputerSystem
CSDVersion: Service Pack 1
CSName: SOME_NAME [MODIFIED]
CurrentTimeZone: 120
DataExecutionPrevention_32BitApplications: True
DataExecutionPrevention_Available: True
DataExecutionPrevention_Drivers: True
DataExecutionPrevention_SupportPolicy: 3
Debug: False
Description: Development Server
Distributed: False
EncryptionLevel: 256
ForegroundApplicationBoost: 2
FreePhysicalMemory: 5.366.516 K
FreeSpaceInPagingFiles: 8.368.456 K
FreeVirtualMemory: 12.985.412 K
InstallDate: 17-01-2011 15:01:55
LargeSystemCache: null
LastBootUpTime: 11-09-2012 14:44:33
LocalDateTime: 03-10-2012 14:13:47
Locale: 0406
Manufacturer: Microsoft Corporation
MaxNumberOfProcesses: 4294967295
MaxProcessMemorySize: 8.589.934.464 K
MUILanguages: System.Object[]
Name: Microsoft Windows Server 2008 R2 Standard |C:\Windows|\Device\Harddisk0\Partition2
NumberOfLicensedUsers: null
NumberOfProcesses: 70
NumberOfUsers: 7
OperatingSystemSKU: 7
Organization:
OSArchitecture: 64-bit
OSLanguage: 1033
OSProductSuite: 272
OSType: 18
OtherTypeDescription: null
PAEEnabled: null
PlusProductID: null
PlusVersionNumber: null
Primary: True
ProductType: 3
RegisteredUser: Windows User
SerialNumber: 11111-111-1111111-11111 [MODIFIED]
ServicePackMajorVersion: 1
ServicePackMinorVersion: 0
SizeStoredInPagingFiles: 8.388.152 K
Status: OK
SuiteMask: 272
SystemDevice: \Device\HarddiskVolume2
SystemDirectory: C:\Windows\system32
SystemDrive: C:
TotalSwapSpaceSize: null
TotalVirtualMemorySize: 16.774.452 K
TotalVisibleMemorySize: 8.388.152 K
Version: 6.1.7601
WindowsDirectory: C:\Windows
来自Development Server的VMMap屏幕截图
对虚拟大小感到好奇,我从Sysinternals下载VMMap:http://technet.microsoft.com/en-us/sysinternals/dd535533
来自我的本地计算机的VMMap屏幕截图(MemoryFailPoint按预期工作)
我仍然不清楚为什么MemoryFailPoint失败 - 重要的是(特别是在谈论Windows Azure或其他云提供商时)以确保有足够的内存用于操作。举个例子,然后ExtraSmall实例会因为资源非常有限而在进程中的某个地方失败(即使是x64架构) - 这可以通过MemoryFailPoint来防止,从而确保数据的有效状态。
非常感谢任何帮助。
答案 0 :(得分:5)
.net 4.5和IIS中存在错误(或功能)。
MemoryFailPoint类依赖于内部GC相关变量,但在x64下,此变量可以为0,除以零可以生成正无穷大(双精度)和一些运算(甚至乘以零),这个极值可以生成一个大的double值(取决于实际的FPU设置)。
您可以非常轻松地检查此错误。 创建一个简单的Web应用程序并添加页面然后粘贴此代码:
<%
var memoryFailPointType = typeof(System.Runtime.MemoryFailPoint);
var staticPrivateFieldFlags =
System.Reflection.BindingFlags.NonPublic |
System.Reflection.BindingFlags.Static;
var gCSegmentSizeField = memoryFailPointType
.GetField("GCSegmentSize", staticPrivateFieldFlags);
var gCSegmentSize = (uint)gCSegmentSizeField.GetValue(null);
string allocationResult;
try
{
using (var mfp = new System.Runtime.MemoryFailPoint(1))
{
allocationResult = "Ok";
}
}
catch (Exception exception)
{
allocationResult = exception.Message;
}
%>
<%= gCSegmentSize %><br /> <!-- it can be 0 -->
<%= allocationResult %><br /> <!-- Insufficient memory to meet the expected demands of an operation, and this system is likely to never satisfy this request. If this is a 32 bit system, consider booting in 3 GB mode. -->
因此,如果gCSegmentSize为零且(ulong)(double.PositiveInfinite *(uint)0)不等于0,则不应使用MemoryFailPoint类。
在控制台应用程序中,FPU控制字为0x09001F,但在IIS下,FPU控制字为0x08001F。
答案 1 :(得分:1)
针对.NET 4.5的修补程序汇总2828842中有针对此问题的修复程序 http://support.microsoft.com/kb/2828842/en-us