# # 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 # MIInqgYJKoZIhvcNAQcCoIInmzCCJ5cCAQExDzANBglghkgBZQMEAgEFADB5Bgor # 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/BvW1taslScxMNelDNMYIZfzCCGXsCAQEwgZUwfjELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9z # b2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAxMQITMwAAAsyOtZamvdHJTgAAAAACzDAN # BglghkgBZQMEAgEFAKCBrjAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgor # BgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQg5h3Qwm5V # OArzVlV/7fOmvIyUpkfgG68yf2X0Bv+s8MowQgYKKwYBBAGCNwIBDDE0MDKgFIAS # AE0AaQBjAHIAbwBzAG8AZgB0oRqAGGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbTAN # BgkqhkiG9w0BAQEFAASCAQB/VRvv/HQ1ehvArKqPJqCF5M2ANF2G84sjmnhOI3wD # CmD42A3nE4UUUJHph15Qn1FECF/zr7Ny/hxlxx12+HCu29QJO18RwJs9cBdLoATG # evc8B5YE2MJkkKsshJEUGGOmL1t+wsibDiu8NaZSHdOx/fF0dxAxDvu1AXjSGreg # MAkqyeda6Wv7GbtgvEMt1d9TiFTRcMy2DdloPDvFnqIwvKMmIV9INXtxL7LeJ0VZ # MOHL7uRMqnK0NQ5MriuRcj5qOURGFq/E8g721uIjiubkDCbPzTqY6NjrqfOcdB7l # AZ66Dv4KEvGUODL0Obc/lyNUxBHZ10QNTIre/9BAD0YzoYIXCTCCFwUGCisGAQQB # gjcDAwExghb1MIIW8QYJKoZIhvcNAQcCoIIW4jCCFt4CAQMxDzANBglghkgBZQME # AgEFADCCAVUGCyqGSIb3DQEJEAEEoIIBRASCAUAwggE8AgEBBgorBgEEAYRZCgMB # MDEwDQYJYIZIAWUDBAIBBQAEIB9ceOiijO4q0WCrPmMS917mZaenNpLglZeFG2+i # JEvqAgZjKamuK4QYEzIwMjIxMDA1MjA1MDIwLjYxOVowBIACAfSggdSkgdEwgc4x # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKTAnBgNVBAsTIE1p # Y3Jvc29mdCBPcGVyYXRpb25zIFB1ZXJ0byBSaWNvMSYwJAYDVQQLEx1UaGFsZXMg # VFNTIEVTTjpDNEJELUUzN0YtNUZGQzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUt # U3RhbXAgU2VydmljZaCCEVwwggcQMIIE+KADAgECAhMzAAABo/uas457hkNPAAEA # AAGjMA0GCSqGSIb3DQEBCwUAMHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNo # aW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29y # cG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBDQSAyMDEw # MB4XDTIyMDMwMjE4NTExNloXDTIzMDUxMTE4NTExNlowgc4xCzAJBgNVBAYTAlVT # MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xKTAnBgNVBAsTIE1pY3Jvc29mdCBPcGVy # YXRpb25zIFB1ZXJ0byBSaWNvMSYwJAYDVQQLEx1UaGFsZXMgVFNTIEVTTjpDNEJE # LUUzN0YtNUZGQzElMCMGA1UEAxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2Vydmlj # ZTCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAO+9TcrLeyoKcCqLbNtz # 7Nt2JbP1TEzzMhi84gS6YLI7CF6dVSA5I1bFCHcw6ZF2eF8Qiaf0o2XSXf/jp5sg # mUYtMbGi4neAtWSNK5yht4iyQhBxn0TIQqF+NisiBxW+ehMYWEbFI+7cSdX/dWw+ # /Y8/Mu9uq3XCK5P2G+ZibVwOVH95+IiTGnmocxWgds0qlBpa1rYg3bl8XVe5L2qT # UmJBvnQpx2bUru70lt2/HoU5bBbLKAhCPpxy4nmsrdOR3Gv4UbfAmtpQntP758NR # Phg1bACH06FlvbIyP8/uRs3x2323daaGpJQYQoZpABg62rFDTJ4+e06tt+xbfvp8 # M9lo8a1agfxZQ1pIT1VnJdaO98gWMiMW65deFUiUR+WngQVfv2gLsv6o7+Ocpzy6 # RHZIm6WEGZ9LBt571NfCsx5z0Ilvr6SzN0QbaWJTLIWbXwbUVKYebrXEVFMyhuVG # QHesZB+VwV386hYonMxs0jvM8GpOcx0xLyym42XA99VSpsuivTJg4o8a1ACJbTBV # FoEA3VrFSYzOdQ6vzXxrxw6i/T138m+XF+yKtAEnhp+UeAMhlw7jP99EAlgGUl0K # kcBjTYTz+jEyPgKadrU1of5oFi/q9YDlrVv9H4JsVe8GHMOkPTNoB4028j88OEe4 # 26BsfcXLki0phPp7irW0AbRdAgMBAAGjggE2MIIBMjAdBgNVHQ4EFgQUUFH7szwm # CLHPTS9Bo2irLnJji6owHwYDVR0jBBgwFoAUn6cVXQBeYl2D9OXSZacbUzUZ6XIw # XwYDVR0fBFgwVjBUoFKgUIZOaHR0cDovL3d3dy5taWNyb3NvZnQuY29tL3BraW9w # cy9jcmwvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAoMSkuY3Js # MGwGCCsGAQUFBwEBBGAwXjBcBggrBgEFBQcwAoZQaHR0cDovL3d3dy5taWNyb3Nv # ZnQuY29tL3BraW9wcy9jZXJ0cy9NaWNyb3NvZnQlMjBUaW1lLVN0YW1wJTIwUENB # JTIwMjAxMCgxKS5jcnQwDAYDVR0TAQH/BAIwADATBgNVHSUEDDAKBggrBgEFBQcD # CDANBgkqhkiG9w0BAQsFAAOCAgEAWvLep2mXw6iuBxGu0PsstmXI5gLmgPkTKQnj # gZlsoeipsta9oku0MTVxlHVdcdBbFcVHMLRRkUFIkfKnaclyl5eyj03weD6b/pUf # FyDZB8AZpGUXhTYLNR8PepM6yD6g+0E1nH0MhOGoE6XFufkbn6eIdNTGuWwBeEr2 # DNiGhDGlwaUH5ELz3htuyMyWKAgYF28C4iyyhYdvlG9VN6JnC4mc/EIt50BCHp8Z # QAk7HC3ROltg1gu5NjGaSVdisai5OJWf6e5sYQdDBNYKXJdiHei1N7K+L5s1vV+C # 6d3TsF9+ANpioBDAOGnFSYt4P+utW11i37iLLLb926pCL4Ly++GU0wlzYfn7n22R # yQmvD11oyiZHhmRssDBqsA+nvCVtfnH183Df5oBBVskzZcJTUjCxaagDK7AqB6QA # 3H7l/2SFeeqfX/Dtdle4B+vPV4lq1CCs0A1LB9lmzS0vxoRDusY80DQi10K3SfZK # 1hyyaj9a8pbZG0BsBp2Nwc4xtODEeBTWoAzF9ko4V6d09uFFpJrLoV+e8cJU/hT3 # +SlW7dnr5dtYvziHTpZuuRv4KU6F3OQzNpHf7cBLpWKRXRjGYdVnAGb8NzW6wWTj # ZjMCNdCFG7pkKLMOGdqPDFdfk+EYE5RSG9yxS76cPfXqRKVtJZScIF64ejnXbFIs # 5bh8KwEwggdxMIIFWaADAgECAhMzAAAAFcXna54Cm0mZAAAAAAAVMA0GCSqGSIb3 # DQEBCwUAMIGIMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4G # A1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMTIw # MAYDVQQDEylNaWNyb3NvZnQgUm9vdCBDZXJ0aWZpY2F0ZSBBdXRob3JpdHkgMjAx # MDAeFw0yMTA5MzAxODIyMjVaFw0zMDA5MzAxODMyMjVaMHwxCzAJBgNVBAYTAlVT # MRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQK # ExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1l # LVN0YW1wIFBDQSAyMDEwMIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA # 5OGmTOe0ciELeaLL1yR5vQ7VgtP97pwHB9KpbE51yMo1V/YBf2xK4OK9uT4XYDP/ # XE/HZveVU3Fa4n5KWv64NmeFRiMMtY0Tz3cywBAY6GB9alKDRLemjkZrBxTzxXb1 # hlDcwUTIcVxRMTegCjhuje3XD9gmU3w5YQJ6xKr9cmmvHaus9ja+NSZk2pg7uhp7 # M62AW36MEBydUv626GIl3GoPz130/o5Tz9bshVZN7928jaTjkY+yOSxRnOlwaQ3K # Ni1wjjHINSi947SHJMPgyY9+tVSP3PoFVZhtaDuaRr3tpK56KTesy+uDRedGbsoy # 1cCGMFxPLOJiss254o2I5JasAUq7vnGpF1tnYN74kpEeHT39IM9zfUGaRnXNxF80 # 3RKJ1v2lIH1+/NmeRd+2ci/bfV+AutuqfjbsNkz2K26oElHovwUDo9Fzpk03dJQc # NIIP8BDyt0cY7afomXw/TNuvXsLz1dhzPUNOwTM5TI4CvEJoLhDqhFFG4tG9ahha # YQFzymeiXtcodgLiMxhy16cg8ML6EgrXY28MyTZki1ugpoMhXV8wdJGUlNi5UPkL # iWHzNgY1GIRH29wb0f2y1BzFa/ZcUlFdEtsluq9QBXpsxREdcu+N+VLEhReTwDwV # 2xo3xwgVGD94q0W29R6HXtqPnhZyacaue7e3PmriLq0CAwEAAaOCAd0wggHZMBIG # CSsGAQQBgjcVAQQFAgMBAAEwIwYJKwYBBAGCNxUCBBYEFCqnUv5kxJq+gpE8RjUp # zxD/LwTuMB0GA1UdDgQWBBSfpxVdAF5iXYP05dJlpxtTNRnpcjBcBgNVHSAEVTBT # MFEGDCsGAQQBgjdMg30BATBBMD8GCCsGAQUFBwIBFjNodHRwOi8vd3d3Lm1pY3Jv # c29mdC5jb20vcGtpb3BzL0RvY3MvUmVwb3NpdG9yeS5odG0wEwYDVR0lBAwwCgYI # KwYBBQUHAwgwGQYJKwYBBAGCNxQCBAweCgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGG # MA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAU1fZWy4/oolxiaNE9lJBb186a # GMQwVgYDVR0fBE8wTTBLoEmgR4ZFaHR0cDovL2NybC5taWNyb3NvZnQuY29tL3Br # aS9jcmwvcHJvZHVjdHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3JsMFoGCCsG # AQUFBwEBBE4wTDBKBggrBgEFBQcwAoY+aHR0cDovL3d3dy5taWNyb3NvZnQuY29t # L3BraS9jZXJ0cy9NaWNSb29DZXJBdXRfMjAxMC0wNi0yMy5jcnQwDQYJKoZIhvcN # AQELBQADggIBAJ1VffwqreEsH2cBMSRb4Z5yS/ypb+pcFLY+TkdkeLEGk5c9MTO1 # OdfCcTY/2mRsfNB1OW27DzHkwo/7bNGhlBgi7ulmZzpTTd2YurYeeNg2LpypglYA # A7AFvonoaeC6Ce5732pvvinLbtg/SHUB2RjebYIM9W0jVOR4U3UkV7ndn/OOPcbz # aN9l9qRWqveVtihVJ9AkvUCgvxm2EhIRXT0n4ECWOKz3+SmJw7wXsFSFQrP8DJ6L # GYnn8AtqgcKBGUIZUnWKNsIdw2FzLixre24/LAl4FOmRsqlb30mjdAy87JGA0j3m # Sj5mO0+7hvoyGtmW9I/2kQH2zsZ0/fZMcm8Qq3UwxTSwethQ/gpY3UA8x1RtnWN0 # SCyxTkctwRQEcb9k+SS+c23Kjgm9swFXSVRk2XPXfx5bRAGOWhmRaw2fpCjcZxko # JLo4S5pu+yFUa2pFEUep8beuyOiJXk+d0tBMdrVXVAmxaQFEfnyhYWxz/gq77EFm # PWn9y8FBSX5+k77L+DvktxW/tM4+pTFRhLy/AsGConsXHRWJjXD+57XQKBqJC482 # 2rpM+Zv/Cuk0+CQ1ZyvgDbjmjJnW4SLq8CdCPSWU5nR0W2rRnj7tfqAxM328y+l7 # vzhwRNGQ8cirOoo6CGJ/2XBjU02N7oJtpQUQwXEGahC0HVUzWLOhcGbyoYICzzCC # AjgCAQEwgfyhgdSkgdEwgc4xCzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5n # dG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9y # YXRpb24xKTAnBgNVBAsTIE1pY3Jvc29mdCBPcGVyYXRpb25zIFB1ZXJ0byBSaWNv # MSYwJAYDVQQLEx1UaGFsZXMgVFNTIEVTTjpDNEJELUUzN0YtNUZGQzElMCMGA1UE # AxMcTWljcm9zb2Z0IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIaAxUA # Hl/pXkLMAbPapCwa+GXc3SlDDROggYMwgYCkfjB8MQswCQYDVQQGEwJVUzETMBEG # A1UECBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWlj # cm9zb2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFt # cCBQQ0EgMjAxMDANBgkqhkiG9w0BAQUFAAIFAObn7c4wIhgPMjAyMjEwMDUxNTQ5 # MzRaGA8yMDIyMTAwNjE1NDkzNFowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA5uft # zgIBADAHAgEAAgIaNTAHAgEAAgIRRzAKAgUA5uk/TgIBADA2BgorBgEEAYRZCgQC # MSgwJjAMBgorBgEEAYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqG # SIb3DQEBBQUAA4GBAIDi+c/Nh2A84CsPYPLNKkkT5Bxv6kK9+1kOISXyRAKJ5vv9 # vB02c2chf7/zX2nHg+RkWcY8P6FaLgqWWhnDn8I0xIzrIXwTe+igU8V9RL9Kzj0X # 8Cd3EILKl6yfZbMjSmAlXC1fJUQ0QTBlvVEDd/4jss7rXrgplnPbLjt5knd6MYIE # DTCCBAkCAQEwgZMwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24x # EDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv # bjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAGj # +5qzjnuGQ08AAQAAAaMwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzEN # BgsqhkiG9w0BCRABBDAvBgkqhkiG9w0BCQQxIgQgsD9LwgaK5KhBJABOtJHv9nKJ # Tz05D3+qXWXad3hhD2AwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCCM+Liw # BnHMMoOd/sgbaYxpwvEJlREZl/pTPklz6euN/jCBmDCBgKR+MHwxCzAJBgNVBAYT # AlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYD # VQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBU # aW1lLVN0YW1wIFBDQSAyMDEwAhMzAAABo/uas457hkNPAAEAAAGjMCIEIHVAojP8 # NrMzVfgw+ZgwF370fdd3BaladQ/E2x8aY1WUMA0GCSqGSIb3DQEBCwUABIICAK74 # r/8XM36qmym7dv9jSf+fUyl9r7E3OdOCrUcJfsotJBlQR4GiMj8572Bbel6biGC6 # cKClMfnwIWZhck8rE+UvxpQp/bf9c7BZVGNuSG9NTaGHP2EulwS8orl5sQroYN4T # MMjkkg6hoRYkQbKcrzV5HPRjuETMAMHc8j2b/3YlG7W9Yspi0NIhXOApo15e5Nyb # ymqPn3r4CZ9sh9k8lzOrbzlVtrmkL3JYCWWcIc8embXrvSTHZT94vpDtCJs2Ww/9 # lpl8N0zuKOq4P+RqstrleAHUc2UjEFPKJpmndrMs4ZIUMWsarsF/sXNc3ee0iMt0 # ZLwMhrtOGB/E0OlkHBlAO95Sfb+w5hXblldKnfzQ2DMklroqxLwU18ysLAUNPjKR # howaEogWRzdgpQMyWPn6gHY3+Fgh399vOHWUK24OooqS8J5m+V4DssroVnygy0Um # tKtYgayMicT0/ANDJ4muKxghlPlhGSKO+CAYsknEbheFsUiO6CxklvGu3onkpigj # 98f2PxemkFohRfOBiQt/pxhS+1U3xhwZrqOTQSP1t79CPNfue4ArzhHwknqlfqg8 # q6ZhZLi25vY8UtA86SjzUgC8cdjq7jB4krLLr8dktlcTvpDbIDpMUSuJGiANq6t3 # mfKeoapgnrQCRIEMG7tJox2UvCztorEg/FqURCt1 # SIG # End signature block