# # 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 # MIInngYJKoZIhvcNAQcCoIInjzCCJ4sCAQExDzANBglghkgBZQMEAgEFADB5Bgor # 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/BvW1taslScxMNelDNMYIZczCCGW8CAQEwgZUwfjELMAkG # A1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx # HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEoMCYGA1UEAxMfTWljcm9z # b2Z0IENvZGUgU2lnbmluZyBQQ0EgMjAxMQITMwAAAsyOtZamvdHJTgAAAAACzDAN # BglghkgBZQMEAgEFAKCBrjAZBgkqhkiG9w0BCQMxDAYKKwYBBAGCNwIBBDAcBgor # BgEEAYI3AgELMQ4wDAYKKwYBBAGCNwIBFTAvBgkqhkiG9w0BCQQxIgQg5h3Qwm5V # OArzVlV/7fOmvIyUpkfgG68yf2X0Bv+s8MowQgYKKwYBBAGCNwIBDDE0MDKgFIAS # AE0AaQBjAHIAbwBzAG8AZgB0oRqAGGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbTAN # BgkqhkiG9w0BAQEFAASCAQB/VRvv/HQ1ehvArKqPJqCF5M2ANF2G84sjmnhOI3wD # CmD42A3nE4UUUJHph15Qn1FECF/zr7Ny/hxlxx12+HCu29QJO18RwJs9cBdLoATG # evc8B5YE2MJkkKsshJEUGGOmL1t+wsibDiu8NaZSHdOx/fF0dxAxDvu1AXjSGreg # MAkqyeda6Wv7GbtgvEMt1d9TiFTRcMy2DdloPDvFnqIwvKMmIV9INXtxL7LeJ0VZ # MOHL7uRMqnK0NQ5MriuRcj5qOURGFq/E8g721uIjiubkDCbPzTqY6NjrqfOcdB7l # AZ66Dv4KEvGUODL0Obc/lyNUxBHZ10QNTIre/9BAD0YzoYIW/TCCFvkGCisGAQQB # gjcDAwExghbpMIIW5QYJKoZIhvcNAQcCoIIW1jCCFtICAQMxDzANBglghkgBZQME # AgEFADCCAVEGCyqGSIb3DQEJEAEEoIIBQASCATwwggE4AgEBBgorBgEEAYRZCgMB # MDEwDQYJYIZIAWUDBAIBBQAEIB9ceOiijO4q0WCrPmMS917mZaenNpLglZeFG2+i # JEvqAgZjI3hHRy8YEzIwMjIxMDA1MjExNzQ3Ljc0MlowBIACAfSggdCkgc0wgcox # CzAJBgNVBAYTAlVTMRMwEQYDVQQIEwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRt # b25kMR4wHAYDVQQKExVNaWNyb3NvZnQgQ29ycG9yYXRpb24xJTAjBgNVBAsTHE1p # Y3Jvc29mdCBBbWVyaWNhIE9wZXJhdGlvbnMxJjAkBgNVBAsTHVRoYWxlcyBUU1Mg # RVNOOjhBODItRTM0Ri05RERBMSUwIwYDVQQDExxNaWNyb3NvZnQgVGltZS1TdGFt # cCBTZXJ2aWNloIIRVDCCBwwwggT0oAMCAQICEzMAAAGZyI+vrbZ9vosAAQAAAZkw # DQYJKoZIhvcNAQELBQAwfDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0 # b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3Jh # dGlvbjEmMCQGA1UEAxMdTWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTAwHhcN # MjExMjAyMTkwNTE2WhcNMjMwMjI4MTkwNTE2WjCByjELMAkGA1UEBhMCVVMxEzAR # BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQxHjAcBgNVBAoTFU1p # Y3Jvc29mdCBDb3Jwb3JhdGlvbjElMCMGA1UECxMcTWljcm9zb2Z0IEFtZXJpY2Eg # T3BlcmF0aW9uczEmMCQGA1UECxMdVGhhbGVzIFRTUyBFU046OEE4Mi1FMzRGLTlE # REExJTAjBgNVBAMTHE1pY3Jvc29mdCBUaW1lLVN0YW1wIFNlcnZpY2UwggIiMA0G # CSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC4E/lXXKMsy9rVa2a8bRb0Ar/Pj4+b # KiAgMgKayvCMFn3ddGof8eWFgJWp5JdKjWjrnmW1r9tHpcP2kFpjXp2Udrj55jt5 # NYi1MERcoIo+E29XuCwFAMJftGdvsWea/OTQPIFsZEWqEteXdRncyVwct5xFzBIC # 1JWCdmfc7R59RMIyvgWjIz8356mweowkOstN1fe53KIJ8flrYILIQWsNRMOT3znA # GwIb9kyL54C6jZjFxOSusGYmVQ+Gr/qZQELw1ipx9s5jNP1LSpOpfTEBFu+y9KLN # BmMBARkSPpTFkGEyGSwGGgSdOi6BU6FPK+6urZ830jrRemK4JkIJ9tQhlGcIhAjh # cqZStn+38lRjVvrfbBI5EpI2NwlVIK2ibGW7sWeTAz/yNPNISUbQhGAJse/OgGj/ # 1qz/Ha9mqfYZ8BHchNxn08nWkqyrjrKicQyxuD8mCatTrVSbOJYfQyZdHR9a4vgy # GeZEXBYQNAlIuB37QCOAgs/VeDU8M4dc/IlrTyC0uV1SS4Gk8zV+5X5eRu+XORN8 # FWqzI6k/9y6cWwOWMK6aUN1XqLcaF/sm9rX84eKW2lhDc3C31WLjp8UOfOHZfPuy # y54xfilnhhCPy4QKJ9jggoqqeeEhCEfgDYjy+PByV/e5HDB2xHdtlL93wltAkI3a # Cxo84kVPBCa0OwIDAQABo4IBNjCCATIwHQYDVR0OBBYEFI26Vrg+nGWvrvIh0dQP # EonENR0QMB8GA1UdIwQYMBaAFJ+nFV0AXmJdg/Tl0mWnG1M1GelyMF8GA1UdHwRY # MFYwVKBSoFCGTmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2lvcHMvY3JsL01p # Y3Jvc29mdCUyMFRpbWUtU3RhbXAlMjBQQ0ElMjAyMDEwKDEpLmNybDBsBggrBgEF # BQcBAQRgMF4wXAYIKwYBBQUHMAKGUGh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9w # a2lvcHMvY2VydHMvTWljcm9zb2Z0JTIwVGltZS1TdGFtcCUyMFBDQSUyMDIwMTAo # MSkuY3J0MAwGA1UdEwEB/wQCMAAwEwYDVR0lBAwwCgYIKwYBBQUHAwgwDQYJKoZI # hvcNAQELBQADggIBAHGzWh29ibBNro3ns8E3EOHGsLB1Gzk90SFYUKBilIu4jDbR # 7qbvXNd8nnl/z5D9LKgw3T81jqy5tMiWp+p4jYBBk3PRx1ySqLUfhF5ZMWolRzW+ # cQZGXV38iSmdAUG0CpR5x1rMdPIrTczVUFsOYGqmkoUQ/dRiVL4iAXJLCNTj4x3Y # wIQcCPt0ijJVinPIMAYzA8f99BbeiskyI0BHGAd0kGUX2I2/puYnlyS8toBnANjh # 21xgvEuaZ2dvRqvWk/i1XIlO67au/XCeMTvXhPOIUmq80U32Tifw3SSiBKTyir7m # oWH1i7H2q5QAnrBxuyy//ZsDfARDV/Atmj5jr6ATfRHDdUanQpeoBS+iylNU6RAR # u8g+TMCu/ZndZmrs9w+8galUIGg+GmlNk07fXJ58Oc+qFqgNAsNkMi+dSzKkWGA4 # /klJFn0XichXL8+t7KOayXKGzQja6CdtCjisnyS8hbv4PKhaeMtf68wJWKKOs0tt # 2AJfYC5vSbH9ck8BGj2e/yQXEZEu88L5/fHK5XUk/IKXx3zaLkxXTSZ43Ea/WKXV # BzMasHZ3Pmny0moEekAXx1UhLNNYv4Vum33VirxSB6r/GKQxFSHu7yFfrWQpYyyD # H119TmhAedS8T1VabqdtO5ZP2E14TK82Vyxy3xEPelOo4dRIlhm7XY6k9B68MIIH # cTCCBVmgAwIBAgITMwAAABXF52ueAptJmQAAAAAAFTANBgkqhkiG9w0BAQsFADCB # iDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEyMDAGA1UEAxMp # TWljcm9zb2Z0IFJvb3QgQ2VydGlmaWNhdGUgQXV0aG9yaXR5IDIwMTAwHhcNMjEw # OTMwMTgyMjI1WhcNMzAwOTMwMTgzMjI1WjB8MQswCQYDVQQGEwJVUzETMBEGA1UE # CBMKV2FzaGluZ3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9z # b2Z0IENvcnBvcmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQ # Q0EgMjAxMDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoCggIBAOThpkzntHIh # C3miy9ckeb0O1YLT/e6cBwfSqWxOdcjKNVf2AX9sSuDivbk+F2Az/1xPx2b3lVNx # WuJ+Slr+uDZnhUYjDLWNE893MsAQGOhgfWpSg0S3po5GawcU88V29YZQ3MFEyHFc # UTE3oAo4bo3t1w/YJlN8OWECesSq/XJprx2rrPY2vjUmZNqYO7oaezOtgFt+jBAc # nVL+tuhiJdxqD89d9P6OU8/W7IVWTe/dvI2k45GPsjksUZzpcGkNyjYtcI4xyDUo # veO0hyTD4MmPfrVUj9z6BVWYbWg7mka97aSueik3rMvrg0XnRm7KMtXAhjBcTyzi # YrLNueKNiOSWrAFKu75xqRdbZ2De+JKRHh09/SDPc31BmkZ1zcRfNN0Sidb9pSB9 # fvzZnkXftnIv231fgLrbqn427DZM9ituqBJR6L8FA6PRc6ZNN3SUHDSCD/AQ8rdH # GO2n6Jl8P0zbr17C89XYcz1DTsEzOUyOArxCaC4Q6oRRRuLRvWoYWmEBc8pnol7X # KHYC4jMYctenIPDC+hIK12NvDMk2ZItboKaDIV1fMHSRlJTYuVD5C4lh8zYGNRiE # R9vcG9H9stQcxWv2XFJRXRLbJbqvUAV6bMURHXLvjflSxIUXk8A8FdsaN8cIFRg/ # eKtFtvUeh17aj54WcmnGrnu3tz5q4i6tAgMBAAGjggHdMIIB2TASBgkrBgEEAYI3 # FQEEBQIDAQABMCMGCSsGAQQBgjcVAgQWBBQqp1L+ZMSavoKRPEY1Kc8Q/y8E7jAd # BgNVHQ4EFgQUn6cVXQBeYl2D9OXSZacbUzUZ6XIwXAYDVR0gBFUwUzBRBgwrBgEE # AYI3TIN9AQEwQTA/BggrBgEFBQcCARYzaHR0cDovL3d3dy5taWNyb3NvZnQuY29t # L3BraW9wcy9Eb2NzL1JlcG9zaXRvcnkuaHRtMBMGA1UdJQQMMAoGCCsGAQUFBwMI # MBkGCSsGAQQBgjcUAgQMHgoAUwB1AGIAQwBBMAsGA1UdDwQEAwIBhjAPBgNVHRMB # Af8EBTADAQH/MB8GA1UdIwQYMBaAFNX2VsuP6KJcYmjRPZSQW9fOmhjEMFYGA1Ud # HwRPME0wS6BJoEeGRWh0dHA6Ly9jcmwubWljcm9zb2Z0LmNvbS9wa2kvY3JsL3By # b2R1Y3RzL01pY1Jvb0NlckF1dF8yMDEwLTA2LTIzLmNybDBaBggrBgEFBQcBAQRO # MEwwSgYIKwYBBQUHMAKGPmh0dHA6Ly93d3cubWljcm9zb2Z0LmNvbS9wa2kvY2Vy # dHMvTWljUm9vQ2VyQXV0XzIwMTAtMDYtMjMuY3J0MA0GCSqGSIb3DQEBCwUAA4IC # AQCdVX38Kq3hLB9nATEkW+Geckv8qW/qXBS2Pk5HZHixBpOXPTEztTnXwnE2P9pk # bHzQdTltuw8x5MKP+2zRoZQYIu7pZmc6U03dmLq2HnjYNi6cqYJWAAOwBb6J6Gng # ugnue99qb74py27YP0h1AdkY3m2CDPVtI1TkeFN1JFe53Z/zjj3G82jfZfakVqr3 # lbYoVSfQJL1AoL8ZthISEV09J+BAljis9/kpicO8F7BUhUKz/AyeixmJ5/ALaoHC # gRlCGVJ1ijbCHcNhcy4sa3tuPywJeBTpkbKpW99Jo3QMvOyRgNI95ko+ZjtPu4b6 # MhrZlvSP9pEB9s7GdP32THJvEKt1MMU0sHrYUP4KWN1APMdUbZ1jdEgssU5HLcEU # BHG/ZPkkvnNtyo4JvbMBV0lUZNlz138eW0QBjloZkWsNn6Qo3GcZKCS6OEuabvsh # VGtqRRFHqfG3rsjoiV5PndLQTHa1V1QJsWkBRH58oWFsc/4Ku+xBZj1p/cvBQUl+ # fpO+y/g75LcVv7TOPqUxUYS8vwLBgqJ7Fx0ViY1w/ue10CgaiQuPNtq6TPmb/wrp # NPgkNWcr4A245oyZ1uEi6vAnQj0llOZ0dFtq0Z4+7X6gMTN9vMvpe784cETRkPHI # qzqKOghif9lwY1NNje6CbaUFEMFxBmoQtB1VM1izoXBm8qGCAsswggI0AgEBMIH4 # oYHQpIHNMIHKMQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGluZ3RvbjEQMA4G # A1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBvcmF0aW9uMSUw # IwYDVQQLExxNaWNyb3NvZnQgQW1lcmljYSBPcGVyYXRpb25zMSYwJAYDVQQLEx1U # aGFsZXMgVFNTIEVTTjo4QTgyLUUzNEYtOUREQTElMCMGA1UEAxMcTWljcm9zb2Z0 # IFRpbWUtU3RhbXAgU2VydmljZaIjCgEBMAcGBSsOAwIaAxUAku/zYujnqapN6BJ9 # MJ5jtgDrlOuggYMwgYCkfjB8MQswCQYDVQQGEwJVUzETMBEGA1UECBMKV2FzaGlu # Z3RvbjEQMA4GA1UEBxMHUmVkbW9uZDEeMBwGA1UEChMVTWljcm9zb2Z0IENvcnBv # cmF0aW9uMSYwJAYDVQQDEx1NaWNyb3NvZnQgVGltZS1TdGFtcCBQQ0EgMjAxMDAN # BgkqhkiG9w0BAQUFAAIFAOboVEQwIhgPMjAyMjEwMDYwMzA2NDRaGA8yMDIyMTAw # NzAzMDY0NFowdDA6BgorBgEEAYRZCgQBMSwwKjAKAgUA5uhURAIBADAHAgEAAgIO # vjAHAgEAAgIUizAKAgUA5umlxAIBADA2BgorBgEEAYRZCgQCMSgwJjAMBgorBgEE # AYRZCgMCoAowCAIBAAIDB6EgoQowCAIBAAIDAYagMA0GCSqGSIb3DQEBBQUAA4GB # AK9MHNpDcSeUPlEZToRfCiSj2ELzNhJqftK+ZuFIql1eUlVkzBO43MCum3f8yVpS # 01E0mxXsq0cIkdlbLRZllH1OeaEwmswzQREm6JCPEtthPfVUZww+8q5L1+32lBQM # r4jCgDbHUpbF0Y35FwcrMpt3p5D8rEYr0OK9OtxH32MGMYIEDTCCBAkCAQEwgZMw # fDELMAkGA1UEBhMCVVMxEzARBgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1Jl # ZG1vbmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEmMCQGA1UEAxMd # TWljcm9zb2Z0IFRpbWUtU3RhbXAgUENBIDIwMTACEzMAAAGZyI+vrbZ9vosAAQAA # AZkwDQYJYIZIAWUDBAIBBQCgggFKMBoGCSqGSIb3DQEJAzENBgsqhkiG9w0BCRAB # BDAvBgkqhkiG9w0BCQQxIgQgbyZwbN3XdYuw9m1BZeh5jcgadTXHkEk2pLgsqCT0 # bcYwgfoGCyqGSIb3DQEJEAIvMYHqMIHnMIHkMIG9BCBmfWn58qKN7WWpBTYOrUO1 # BSCSnKPLC/G7wCOIc2JsfjCBmDCBgKR+MHwxCzAJBgNVBAYTAlVTMRMwEQYDVQQI # EwpXYXNoaW5ndG9uMRAwDgYDVQQHEwdSZWRtb25kMR4wHAYDVQQKExVNaWNyb3Nv # ZnQgQ29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUaW1lLVN0YW1wIFBD # QSAyMDEwAhMzAAABmciPr622fb6LAAEAAAGZMCIEIPP2dt6ou42BenN3N+GeG0hc # vhvhkAAtOrhzpsOyl0BQMA0GCSqGSIb3DQEBCwUABIICAKVrNI0+1rEEg8HydIhL # HVC4mYotx7S4s2KIj+cEiQPYecqd0GkOZnBSEQ85l/I2T0f/DHqMluUZrLPisott # YTtaWJ60nmR2QGoy8coIZVQX52SdsPQo3VM5nn7PVontIpZnl6y0dPKnGEM7kUPs # 1b1SMhqYrjcsurCPtHg/nMpv2gblST13z+FobHzI+U2PaQME/PHzsfbPcj6jDd9W # OtjCmQW0TFrw3RIfKiI+Nbkw8xubrS/L2yUo4d48sZyP6sNjrVZdQOWzBRq5FADw # jlQjSKWDXUewPgDo/TOMpf9fExkl5DpMOvCuYgEzwVX0Sf/wHvqxKVEXDJ0g4J2l # +jGd10xUjynP9Yz6hU6mCaWGXOdZdfb8xnXhS3F++7vNNU+GeG4hRYxCy3SlbR3o # 3jZDDaEUz42cs6RZ9W9CfUcovD+9LWEYxSQfLbHpW3QO99+/EbMOTAVjdNebujaO # AHV0LrGXnrJGlq++LDEFUX+xNLnZheubWy+lHDaQ8JiPQu9gg4WwSe8Coz9WUyGo # bplJxmvV/myQebxV76ybs2y54b92rYRpzA+sAk2+nSc+DBrDIIZ/bzmRvOV+NVsJ # ZfK6nxPwfNRSEqK7w02HLzunD4S5PXtX2FvKoXb6Ezi7U3iOqY3IsTuljSBrxkvx # wJkYIMO7g2FHGrx8BgWLGw/3 # SIG # End signature block