# -*- coding: utf-8 -*-
#-------------------------------------------------------------------------
# drawElements Quality Program utilities
# --------------------------------------
#
# Copyright 2016 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 argparse
import tempfile
import sys
from build.common import *
from build.build import *
pythonExecutable = sys.executable or "python"
class Environment:
def __init__ (self, srcDir, tmpDir):
self.srcDir = srcDir
self.tmpDir = tmpDir
class BuildTestStep:
def getName (self):
return "<unknown>"
def isAvailable (self, env):
return True
def run (self, env):
raise Exception("Not implemented")
class RunScript(BuildTestStep):
def __init__ (self, scriptPath, getExtraArgs = None):
self.scriptPath = scriptPath
self.getExtraArgs = getExtraArgs
def getName (self):
return self.scriptPath
def run (self, env):
args = [pythonExecutable, os.path.join(env.srcDir, self.scriptPath)]
if self.getExtraArgs != None:
args += self.getExtraArgs(env)
execute(args)
def makeCflagsArgs (cflags):
cflagsStr = " ".join(cflags)
return ["-DCMAKE_C_FLAGS=%s" % cflagsStr, "-DCMAKE_CXX_FLAGS=%s" % cflagsStr]
def makeBuildArgs (target, cc, cpp, cflags):
return ["-DDEQP_TARGET=%s" % target, "-DCMAKE_C_COMPILER=%s" % cc, "-DCMAKE_CXX_COMPILER=%s" % cpp] + makeCflagsArgs(cflags)
class BuildConfigGen:
def isAvailable (self, env):
return True
class UnixConfig(BuildConfigGen):
def __init__ (self, target, buildType, cc, cpp, cflags):
self.target = target
self.buildType = buildType
self.cc = cc
self.cpp = cpp
self.cflags = cflags
def isAvailable (self, env):
return which(self.cc) != None and which(self.cpp) != None
def getBuildConfig (self, env, buildDir):
args = makeBuildArgs(self.target, self.cc, self.cpp, self.cflags)
return BuildConfig(buildDir, self.buildType, args, env.srcDir)
class VSConfig(BuildConfigGen):
def __init__ (self, buildType):
self.buildType = buildType
def getBuildConfig (self, env, buildDir):
args = ["-DCMAKE_C_FLAGS=/WX -DCMAKE_CXX_FLAGS=/WX"]
return BuildConfig(buildDir, self.buildType, args, env.srcDir)
class Build(BuildTestStep):
def __init__ (self, buildDir, configGen, generator):
self.buildDir = buildDir
self.configGen = configGen
self.generator = generator
def getName (self):
return self.buildDir
def isAvailable (self, env):
return self.configGen.isAvailable(env) and self.generator != None and self.generator.isAvailable()
def run (self, env):
# specialize config for env
buildDir = os.path.join(env.tmpDir, self.buildDir)
curConfig = self.configGen.getBuildConfig(env, buildDir)
build(curConfig, self.generator)
class CheckSrcChanges(BuildTestStep):
def getName (self):
return "check for changes"
def run (self, env):
pushWorkingDir(env.srcDir)
execute(["git", "diff", "--exit-code"])
popWorkingDir()
def getClangVersion ():
knownVersions = ["4.0", "3.9", "3.8", "3.7", "3.6", "3.5"]
for version in knownVersions:
if which("clang-" + version) != None:
return "-" + version
return ""
def runSteps (steps):
for step in steps:
if step.isAvailable(env):
print "Run: %s" % step.getName()
step.run(env)
else:
print "Skip: %s" % step.getName()
COMMON_CFLAGS = ["-Werror", "-Wno-error=unused-function"]
COMMON_GCC_CFLAGS = COMMON_CFLAGS + ["-Wno-implicit-fallthrough"]
COMMON_CLANG_CFLAGS = COMMON_CFLAGS + ["-Wno-error=unused-command-line-argument"]
GCC_32BIT_CFLAGS = COMMON_GCC_CFLAGS + ["-m32"]
CLANG_32BIT_CFLAGS = COMMON_CLANG_CFLAGS + ["-m32"]
GCC_64BIT_CFLAGS = COMMON_GCC_CFLAGS + ["-m64"]
CLANG_64BIT_CFLAGS = COMMON_CLANG_CFLAGS + ["-m64"]
CLANG_VERSION = getClangVersion()
# Always ran before any receipe
PREREQUISITES = [
RunScript(os.path.join("external", "fetch_sources.py"))
]
# Always ran after any receipe
POST_CHECKS = [
CheckSrcChanges()
]
BUILD_TARGETS = [
Build("clang-64-debug",
UnixConfig("null",
"Debug",
"clang" + CLANG_VERSION,
"clang++" + CLANG_VERSION,
CLANG_64BIT_CFLAGS),
ANY_UNIX_GENERATOR),
Build("gcc-32-debug",
UnixConfig("null",
"Debug",
"gcc",
"g++",
GCC_32BIT_CFLAGS),
ANY_UNIX_GENERATOR),
Build("gcc-64-release",
UnixConfig("null",
"Release",
"gcc",
"g++",
GCC_64BIT_CFLAGS),
ANY_UNIX_GENERATOR),
Build("vs-64-debug",
VSConfig("Debug"),
ANY_VS_X64_GENERATOR),
]
EARLY_SPECIAL_RECIPES = [
('gen-inl-files', [
RunScript(os.path.join("scripts", "gen_egl.py")),
RunScript(os.path.join("scripts", "opengl", "gen_all.py")),
RunScript(os.path.join("external", "vulkancts", "scripts", "gen_framework.py")),
RunScript(os.path.join("external", "vulkancts", "scripts", "gen_framework_c.py")),
RunScript(os.path.join("scripts", "gen_android_bp.py")),
]),
('gen-ext-deps', [
RunScript(os.path.join("external", "vulkancts", "scripts", "gen_ext_deps.py"))
]),
]
LATE_SPECIAL_RECIPES = [
('android-mustpass', [
RunScript(os.path.join("scripts", "build_android_mustpass.py"),
lambda env: ["--build-dir", os.path.join(env.tmpDir, "android-mustpass")]),
]),
('vulkan-mustpass', [
RunScript(os.path.join("external", "vulkancts", "scripts", "build_mustpass.py"),
lambda env: ["--build-dir", os.path.join(env.tmpDir, "vulkan-mustpass")]),
]),
('spirv-binaries', [
RunScript(os.path.join("external", "vulkancts", "scripts", "build_spirv_binaries.py"),
lambda env: ["--build-dir", os.path.join(env.tmpDir, "spirv-binaries"),
"--dst-path", os.path.join(env.tmpDir, "spirv-binaries")]),
]),
('check-all', [
RunScript(os.path.join("scripts", "src_util", "check_all.py")),
])
]
def getBuildRecipes ():
return [(b.getName(), [b]) for b in BUILD_TARGETS]
def getAllRecipe (recipes):
allSteps = []
for name, steps in recipes:
allSteps += steps
return ("all", allSteps)
def getRecipes ():
recipes = EARLY_SPECIAL_RECIPES + getBuildRecipes() + LATE_SPECIAL_RECIPES
return recipes
def getRecipe (recipes, recipeName):
for curName, steps in recipes:
if curName == recipeName:
return (curName, steps)
return None
RECIPES = getRecipes()
def parseArgs ():
parser = argparse.ArgumentParser(description = "Build and test source",
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument("-s",
"--src-dir",
dest="srcDir",
default=DEQP_DIR,
help="Source directory")
parser.add_argument("-t",
"--tmp-dir",
dest="tmpDir",
default=os.path.join(tempfile.gettempdir(), "deqp-build-test"),
help="Temporary directory")
parser.add_argument("-r",
"--recipe",
dest="recipe",
choices=[n for n, s in RECIPES] + ["all"],
default="all",
help="Build / test recipe")
parser.add_argument("-d",
"--dump-recipes",
dest="dumpRecipes",
action="store_true",
help="Print out recipes that have any available actions")
parser.add_argument("--skip-prerequisites",
dest="skipPrerequisites",
action="store_true",
help="Skip external dependency fetch")
return parser.parse_args()
if __name__ == "__main__":
args = parseArgs()
env = Environment(args.srcDir, args.tmpDir)
if args.dumpRecipes:
for name, steps in RECIPES:
for step in steps:
if step.isAvailable(env):
print name
break
else:
name, steps = getAllRecipe(RECIPES) if args.recipe == "all" \
else getRecipe(RECIPES, args.recipe)
print "Running %s" % name
allSteps = (PREREQUISITES if (args.skipPrerequisites == False) else []) + steps + POST_CHECKS
runSteps(allSteps)
print "All steps completed successfully"