1 # Copyright (c) 2015-2016 The Khronos Group Inc.
2 # Copyright (c) 2015-2016 Valve Corporation
3 # Copyright (c) 2015-2016 LunarG, Inc.
4 #
5 # Permission is hereby granted, free of charge, to any person obtaining a copy
6 # of this software and/or associated documentation files (the "Materials"), to
7 # deal in the Materials without restriction, including without limitation the
8 # rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
9 # sell copies of the Materials, and to permit persons to whom the Materials are
10 # furnished to do so, subject to the following conditions:
11 #
12 # The above copyright notice(s) and this permission notice shall be included in
13 # all copies or substantial portions of the Materials.
14 #
15 # THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
18 #
19 # IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
20 # DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
21 # OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE MATERIALS OR THE
22 # USE OR OTHER DEALINGS IN THE MATERIALS.
23 #
24 # Author: David Pinedo <david@LunarG.com>
25 # Author: Mark Young <mark@LunarG.com>
26 #
27
28
29 # This Powershell script is used by the Vulkan Run Time Installer/Uninstaller to:
30 # - Copy the most recent vulkan-<majorabi>-*.dll in C:\Windows\System32
31 # to vulkan-<majorabi>.dll
32 # - Copy the most recent version of vulkaninfo-<abimajor>-*.exe in
33 # C:\Windows\System32 to vulkaninfo.exe
34 # - The same thing is done for those files in C:\Windows\SysWOW64 on a 64-bit
35 # target.
36 # - Set the layer registry entries to point to the layer json files
37 # in the Vulkan SDK associated with the most recent vulkan*dll.
38 #
39 # This script takes the following parameters:
40 # $majorabi : a single string number specifying the major abi version.
41 # $ossize : an integer indicating if the target is a 64 (64) or 32 (32) bit OS.
42 #
43
44 Param(
45 [string]$majorabi,
46 [int]$ossize
47 )
48
49 $vulkandll = "vulkan-"+$majorabi+".dll"
50 $windrive = $env:SYSTEMDRIVE
51 $winfolder = $env:SYSTEMROOT
52 $script:VulkanDllList=@()
53
notNumeric($x)54 function notNumeric ($x) {
55 try {
56 0 + $x | Out-Null
57 return $false
58 } catch {
59 return $true
60 }
61 }
62
63 # The name of the versioned vulkan dll file is one of the following:
64 #
65 # vulkan-<majorabi>-<major>-<minor>-<patch>-<buildno>-<prerelease>-<prebuildno>
66 # vulkan-<majorabi>-<major>-<minor>-<patch>-<buildno>-<prerelease>
67 # vulkan-<majorabi>-<major>-<minor>-<patch>-<buildno>-<prebuildno>
68 # vulkan-<majorabi>-<major>-<minor>-<patch>-<buildno>.dll
69 #
70 # <major>, <minor>, <patch>, <buildno> and <prebuildno> are 1 to 10 numeric digits.
71 # <prerelease> is any combination of alpha and numeric characters.
72 # If <prerelease> and/or <prebuildno> are present, this identifies a prerelease,
73 # and the vulkan dll file will be considered less recent than one with the same
74 # <major>, <minor>, <patch>, <buildno> numbers without the <prerelease> and/or
75 # <prebuildno>.
76
77
78 # We first create an array, with one array element for each vulkan-*dll in
79 # C:\Windows\System32 (and C:\Windows\SysWOW64 on 64-bit systems), with each element
80 # containing:
81 # <major>=<minor>=<patch>=<buildno>=<prerelease>=<prebuildno>=
82 # filename
83 # @<major>@<minor>@<patch>@<buildno>@<prerelease>@<prebuildno>@
84 # [Note that the above three lines are one element in the array.]
85 # The build identifiers separated by "=" are suitable for sorting, i.e.
86 # expanded to 10 digits with leading 0s. If <prerelease> or <prebuildno> are
87 # not specified, "zzzzzzzzzz" is substituted for them, so that they sort
88 # to a position after those that do specify them.
89 # The build identifiers separated by "@" are the original values extracted
90 # from the file name. They are used later to find the path to the SDK
91 # install directory for the given filename.
92
93
UpdateVulkanSysFolder([string]$dir, [int]$writeSdkName)94 function UpdateVulkanSysFolder([string]$dir, [int]$writeSdkName)
95 {
96 # Push the current path on the stack and go to $dir
97 Push-Location -Path $dir
98
99 # Create a list for all the DLLs in the folder.
100 # First Initialize the list to empty
101 $script:VulkanDllList = @()
102
103 # Find all DLL objects in this directory
104 dir -name vulkan-$majorabi-*.dll |
105 ForEach-Object {
106 if ($_ -match "=" -or
107 $_ -match "@" -or
108 $_ -match " " -or
109 ($_.Split('-').count -lt 6) -or
110 ($_.Split('-').count -gt 8))
111 {
112 # If a file name contains "=", "@", or " ", or it contains less then 5 dashes or more than
113 # 7 dashes, it wasn't installed by the Vulkan Run Time.
114 # Note that we need to use return inside of ForEach-Object is to continue with iteration.
115 return
116 }
117 $major=$_.Split('-')[2]
118 $majorOrig=$major
119 $minor=$_.Split('-')[3]
120 $minorOrig=$minor
121 $patch=$_.Split('-')[4]
122 $patchOrig=$patch
123 $buildno=$_.Split('-')[5]
124
125 if ($buildno -match ".dll") {
126 # prerelease and prebuildno are not in the name
127 # Extract buildno, and set prerelease and prebuildno to "z"s
128 $buildno=$buildno -replace ".dll",""
129 $buildnoOrig=$buildno
130 $prerelease="z"*10
131 $prereleaseOrig=""
132 $prebuildno="z"*10
133 $prebuildnoOrig=""
134 } else {
135 # Extract buildno, prerelease, and prebuildno
136 $f=$_ -replace ".dll",""
137 $buildno=$f.Split('-')[5]
138 $buildnoOrig=$buildno
139 $prerelease=$f.Split('-')[6]
140 $prebuildno=$f.Split('-')[7]
141 if ($prebuildno.Length -eq 0) {
142 if ($prerelease -match "^[0-9]") {
143 # prerelease starts with a digit, it must be the prebuildno
144 $prebuildno=$prerelease
145 $prerelease=""
146 }
147 }
148 $prereleaseOrig=$prerelease
149 $prebuildnoOrig=$prebuildno
150
151 if ($prerelease.Length -eq 0) {
152 $prerelease="z"*10
153 }
154 if ($prebuildno.Length -eq 0) {
155 $prebuildno="z"*10
156 }
157 }
158
159 # Make sure fields that are supposed to be numbers are numbers
160 if (notNumeric($major)) {return}
161 if (notNumeric($minor)) {return}
162 if (notNumeric($patch)) {return}
163 if (notNumeric($buildno)) {return}
164 if (notNumeric($prebuildno)) {
165 if ($prebuildno -ne "z"*10) {return}
166 }
167
168 $major = $major.padleft(10,'0')
169 $minor = $minor.padleft(10,'0')
170 $patch = $patch.padleft(10,'0')
171 $buildno = $buildno.padleft(10,'0')
172 $prerelease = $prerelease.padleft(10,'0')
173 $prebuildno = $prebuildno.padleft(10,'0')
174
175 # Add a new element to the $VulkanDllList array
176 $script:VulkanDllList+="$major=$minor=$patch=$buildno=$prerelease=$prebuildno= $_ @$majorOrig@$minorOrig@$patchOrig@$buildnoOrig@$prereleaseOrig@$prebuildnoOrig@"
177 }
178
179 # If $VulkanDllList contains at least one element, there's at least one vulkan*.dll file.
180 # Copy the most recent vulkan*.dll (named in the last element of $VulkanDllList) to vulkan-$majorabi.dll.
181
182 if ($script:VulkanDllList.Length -gt 0) {
183
184 # Sort the list. The most recent vulkan-*.dll will be in the last element of the list.
185 [array]::sort($script:VulkanDllList)
186
187 # Put the name of the most recent vulkan-*.dll in $mrVulkanDLL.
188 # The most recent vulkanDLL is the second word in the last element of the
189 # sorted $VulkanDllList. Copy it to $vulkandll.
190 $mrVulkanDll=$script:VulkanDllList[-1].Split(' ')[1]
191 copy $mrVulkanDll $vulkandll
192
193 # Copy the most recent version of vulkaninfo-<abimajor>-*.exe to vulkaninfo.exe.
194 # We create the source file name for the copy from $mrVulkanDll.
195 $mrVulkaninfo=$mrVulkanDll -replace ".dll",".exe"
196 $mrVulkaninfo=$mrVulkaninfo -replace "vulkan","vulkaninfo"
197 copy $mrVulkaninfo vulkaninfo.exe
198
199 # Create the name used in the registry for the SDK associated with $mrVulkanDll.
200 $major=$script:VulkanDllList[-1].Split('@')[1]
201 $minor=$script:VulkanDllList[-1].Split('@')[2]
202 $patch=$script:VulkanDllList[-1].Split('@')[3]
203 $buildno=$script:VulkanDllList[-1].Split('@')[4]
204 $prerelease=$script:VulkanDllList[-1].Split('@')[5]
205 $prebuildno=$script:VulkanDllList[-1].Split('@')[6]
206
207 $sdktempname="VulkanSDK"+$major + "." + $minor + "." + $patch + "." + $buildno
208 if ($prerelease -ne "") {
209 $sdktempname=$sdktempname + "." + $prerelease
210 }
211 if ($prebuildno -ne "") {
212 $sdktempname=$sdktempname + "." + $prebuildno
213 }
214 }
215
216 # Return to our previous folder
217 Pop-Location
218
219 # Only update the overall script-scope SDK name if we're told to
220 if ($writeSdkName -ne 0) {
221 $script:sdkname = $sdktempname
222 }
223
224 return
225 }
226
227 # We only care about SYSWOW64 if we're targeting a 64-bit OS
228 if ($ossize -eq 64) {
229 # Update the SYSWOW64 Vulkan DLLS/EXEs
230 UpdateVulkanSysFolder $winfolder\SYSWOW64 0
231 }
232
233 # Update the SYSTEM32 Vulkan DLLS/EXEs
234 UpdateVulkanSysFolder $winfolder\SYSTEM32 1
235
236 # Create an array of vulkan sdk install dirs
237
238 $mrVulkanDllInstallDir=""
239 $VulkanSdkDirs=@()
240 Get-ChildItem -Path Registry::HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall |
241 ForEach-Object {
242 $regkey=$_ -replace ".*\\",""
243 if ($_ -match "\\VulkanSDK") {
244 # Get the install path from UninstallString
245 $tmp=Get-ItemProperty -Path Registry::HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$regkey -Name UninstallString
246 $tmp=$tmp -replace "\\Uninstall.exe.*",""
247 $tmp=$tmp -replace ".*=.",""
248 $VulkanSdkDirs+=$tmp
249 if ($regkey -eq $script:sdkname) {
250 # Save away the sdk install dir for the the most recent vulkandll
251 $mrVulkanDllInstallDir=$tmp
252 }
253 }
254 }
255
256
257 # Search list of sdk install dirs for an sdk compatible with $script:sdkname.
258 # We go backwards through VulkanDllList to generate SDK names, because we want the most recent SDK.
259 if ($mrVulkanDllInstallDir -eq "") {
260 ForEach ($idx in ($script:VulkanDllList.Length-1)..0) {
261 $vulkanDllMajor=$script:VulkanDllList[$idx].Split('@')[1]
262 $vulkanDllMinor=$script:VulkanDllList[$idx].Split('@')[2]
263 $vulkanDllPatch=$script:VulkanDllList[$idx].Split('@')[3]
264 $vulkanDllBuildno=$script:VulkanDllList[$idx].Split('@')[4]
265 $vulkanDllPrerelease=$script:VulkanDllList[$idx].Split('@')[5]
266 $vulkanDllPrebuildno=$script:VulkanDllList[$idx].Split('@')[6]
267 $regEntry="VulkanSDK"+$vulkanDllMajor+"."+$vulkanDllMinor+"."+$vulkanDllPatch+"."+$vulkanDllBuildno
268 if ($vulkanDllPrerelease) {
269 $regEntry=$regEntry+"."+$vulkanDllPrerelease
270 }
271 if ($vulkanDllPrebuildno) {
272 $regEntry=$regEntry+"."+$vulkanDllPrebuildno
273 }
274 $rval=Get-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Uninstall\$regEntry -ErrorAction SilentlyContinue
275 $instDir=$rval
276 $instDir=$instDir -replace "\\Uninstall.exe.*",""
277 $instDir=$instDir -replace ".*=.",""
278 if ($rval) {
279 $rval=$rval -replace ".* DisplayVersion=",""
280 $rval=$rval -replace ";.*",""
281 $reMajor=$rval.Split('.')[0]
282 $reMinor=$rval.Split('.')[1]
283 $rePatch=$rval.Split('.')[2]
284 if ($reMajor+$reMinor+$rePatch -eq $vulkanDllMajor+$vulkanDllMinor+$vulkanDllPatch) {
285 $mrVulkanDllInstallDir=$instDir
286 break
287 }
288 }
289 }
290 }
291
292 # Add C:\Vulkan\SDK\0.9.3 to list of SDK install dirs.
293 # We do this because there is in a bug in SDK 0.9.3 in which layer
294 # reg entries were not removed on uninstall. So we'll try to clean up
295 # and remove them now.
296 # This works only if 0.9.3 was installed to the default location.
297 # If it was not installed to the default location, those entries will
298 # need to be cleaned up manually.
299
300 $VulkanSdkDirs+="C:\VulkanSDK\0.9.3"
301 $VulkanSdkDirs+="$windrive\VulkanSDK\0.9.3"
302
303 # Remove layer registry entries associated with all installed Vulkan SDKs.
304 # Note that we remove only those entries created by Vulkan SDKs. If other
305 # layers were installed that are not from an SDK, we don't mess with them.
306
307 Get-Item -Path Registry::HKEY_LOCAL_MACHINE\SOFTWARE\Khronos\Vulkan\ExplicitLayers | Select-Object -ExpandProperty Property |
308 ForEach-Object {
309 $regval=$_
310 ForEach ($sdkdir in $VulkanSdkDirs) {
311 if ($regval -like "$sdkdir\*.json") {
312 Remove-ItemProperty -ErrorAction SilentlyContinue -Path HKLM:\SOFTWARE\Khronos\Vulkan\ExplicitLayers -name $regval
313 }
314 }
315 }
316 # Remove 32-bit layer registry entries if we're targeting a 64-bit OS
317 if ($ossize -eq 64) {
318 Get-Item -Path Registry::HKEY_LOCAL_MACHINE\SOFTWARE\WOW6432Node\Khronos\Vulkan\ExplicitLayers | Select-Object -ExpandProperty Property |
319 ForEach-Object {
320 $regval=$_
321 ForEach ($sdkdir in $VulkanSdkDirs) {
322 if ($regval -like "$sdkdir\*.json") {
323 Remove-ItemProperty -ErrorAction SilentlyContinue -Path HKLM:\SOFTWARE\WOW6432Node\Khronos\Vulkan\ExplicitLayers -name $regval
324 }
325 }
326 }
327 }
328
329
330 # Create layer registry entries associated with Vulkan SDK from which $mrVulkanDll is from
331
332 if ($mrVulkanDllInstallDir -ne "") {
333 if ($ossize -eq 64) {
334
335 # Create registry entires in normal registry location for 64-bit items on a 64-bit OS
336 New-Item -Force -ErrorAction SilentlyContinue -Path HKLM:\SOFTWARE\Khronos\Vulkan\ExplicitLayers | out-null
337 Get-ChildItem $mrVulkanDllInstallDir\Bin -Filter VkLayer*json |
338 ForEach-Object {
339 New-ItemProperty -Path HKLM:\SOFTWARE\Khronos\Vulkan\ExplicitLayers -Name $mrVulkanDllInstallDir\Bin\$_ -PropertyType DWord -Value 0 | out-null
340 }
341
342 # Create registry entires for the WOW6432Node registry location for 32-bit items on a 64-bit OS
343 New-Item -Force -ErrorAction SilentlyContinue -Path HKLM:\SOFTWARE\WOW6432Node\Khronos\Vulkan\ExplicitLayers | out-null
344 Get-ChildItem $mrVulkanDllInstallDir\Bin32 -Filter VkLayer*json |
345 ForEach-Object {
346 New-ItemProperty -Path HKLM:\SOFTWARE\WOW6432Node\Khronos\Vulkan\ExplicitLayers -Name $mrVulkanDllInstallDir\Bin32\$_ -PropertyType DWord -Value 0 | out-null
347 }
348
349 } else {
350
351 # Create registry entires in normal registry location for 32-bit items on a 32-bit OS
352 New-Item -Force -ErrorAction SilentlyContinue -Path HKLM:\SOFTWARE\Khronos\Vulkan\ExplicitLayers | out-null
353 Get-ChildItem $mrVulkanDllInstallDir\Bin32 -Filter VkLayer*json |
354 ForEach-Object {
355 New-ItemProperty -Path HKLM:\SOFTWARE\Khronos\Vulkan\ExplicitLayers -Name $mrVulkanDllInstallDir\Bin32\$_ -PropertyType DWord -Value 0 | out-null
356 }
357
358 }
359 }
360
361 # SIG # Begin signature block
362 # MIIcZgYJKoZIhvcNAQcCoIIcVzCCHFMCAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB
363 # gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR
364 # AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUdeZMvyfevbCm2d9Sn02g0L39
365 # 6EKggheVMIIFHjCCBAagAwIBAgIQDmYEpPtQ2iBY4vC2AGq6uzANBgkqhkiG9w0B
366 # AQsFADByMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYD
367 # VQQLExB3d3cuZGlnaWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2VydCBTSEEyIEFz
368 # c3VyZWQgSUQgQ29kZSBTaWduaW5nIENBMB4XDTE1MDQzMDAwMDAwMFoXDTE2MDcw
369 # NjEyMDAwMFowZTELMAkGA1UEBhMCVVMxETAPBgNVBAgTCENvbG9yYWRvMRUwEwYD
370 # VQQHEwxGb3J0IENvbGxpbnMxFTATBgNVBAoTDEx1bmFyRywgSW5jLjEVMBMGA1UE
371 # AxMMTHVuYXJHLCBJbmMuMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA
372 # jaLJm/Joxsn/IeopiGY3XPeSZeIhjSjSlQUkDyIyDGcBG7CvfiSsXIw3EgdGkjQg
373 # yBcW5YsPz9bPGPUjo5go7CwZaRkhW7/LmSkAlx0UAv8EMLuJrAZ3jBNZvpPPqfWd
374 # zgi/Rkm2gWQ6eSKouy7IjcLk+EwkeBbB+UBnYfMp0BfCPzR3mPgGAJH6efAmEaqQ
375 # FBCrX97joYgDqp3v8u42jALLl/Ict/GNMHLxP+QWagIHIICCRgS6s02OsildLF6R
376 # nqJOOG/43f2qUD4Cab65kzlI+0+uQyOl1UlxNxp0XareghGTqECsYA03j64Esxyo
377 # 2xrNbV2LJm9crTX6QthxywIDAQABo4IBuzCCAbcwHwYDVR0jBBgwFoAUWsS5eyoK
378 # o6XqcQPAYPkt9mV1DlgwHQYDVR0OBBYEFOSdVsqodGWApfCjHtAcn8sAzLBGMA4G
379 # A1UdDwEB/wQEAwIHgDATBgNVHSUEDDAKBggrBgEFBQcDAzB3BgNVHR8EcDBuMDWg
380 # M6Axhi9odHRwOi8vY3JsMy5kaWdpY2VydC5jb20vc2hhMi1hc3N1cmVkLWNzLWcx
381 # LmNybDA1oDOgMYYvaHR0cDovL2NybDQuZGlnaWNlcnQuY29tL3NoYTItYXNzdXJl
382 # ZC1jcy1nMS5jcmwwQgYDVR0gBDswOTA3BglghkgBhv1sAwEwKjAoBggrBgEFBQcC
383 # ARYcaHR0cHM6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzCBhAYIKwYBBQUHAQEEeDB2
384 # MCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5jb20wTgYIKwYBBQUH
385 # MAKGQmh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydFNIQTJBc3N1
386 # cmVkSURDb2RlU2lnbmluZ0NBLmNydDAMBgNVHRMBAf8EAjAAMA0GCSqGSIb3DQEB
387 # CwUAA4IBAQCIt1S8zfvzMQEVmdAssjrwqBaq78xhtGPLjkNF06EvtWoV6VMLI/A6
388 # 45KoULsaXeYuszLxNI+OT/b4HfD0e2LxImaTDZRmCLeIs+2pMLSlWDSV4okm8Vk2
389 # rObLBlgiI1x0PiMa1le9D832COWM4EJcH7pxM+9JfiHYMLlZbcfNEVgv6Dhhl4MG
390 # mOTMTl7vQNNQaJ1coNVf9m5Bez1DV9Iu2Cgd8BHp1oLVCQCHjVv0Ifj48RIPi4SQ
391 # khzalrnrf+L/BWRDhpLnxYasazdV5WfrMHurPuBvYUiLQNkU9SqKgRk9XrzDAfMe
392 # gPbGybMr0kqtbE/A/cDcTVnvRuTZnhXSMIIFMDCCBBigAwIBAgIQBAkYG1/Vu2Z1
393 # U0O1b5VQCDANBgkqhkiG9w0BAQsFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMM
394 # RGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQD
395 # ExtEaWdpQ2VydCBBc3N1cmVkIElEIFJvb3QgQ0EwHhcNMTMxMDIyMTIwMDAwWhcN
396 # MjgxMDIyMTIwMDAwWjByMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQg
397 # SW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMTEwLwYDVQQDEyhEaWdpQ2Vy
398 # dCBTSEEyIEFzc3VyZWQgSUQgQ29kZSBTaWduaW5nIENBMIIBIjANBgkqhkiG9w0B
399 # AQEFAAOCAQ8AMIIBCgKCAQEA+NOzHH8OEa9ndwfTCzFJGc/Q+0WZsTrbRPV/5aid
400 # 2zLXcep2nQUut4/6kkPApfmJ1DcZ17aq8JyGpdglrA55KDp+6dFn08b7KSfH03sj
401 # lOSRI5aQd4L5oYQjZhJUM1B0sSgmuyRpwsJS8hRniolF1C2ho+mILCCVrhxKhwjf
402 # DPXiTWAYvqrEsq5wMWYzcT6scKKrzn/pfMuSoeU7MRzP6vIK5Fe7SrXpdOYr/mzL
403 # fnQ5Ng2Q7+S1TqSp6moKq4TzrGdOtcT3jNEgJSPrCGQ+UpbB8g8S9MWOD8Gi6CxR
404 # 93O8vYWxYoNzQYIH5DiLanMg0A9kczyen6Yzqf0Z3yWT0QIDAQABo4IBzTCCAckw
405 # EgYDVR0TAQH/BAgwBgEB/wIBADAOBgNVHQ8BAf8EBAMCAYYwEwYDVR0lBAwwCgYI
406 # KwYBBQUHAwMweQYIKwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2Nz
407 # cC5kaWdpY2VydC5jb20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2lj
408 # ZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwgYEGA1UdHwR6MHgw
409 # OqA4oDaGNGh0dHA6Ly9jcmw0LmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJ
410 # RFJvb3RDQS5jcmwwOqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdp
411 # Q2VydEFzc3VyZWRJRFJvb3RDQS5jcmwwTwYDVR0gBEgwRjA4BgpghkgBhv1sAAIE
412 # MCowKAYIKwYBBQUHAgEWHGh0dHBzOi8vd3d3LmRpZ2ljZXJ0LmNvbS9DUFMwCgYI
413 # YIZIAYb9bAMwHQYDVR0OBBYEFFrEuXsqCqOl6nEDwGD5LfZldQ5YMB8GA1UdIwQY
414 # MBaAFEXroq/0ksuCMS1Ri6enIZ3zbcgPMA0GCSqGSIb3DQEBCwUAA4IBAQA+7A1a
415 # JLPzItEVyCx8JSl2qB1dHC06GsTvMGHXfgtg/cM9D8Svi/3vKt8gVTew4fbRknUP
416 # UbRupY5a4l4kgU4QpO4/cY5jDhNLrddfRHnzNhQGivecRk5c/5CxGwcOkRX7uq+1
417 # UcKNJK4kxscnKqEpKBo6cSgCPC6Ro8AlEeKcFEehemhor5unXCBc2XGxDI+7qPjF
418 # Emifz0DLQESlE/DmZAwlCEIysjaKJAL+L3J+HNdJRZboWR3p+nRka7LrZkPas7CM
419 # 1ekN3fYBIM6ZMWM9CBoYs4GbT8aTEAb8B4H6i9r5gkn3Ym6hU/oSlBiFLpKR6mhs
420 # RDKyZqHnGKSaZFHvMIIGajCCBVKgAwIBAgIQAwGaAjr/WLFr1tXq5hfwZjANBgkq
421 # hkiG9w0BAQUFADBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5j
422 # MRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2VydCBB
423 # c3N1cmVkIElEIENBLTEwHhcNMTQxMDIyMDAwMDAwWhcNMjQxMDIyMDAwMDAwWjBH
424 # MQswCQYDVQQGEwJVUzERMA8GA1UEChMIRGlnaUNlcnQxJTAjBgNVBAMTHERpZ2lD
425 # ZXJ0IFRpbWVzdGFtcCBSZXNwb25kZXIwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAw
426 # ggEKAoIBAQCjZF38fLPggjXg4PbGKuZJdTvMbuBTqZ8fZFnmfGt/a4ydVfiS457V
427 # WmNbAklQ2YPOb2bu3cuF6V+l+dSHdIhEOxnJ5fWRn8YUOawk6qhLLJGJzF4o9GS2
428 # ULf1ErNzlgpno75hn67z/RJ4dQ6mWxT9RSOOhkRVfRiGBYxVh3lIRvfKDo2n3k5f
429 # 4qi2LVkCYYhhchhoubh87ubnNC8xd4EwH7s2AY3vJ+P3mvBMMWSN4+v6GYeofs/s
430 # jAw2W3rBerh4x8kGLkYQyI3oBGDbvHN0+k7Y/qpA8bLOcEaD6dpAoVk62RUJV5lW
431 # MJPzyWHM0AjMa+xiQpGsAsDvpPCJEY93AgMBAAGjggM1MIIDMTAOBgNVHQ8BAf8E
432 # BAMCB4AwDAYDVR0TAQH/BAIwADAWBgNVHSUBAf8EDDAKBggrBgEFBQcDCDCCAb8G
433 # A1UdIASCAbYwggGyMIIBoQYJYIZIAYb9bAcBMIIBkjAoBggrBgEFBQcCARYcaHR0
434 # cHM6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzCCAWQGCCsGAQUFBwICMIIBVh6CAVIA
435 # QQBuAHkAIAB1AHMAZQAgAG8AZgAgAHQAaABpAHMAIABDAGUAcgB0AGkAZgBpAGMA
436 # YQB0AGUAIABjAG8AbgBzAHQAaQB0AHUAdABlAHMAIABhAGMAYwBlAHAAdABhAG4A
437 # YwBlACAAbwBmACAAdABoAGUAIABEAGkAZwBpAEMAZQByAHQAIABDAFAALwBDAFAA
438 # UwAgAGEAbgBkACAAdABoAGUAIABSAGUAbAB5AGkAbgBnACAAUABhAHIAdAB5ACAA
439 # QQBnAHIAZQBlAG0AZQBuAHQAIAB3AGgAaQBjAGgAIABsAGkAbQBpAHQAIABsAGkA
440 # YQBiAGkAbABpAHQAeQAgAGEAbgBkACAAYQByAGUAIABpAG4AYwBvAHIAcABvAHIA
441 # YQB0AGUAZAAgAGgAZQByAGUAaQBuACAAYgB5ACAAcgBlAGYAZQByAGUAbgBjAGUA
442 # LjALBglghkgBhv1sAxUwHwYDVR0jBBgwFoAUFQASKxOYspkH7R7for5XDStnAs0w
443 # HQYDVR0OBBYEFGFaTSS2STKdSip5GoNL9B6Jwcp9MH0GA1UdHwR2MHQwOKA2oDSG
444 # Mmh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJRENBLTEu
445 # Y3JsMDigNqA0hjJodHRwOi8vY3JsNC5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1
446 # cmVkSURDQS0xLmNybDB3BggrBgEFBQcBAQRrMGkwJAYIKwYBBQUHMAGGGGh0dHA6
447 # Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBBBggrBgEFBQcwAoY1aHR0cDovL2NhY2VydHMu
448 # ZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEQ0EtMS5jcnQwDQYJKoZIhvcN
449 # AQEFBQADggEBAJ0lfhszTbImgVybhs4jIA+Ah+WI//+x1GosMe06FxlxF82pG7xa
450 # FjkAneNshORaQPveBgGMN/qbsZ0kfv4gpFetW7easGAm6mlXIV00Lx9xsIOUGQVr
451 # NZAQoHuXx/Y/5+IRQaa9YtnwJz04HShvOlIJ8OxwYtNiS7Dgc6aSwNOOMdgv420X
452 # Ewbu5AO2FKvzj0OncZ0h3RTKFV2SQdr5D4HRmXQNJsQOfxu19aDxxncGKBXp2JPl
453 # VRbwuwqrHNtcSCdmyKOLChzlldquxC5ZoGHd2vNtomHpigtt7BIYvfdVVEADkitr
454 # wlHCCkivsNRu4PQUCjob4489yq9qjXvc2EQwggbNMIIFtaADAgECAhAG/fkDlgOt
455 # 6gAK6z8nu7obMA0GCSqGSIb3DQEBBQUAMGUxCzAJBgNVBAYTAlVTMRUwEwYDVQQK
456 # EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xJDAiBgNV
457 # BAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0wNjExMTAwMDAwMDBa
458 # Fw0yMTExMTAwMDAwMDBaMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2Vy
459 # dCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lD
460 # ZXJ0IEFzc3VyZWQgSUQgQ0EtMTCCASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoC
461 # ggEBAOiCLZn5ysJClaWAc0Bw0p5WVFypxNJBBo/JM/xNRZFcgZ/tLJz4FlnfnrUk
462 # FcKYubR3SdyJxArar8tea+2tsHEx6886QAxGTZPsi3o2CAOrDDT+GEmC/sfHMUiA
463 # fB6iD5IOUMnGh+s2P9gww/+m9/uizW9zI/6sVgWQ8DIhFonGcIj5BZd9o8dD3QLo
464 # Oz3tsUGj7T++25VIxO4es/K8DCuZ0MZdEkKB4YNugnM/JksUkK5ZZgrEjb7Szgau
465 # rYRvSISbT0C58Uzyr5j79s5AXVz2qPEvr+yJIvJrGGWxwXOt1/HYzx4KdFxCuGh+
466 # t9V3CidWfA9ipD8yFGCV/QcEogkCAwEAAaOCA3owggN2MA4GA1UdDwEB/wQEAwIB
467 # hjA7BgNVHSUENDAyBggrBgEFBQcDAQYIKwYBBQUHAwIGCCsGAQUFBwMDBggrBgEF
468 # BQcDBAYIKwYBBQUHAwgwggHSBgNVHSAEggHJMIIBxTCCAbQGCmCGSAGG/WwAAQQw
469 # ggGkMDoGCCsGAQUFBwIBFi5odHRwOi8vd3d3LmRpZ2ljZXJ0LmNvbS9zc2wtY3Bz
470 # LXJlcG9zaXRvcnkuaHRtMIIBZAYIKwYBBQUHAgIwggFWHoIBUgBBAG4AeQAgAHUA
471 # cwBlACAAbwBmACAAdABoAGkAcwAgAEMAZQByAHQAaQBmAGkAYwBhAHQAZQAgAGMA
472 # bwBuAHMAdABpAHQAdQB0AGUAcwAgAGEAYwBjAGUAcAB0AGEAbgBjAGUAIABvAGYA
473 # IAB0AGgAZQAgAEQAaQBnAGkAQwBlAHIAdAAgAEMAUAAvAEMAUABTACAAYQBuAGQA
474 # IAB0AGgAZQAgAFIAZQBsAHkAaQBuAGcAIABQAGEAcgB0AHkAIABBAGcAcgBlAGUA
475 # bQBlAG4AdAAgAHcAaABpAGMAaAAgAGwAaQBtAGkAdAAgAGwAaQBhAGIAaQBsAGkA
476 # dAB5ACAAYQBuAGQAIABhAHIAZQAgAGkAbgBjAG8AcgBwAG8AcgBhAHQAZQBkACAA
477 # aABlAHIAZQBpAG4AIABiAHkAIAByAGUAZgBlAHIAZQBuAGMAZQAuMAsGCWCGSAGG
478 # /WwDFTASBgNVHRMBAf8ECDAGAQH/AgEAMHkGCCsGAQUFBwEBBG0wazAkBggrBgEF
479 # BQcwAYYYaHR0cDovL29jc3AuZGlnaWNlcnQuY29tMEMGCCsGAQUFBzAChjdodHRw
480 # Oi8vY2FjZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRBc3N1cmVkSURSb290Q0Eu
481 # Y3J0MIGBBgNVHR8EejB4MDqgOKA2hjRodHRwOi8vY3JsMy5kaWdpY2VydC5jb20v
482 # RGlnaUNlcnRBc3N1cmVkSURSb290Q0EuY3JsMDqgOKA2hjRodHRwOi8vY3JsNC5k
483 # aWdpY2VydC5jb20vRGlnaUNlcnRBc3N1cmVkSURSb290Q0EuY3JsMB0GA1UdDgQW
484 # BBQVABIrE5iymQftHt+ivlcNK2cCzTAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYun
485 # pyGd823IDzANBgkqhkiG9w0BAQUFAAOCAQEARlA+ybcoJKc4HbZbKa9Sz1LpMUer
486 # Vlx71Q0LQbPv7HUfdDjyslxhopyVw1Dkgrkj0bo6hnKtOHisdV0XFzRyR4WUVtHr
487 # uzaEd8wkpfMEGVWp5+Pnq2LN+4stkMLA0rWUvV5PsQXSDj0aqRRbpoYxYqioM+Sb
488 # OafE9c4deHaUJXPkKqvPnHZL7V/CSxbkS3BMAIke/MV5vEwSV/5f4R68Al2o/vsH
489 # OE8Nxl2RuQ9nRc3Wg+3nkg2NsWmMT/tZ4CMP0qquAHzunEIOz5HXJ7cW7g/DvXwK
490 # oO4sCFWFIrjrGBpN/CohrUkxg0eVd3HcsRtLSxwQnHcUwZ1PL1qVCCkQJjGCBDsw
491 # ggQ3AgEBMIGGMHIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMx
492 # GTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xMTAvBgNVBAMTKERpZ2lDZXJ0IFNI
493 # QTIgQXNzdXJlZCBJRCBDb2RlIFNpZ25pbmcgQ0ECEA5mBKT7UNogWOLwtgBqursw
494 # CQYFKw4DAhoFAKB4MBgGCisGAQQBgjcCAQwxCjAIoAKAAKECgAAwGQYJKoZIhvcN
495 # AQkDMQwGCisGAQQBgjcCAQQwHAYKKwYBBAGCNwIBCzEOMAwGCisGAQQBgjcCARUw
496 # IwYJKoZIhvcNAQkEMRYEFCQBfl/Xm3/R6yW/EO6kbSmkdowDMA0GCSqGSIb3DQEB
497 # AQUABIIBADCbC3HqswOLfqwjX9+TM0hW9sG02WMHPbz0fFBTH5J/tck4wZECl9ct
498 # DK0pUzHoJBY9EuBnH9OD46MiVCIYwYHQ9w/xiaypUNRbfXYEwSVL9EXCIcYkkqAN
499 # pSpDrQJu0TzmGyvN1fSvYj/qahvIVKz/cxbzzQbYl4NqNXRfiD26Pa5JOdNABP8g
500 # WL5Ruk/MPvMJE0dIW3em40hoanGKQhP0xgQ/BGJygumYrZsigENfhQkRVngH/aUP
501 # f5k78VKL3DFoCMmneIxAfIwspTC37izb/AjlqDNUbqEmfBBIsbLgu6teZVIyPBI/
502 # nktk5kwOOhzuyeQxLAcn0z+8ToF5frKhggIPMIICCwYJKoZIhvcNAQkGMYIB/DCC
503 # AfgCAQEwdjBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkw
504 # FwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2VydCBBc3N1
505 # cmVkIElEIENBLTECEAMBmgI6/1ixa9bV6uYX8GYwCQYFKw4DAhoFAKBdMBgGCSqG
506 # SIb3DQEJAzELBgkqhkiG9w0BBwEwHAYJKoZIhvcNAQkFMQ8XDTE2MDMyMjIwMjM0
507 # N1owIwYJKoZIhvcNAQkEMRYEFM6NeSjPd00j7j25copMrjENL7GXMA0GCSqGSIb3
508 # DQEBAQUABIIBAHJbUlt2mxIX5hbiigRw3kIoug57G5sDYWQK8rcTjHUif6PAdEqj
509 # 5c1UhxQHJxEasddUAqbEtCsG8qiz1lq76KKiwaWxffSRQ2JwjYEvnYQ2TK9rtnMs
510 # zeYnQajrIUP44z7ysqoikB0bEgup0QVDScm4SSa1SmqQzHMsUX5rCygsM3PlpF5K
511 # dH2u3eSK4zDhGiye6/SQkcddvsI2lLFRcxQIyfUD4+W9oFdXuYkKhNBGPLUlOH9V
512 # DEDQG9zH6CAzvla/r1iYnX8RZ4rz7yacdrMBq5g92HAEcuXFTBQfaeAZSGQBhNSn
513 # p1rVWgLb0T3a/5zlOtZvp+bLyDRbms+w8BY=
514 # SIG # End signature block
515