# # This script launches the VC redist executable and handles certain error scenarios. # param( [Parameter(Mandatory=$true)][String][ValidateNotNullOrEmpty()]$PayloadDirectory, [Parameter(Mandatory=$true)][String][ValidateSet("arm64","x64","x86")]$Architecture, [Parameter(Mandatory=$true)][String][ValidateNotNullOrEmpty()]$LogFile, [Parameter(Mandatory=$false)][String]$VersionToInstall ) # # Variables and helper functions # # Use transcript to log all output $envTemp = [Environment]::GetEnvironmentVariable('TEMP', 'User') $date = Get-Date -Format "yyyyMMddHHmmss" $transcriptPath = "$envTemp\dd_vcredist_wrapper_" + $Architecture + "_" + $date + ".log" Start-Transcript -Path $transcriptPath # Gather VC runtime version information in registry and files, generating appropriate output Function Collect-Version-Information($outputString) { # Get registry version information $vcrtRegVersionNative = $missingVersion $vcrtRegVersionWOW = $missingVersion if (![string]::IsNullOrEmpty($vcrtRegKeyPathNative) -and (Test-Path $vcrtRegKeyPathNative)) { $vcrtRegKeyNative = Get-ItemProperty -Path $vcrtRegKeyPathNative $vcrtRegVersionNative = "$($vcrtRegKeyNative.Major).$($vcrtRegKeyNative.Minor).$($vcrtRegKeyNative.Bld).$($vcrtRegKeyNative.Rbld)" } if (![string]::IsNullOrEmpty($vcrtRegKeyPathWOW) -and (Test-Path $vcrtRegKeyPathWOW)) { $vcrtRegKeyWOW = Get-ItemProperty -Path $vcrtRegKeyPathWOW $vcrtRegVersionWOW = "$($vcrtRegKeyWOW.Major).$($vcrtRegKeyWOW.Minor).$($vcrtRegKeyWOW.Bld).$($vcrtRegKeyWOW.Rbld)" } $regVersionToCheck = $vcrtRegVersionNative if (($installType -eq [InstallType]::X64Native) -or ($installType -eq [InstallType]::Arm64Native) -and ($vcrtRegVersionNative -ne $vcrtRegVersionWOW)) { $vcrtRegVersion = "registry version mismatch" Write-Host "$outputString registry versions" if ($vcrtRegVersionNative -ne $missingVersion) { Write-Host " native: $vcrtRegVersionNative" } else { Write-Host " native: $missingVersionOutput" } if ($vcrtRegVersionWOW -ne $missingVersion) { Write-Host " WOW: $vcrtRegVersionWOW" } else { Write-Host " WOW: $missingVersionOutput" } $regVersionToCheck = $null } elseif ($installType -eq [InstallType]::X86OnX64) { $regVersionToCheck = $vcrtRegVersionWOW } if ($regVersionToCheck -ne $null) { if ($regVersionToCheck -ne $missingVersion) { $vcrtRegVersion = $regVersionToCheck Write-Host "$outputString registered version: $vcrtRegVersion" } else { Write-Host "No version registered" } } # Get file version information $vcrtFileVersion = $null Foreach ($redistFile in $allRedistFiles) { if (Test-Path -path "$systemDir\$redistFile") { $redistFileVersion = (Get-Item -LiteralPath "$systemDir\$redistFile").VersionInfo.FileVersionRaw $redistFileVersions += @{ $redistFile = $redistFileVersion } if ($vcrtFileVersion -eq $null) { $vcrtFileVersion = $redistFileVersion } elseif ($redistFileVersion -ne $vcrtFileVersion) { $vcrtFileVersion = "multiple versions found" } } else { $redistFileVersions += @{ $redistFile = $missingVersion } } } if ($vcrtFileVersion -eq $null) { $vcrtFileVersion = "no VC runtime files installed" } Write-Host "$outputString file version: $vcrtFileVersion" # If the registry and all file versions do not agree, list each file and its version $vcrtVersionConsistent = $true $outputStringLowerCase = $outputString.ToLower() if ($vcrtRegVersion -ne $vcrtFileVersion) { Write-Host "Inconsistent VC runtime install detected: registry version does not match file version" $vcrtVersionConsistent = $false $skipSoftRebootCheckReasons += "$outputStringLowerCase registry version does not match $outputStringLowerCase file version" } if ($vcrtFileVersion -eq "multiple versions found") { Write-Host "Inconsistent VC runtime install detected: not all files have same version" $vcrtVersionConsistent = $false $skipSoftRebootCheckReasons += "not all $outputStringLowerCase files have same version" } if (! $vcrtVersionConsistent ) { Foreach ($redistFile in $allRedistFiles) { $redistFileVersionValue = $redistFileVersions[$redistFile] if ($redistFileVersionValue -eq $missingVersion) { $redistFileVersionValue = "not installed"} Write-Host " ${redistFile}: $redistFileVersionValue" } $vcrtVersionConsistent = $false } $vcrtVersion = $vcrtRegVersion if ($vcrtVersionConsistent -eq $false) { $vcrtVersion = $null } return $vcrtVersionConsistent, $vcrtVersion } # # Main execution sequence # # Initialize variables $envMachineArch = [Environment]::GetEnvironmentVariable("PROCESSOR_ARCHITECTURE", 'Machine'); $envWINDIR = [Environment]::GetEnvironmentVariable('WINDIR', 'Machine') # These variables are initialized to default x64 native values and overridden as appropriate: # $installType = one of { X64Native, X86OnX64, X86Native } # $systemDir = where to find installed files # $registryPath[Native,WOW] = the paths to the Native and WOW registry keys enum InstallType { Arm64Native X64Native X86OnX64 X86Native } # Setting $systemDir needs to account for System32 redirection on 64-bit processors: # # MachineArch Process VCRTArch TargetDirectory SystemDirValue # x64 64-bit x64 System32 System32 # x64 32-bit x64 System32 Sysnative (avoids redirection to SysWOW64) # x64 64-bit x86 SysWOW64 SysWOW64 # x64 32-bit x86 SysWOW64 System32 (redirects to SysWOW64) # x86 32-bit x86 System32 System32 # arm64 64-bit arm64 System32 System32 $installType = [InstallType]::X64Native $is64BitPS = [IntPtr]::Size -eq 8 $systemDir = "$envWINDIR\System32" if ($Architecture -eq "x64" -and $is64BitPS -eq $false) { $systemDir = "$envWINDIR\Sysnative" } elseif ($Architecture -eq "x86" -and $is64BitPS -eq $true) { $systemDir = "$envWINDIR\SysWOW64" } # Registry keys $vcrtRegKeyPathNative = "HKLM:\Software\Microsoft\VisualStudio\14.0\VC\Runtimes\$Architecture" $vcrtRegKeyPathWOW = "HKLM:\Software\WOW6432Node\Microsoft\VisualStudio\14.0\VC\Runtimes\$Architecture" if ($envMachineArch -eq 'AMD64' -and $Architecture -eq 'x86') { $installType = [InstallType]::X86OnX64 } elseif ($envMachineArch -eq 'X86' -and $Architecture -eq 'x86') { $installType = [InstallType]::X86Native $vcrtRegKeyPathWOW = "" } elseif ($envMachineArch -eq 'ARM64') { $installType = [InstallType]::Arm64Native } # The value "0.0.0.0" indicates the registry key is empty or the file is not present, # which typically indicates the VCRT is not installed or only partially installed. $missingVersion = "0.0.0.0" $missingVersionOutput = "not found" # List of VCRT files $redistSharedFiles = @( "concrt140.dll", "mfc140.dll", "mfc140chs.dll", "mfc140cht.dll", "mfc140deu.dll", "mfc140enu.dll", "mfc140esn.dll", "mfc140fra.dll", "mfc140ita.dll", "mfc140jpn.dll", "mfc140kor.dll", "mfc140rus.dll", "mfc140u.dll", "msvcp140.dll", "msvcp140_1.dll", "msvcp140_2.dll", "msvcp140_atomic_wait.dll", "msvcp140_codecvt_ids.dll", "vcamp140.dll", "vccorlib140.dll", "vcomp140.dll", "vcruntime140.dll" ) $redistIntelFiles = @( "mfcm140.dll", "mfcm140u.dll" ) $redistX64Files = @( "vcruntime140_1.dll" ) if ($Architecture -eq 'x64') { $allRedistFiles = @($redistSharedFiles) + @($redistIntelFiles) + @($redistX64Files) } elseif ($Architecture -eq 'arm64') { $allRedistFiles = @($redistSharedFiles) + @($redistX64Files) } else { $allRedistFiles = @($redistSharedFiles) + @($redistIntelFiles) } # Begin output Write-Host "Installing Visual C++ Redist" Write-Host "Inputs:" Write-Host " PayloadDirectory: $PayloadDirectory" Write-Host " Architecture: $Architecture" Write-Host " LogFile: $LogFile" Write-Host " VersionToInstall: $VersionToInstall" Write-Host "Environment:" Write-Host " WINDIR: $envWINDIR" Write-Host " TEMP: $envTemp" Write-Host " system directory: $systemDir" Write-Host " native reg. key: $vcrtRegKeyPathNative" Write-Host " WOW64 reg. key: $vcrtRegKeyPathWOW" Write-Host "Installing $Architecture Visual C++ Runtime on $envMachineArch system" $checkForSoftReboot = $true $skipSoftRebootCheckReasons = @() # Lookup current versions in registry and files to determine if there is # a consistent VC Runtime installation. An inconsistent initial install does not # disable soft-reboot since it can be a valid state (vc_redist installs a full # set of runtime files, then an MSM install updates only a subset of files). $vcrtIsInitialInstallConsistent, $vcrtUpdatedVersion = Collect-Version-Information("Initial") if (! $vcrtIsInitialInstallConsistent ) { Write-Host "Information: initial VC runtime install is not consistent" } # Run VCRT install command with parameters used for VS installs # command: $PayloadDirectory\vc_redist.$Architecture.exe /q /norestart /log $LogFile # example: VC_redist.x64.exe /q /norestart /log # Use Start-Process to wait for exit and pass-thru the exit code $vcrtExePath=-join($PayloadDirectory, "\\VC_redist.", $Architecture, ".exe") $vcrtInstallerVersion = (Get-Item -LiteralPath $vcrtExePath).VersionInfo.FileVersionRaw Write-Host "Redist Installer Version: $vcrtInstallerVersion" $vcrtProcess = Start-Process -FilePath $vcrtExePath -ArgumentList "/q /norestart /log $LogFile" -Wait -PassThru $vcrtProcessExitCode = $vcrtProcess.ExitCode Write-Host "Redist Exit Code: $vcrtProcessExitCode" # Gather updated versions in registry and files $vcrtIsUpdatedInstallConsistent, $vcrtUpdatedVersion = Collect-Version-Information("Updated") if (! $vcrtIsUpdatedInstallConsistent ) { Write-Host "Skipping soft-reboot check: updated VC runtime install is not consistent" $checkForSoftReboot = $false } if ((! [string]::IsNullOrEmpty($VersionToInstall)) -and ($vcrtUpdatedVersion -ne $VersionToInstall)) { $msg = "updated VC runtime version $vcrtUpdatedVersion does not match expected version $VersionToInstall" Write-Host "Skipping soft-reboot check: $msg" $checkForSoftReboot = $false $skipSoftRebootCheckReasons += $msg } $returnCode=$vcrtProcessExitCode if ($vcrtProcessExitCode -ne 3010) { $msg = "redist exit code was not ERROR_SUCCESS_REBOOT_REQUIRED (3010)" Write-Host "Skipping soft-reboot check: $msg" $checkForSoftReboot = $false $skipSoftRebootCheckReasons += $msg } if ($checkForSoftReboot -and ($vcrtIsUpdatedInstallConsistent -eq $true)) { # The soft-reboot case occurs when the installer returns ERROR_SUCCESS_REBOOT_REQUIRED (3010), # but all files were actuall updated and the VCRT install is consistent. In this case, convert # the return code to NS_S_REBOOT_RECOMMENDED (862968) for soft reboot. This code is defined in # files nserror.h and zunenserror.h and can be identified by http://errors. Write-Host "Soft-reboot conditions detected, changing exit code from 3010 to 862968" $returnCode = 862968 } else { Write-Host "Reasons for skipping soft-reboot check:" Foreach ($reason in $skipSoftRebootCheckReasons) { Write-Host " $reason" } } Write-Host "Wrapper Exit Code: $returnCode" Stop-Transcript exit $returnCode # SIG # Begin signature block # MIInygYJKoZIhvcNAQcCoIInuzCCJ7cCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCCe0r/vqGl/Mocv # MXjqqOgoSSIdqX08ddteSBaJH+kc0qCCDYEwggX/MIID56ADAgECAhMzAAACzI61 # lqa90clOAAAAAALMMA0GCSqGSIb3DQEBCwUAMH4xCzAJBgNVBAYTAlVTMRMwEQYD # VQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMTH01pY3Jvc29mdCBDb2RlIFNpZ25p # bmcgUENBIDIwMTEwHhcNMjIwNTEyMjA0NjAxWhcNMjMwNTExMjA0NjAxWjB0MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMR4wHAYDVQQDExVNaWNy # b3NvZnQgQ29ycG9yYXRpb24wggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIB # AQCiTbHs68bADvNud97NzcdP0zh0mRr4VpDv68KobjQFybVAuVgiINf9aG2zQtWK # No6+2X2Ix65KGcBXuZyEi0oBUAAGnIe5O5q/Y0Ij0WwDyMWaVad2Te4r1Eic3HWH # UfiiNjF0ETHKg3qa7DCyUqwsR9q5SaXuHlYCwM+m59Nl3jKnYnKLLfzhl13wImV9 # DF8N76ANkRyK6BYoc9I6hHF2MCTQYWbQ4fXgzKhgzj4zeabWgfu+ZJCiFLkogvc0 # RVb0x3DtyxMbl/3e45Eu+sn/x6EVwbJZVvtQYcmdGF1yAYht+JnNmWwAxL8MgHMz # xEcoY1Q1JtstiY3+u3ulGMvhAgMBAAGjggF+MIIBejAfBgNVHSUEGDAWBgorBgEE # AYI3TAgBBggrBgEFBQcDAzAdBgNVHQ4EFgQUiLhHjTKWzIqVIp+sM2rOHH11rfQw # UAYDVR0RBEkwR6RFMEMxKTAnBgNVBAsTIE1pY3Jvc29mdCBPcGVyYXRpb25zIFB1 # ZXJ0byBSaWNvMRYwFAYDVQQFEw0yMzAwMTIrNDcwNTI5MB8GA1UdIwQYMBaAFEhu # ZOVQBdOCqhc3NyK1bajKdQKVMFQGA1UdHwRNMEswSaBHoEWGQ2h0dHA6Ly93d3cu # bWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01pY0NvZFNpZ1BDQTIwMTFfMjAxMS0w # Ny0wOC5jcmwwYQYIKwYBBQUHAQEEVTBTMFEGCCsGAQUFBzAChkVodHRwOi8vd3d3 # Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01pY0NvZFNpZ1BDQTIwMTFfMjAx # MS0wNy0wOC5jcnQwDAYDVR0TAQH/BAIwADANBgkqhkiG9w0BAQsFAAOCAgEAeA8D # sOAHS53MTIHYu8bbXrO6yQtRD6JfyMWeXaLu3Nc8PDnFc1efYq/F3MGx/aiwNbcs # J2MU7BKNWTP5JQVBA2GNIeR3mScXqnOsv1XqXPvZeISDVWLaBQzceItdIwgo6B13 # vxlkkSYMvB0Dr3Yw7/W9U4Wk5K/RDOnIGvmKqKi3AwyxlV1mpefy729FKaWT7edB # d3I4+hldMY8sdfDPjWRtJzjMjXZs41OUOwtHccPazjjC7KndzvZHx/0VWL8n0NT/ # 404vftnXKifMZkS4p2sB3oK+6kCcsyWsgS/3eYGw1Fe4MOnin1RhgrW1rHPODJTG # AUOmW4wc3Q6KKr2zve7sMDZe9tfylonPwhk971rX8qGw6LkrGFv31IJeJSe/aUbG # dUDPkbrABbVvPElgoj5eP3REqx5jdfkQw7tOdWkhn0jDUh2uQen9Atj3RkJyHuR0 # GUsJVMWFJdkIO/gFwzoOGlHNsmxvpANV86/1qgb1oZXdrURpzJp53MsDaBY/pxOc # J0Cvg6uWs3kQWgKk5aBzvsX95BzdItHTpVMtVPW4q41XEvbFmUP1n6oL5rdNdrTM # j/HXMRk1KCksax1Vxo3qv+13cCsZAaQNaIAvt5LvkshZkDZIP//0Hnq7NnWeYR3z # 4oFiw9N2n3bb9baQWuWPswG0Dq9YT9kb+Cs4qIIwggd6MIIFYqADAgECAgphDpDS # AAAAAAADMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMK # V2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0 # IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0 # ZSBBdXRob3JpdHkgMjAxMTAeFw0xMTA3MDgyMDU5MDlaFw0yNjA3MDgyMTA5MDla # MH4xCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdS # ZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKDAmBgNVBAMT # H01pY3Jvc29mdCBDb2RlIFNpZ25pbmcgUENBIDIwMTEwggIiMA0GCSqGSIb3DQEB # AQUAA4ICDwAwggIKAoICAQCr8PpyEBwurdhuqoIQTTS68rZYIZ9CGypr6VpQqrgG # OBoESbp/wwwe3TdrxhLYC/A4wpkGsMg51QEUMULTiQ15ZId+lGAkbK+eSZzpaF7S # 35tTsgosw6/ZqSuuegmv15ZZymAaBelmdugyUiYSL+erCFDPs0S3XdjELgN1q2jz # y23zOlyhFvRGuuA4ZKxuZDV4pqBjDy3TQJP4494HDdVceaVJKecNvqATd76UPe/7 # 4ytaEB9NViiienLgEjq3SV7Y7e1DkYPZe7J7hhvZPrGMXeiJT4Qa8qEvWeSQOy2u # M1jFtz7+MtOzAz2xsq+SOH7SnYAs9U5WkSE1JcM5bmR/U7qcD60ZI4TL9LoDho33 # X/DQUr+MlIe8wCF0JV8YKLbMJyg4JZg5SjbPfLGSrhwjp6lm7GEfauEoSZ1fiOIl # XdMhSz5SxLVXPyQD8NF6Wy/VI+NwXQ9RRnez+ADhvKwCgl/bwBWzvRvUVUvnOaEP # 6SNJvBi4RHxF5MHDcnrgcuck379GmcXvwhxX24ON7E1JMKerjt/sW5+v/N2wZuLB # l4F77dbtS+dJKacTKKanfWeA5opieF+yL4TXV5xcv3coKPHtbcMojyyPQDdPweGF # RInECUzF1KVDL3SV9274eCBYLBNdYJWaPk8zhNqwiBfenk70lrC8RqBsmNLg1oiM # CwIDAQABo4IB7TCCAekwEAYJKwYBBAGCNxUBBAMCAQAwHQYDVR0OBBYEFEhuZOVQ # BdOCqhc3NyK1bajKdQKVMBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMAsGA1Ud # DwQEAwIBhjAPBgNVHRMBAf8EBTADAQH/MB8GA1UdIwQYMBaAFHItOgIxkEO5FAVO # 4eqnxzHRI4k0MFoGA1UdHwRTMFEwT6BNoEuGSWh0dHA6Ly9jcmwubWljcm9zb2Z0 # LmNvbS9wa2kvY3JsL3Byb2R1Y3RzL01pY1Jvb0NlckF1dDIwMTFfMjAxMV8wM18y # Mi5jcmwwXgYIKwYBBQUHAQEEUjBQME4GCCsGAQUFBzAChkJodHRwOi8vd3d3Lm1p # Y3Jvc29mdC5jb20vcGtpL2NlcnRzL01pY1Jvb0NlckF1dDIwMTFfMjAxMV8wM18y # Mi5jcnQwgZ8GA1UdIASBlzCBlDCBkQYJKwYBBAGCNy4DMIGDMD8GCCsGAQUFBwIB # FjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2RvY3MvcHJpbWFyeWNw # cy5odG0wQAYIKwYBBQUHAgIwNB4yIB0ATABlAGcAYQBsAF8AcABvAGwAaQBjAHkA # XwBzAHQAYQB0AGUAbQBlAG4AdAAuIB0wDQYJKoZIhvcNAQELBQADggIBAGfyhqWY # 4FR5Gi7T2HRnIpsLlhHhY5KZQpZ90nkMkMFlXy4sPvjDctFtg/6+P+gKyju/R6mj # 82nbY78iNaWXXWWEkH2LRlBV2AySfNIaSxzzPEKLUtCw/WvjPgcuKZvmPRul1LUd # d5Q54ulkyUQ9eHoj8xN9ppB0g430yyYCRirCihC7pKkFDJvtaPpoLpWgKj8qa1hJ # Yx8JaW5amJbkg/TAj/NGK978O9C9Ne9uJa7lryft0N3zDq+ZKJeYTQ49C/IIidYf # wzIY4vDFLc5bnrRJOQrGCsLGra7lstnbFYhRRVg4MnEnGn+x9Cf43iw6IGmYslmJ # aG5vp7d0w0AFBqYBKig+gj8TTWYLwLNN9eGPfxxvFX1Fp3blQCplo8NdUmKGwx1j # NpeG39rz+PIWoZon4c2ll9DuXWNB41sHnIc+BncG0QaxdR8UvmFhtfDcxhsEvt9B # xw4o7t5lL+yX9qFcltgA1qFGvVnzl6UJS0gQmYAf0AApxbGbpT9Fdx41xtKiop96 # eiL6SJUfq/tHI4D1nvi/a7dLl+LrdXga7Oo3mXkYS//WsyNodeav+vyL6wuA6mk7 # r/ww7QRMjt/fdW1jkT3RnVZOT7+AVyKheBEyIXrvQQqxP/uozKRdwaGIm1dxVk5I # RcBCyZt2WwqASGv9eZ/BvW1taslScxMNelDNMYIZnzCCGZsCAQEwgZUwfjELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9z # b2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAxMQITMwAAAsyOtZamvdHJTgAAAAACzDAN # BglghkgBZQMEAgEFAKCBrjAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgor # BgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQg5h3Qwm5V # OArzVlV/7fOmvIyUpkfgG68yf2X0Bv+s8MowQgYKKwYBBAGCNwIBDDE0MDKgFIAS # AE0AaQBjAHIAbwBzAG8AZgB0oRqAGGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbTAN # BgkqhkiG9w0BAQEFAASCAQB/VRvv/HQ1ehvArKqPJqCF5M2ANF2G84sjmnhOI3wD # CmD42A3nE4UUUJHph15Qn1FECF/zr7Ny/hxlxx12+HCu29QJO18RwJs9cBdLoATG # evc8B5YE2MJkkKsshJEUGGOmL1t+wsibDiu8NaZSHdOx/fF0dxAxDvu1AXjSGreg # MAkqyeda6Wv7GbtgvEMt1d9TiFTRcMy2DdloPDvFnqIwvKMmIV9INXtxL7LeJ0VZ # MOHL7uRMqnK0NQ5MriuRcj5qOURGFq/E8g721uIjiubkDCbPzTqY6NjrqfOcdB7l # AZ66Dv4KEvGUODL0Obc/lyNUxBHZ10QNTIre/9BAD0YzoYIXKTCCFyUGCisGAQQB # gjcDAwExghcVMIIXEQYJKoZIhvcNAQcCoIIXAjCCFv4CAQMxDzANBglghkgBZQME # AgEFADCCAVkGCyqGSIb3DQEJEAEEoIIBSASCAUQwggFAAgEBBgorBgEEAYRZCgMB # MDEwDQYJYIZIAWUDBAIBBQAEIB9ceOiijO4q0WCrPmMS917mZaenNpLglZeFG2+i # JEvqAgZjN1zTXBcYEzIwMjIxMDA1MjA1MTQ0Ljk1NFowBIACAfSggdikgdUwgdIx # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xLTArBgNVBAsTJE1p # Y3Jvc29mdCBJcmVsYW5kIE9wZXJhdGlvbnMgTGltaXRlZDEmMCQGA1UECxMdVGhh # bGVzIFRTUyBFU046RkM0MS00QkQ0LUQyMjAxJTAjBgNVBAMTHE1pY3Jvc29mdCBU # aW1lLVN0YW1wIFNlcnZpY2WgghF4MIIHJzCCBQ+gAwIBAgITMwAAAbn2AA1lVE+8 # AwABAAABuTANBgkqhkiG9w0BAQsFADB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMK # V2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0 # IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0Eg # MjAxMDAeFw0yMjA5MjAyMDIyMTdaFw0yMzEyMTQyMDIyMTdaMIHSMQswCQYDVQQG # EwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwG # A1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNyb3NvZnQg # SXJlbGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJjAkBgNVBAsTHVRoYWxlcyBUU1Mg # RVNOOkZDNDEtNEJENC1EMjIwMSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFt # cCBTZXJ2aWNlMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA40k+yWH1 # FsfJAQJtQgg3EwXm5CTI3TtUhKEhNe5sulacA2AEIu8JwmXuj/Ycc5GexFyZIg0n # +pyUCYsis6OdietuhwCeLGIwRcL5rWxnzirFha0RVjtVjDQsJzNj7zpT/yyGDGqx # p7MqlauI85ylXVKHxKw7F/fTI7uO+V38gEDdPqUczalP8dGNaT+v27LHRDhq3HSa # QtVhL3Lnn+hOUosTTSHv3ZL6Zpp0B3LdWBPB6LCgQ5cPvznC/eH5/Af/BNC0L2WE # DGEw7in44/3zzxbGRuXoGpFZe53nhFPOqnZWv7J6fVDUDq6bIwHterSychgbkHUB # xzhSAmU9D9mIySqDFA0UJZC/PQb2guBI8PwrLQCRfbY9wM5ug+41PhFx5Y9fRRVl # Sxf0hSCztAXjUeJBLAR444cbKt9B2ZKyUBOtuYf/XwzlCuxMzkkg2Ny30bjbGo3x # UX1nxY6IYyM1u+WlwSabKxiXlDKGsQOgWdBNTtsWsPclfR8h+7WxstZ4GpfBunhn # zIAJO2mErZVvM6+Li9zREKZE3O9hBDY+Nns1pNcTga7e+CAAn6u3NRMB8mi285Kp # wyA3AtlrVj4RP+VvRXKOtjAW4e2DRBbJCM/nfnQtOm/TzqnJVSHgDfD86zmFMYVm # AV7lsLIyeljT0zTI90dpD/nqhhSxIhzIrJUCAwEAAaOCAUkwggFFMB0GA1UdDgQW # BBS3sDhx21hDmgmMTVmqtKienjVEUjAfBgNVHSMEGDAWgBSfpxVdAF5iXYP05dJl # pxtTNRnpcjBfBgNVHR8EWDBWMFSgUqBQhk5odHRwOi8vd3d3Lm1pY3Jvc29mdC5j # b20vcGtpb3BzL2NybC9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENBJTIwMjAx # MCgxKS5jcmwwbAYIKwYBBQUHAQEEYDBeMFwGCCsGAQUFBzAChlBodHRwOi8vd3d3 # Lm1pY3Jvc29mdC5jb20vcGtpb3BzL2NlcnRzL01pY3Jvc29mdCUyMFRpbWUtU3Rh # bXAlMjBQQ0ElMjAyMDEwKDEpLmNydDAMBgNVHRMBAf8EAjAAMBYGA1UdJQEB/wQM # MAoGCCsGAQUFBwMIMA4GA1UdDwEB/wQEAwIHgDANBgkqhkiG9w0BAQsFAAOCAgEA # zdxns0VQdEywsrOOXusk8iS/ugn6z2SS63SFmJ/1ZK3rRLNgZQunXOZ0+pz7Dx4d # OSGpfQYoKnZNOpLMFcGHAc6bz6nqFTE2UN7AYxlSiz3nZpNduUBPc4oGd9UEtDJR # q+tKO4kZkBbfRw1jeuNUNSUYP5XKBAfJJoNq+IlBsrr/p9C9RQWioiTeV0Z+OcC2 # d5uxWWqHpZZqZVzkBl2lZHWNLM3+jEpipzUEbhLHGU+1x+sB0HP9xThvFVeoAB/T # Y1mxy8k2lGc4At/mRWjYe6klcKyT1PM/k81baxNLdObCEhCY/GvQTRSo6iNSsElQ # 6FshMDFydJr8gyW4vUddG0tBkj7GzZ5G2485SwpRbvX/Vh6qxgIscu+7zZx4NVBC # 8/sYcQSSnaQSOKh9uNgSsGjaIIRrHF5fhn0e8CADgyxCRufp7gQVB/Xew/4qfdeA # wi8luosl4VxCNr5JR45e7lx+TF7QbNM2iN3IjDNoeWE5+VVFk2vF57cH7JnB3ckc # Mi+/vW5Ij9IjPO31xTYbIdBWrEFKtG0pbpbxXDvOlW+hWwi/eWPGD7s2IZKVdfWz # vNsE0MxSP06fM6Ucr/eas5TxgS5F/pHBqRblQJ4ZqbLkyIq7Zi7IqIYEK/g4aE+y # 017sAuQQ6HwFfXa3ie25i76DD0vrII9jSNZhpC3MA/0wggdxMIIFWaADAgECAhMz # AAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3DQEBCwUAMIGIMQswCQYDVQQGEwJV # UzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UE # ChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIwMAYDVQQDEylNaWNyb3NvZnQgUm9v # dCBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkgMjAxMDAeFw0yMTA5MzAxODIyMjVaFw0z # MDA5MzAxODMyMjVaMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9u # MRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRp # b24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMIICIjAN # BgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA5OGmTOe0ciELeaLL1yR5vQ7VgtP9 # 7pwHB9KpbE51yMo1V/YBf2xK4OK9uT4XYDP/XE/HZveVU3Fa4n5KWv64NmeFRiMM # tY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1hlDcwUTIcVxRMTegCjhuje3XD9gm # U3w5YQJ6xKr9cmmvHaus9ja+NSZk2pg7uhp7M62AW36MEBydUv626GIl3GoPz130 # /o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3KNi1wjjHINSi947SHJMPgyY9+tVSP # 3PoFVZhtaDuaRr3tpK56KTesy+uDRedGbsoy1cCGMFxPLOJiss254o2I5JasAUq7 # vnGpF1tnYN74kpEeHT39IM9zfUGaRnXNxF803RKJ1v2lIH1+/NmeRd+2ci/bfV+A # utuqfjbsNkz2K26oElHovwUDo9Fzpk03dJQcNIIP8BDyt0cY7afomXw/TNuvXsLz # 1dhzPUNOwTM5TI4CvEJoLhDqhFFG4tG9ahhaYQFzymeiXtcodgLiMxhy16cg8ML6 # EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5UPkLiWHzNgY1GIRH29wb0f2y1BzFa/Zc # UlFdEtsluq9QBXpsxREdcu+N+VLEhReTwDwV2xo3xwgVGD94q0W29R6HXtqPnhZy # acaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIGCSsGAQQBgjcVAQQFAgMBAAEwIwYJ # KwYBBAGCNxUCBBYEFCqnUv5kxJq+gpE8RjUpzxD/LwTuMB0GA1UdDgQWBBSfpxVd # AF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBTMFEGDCsGAQQBgjdMg30BATBBMD8G # CCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jvc29mdC5jb20vcGtpb3BzL0RvY3Mv # UmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYIKwYBBQUHAwgwGQYJKwYBBAGCNxQC # BAweCgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGGMA8GA1UdEwEB/wQFMAMBAf8wHwYD # VR0jBBgwFoAU1fZWy4/oolxiaNE9lJBb186aGMQwVgYDVR0fBE8wTTBLoEmgR4ZF # aHR0cDovL2NybC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVjdHMvTWljUm9v # Q2VyQXV0XzIwMTAtMDYtMjMuY3JsMFoGCCsGAQUFBwEBBE4wTDBKBggrBgEFBQcw # AoY+aHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraS9jZXJ0cy9NaWNSb29DZXJB # dXRfMjAxMC0wNi0yMy5jcnQwDQYJKoZIhvcNAQELBQADggIBAJ1VffwqreEsH2cB # MSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1OdfCcTY/2mRsfNB1OW27DzHkwo/7 # bNGhlBgi7ulmZzpTTd2YurYeeNg2LpypglYAA7AFvonoaeC6Ce5732pvvinLbtg/ # SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbzaN9l9qRWqveVtihVJ9AkvUCgvxm2 # EhIRXT0n4ECWOKz3+SmJw7wXsFSFQrP8DJ6LGYnn8AtqgcKBGUIZUnWKNsIdw2Fz # Lixre24/LAl4FOmRsqlb30mjdAy87JGA0j3mSj5mO0+7hvoyGtmW9I/2kQH2zsZ0 # /fZMcm8Qq3UwxTSwethQ/gpY3UA8x1RtnWN0SCyxTkctwRQEcb9k+SS+c23Kjgm9 # swFXSVRk2XPXfx5bRAGOWhmRaw2fpCjcZxkoJLo4S5pu+yFUa2pFEUep8beuyOiJ # Xk+d0tBMdrVXVAmxaQFEfnyhYWxz/gq77EFmPWn9y8FBSX5+k77L+DvktxW/tM4+ # pTFRhLy/AsGConsXHRWJjXD+57XQKBqJC4822rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW # 4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328y+l7vzhwRNGQ8cirOoo6CGJ/2XBjU02N # 7oJtpQUQwXEGahC0HVUzWLOhcGbyoYIC1DCCAj0CAQEwggEAoYHYpIHVMIHSMQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMS0wKwYDVQQLEyRNaWNy # b3NvZnQgSXJlbGFuZCBPcGVyYXRpb25zIExpbWl0ZWQxJjAkBgNVBAsTHVRoYWxl # cyBUU1MgRVNOOkZDNDEtNEJENC1EMjIwMSUwIwYDVQQDExxNaWNyb3NvZnQgVGlt # ZS1TdGFtcCBTZXJ2aWNloiMKAQEwBwYFKw4DAhoDFQDHYh4YeGTnwxCTPNJaScZw # uN+BOqCBgzCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9u # MRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRp # b24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEwMA0GCSqG # SIb3DQEBBQUAAgUA5ufJ6zAiGA8yMDIyMTAwNTE3MTYyN1oYDzIwMjIxMDA2MTcx # NjI3WjB0MDoGCisGAQQBhFkKBAExLDAqMAoCBQDm58nrAgEAMAcCAQACAhEeMAcC # AQACAhIpMAoCBQDm6RtrAgEAMDYGCisGAQQBhFkKBAIxKDAmMAwGCisGAQQBhFkK # AwKgCjAIAgEAAgMHoSChCjAIAgEAAgMBhqAwDQYJKoZIhvcNAQEFBQADgYEAaDsb # wM43iCnFHv7d3Pskbg50fGoEK/hHhvVJH0aCRhXtgRObrkc7/9dM8JGKWLSM7Z45 # WLcrV43RT/GGvDcZUgJyWp5ZmMreketDCGArsFMcXG+RUO50mqogbdPhQeGlwQZc # Zk9alTv5+Op2xi1izVjXxbwaxsauEcP5vX46vUExggQNMIIECQIBATCBkzB8MQsw # CQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9u # ZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNy # b3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMAITMwAAAbn2AA1lVE+8AwABAAABuTAN # BglghkgBZQMEAgEFAKCCAUowGgYJKoZIhvcNAQkDMQ0GCyqGSIb3DQEJEAEEMC8G # CSqGSIb3DQEJBDEiBCCrVQdqK1YLGbYNHUxL2UWAhqz3ep5+06qRtZDatMCItDCB # +gYLKoZIhvcNAQkQAi8xgeowgecwgeQwgb0EIGTrRs7xbzm5MB8lUQ7e9fZotpAV # yBwal3Cw6iL5+g/0MIGYMIGApH4wfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldh # c2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBD # b3Jwb3JhdGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIw # MTACEzMAAAG59gANZVRPvAMAAQAAAbkwIgQgNv5LZGfEcpOf2K4lbWez59zJgeEx # TrNd4hOwcFiPzdswDQYJKoZIhvcNAQELBQAEggIAwME4ZSiJNcf1VX+oDv1wCNM8 # drpREsv0S9bleZewFNawrY+MT0IgJa48T8jV1kN54o5B6m9qvWtZ+IDejO4EB+f0 # fiSdj1D4b6tVwM8J3dX3Pgf66zH5l2PMx8nmDDp0cyDuxBxW4crstKqJVErvYr3e # RuWsg0ttpLOZYuQYgyNjVnbMwecMB7AgDXo2i9ey1S5UTOgh836jvW/qtBucTF4R # 14KOPE5pidelXYe/EPNcijalKNqJL7ZPkKiFm6JNDI3mLUZ5TMDQhJaZe2aTlO88 # m8dzlB+OUxP8oCtS47o4njTeo7ETyiCoMpZP0PTqpy1auIljN7L+2/bdreaAv3mR # lmrFAOjBqSLXU4zyyILbqTSIjx5iuM6adDXDrL4vSrLDt6emwhz0US7g8cly9G/B # X8MiuFwcvxLANnYEcfZgqFvVirzupywukfYdA+YengOzTfQ3PCHH43Z98W2ygH7I # 2VnKugd0ZXmv9GlFUdxhXcqh0/RAcZtLOPLV3fAEvGfwh5u7x4WhbMxelQE86H4/ # Nj26RMgmc4OrZOeFmPjph/0LZ0p4b/Xu+Dgy8Cj5SzEFCtbT24k2fiFQ9iuFjVTD # zTJe2xynWvjXWEebkq5N6Vt6ovrDZ2/XQUcEP7ymbikZeLEhYaacthKTlxYNLvLa # 4iN2Q/VQl0uJhBsQLT0= # SIG # End signature block