blob: 8033b48e5cfcc2cb2f64037aca44c129f54a969b [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
Colin Crossf1a035e2020-11-16 17:32:30 -080030 "github.com/google/blueprint"
Colin Cross43f08db2018-11-12 10:13:39 -080031 "github.com/google/blueprint/pathtools"
32)
33
34var (
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000035 dexpreoptScriptPath = flag.String("dexpreopt_script", "", "path to output dexpreopt script")
36 globalSoongConfigPath = flag.String("global_soong", "", "path to global configuration file for settings originating from Soong")
37 globalConfigPath = flag.String("global", "", "path to global configuration file")
38 moduleConfigPath = flag.String("module", "", "path to module configuration file")
39 outDir = flag.String("out_dir", "", "path to output directory")
Jeongik Cha4dda75e2021-04-27 23:56:44 +090040 // If uses_target_files is true, dexpreopt_gen will be running on extracted target_files.zip files.
41 // In this case, the tool replace output file path with $(basePath)/$(on-device file path).
42 // The flag is useful when running dex2oat on system image and vendor image which are built separately.
43 usesTargetFiles = flag.Bool("uses_target_files", false, "whether or not dexpreopt is running on target_files")
44 // basePath indicates the path where target_files.zip is extracted.
Jiakai Zhanga4496782023-05-17 16:57:30 +010045 basePath = flag.String("base_path", ".", "base path where images and tools are extracted")
46 productPackagesPath = flag.String("product_packages", "", "path to product_packages.txt")
Colin Cross43f08db2018-11-12 10:13:39 -080047)
48
Colin Crossf1a035e2020-11-16 17:32:30 -080049type builderContext struct {
Colin Cross69f59a32019-02-15 10:39:37 -080050 config android.Config
51}
52
Colin Crossf1a035e2020-11-16 17:32:30 -080053func (x *builderContext) Config() android.Config { return x.config }
54func (x *builderContext) AddNinjaFileDeps(...string) {}
55func (x *builderContext) Build(android.PackageContext, android.BuildParams) {}
56func (x *builderContext) Rule(android.PackageContext, string, blueprint.RuleParams, ...string) blueprint.Rule {
57 return nil
58}
Colin Cross69f59a32019-02-15 10:39:37 -080059
Colin Cross43f08db2018-11-12 10:13:39 -080060func main() {
61 flag.Parse()
62
63 usage := func(err string) {
64 if err != "" {
65 fmt.Println(err)
66 flag.Usage()
67 os.Exit(1)
68 }
69 }
70
71 if flag.NArg() > 0 {
72 usage("unrecognized argument " + flag.Arg(0))
73 }
74
75 if *dexpreoptScriptPath == "" {
76 usage("path to output dexpreopt script is required")
77 }
78
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000079 if *globalSoongConfigPath == "" {
80 usage("--global_soong configuration file is required")
81 }
82
Colin Cross43f08db2018-11-12 10:13:39 -080083 if *globalConfigPath == "" {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000084 usage("--global configuration file is required")
Colin Cross43f08db2018-11-12 10:13:39 -080085 }
86
87 if *moduleConfigPath == "" {
Martin Stjernholmc52aaf12020-01-06 23:11:37 +000088 usage("--module configuration file is required")
Colin Cross43f08db2018-11-12 10:13:39 -080089 }
90
Jiakai Zhanga4496782023-05-17 16:57:30 +010091 if *productPackagesPath == "" {
92 usage("--product_packages configuration file is required")
93 }
94
Lukacs T. Berkid6cee7e2021-09-01 16:25:51 +020095 // NOTE: duplicating --out_dir here is incorrect (one should be the another
96 // plus "/soong" but doing so apparently breaks dexpreopt
97 ctx := &builderContext{android.NullConfig(*outDir, *outDir)}
Colin Cross69f59a32019-02-15 10:39:37 -080098
Colin Cross988414c2020-01-11 01:11:46 +000099 globalSoongConfigData, err := ioutil.ReadFile(*globalSoongConfigPath)
100 if err != nil {
Martin Stjernholm75a48d82020-01-10 20:32:59 +0000101 fmt.Fprintf(os.Stderr, "error reading global Soong config %q: %s\n", *globalSoongConfigPath, err)
Colin Cross988414c2020-01-11 01:11:46 +0000102 os.Exit(2)
103 }
104
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000105 globalSoongConfig, err := dexpreopt.ParseGlobalSoongConfig(ctx, globalSoongConfigData)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000106 if err != nil {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000107 fmt.Fprintf(os.Stderr, "error parsing global Soong config %q: %s\n", *globalSoongConfigPath, err)
Martin Stjernholmc52aaf12020-01-06 23:11:37 +0000108 os.Exit(2)
109 }
110
Colin Cross988414c2020-01-11 01:11:46 +0000111 globalConfigData, err := ioutil.ReadFile(*globalConfigPath)
Colin Cross43f08db2018-11-12 10:13:39 -0800112 if err != nil {
Colin Cross988414c2020-01-11 01:11:46 +0000113 fmt.Fprintf(os.Stderr, "error reading global config %q: %s\n", *globalConfigPath, err)
Colin Cross43f08db2018-11-12 10:13:39 -0800114 os.Exit(2)
115 }
116
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000117 globalConfig, err := dexpreopt.ParseGlobalConfig(ctx, globalConfigData)
Colin Cross988414c2020-01-11 01:11:46 +0000118 if err != nil {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000119 fmt.Fprintf(os.Stderr, "error parsing global config %q: %s\n", *globalConfigPath, err)
Colin Cross988414c2020-01-11 01:11:46 +0000120 os.Exit(2)
121 }
122
123 moduleConfigData, err := ioutil.ReadFile(*moduleConfigPath)
124 if err != nil {
125 fmt.Fprintf(os.Stderr, "error reading module config %q: %s\n", *moduleConfigPath, err)
126 os.Exit(2)
127 }
128
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000129 moduleConfig, err := dexpreopt.ParseModuleConfig(ctx, moduleConfigData)
Colin Cross43f08db2018-11-12 10:13:39 -0800130 if err != nil {
Martin Stjernholm40f9f3c2020-01-20 18:12:23 +0000131 fmt.Fprintf(os.Stderr, "error parsing module config %q: %s\n", *moduleConfigPath, err)
Colin Cross43f08db2018-11-12 10:13:39 -0800132 os.Exit(2)
133 }
134
Colin Cross69f59a32019-02-15 10:39:37 -0800135 moduleConfig.DexPath = android.PathForTesting("$1")
136
Colin Cross43f08db2018-11-12 10:13:39 -0800137 defer func() {
138 if r := recover(); r != nil {
139 switch x := r.(type) {
140 case runtime.Error:
141 panic(x)
142 case error:
143 fmt.Fprintln(os.Stderr, "error:", r)
144 os.Exit(3)
145 default:
146 panic(x)
147 }
148 }
149 }()
Jeongik Cha4dda75e2021-04-27 23:56:44 +0900150 if *usesTargetFiles {
151 moduleConfig.ManifestPath = android.OptionalPath{}
152 prefix := "dex2oat_result"
153 moduleConfig.BuildPath = android.PathForOutput(ctx, filepath.Join(prefix, moduleConfig.DexLocation))
154 for i, location := range moduleConfig.PreoptBootClassPathDexLocations {
155 moduleConfig.PreoptBootClassPathDexFiles[i] = android.PathForSource(ctx, *basePath+location)
156 }
157 for i := range moduleConfig.ClassLoaderContexts {
158 for _, v := range moduleConfig.ClassLoaderContexts[i] {
159 v.Host = android.PathForSource(ctx, *basePath+v.Device)
160 }
161 }
162 moduleConfig.EnforceUsesLibraries = false
163 for i, location := range moduleConfig.DexPreoptImageLocationsOnDevice {
164 moduleConfig.DexPreoptImageLocationsOnHost[i] = *basePath + location
165 }
166 }
Jiakai Zhanga4496782023-05-17 16:57:30 +0100167 writeScripts(ctx, globalSoongConfig, globalConfig, moduleConfig, *dexpreoptScriptPath, *productPackagesPath)
Colin Cross43f08db2018-11-12 10:13:39 -0800168}
169
Colin Crossf1a035e2020-11-16 17:32:30 -0800170func writeScripts(ctx android.BuilderContext, globalSoong *dexpreopt.GlobalSoongConfig,
Jiakai Zhanga4496782023-05-17 16:57:30 +0100171 global *dexpreopt.GlobalConfig, module *dexpreopt.ModuleConfig, dexpreoptScriptPath string,
172 productPackagesPath string) {
Colin Crossfeec25b2019-01-30 17:32:39 -0800173 write := func(rule *android.RuleBuilder, file string) {
Colin Cross43f08db2018-11-12 10:13:39 -0800174 script := &bytes.Buffer{}
175 script.WriteString(scriptHeader)
176 for _, c := range rule.Commands() {
177 script.WriteString(c)
178 script.WriteString("\n\n")
179 }
180
181 depFile := &bytes.Buffer{}
182
183 fmt.Fprint(depFile, `: \`+"\n")
Colin Cross25397f52019-02-15 16:03:58 -0800184 for _, tool := range rule.Tools() {
Colin Cross43f08db2018-11-12 10:13:39 -0800185 fmt.Fprintf(depFile, ` %s \`+"\n", tool)
186 }
Colin Cross25397f52019-02-15 16:03:58 -0800187 for _, input := range rule.Inputs() {
Colin Cross43f08db2018-11-12 10:13:39 -0800188 // Assume the rule that ran the script already has a dependency on the input file passed on the
189 // command line.
Colin Cross69f59a32019-02-15 10:39:37 -0800190 if input.String() != "$1" {
Colin Cross43f08db2018-11-12 10:13:39 -0800191 fmt.Fprintf(depFile, ` %s \`+"\n", input)
192 }
193 }
194 depFile.WriteString("\n")
195
196 fmt.Fprintln(script, "rm -f $2.d")
197 // Write the output path unescaped so the $2 gets expanded
198 fmt.Fprintln(script, `echo -n $2 > $2.d`)
199 // Write the rest of the depsfile using cat <<'EOF', which will not do any shell expansion on
200 // the contents to preserve backslashes and special characters in filenames.
201 fmt.Fprintf(script, "cat >> $2.d <<'EOF'\n%sEOF\n", depFile.String())
202
203 err := pathtools.WriteFileIfChanged(file, script.Bytes(), 0755)
204 if err != nil {
205 panic(err)
206 }
207 }
Jiakai Zhanga4496782023-05-17 16:57:30 +0100208 dexpreoptRule, err := dexpreopt.GenerateDexpreoptRule(
209 ctx, globalSoong, global, module, android.PathForTesting(productPackagesPath))
Jeongik Cha4dda75e2021-04-27 23:56:44 +0900210 if err != nil {
211 panic(err)
212 }
213 // When usesTargetFiles is true, only odex/vdex files are necessary.
214 // So skip redunant processes(such as copying the result to the artifact path, and zipping, and so on.)
215 if *usesTargetFiles {
216 write(dexpreoptRule, dexpreoptScriptPath)
217 return
218 }
219 installDir := module.BuildPath.InSameDir(ctx, "dexpreopt_install")
220
221 dexpreoptRule.Command().FlagWithArg("rm -rf ", installDir.String())
222 dexpreoptRule.Command().FlagWithArg("mkdir -p ", installDir.String())
223
224 for _, install := range dexpreoptRule.Installs() {
225 installPath := installDir.Join(ctx, strings.TrimPrefix(install.To, "/"))
226 dexpreoptRule.Command().Text("mkdir -p").Flag(filepath.Dir(installPath.String()))
227 dexpreoptRule.Command().Text("cp -f").Input(install.From).Output(installPath)
228 }
229 dexpreoptRule.Command().Tool(globalSoong.SoongZip).
230 FlagWithArg("-o ", "$2").
231 FlagWithArg("-C ", installDir.String()).
232 FlagWithArg("-D ", installDir.String())
Colin Cross43f08db2018-11-12 10:13:39 -0800233
234 // The written scripts will assume the input is $1 and the output is $2
Colin Cross69f59a32019-02-15 10:39:37 -0800235 if module.DexPath.String() != "$1" {
Colin Cross43f08db2018-11-12 10:13:39 -0800236 panic(fmt.Errorf("module.DexPath must be '$1', was %q", module.DexPath))
237 }
Colin Cross43f08db2018-11-12 10:13:39 -0800238
239 write(dexpreoptRule, dexpreoptScriptPath)
Colin Cross43f08db2018-11-12 10:13:39 -0800240}
241
242const scriptHeader = `#!/bin/bash
243
244err() {
245 errno=$?
246 echo "error: $0:$1 exited with status $errno" >&2
247 echo "error in command:" >&2
248 sed -n -e "$1p" $0 >&2
249 if [ "$errno" -ne 0 ]; then
250 exit $errno
251 else
252 exit 1
253 fi
254}
255
256trap 'err $LINENO' ERR
257
258`