219 lines
6.1 KiB
Go
219 lines
6.1 KiB
Go
// Copyright 2018 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.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"flag"
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strings"
|
|
|
|
"android/soong/android"
|
|
"android/soong/dexpreopt"
|
|
|
|
"github.com/google/blueprint/pathtools"
|
|
)
|
|
|
|
var (
|
|
dexpreoptScriptPath = flag.String("dexpreopt_script", "", "path to output dexpreopt script")
|
|
globalSoongConfigPath = flag.String("global_soong", "", "path to global configuration file for settings originating from Soong")
|
|
globalConfigPath = flag.String("global", "", "path to global configuration file")
|
|
moduleConfigPath = flag.String("module", "", "path to module configuration file")
|
|
outDir = flag.String("out_dir", "", "path to output directory")
|
|
)
|
|
|
|
type pathContext struct {
|
|
config android.Config
|
|
}
|
|
|
|
func (x *pathContext) Config() android.Config { return x.config }
|
|
func (x *pathContext) AddNinjaFileDeps(...string) {}
|
|
|
|
func main() {
|
|
flag.Parse()
|
|
|
|
usage := func(err string) {
|
|
if err != "" {
|
|
fmt.Println(err)
|
|
flag.Usage()
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
if flag.NArg() > 0 {
|
|
usage("unrecognized argument " + flag.Arg(0))
|
|
}
|
|
|
|
if *dexpreoptScriptPath == "" {
|
|
usage("path to output dexpreopt script is required")
|
|
}
|
|
|
|
if *globalSoongConfigPath == "" {
|
|
usage("--global_soong configuration file is required")
|
|
}
|
|
|
|
if *globalConfigPath == "" {
|
|
usage("--global configuration file is required")
|
|
}
|
|
|
|
if *moduleConfigPath == "" {
|
|
usage("--module configuration file is required")
|
|
}
|
|
|
|
ctx := &pathContext{android.NullConfig(*outDir)}
|
|
|
|
globalSoongConfigData, err := ioutil.ReadFile(*globalSoongConfigPath)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error reading global Soong config %q: %s\n", *globalSoongConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
globalSoongConfig, err := dexpreopt.ParseGlobalSoongConfig(ctx, globalSoongConfigData)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error parsing global Soong config %q: %s\n", *globalSoongConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
globalConfigData, err := ioutil.ReadFile(*globalConfigPath)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error reading global config %q: %s\n", *globalConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
globalConfig, err := dexpreopt.ParseGlobalConfig(ctx, globalConfigData)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error parsing global config %q: %s\n", *globalConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
moduleConfigData, err := ioutil.ReadFile(*moduleConfigPath)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error reading module config %q: %s\n", *moduleConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
moduleConfig, err := dexpreopt.ParseModuleConfig(ctx, moduleConfigData)
|
|
if err != nil {
|
|
fmt.Fprintf(os.Stderr, "error parsing module config %q: %s\n", *moduleConfigPath, err)
|
|
os.Exit(2)
|
|
}
|
|
|
|
moduleConfig.DexPath = android.PathForTesting("$1")
|
|
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
switch x := r.(type) {
|
|
case runtime.Error:
|
|
panic(x)
|
|
case error:
|
|
fmt.Fprintln(os.Stderr, "error:", r)
|
|
os.Exit(3)
|
|
default:
|
|
panic(x)
|
|
}
|
|
}
|
|
}()
|
|
|
|
writeScripts(ctx, globalSoongConfig, globalConfig, moduleConfig, *dexpreoptScriptPath)
|
|
}
|
|
|
|
func writeScripts(ctx android.PathContext, globalSoong *dexpreopt.GlobalSoongConfig,
|
|
global *dexpreopt.GlobalConfig, module *dexpreopt.ModuleConfig, dexpreoptScriptPath string) {
|
|
dexpreoptRule, err := dexpreopt.GenerateDexpreoptRule(ctx, globalSoong, global, module)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
installDir := module.BuildPath.InSameDir(ctx, "dexpreopt_install")
|
|
|
|
dexpreoptRule.Command().FlagWithArg("rm -rf ", installDir.String())
|
|
dexpreoptRule.Command().FlagWithArg("mkdir -p ", installDir.String())
|
|
|
|
for _, install := range dexpreoptRule.Installs() {
|
|
installPath := installDir.Join(ctx, strings.TrimPrefix(install.To, "/"))
|
|
dexpreoptRule.Command().Text("mkdir -p").Flag(filepath.Dir(installPath.String()))
|
|
dexpreoptRule.Command().Text("cp -f").Input(install.From).Output(installPath)
|
|
}
|
|
dexpreoptRule.Command().Tool(globalSoong.SoongZip).
|
|
FlagWithArg("-o ", "$2").
|
|
FlagWithArg("-C ", installDir.String()).
|
|
FlagWithArg("-D ", installDir.String())
|
|
|
|
write := func(rule *android.RuleBuilder, file string) {
|
|
script := &bytes.Buffer{}
|
|
script.WriteString(scriptHeader)
|
|
for _, c := range rule.Commands() {
|
|
script.WriteString(c)
|
|
script.WriteString("\n\n")
|
|
}
|
|
|
|
depFile := &bytes.Buffer{}
|
|
|
|
fmt.Fprint(depFile, `: \`+"\n")
|
|
for _, tool := range rule.Tools() {
|
|
fmt.Fprintf(depFile, ` %s \`+"\n", tool)
|
|
}
|
|
for _, input := range rule.Inputs() {
|
|
// Assume the rule that ran the script already has a dependency on the input file passed on the
|
|
// command line.
|
|
if input.String() != "$1" {
|
|
fmt.Fprintf(depFile, ` %s \`+"\n", input)
|
|
}
|
|
}
|
|
depFile.WriteString("\n")
|
|
|
|
fmt.Fprintln(script, "rm -f $2.d")
|
|
// Write the output path unescaped so the $2 gets expanded
|
|
fmt.Fprintln(script, `echo -n $2 > $2.d`)
|
|
// Write the rest of the depsfile using cat <<'EOF', which will not do any shell expansion on
|
|
// the contents to preserve backslashes and special characters in filenames.
|
|
fmt.Fprintf(script, "cat >> $2.d <<'EOF'\n%sEOF\n", depFile.String())
|
|
|
|
err := pathtools.WriteFileIfChanged(file, script.Bytes(), 0755)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// The written scripts will assume the input is $1 and the output is $2
|
|
if module.DexPath.String() != "$1" {
|
|
panic(fmt.Errorf("module.DexPath must be '$1', was %q", module.DexPath))
|
|
}
|
|
|
|
write(dexpreoptRule, dexpreoptScriptPath)
|
|
}
|
|
|
|
const scriptHeader = `#!/bin/bash
|
|
|
|
err() {
|
|
errno=$?
|
|
echo "error: $0:$1 exited with status $errno" >&2
|
|
echo "error in command:" >&2
|
|
sed -n -e "$1p" $0 >&2
|
|
if [ "$errno" -ne 0 ]; then
|
|
exit $errno
|
|
else
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
trap 'err $LINENO' ERR
|
|
|
|
`
|