blob: e2818bb6188405031f46e5f16613cf25f58babc5 [file] [log] [blame]
Colin Cross43f08db2018-11-12 10:13:39 -08001// Copyright 2018 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package main
16
17import (
18 "bytes"
19 "flag"
20 "fmt"
Colin Cross988414c2020-01-11 01:11:46 +000021 "io/ioutil"
Colin Cross43f08db2018-11-12 10:13:39 -080022 "os"
23 "path/filepath"
24 "runtime"
Colin Cross69f59a32019-02-15 10:39:37 -080025 "strings"
Colin Cross43f08db2018-11-12 10:13:39 -080026
Colin Crossfeec25b2019-01-30 17:32:39 -080027 "android/soong/android"
Colin Cross43f08db2018-11-12 10:13:39 -080028 "android/soong/dexpreopt"
29
30 "github.com/google/blueprint/pathtools"
31)
32
33var (
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000034 dexpreoptScriptPath = flag.String("dexpreopt_script", "", "path to output dexpreopt script")
35 globalSoongConfigPath = flag.String("global_soong", "", "path to global configuration file for settings originating from Soong")
36 globalConfigPath = flag.String("global", "", "path to global configuration file")
37 moduleConfigPath = flag.String("module", "", "path to module configuration file")
38 outDir = flag.String("out_dir", "", "path to output directory")
Colin Cross43f08db2018-11-12 10:13:39 -080039)
40
Colin Cross69f59a32019-02-15 10:39:37 -080041type pathContext struct {
42 config android.Config
43}
44
Colin Cross69f59a32019-02-15 10:39:37 -080045func (x *pathContext) Config() android.Config { return x.config }
46func (x *pathContext) AddNinjaFileDeps(...string) {}
47
Colin Cross43f08db2018-11-12 10:13:39 -080048func main() {
49 flag.Parse()
50
51 usage := func(err string) {
52 if err != "" {
53 fmt.Println(err)
54 flag.Usage()
55 os.Exit(1)
56 }
57 }
58
59 if flag.NArg() > 0 {
60 usage("unrecognized argument " + flag.Arg(0))
61 }
62
63 if *dexpreoptScriptPath == "" {
64 usage("path to output dexpreopt script is required")
65 }
66
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000067 if *globalSoongConfigPath == "" {
68 usage("--global_soong configuration file is required")
69 }
70
Colin Cross43f08db2018-11-12 10:13:39 -080071 if *globalConfigPath == "" {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000072 usage("--global configuration file is required")
Colin Cross43f08db2018-11-12 10:13:39 -080073 }
74
75 if *moduleConfigPath == "" {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000076 usage("--module configuration file is required")
Colin Cross43f08db2018-11-12 10:13:39 -080077 }
78
Colin Cross988414c2020-01-11 01:11:46 +000079 ctx := &pathContext{android.NullConfig(*outDir)}
Colin Cross69f59a32019-02-15 10:39:37 -080080
Colin Cross988414c2020-01-11 01:11:46 +000081 globalSoongConfigData, err := ioutil.ReadFile(*globalSoongConfigPath)
82 if err != nil {
Hans Boehme4b53422020-01-25 01:44:30 +000083 fmt.Fprintf(os.Stderr, "error reading global config %q: %s\n", *globalSoongConfigPath, err)
Colin Cross988414c2020-01-11 01:11:46 +000084 os.Exit(2)
85 }
86
Hans Boehm453bf092020-01-25 01:44:30 +000087 globalSoongConfig, err := dexpreopt.LoadGlobalSoongConfig(ctx, globalSoongConfigData)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000088 if err != nil {
Hans Boehme4b53422020-01-25 01:44:30 +000089 fmt.Fprintf(os.Stderr, "error loading global config %q: %s\n", *globalSoongConfigPath, err)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000090 os.Exit(2)
91 }
92
Colin Cross988414c2020-01-11 01:11:46 +000093 globalConfigData, err := ioutil.ReadFile(*globalConfigPath)
Colin Cross43f08db2018-11-12 10:13:39 -080094 if err != nil {
Colin Cross988414c2020-01-11 01:11:46 +000095 fmt.Fprintf(os.Stderr, "error reading global config %q: %s\n", *globalConfigPath, err)
Colin Cross43f08db2018-11-12 10:13:39 -080096 os.Exit(2)
97 }
98
Hans Boehme4b53422020-01-25 01:44:30 +000099 globalConfig, err := dexpreopt.LoadGlobalConfig(ctx, globalConfigData, globalSoongConfig)
Colin Cross988414c2020-01-11 01:11:46 +0000100 if err != nil {
Hans Boehme4b53422020-01-25 01:44:30 +0000101 fmt.Fprintf(os.Stderr, "error parse global config %q: %s\n", *globalConfigPath, err)
Colin Cross988414c2020-01-11 01:11:46 +0000102 os.Exit(2)
103 }
104
105 moduleConfigData, err := ioutil.ReadFile(*moduleConfigPath)
106 if err != nil {
107 fmt.Fprintf(os.Stderr, "error reading module config %q: %s\n", *moduleConfigPath, err)
108 os.Exit(2)
109 }
110
Hans Boehm453bf092020-01-25 01:44:30 +0000111 moduleConfig, err := dexpreopt.LoadModuleConfig(ctx, moduleConfigData)
Colin Cross43f08db2018-11-12 10:13:39 -0800112 if err != nil {
Hans Boehm453bf092020-01-25 01:44:30 +0000113 fmt.Fprintf(os.Stderr, "error loading module config %q: %s\n", *moduleConfigPath, err)
Colin Cross43f08db2018-11-12 10:13:39 -0800114 os.Exit(2)
115 }
116
Colin Cross69f59a32019-02-15 10:39:37 -0800117 moduleConfig.DexPath = android.PathForTesting("$1")
118
Colin Cross43f08db2018-11-12 10:13:39 -0800119 defer func() {
120 if r := recover(); r != nil {
121 switch x := r.(type) {
122 case runtime.Error:
123 panic(x)
124 case error:
125 fmt.Fprintln(os.Stderr, "error:", r)
126 os.Exit(3)
127 default:
128 panic(x)
129 }
130 }
131 }()
132
Hans Boehme4b53422020-01-25 01:44:30 +0000133 writeScripts(ctx, globalConfig, moduleConfig, *dexpreoptScriptPath)
Colin Cross43f08db2018-11-12 10:13:39 -0800134}
135
Hans Boehme4b53422020-01-25 01:44:30 +0000136func writeScripts(ctx android.PathContext, global dexpreopt.GlobalConfig, module dexpreopt.ModuleConfig,
137 dexpreoptScriptPath string) {
138 dexpreoptRule, err := dexpreopt.GenerateDexpreoptRule(ctx, global, module)
Colin Cross43f08db2018-11-12 10:13:39 -0800139 if err != nil {
140 panic(err)
141 }
142
Colin Cross69f59a32019-02-15 10:39:37 -0800143 installDir := module.BuildPath.InSameDir(ctx, "dexpreopt_install")
Colin Cross43f08db2018-11-12 10:13:39 -0800144
Colin Cross69f59a32019-02-15 10:39:37 -0800145 dexpreoptRule.Command().FlagWithArg("rm -rf ", installDir.String())
146 dexpreoptRule.Command().FlagWithArg("mkdir -p ", installDir.String())
Colin Cross43f08db2018-11-12 10:13:39 -0800147
148 for _, install := range dexpreoptRule.Installs() {
Colin Cross69f59a32019-02-15 10:39:37 -0800149 installPath := installDir.Join(ctx, strings.TrimPrefix(install.To, "/"))
150 dexpreoptRule.Command().Text("mkdir -p").Flag(filepath.Dir(installPath.String()))
Colin Cross43f08db2018-11-12 10:13:39 -0800151 dexpreoptRule.Command().Text("cp -f").Input(install.From).Output(installPath)
152 }
Hans Boehme4b53422020-01-25 01:44:30 +0000153 dexpreoptRule.Command().Tool(global.SoongConfig.SoongZip).
Colin Cross69f59a32019-02-15 10:39:37 -0800154 FlagWithArg("-o ", "$2").
155 FlagWithArg("-C ", installDir.String()).
156 FlagWithArg("-D ", installDir.String())
Colin Cross43f08db2018-11-12 10:13:39 -0800157
Colin Crossfeec25b2019-01-30 17:32:39 -0800158 write := func(rule *android.RuleBuilder, file string) {
Colin Cross43f08db2018-11-12 10:13:39 -0800159 script := &bytes.Buffer{}
160 script.WriteString(scriptHeader)
161 for _, c := range rule.Commands() {
162 script.WriteString(c)
163 script.WriteString("\n\n")
164 }
165
166 depFile := &bytes.Buffer{}
167
168 fmt.Fprint(depFile, `: \`+"\n")
Colin Cross25397f52019-02-15 16:03:58 -0800169 for _, tool := range rule.Tools() {
Colin Cross43f08db2018-11-12 10:13:39 -0800170 fmt.Fprintf(depFile, ` %s \`+"\n", tool)
171 }
Colin Cross25397f52019-02-15 16:03:58 -0800172 for _, input := range rule.Inputs() {
Colin Cross43f08db2018-11-12 10:13:39 -0800173 // Assume the rule that ran the script already has a dependency on the input file passed on the
174 // command line.
Colin Cross69f59a32019-02-15 10:39:37 -0800175 if input.String() != "$1" {
Colin Cross43f08db2018-11-12 10:13:39 -0800176 fmt.Fprintf(depFile, ` %s \`+"\n", input)
177 }
178 }
179 depFile.WriteString("\n")
180
181 fmt.Fprintln(script, "rm -f $2.d")
182 // Write the output path unescaped so the $2 gets expanded
183 fmt.Fprintln(script, `echo -n $2 > $2.d`)
184 // Write the rest of the depsfile using cat <<'EOF', which will not do any shell expansion on
185 // the contents to preserve backslashes and special characters in filenames.
186 fmt.Fprintf(script, "cat >> $2.d <<'EOF'\n%sEOF\n", depFile.String())
187
188 err := pathtools.WriteFileIfChanged(file, script.Bytes(), 0755)
189 if err != nil {
190 panic(err)
191 }
192 }
193
194 // The written scripts will assume the input is $1 and the output is $2
Colin Cross69f59a32019-02-15 10:39:37 -0800195 if module.DexPath.String() != "$1" {
Colin Cross43f08db2018-11-12 10:13:39 -0800196 panic(fmt.Errorf("module.DexPath must be '$1', was %q", module.DexPath))
197 }
Colin Cross43f08db2018-11-12 10:13:39 -0800198
199 write(dexpreoptRule, dexpreoptScriptPath)
Colin Cross43f08db2018-11-12 10:13:39 -0800200}
201
202const scriptHeader = `#!/bin/bash
203
204err() {
205 errno=$?
206 echo "error: $0:$1 exited with status $errno" >&2
207 echo "error in command:" >&2
208 sed -n -e "$1p" $0 >&2
209 if [ "$errno" -ne 0 ]; then
210 exit $errno
211 else
212 exit 1
213 fi
214}
215
216trap 'err $LINENO' ERR
217
218`