| # -*- coding: utf-8 -*- |
| |
| #------------------------------------------------------------------------- |
| # drawElements Quality Program utilities |
| # -------------------------------------- |
| # |
| # Copyright 2015 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| # |
| #------------------------------------------------------------------------- |
| |
| import os |
| import sys |
| import shutil |
| import tarfile |
| import zipfile |
| import hashlib |
| import argparse |
| import subprocess |
| import ssl |
| import stat |
| import platform |
| import logging |
| |
| scriptPath = os.path.join(os.path.dirname(__file__), "..", "scripts") |
| sys.path.insert(0, scriptPath) |
| |
| from ctsbuild.common import * |
| |
| EXTERNAL_DIR = os.path.realpath(os.path.normpath(os.path.dirname(__file__))) |
| |
| SYSTEM_NAME = platform.system() |
| |
| def computeChecksum (data): |
| return hashlib.sha256(data).hexdigest() |
| |
| def onReadonlyRemoveError (func, path, exc_info): |
| os.chmod(path, stat.S_IWRITE) |
| os.unlink(path) |
| |
| class Source: |
| def __init__(self, baseDir, extractDir): |
| self.baseDir = baseDir |
| self.extractDir = extractDir |
| |
| def clean (self): |
| fullDstPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.extractDir) |
| # Remove read-only first |
| readonlydir = os.path.join(fullDstPath, ".git") |
| if os.path.exists(readonlydir): |
| logging.debug("Deleting " + readonlydir) |
| shutil.rmtree(readonlydir, onerror = onReadonlyRemoveError) |
| if os.path.exists(fullDstPath): |
| logging.debug("Deleting " + fullDstPath) |
| shutil.rmtree(fullDstPath, ignore_errors=False) |
| |
| class SourcePackage (Source): |
| def __init__(self, url, checksum, baseDir, extractDir = "src", postExtract=None): |
| Source.__init__(self, baseDir, extractDir) |
| self.url = url |
| self.filename = os.path.basename(self.url) |
| self.checksum = checksum |
| self.archiveDir = "packages" |
| self.postExtract = postExtract |
| |
| def clean (self): |
| Source.clean(self) |
| self.removeArchives() |
| |
| def update (self, cmdProtocol = None, force = False): |
| if not self.isArchiveUpToDate(): |
| self.fetchAndVerifyArchive() |
| |
| if self.getExtractedChecksum() != self.checksum: |
| Source.clean(self) |
| self.extract() |
| self.storeExtractedChecksum(self.checksum) |
| |
| def removeArchives (self): |
| archiveDir = os.path.join(EXTERNAL_DIR, pkg.baseDir, pkg.archiveDir) |
| if os.path.exists(archiveDir): |
| logging.debug("Deleting " + archiveDir) |
| shutil.rmtree(archiveDir, ignore_errors=False) |
| |
| def isArchiveUpToDate (self): |
| archiveFile = os.path.join(EXTERNAL_DIR, pkg.baseDir, pkg.archiveDir, pkg.filename) |
| if os.path.exists(archiveFile): |
| return computeChecksum(readBinaryFile(archiveFile)) == self.checksum |
| else: |
| return False |
| |
| def getExtractedChecksumFilePath (self): |
| return os.path.join(EXTERNAL_DIR, pkg.baseDir, pkg.archiveDir, "extracted") |
| |
| def getExtractedChecksum (self): |
| extractedChecksumFile = self.getExtractedChecksumFilePath() |
| |
| if os.path.exists(extractedChecksumFile): |
| return readFile(extractedChecksumFile) |
| else: |
| return None |
| |
| def storeExtractedChecksum (self, checksum): |
| checksum_bytes = checksum.encode("utf-8") |
| writeBinaryFile(self.getExtractedChecksumFilePath(), checksum_bytes) |
| |
| def connectToUrl (self, url): |
| result = None |
| |
| if sys.version_info < (3, 0): |
| from urllib2 import urlopen |
| else: |
| from urllib.request import urlopen |
| |
| if args.insecure: |
| print("Ignoring certificate checks") |
| ssl_context = ssl._create_unverified_context() |
| result = urlopen(url, context=ssl_context) |
| else: |
| result = urlopen(url) |
| |
| return result |
| |
| def fetchAndVerifyArchive (self): |
| print("Fetching %s" % self.url) |
| |
| req = self.connectToUrl(self.url) |
| data = req.read() |
| checksum = computeChecksum(data) |
| dstPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.archiveDir, self.filename) |
| |
| if checksum != self.checksum: |
| raise Exception("Checksum mismatch for %s, expected %s, got %s" % (self.filename, self.checksum, checksum)) |
| |
| if not os.path.exists(os.path.dirname(dstPath)): |
| os.makedirs(os.path.dirname(dstPath)) |
| |
| writeBinaryFile(dstPath, data) |
| |
| def extract (self): |
| print("Extracting %s to %s/%s" % (self.filename, self.baseDir, self.extractDir)) |
| |
| srcPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.archiveDir, self.filename) |
| tmpPath = os.path.join(EXTERNAL_DIR, ".extract-tmp-%s" % self.baseDir) |
| dstPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.extractDir) |
| |
| if self.filename.endswith(".zip"): |
| archive = zipfile.ZipFile(srcPath) |
| else: |
| archive = tarfile.open(srcPath) |
| |
| if os.path.exists(tmpPath): |
| shutil.rmtree(tmpPath, ignore_errors=False) |
| |
| os.mkdir(tmpPath) |
| |
| archive.extractall(tmpPath) |
| archive.close() |
| |
| extractedEntries = os.listdir(tmpPath) |
| if len(extractedEntries) != 1 or not os.path.isdir(os.path.join(tmpPath, extractedEntries[0])): |
| raise Exception("%s doesn't contain single top-level directory" % self.filename) |
| |
| topLevelPath = os.path.join(tmpPath, extractedEntries[0]) |
| |
| if not os.path.exists(dstPath): |
| os.mkdir(dstPath) |
| |
| for entry in os.listdir(topLevelPath): |
| if os.path.exists(os.path.join(dstPath, entry)): |
| raise Exception("%s exists already" % entry) |
| |
| shutil.move(os.path.join(topLevelPath, entry), dstPath) |
| |
| shutil.rmtree(tmpPath, ignore_errors=True) |
| |
| if self.postExtract != None: |
| self.postExtract(dstPath) |
| |
| class SourceFile (Source): |
| def __init__(self, url, filename, checksum, baseDir, extractDir = "src"): |
| Source.__init__(self, baseDir, extractDir) |
| self.url = url |
| self.filename = filename |
| self.checksum = checksum |
| |
| def update (self, cmdProtocol = None, force = False): |
| if not self.isFileUpToDate(): |
| Source.clean(self) |
| self.fetchAndVerifyFile() |
| |
| def isFileUpToDate (self): |
| file = os.path.join(EXTERNAL_DIR, pkg.baseDir, pkg.extractDir, pkg.filename) |
| if os.path.exists(file): |
| data = readFile(file) |
| return computeChecksum(data.encode('utf-8')) == self.checksum |
| else: |
| return False |
| |
| def connectToUrl (self, url): |
| result = None |
| |
| if sys.version_info < (3, 0): |
| from urllib2 import urlopen |
| else: |
| from urllib.request import urlopen |
| |
| if args.insecure: |
| print("Ignoring certificate checks") |
| ssl_context = ssl._create_unverified_context() |
| result = urlopen(url, context=ssl_context) |
| else: |
| result = urlopen(url) |
| |
| return result |
| |
| def fetchAndVerifyFile (self): |
| print("Fetching %s" % self.url) |
| |
| req = self.connectToUrl(self.url) |
| data = req.read() |
| checksum = computeChecksum(data) |
| dstPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.extractDir, self.filename) |
| |
| if checksum != self.checksum: |
| raise Exception("Checksum mismatch for %s, expected %s, got %s" % (self.filename, self.checksum, checksum)) |
| |
| if not os.path.exists(os.path.dirname(dstPath)): |
| os.mkdir(os.path.dirname(dstPath)) |
| |
| writeBinaryFile(dstPath, data) |
| |
| class GitRepo (Source): |
| def __init__(self, httpsUrl, sshUrl, revision, baseDir, extractDir = "src", removeTags = [], patch = ""): |
| Source.__init__(self, baseDir, extractDir) |
| self.httpsUrl = httpsUrl |
| self.sshUrl = sshUrl |
| self.revision = revision |
| self.removeTags = removeTags |
| self.patch = patch |
| |
| def checkout(self, url, fullDstPath, force): |
| if not os.path.exists(os.path.join(fullDstPath, '.git')): |
| run(["git", "clone", "--no-checkout", url, fullDstPath]) |
| |
| pushWorkingDir(fullDstPath) |
| print("Directory: " + fullDstPath) |
| try: |
| for tag in self.removeTags: |
| proc = subprocess.Popen(['git', 'tag', '-l', tag], stdout=subprocess.PIPE) |
| (stdout, stderr) = proc.communicate() |
| if len(stdout) > 0: |
| run(["git", "tag", "-d",tag]) |
| force_arg = ['--force'] if force else [] |
| run(["git", "fetch"] + force_arg + ["--tags", url, "+refs/heads/*:refs/remotes/origin/*"]) |
| run(["git", "checkout"] + force_arg + [self.revision]) |
| |
| if(self.patch != ""): |
| patchFile = os.path.join(EXTERNAL_DIR, self.patch) |
| run(["git", "reset", "--hard", "HEAD"]) |
| run(["git", "apply", patchFile]) |
| except: |
| # This might be a KeyboardInterrupt or other error, propagate. |
| raise |
| finally: |
| popWorkingDir() |
| |
| def update (self, cmdProtocol, force = False): |
| fullDstPath = os.path.join(EXTERNAL_DIR, self.baseDir, self.extractDir) |
| url = self.httpsUrl |
| backupUrl = self.sshUrl |
| |
| # If url is none then start with ssh |
| if cmdProtocol == 'ssh' or url == None: |
| url = self.sshUrl |
| backupUrl = self.httpsUrl |
| |
| try: |
| self.checkout(url, fullDstPath, force) |
| except KeyboardInterrupt: |
| # Propagate the exception to stop the process if possible. |
| raise |
| except: |
| # For any other kind of exception, including subprocess errors, we |
| # try the backup URL. |
| if backupUrl != None: |
| self.checkout(backupUrl, fullDstPath, force) |
| |
| def postExtractLibpng (path): |
| shutil.copy(os.path.join(path, "scripts", "pnglibconf.h.prebuilt"), |
| os.path.join(path, "pnglibconf.h")) |
| |
| PACKAGES = [ |
| SourcePackage( |
| "https://github.com/madler/zlib/releases/download/v1.2.13/zlib-1.2.13.tar.gz", |
| "b3a24de97a8fdbc835b9833169501030b8977031bcb54b3b3ac13740f846ab30", |
| "zlib"), |
| SourcePackage( |
| "http://prdownloads.sourceforge.net/libpng/libpng-1.6.27.tar.gz", |
| "c9d164ec247f426a525a7b89936694aefbc91fb7a50182b198898b8fc91174b4", |
| "libpng", |
| postExtract = postExtractLibpng), |
| SourceFile( |
| "https://raw.githubusercontent.com/baldurk/renderdoc/v1.1/renderdoc/api/app/renderdoc_app.h", |
| "renderdoc_app.h", |
| "e7b5f0aa5b1b0eadc63a1c624c0ca7f5af133aa857d6a4271b0ef3d0bdb6868e", |
| "renderdoc"), |
| GitRepo( |
| "https://github.com/KhronosGroup/SPIRV-Tools.git", |
| "git@github.com:KhronosGroup/SPIRV-Tools.git", |
| "4c7e1fa5c3d988cca0e626d359d30b117b9c2822", |
| "spirv-tools"), |
| GitRepo( |
| "https://github.com/KhronosGroup/glslang.git", |
| "git@github.com:KhronosGroup/glslang.git", |
| "4da479aa6afa43e5a2ce4c4148c572a03123faf3", |
| "glslang", |
| removeTags = ["main-tot", "master-tot"]), |
| GitRepo( |
| "https://github.com/KhronosGroup/SPIRV-Headers.git", |
| "git@github.com:KhronosGroup/SPIRV-Headers.git", |
| "db5a00f8cebe81146cafabf89019674a3c4bf03d", |
| "spirv-headers"), |
| GitRepo( |
| "https://github.com/KhronosGroup/Vulkan-Docs.git", |
| "git@github.com:KhronosGroup/Vulkan-Docs.git", |
| "7bb606eb87cde1d34f65f36f4d4c6f2c78f072c8", |
| "vulkan-docs"), |
| GitRepo( |
| "https://github.com/KhronosGroup/Vulkan-ValidationLayers.git", |
| "git@github.com:KhronosGroup/Vulkan-ValidationLayers.git", |
| "f589bc456545fbab97caf49380b102b8aafe1f40", |
| "vulkan-validationlayers"), |
| GitRepo( |
| "https://github.com/google/amber.git", |
| "git@github.com:google/amber.git", |
| "6bb8b9979d12122c3ac2e627bcad965129556d12", |
| "amber"), |
| GitRepo( |
| "https://github.com/open-source-parsers/jsoncpp.git", |
| "git@github.com:open-source-parsers/jsoncpp.git", |
| "9059f5cad030ba11d37818847443a53918c327b1", |
| "jsoncpp"), |
| # NOTE: The samples application is not well suited to external |
| # integration, this fork contains the small fixes needed for use |
| # by the CTS. |
| GitRepo( |
| "https://github.com/Igalia/vk_video_samples.git", |
| "git@github.com:Igalia/vk_video_samples.git", |
| "6821adf11eb4f84a2168264b954c170d03237699", |
| "nvidia-video-samples"), |
| ] |
| |
| def parseArgs (): |
| versionsForInsecure = ((2,7,9), (3,4,3)) |
| versionsForInsecureStr = ' or '.join(('.'.join(str(x) for x in v)) for v in versionsForInsecure) |
| |
| parser = argparse.ArgumentParser(description = "Fetch external sources") |
| parser.add_argument('--clean', dest='clean', action='store_true', default=False, |
| help='Remove sources instead of fetching') |
| parser.add_argument('--insecure', dest='insecure', action='store_true', default=False, |
| help="Disable certificate check for external sources." |
| " Minimum python version required " + versionsForInsecureStr) |
| parser.add_argument('--protocol', dest='protocol', default='https', choices=['ssh', 'https'], |
| help="Select protocol to checkout git repositories.") |
| parser.add_argument('--force', dest='force', action='store_true', default=False, |
| help="Pass --force to git fetch and checkout commands") |
| parser.add_argument("-v", "--verbose", |
| dest="verbose", |
| action="store_true", |
| help="Enable verbose logging") |
| args = parser.parse_args() |
| |
| if args.insecure: |
| for versionItem in versionsForInsecure: |
| if (sys.version_info.major == versionItem[0]): |
| if sys.version_info < versionItem: |
| parser.error("For --insecure minimum required python version is " + |
| versionsForInsecureStr) |
| break; |
| |
| return args |
| |
| def run(*popenargs, **kwargs): |
| process = subprocess.Popen(*popenargs, **kwargs) |
| |
| try: |
| stdout, stderr = process.communicate(None) |
| except KeyboardInterrupt: |
| # Terminate the process, wait and propagate. |
| process.terminate() |
| process.wait() |
| raise |
| except: |
| # With any other exception, we _kill_ the process and propagate. |
| process.kill() |
| process.wait() |
| raise |
| else: |
| # Everything good, fetch the retcode and raise exception if needed. |
| retcode = process.poll() |
| if retcode: |
| raise subprocess.CalledProcessError(retcode, process.args, output=stdout, stderr=stderr) |
| |
| if __name__ == "__main__": |
| args = parseArgs() |
| initializeLogger(args.verbose) |
| |
| try: |
| for pkg in PACKAGES: |
| if args.clean: |
| pkg.clean() |
| else: |
| pkg.update(args.protocol, args.force) |
| except KeyboardInterrupt: |
| sys.exit("") # Returns 1. |