问:我有一个产生大量子进程的 Windows 服务。似乎有一些限制,大约 100 个,无法启动该进程。CreateProcess() 调用返回一个 pid,但随后该过程无法实现。如果我将我们的服务器作为控制台应用程序运行,那么这个限制就会消失。此外,如果我设置 DETACHED_PROCESS 标志,这个限制会增加一倍以上。但是,如果我设置 DETACHED_PROCESS 并调用 CreateProcesssWithLogonW(),则会失败。
2 回答
这是一个“桌面堆”问题。可以在这里找到一个很好的讨论:
请注意,这仅适用于作为服务运行的程序,因为服务的默认桌面堆大小比应用程序的小得多。
在我们的案例中,我们能够在资源耗尽之前启动大约 100 个子进程而无需进行更改。随着变化,这个数字可以大大增加。
这是我们在知识库中为最终用户提供的答案:
警告:这会影响所有服务的桌面堆!不要让它比必要的大,否则你会推动系统消耗更多的资源,并且你可能会遇到可用桌面堆大小的问题。
如果你发现你不能打开超过 100 个项目,即使在非常大的 RAM 服务器上,你也可能遇到了 Windows“桌面堆大小”的限制。
问题是windows下的服务会话(服务运行的地方)有更少的这个“桌面堆”空间可用于创建windows。
简短的版本是:
与交互式会话相比,服务获得的桌面堆更小。
桌面堆大小限制窗口数量
即使我们看不到它们,每个子服务器也会创建一个或多个“窗口”。
解决方案:
在进行任何更改之前备份您的注册表!
以管理员身份运行 regedit.exe
编辑注册表值:
HKEY_LOCAL_MACHINE\System\CurrentControlSet\Control\Session Manager\SubSystems\Windows
您将看到如下字符串:
%SystemRoot%\system32\csrss.exe ObjectDirectory=\Windows SharedSection=1024,20480,768 Windows=On SubSystemType=Windows ServerDll=basesrv,1 ServerDll=winsrv:UserServerDllInitialization,3 ServerDll=winsrv:ConServerDllInitialization,2 ServerDll=sxssrv,4 ProfileControl=Off MaxRequestThreads=16
关键是:
SharedSection=1024,20480,768
第二个数字 (20480) 是交互式会话的大小。第三个数字 (768) 是非交互式(服务)会话的大小。请注意第三个数字比第二个数字小 26 倍。通过实验,我们发现将其更改为:
SharedSection=1024,20480,2048
将项目限制从 106 增加到 270,几乎完美地随堆大小缩放。选择一个值,该值反映您希望系统上的所有用户同时打开的最大项目数。不要使这个值大于必要的值,并且不大于 8192,因为系统中的每个服务都会消耗更多的宝贵资源。
您需要重新启动才能使这些新设置生效。
我们需要在许多远程桌面服务器上进行测试,因此我们编写了一个 powershell 脚本来查询我们的 RD 服务器的 AD,然后应用此注册表更改。享受
$newValue = "%SystemRoot%\system32\csrss.exe ObjectDirectory=\Windows SharedSection=1024,20480,2048 Windows=On SubSystemType=Windows ServerDll=basesrv,1 ServerDll=winsrv:UserServerDllInitialization,3 ServerDll=sxssrv,4 ProfileControl=Off MaxRequestThreads=16"
$origValue = "%SystemRoot%\system32\csrss.exe ObjectDirectory=\Windows SharedSection=1024,20480,768 Windows=On SubSystemType=Windows ServerDll=basesrv,1 ServerDll=winsrv:UserServerDllInitialization,3 ServerDll=sxssrv,4 ProfileControl=Off MaxRequestThreads=16"
if ($update) {
Clear-Variable update
}
$updateConfirm= [System.Windows.Forms.MessageBox]::Show("Update Desktop Heap Limitation to 2048?" , "Status" , 4)
if ($updateConfirm -eq "YES" ) {
$update = $true
} else {
$revertConfirm= [System.Windows.Forms.MessageBox]::Show("Revert Desktop Heap Limitation to 768?" , "Status" , 4)
if ($revertConfirm -eq "YES" ) {
$update = $false
}
}
if (($updateConfirm -ne "YES") -and ($revertConfirm -ne "YES")) {
Write-Host "User did not specify whether to update or revert Desktop Heap Limitation. Exiting Setup."
Read-Host "Press Enter to exit."
break
}
#Import Active Directory PowerShell module
if (Test-Path C:\Users\${env:USERNAME}\Documents\WindowsPowerShell\Modules\ActiveDirectory\ActiveDirectory.psm1) {
Import-Module ActiveDirectory -prefix AD
} else {
$s = New-PSSession -computerName DC01WDC01
Invoke-command { import-module ActiveDirectory } -session $s
Export-PSSession -session $s -commandname *-AD* -outputmodule ActiveDirectory -allowclobber
Import-Module ActiveDirectory -prefix AD
Remove-PSSession -session $s
}
$servers = Get-ADADComputer -Filter {(Name -Like "RDS*")} | Select -Expand Name
foreach ($server in $servers) {
Write-Host "Working on $server" -ForegroundColor Magenta
if(!(Test-Connection -ComputerName $server -Count 1 -quiet)) {
Write-Warning "$server : Offline"
Continue
}
if ($update -eq $true) {
Invoke-Command -ComputerName $server -ArgumentList $newValue -ScriptBlock {
Set-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\SubSystems" -Name "Windows" -Value $args[0]
$result = Get-ItemProperty "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\SubSystems" | SELECT Windows
if ($result.Windows -like "*SharedSection=1024,20480,2048*") {
Write-Host "Successfully reverted Desktop Heap Limit to 2048" -ForegroundColor Green
} else {
Write-Warning "Update to registry value unsuccessful on $env:ComputerName"
}
}
} elseif ($update -eq $false) {
Invoke-Command -ComputerName $server -ArgumentList $origValue -ScriptBlock {
Set-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\SubSystems" -Name "Windows" -Value $args[0]
$result = Get-ItemProperty "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\SubSystems" | SELECT Windows
if ($result.Windows -like "*SharedSection=1024,20480,768*") {
Write-Host "Successfully reverted Desktop Heap Limit to 768" -ForegroundColor Green
} else {
Write-Warning "Update to registry value unsuccessful on $env:ComputerName"
}
}
}
}