Merge changes I9512642d,I6548889c,I8db5198f

* changes:
  Add build_test.sh, split common parts of soong_ui.bash
  Allow specifying a build variant
  Improve multiproduct_kati output
This commit is contained in:
Treehugger Robot
2017-05-16 19:45:06 +00:00
committed by Gerrit Code Review
5 changed files with 216 additions and 77 deletions

34
build_test.bash Executable file
View File

@@ -0,0 +1,34 @@
#!/bin/bash -eu
#
# Copyright 2017 Google Inc. All rights reserved.
#
# 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.
#
# This file is used in our continous build infrastructure to run a variety of
# tests related to the build system.
#
# Currently, it's used to build and run multiproduct_kati, so it'll attempt
# to build ninja files for every product in the tree. I expect this to
# evolve as we find interesting things to test or track performance for.
#
# To track how long we took to startup. %N isn't supported on Darwin, but
# that's detected in the Go code, which skips calculating the startup time.
export TRACE_BEGIN_SOONG=$(date +%s%N)
export TOP=$(cd $(dirname ${BASH_SOURCE[0]})/../..; PWD= /bin/pwd)
source "${TOP}/build/soong/cmd/microfactory/microfactory.bash"
build_go multiproduct_kati android/soong/cmd/multiproduct_kati
exec "$(getoutdir)/multiproduct_kati" "$@"

View File

@@ -0,0 +1,88 @@
# Copyright 2017 Google Inc. All rights reserved.
#
# 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.
# Set of utility functions to build and run go code with microfactory
#
# Inputs:
# ${TOP}: The top of the android source tree
# ${OUT_DIR}: The output directory location (defaults to ${TOP}/out)
# ${OUT_DIR_COMMON_BASE}: Change the default out directory to
# ${OUT_DIR_COMMON_BASE}/$(basename ${TOP})
# Ensure GOROOT is set to the in-tree version.
case $(uname) in
Linux)
export GOROOT="${TOP}/prebuilts/go/linux-x86/"
;;
Darwin)
export GOROOT="${TOP}/prebuilts/go/darwin-x86/"
;;
*) echo "unknown OS:" $(uname) >&2 && exit 1;;
esac
# Find the output directory
function getoutdir
{
local out_dir="${OUT_DIR-}"
if [ -z "${out_dir}" ]; then
if [ "${OUT_DIR_COMMON_BASE-}" ]; then
out_dir="${OUT_DIR_COMMON_BASE}/$(basename ${TOP})"
else
out_dir="${TOP}/out"
fi
fi
echo "${out_dir}"
}
# Bootstrap microfactory from source if necessary and use it to build the
# requested binary.
#
# Arguments:
# $1: name of the requested binary
# $2: package name
function build_go
{
# Increment when microfactory changes enough that it cannot rebuild itself.
# For example, if we use a new command line argument that doesn't work on older versions.
local mf_version=2
local mf_src="${TOP}/build/soong/cmd/microfactory"
local out_dir=$(getoutdir)
local mf_bin="${out_dir}/microfactory_$(uname)"
local mf_version_file="${out_dir}/.microfactory_$(uname)_version"
local built_bin="${out_dir}/$1"
local from_src=1
if [ -f "${mf_bin}" ] && [ -f "${mf_version_file}" ]; then
if [ "${mf_version}" -eq "$(cat "${mf_version_file}")" ]; then
from_src=0
fi
fi
local mf_cmd
if [ $from_src -eq 1 ]; then
mf_cmd="${GOROOT}/bin/go run ${mf_src}/microfactory.go"
else
mf_cmd="${mf_bin}"
fi
${mf_cmd} -s "${mf_src}" -b "${mf_bin}" \
-pkg-path "android/soong=${TOP}/build/soong" -trimpath "${TOP}/build/soong" \
-o "${built_bin}" $2
if [ $from_src -eq 1 ]; then
echo "${mf_version}" >"${mf_version_file}"
fi
}

View File

@@ -51,11 +51,86 @@ var outDir = flag.String("out", "", "path to store output directories (defaults
var onlyConfig = flag.Bool("only-config", false, "Only run product config (not Soong or Kati)")
var onlySoong = flag.Bool("only-soong", false, "Only run product config and Soong (not Kati)")
var buildVariant = flag.String("variant", "eng", "build variant to use")
type Product struct {
ctx build.Context
config build.Config
}
type Status struct {
cur int
total int
failed int
ctx build.Context
haveBlankLine bool
smartTerminal bool
lock sync.Mutex
}
func NewStatus(ctx build.Context) *Status {
return &Status{
ctx: ctx,
haveBlankLine: true,
smartTerminal: ctx.IsTerminal(),
}
}
func (s *Status) SetTotal(total int) {
s.total = total
}
func (s *Status) Fail(product string, err error) {
s.Finish(product)
s.lock.Lock()
defer s.lock.Unlock()
if s.smartTerminal && !s.haveBlankLine {
fmt.Fprintln(s.ctx.Stdout())
s.haveBlankLine = true
}
s.failed++
fmt.Fprintln(s.ctx.Stderr(), "FAILED:", product)
s.ctx.Verboseln("FAILED:", product)
s.ctx.Println(err)
}
func (s *Status) Finish(product string) {
s.lock.Lock()
defer s.lock.Unlock()
s.cur++
line := fmt.Sprintf("[%d/%d] %s", s.cur, s.total, product)
if s.smartTerminal {
if max, ok := s.ctx.TermWidth(); ok {
if len(line) > max {
line = line[:max]
}
}
fmt.Fprint(s.ctx.Stdout(), "\r", line, "\x1b[K")
s.haveBlankLine = false
} else {
s.ctx.Println(line)
}
}
func (s *Status) Finished() int {
s.lock.Lock()
defer s.lock.Unlock()
if !s.haveBlankLine {
fmt.Fprintln(s.ctx.Stdout())
s.haveBlankLine = true
}
return s.failed
}
func main() {
log := logger.New(os.Stderr)
defer log.Cleanup()
@@ -80,7 +155,7 @@ func main() {
StdioInterface: build.StdioImpl{},
}}
failed := false
status := NewStatus(buildCtx)
config := build.NewConfig(buildCtx)
if *outDir == "" {
@@ -94,7 +169,7 @@ func main() {
if !*keep {
defer func() {
if !failed {
if status.Finished() == 0 {
os.RemoveAll(*outDir)
}
}()
@@ -114,8 +189,9 @@ func main() {
products := strings.Fields(vars["all_named_products"])
log.Verbose("Got product list:", products)
status.SetTotal(len(products))
var wg sync.WaitGroup
errs := make(chan error, len(products))
productConfigs := make(chan Product, len(products))
// Run the product config for every product in parallel
@@ -124,7 +200,7 @@ func main() {
go func(product string) {
defer wg.Done()
defer logger.Recover(func(err error) {
errs <- fmt.Errorf("Error building %s: %v", product, err)
status.Fail(product, err)
})
productOutDir := filepath.Join(config.OutDir(), product)
@@ -151,7 +227,7 @@ func main() {
productConfig := build.NewConfig(productCtx)
productConfig.Environment().Set("OUT_DIR", productOutDir)
productConfig.Lunch(productCtx, product, "eng")
productConfig.Lunch(productCtx, product, *buildVariant)
build.Build(productCtx, productConfig, build.BuildProductConfig)
productConfigs <- Product{productCtx, productConfig}
@@ -171,7 +247,7 @@ func main() {
for product := range productConfigs {
func() {
defer logger.Recover(func(err error) {
errs <- fmt.Errorf("Error building %s: %v", product.config.TargetProduct(), err)
status.Fail(product.config.TargetProduct(), err)
})
buildWhat := 0
@@ -185,22 +261,14 @@ func main() {
if !*keep {
os.RemoveAll(product.config.OutDir())
}
log.Println("Finished running for", product.config.TargetProduct())
status.Finish(product.config.TargetProduct())
}()
}
}()
}
go func() {
wg2.Wait()
close(errs)
}()
wg2.Wait()
for err := range errs {
failed = true
log.Print(err)
}
if failed {
log.Fatalln("Failed")
if count := status.Finished(); count > 0 {
log.Fatalln(count, "products failed")
}
}

View File

@@ -15,7 +15,7 @@
# limitations under the License.
# To track how long we took to startup. %N isn't supported on Darwin, but
# that's detected in the Go code, and skip calculating the startup time.
# that's detected in the Go code, which skips calculating the startup time.
export TRACE_BEGIN_SOONG=$(date +%s%N)
# Function to find top of the source tree (if $TOP isn't set) by walking up the
@@ -47,63 +47,8 @@ function gettop
fi
}
# Bootstrap microfactory from source if necessary and use it to build the
# soong_ui binary, then run soong_ui.
function run_go
{
# Increment when microfactory changes enough that it cannot rebuild itself.
# For example, if we use a new command line argument that doesn't work on older versions.
local mf_version=2
local mf_src="${TOP}/build/soong/cmd/microfactory"
local out_dir="${OUT_DIR-}"
if [ -z "${out_dir}" ]; then
if [ "${OUT_DIR_COMMON_BASE-}" ]; then
out_dir="${OUT_DIR_COMMON_BASE}/$(basename ${TOP})"
else
out_dir="${TOP}/out"
fi
fi
local mf_bin="${out_dir}/microfactory_$(uname)"
local mf_version_file="${out_dir}/.microfactory_$(uname)_version"
local soong_ui_bin="${out_dir}/soong_ui"
local from_src=1
if [ -f "${mf_bin}" ] && [ -f "${mf_version_file}" ]; then
if [ "${mf_version}" -eq "$(cat "${mf_version_file}")" ]; then
from_src=0
fi
fi
local mf_cmd
if [ $from_src -eq 1 ]; then
mf_cmd="${GOROOT}/bin/go run ${mf_src}/microfactory.go"
else
mf_cmd="${mf_bin}"
fi
${mf_cmd} -s "${mf_src}" -b "${mf_bin}" \
-pkg-path "android/soong=${TOP}/build/soong" -trimpath "${TOP}/build/soong" \
-o "${soong_ui_bin}" android/soong/cmd/soong_ui
if [ $from_src -eq 1 ]; then
echo "${mf_version}" >"${mf_version_file}"
fi
exec "${out_dir}/soong_ui" "$@"
}
export TOP=$(gettop)
case $(uname) in
Linux)
export GOROOT="${TOP}/prebuilts/go/linux-x86/"
;;
Darwin)
export GOROOT="${TOP}/prebuilts/go/darwin-x86/"
;;
*) echo "unknown OS:" $(uname) >&2 && exit 1;;
esac
source build/soong/cmd/microfactory/microfactory.bash
run_go "$@"
build_go soong_ui android/soong/cmd/soong_ui
exec "$(getoutdir)/soong_ui" "$@"

View File

@@ -102,3 +102,7 @@ func (c ContextImpl) IsTerminal() bool {
}
return false
}
func (c ContextImpl) TermWidth() (int, bool) {
return termWidth(c.Stdout())
}