Instead of just depending on the main .art file (boot.art, etc), also expose the dependencies to the .oat/.vdex files (boot.oat/boot.vdex), and all of the module files that get implicitly loading (boot-ext.*, boot-framework.*, etc) This is necessary for RBE, where the rule only gets the files that it depends upon. Test: treehugger Test: build a system image with RBE Change-Id: I0c7051f18582f1891d3398b46763b1521e4326c8
		
			
				
	
	
		
			184 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			184 lines
		
	
	
		
			6.3 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 dexpreopt
 | |
| 
 | |
| import (
 | |
| 	"android/soong/android"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| )
 | |
| 
 | |
| func testModuleConfig(ctx android.PathContext) ModuleConfig {
 | |
| 	return ModuleConfig{
 | |
| 		Name:                            "test",
 | |
| 		DexLocation:                     "/system/app/test/test.apk",
 | |
| 		BuildPath:                       android.PathForOutput(ctx, "test/test.apk"),
 | |
| 		DexPath:                         android.PathForOutput(ctx, "test/dex/test.jar"),
 | |
| 		UncompressedDex:                 false,
 | |
| 		HasApkLibraries:                 false,
 | |
| 		PreoptFlags:                     nil,
 | |
| 		ProfileClassListing:             android.OptionalPath{},
 | |
| 		ProfileIsTextListing:            false,
 | |
| 		EnforceUsesLibraries:            false,
 | |
| 		PresentOptionalUsesLibraries:    nil,
 | |
| 		UsesLibraries:                   nil,
 | |
| 		LibraryPaths:                    nil,
 | |
| 		Archs:                           []android.ArchType{android.Arm},
 | |
| 		DexPreoptImages:                 android.Paths{android.PathForTesting("system/framework/arm/boot.art")},
 | |
| 		DexPreoptImagesDeps:             []android.Paths{android.Paths{}},
 | |
| 		PreoptBootClassPathDexFiles:     nil,
 | |
| 		PreoptBootClassPathDexLocations: nil,
 | |
| 		PreoptExtractedApk:              false,
 | |
| 		NoCreateAppImage:                false,
 | |
| 		ForceCreateAppImage:             false,
 | |
| 		PresignedPrebuilt:               false,
 | |
| 		NoStripping:                     false,
 | |
| 		StripInputPath:                  android.PathForOutput(ctx, "unstripped/test.apk"),
 | |
| 		StripOutputPath:                 android.PathForOutput(ctx, "stripped/test.apk"),
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDexPreopt(t *testing.T) {
 | |
| 	ctx := android.PathContextForTesting(android.TestConfig("out", nil), nil)
 | |
| 	global, module := GlobalConfigForTests(ctx), testModuleConfig(ctx)
 | |
| 
 | |
| 	rule, err := GenerateDexpreoptRule(ctx, global, module)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	wantInstalls := android.RuleBuilderInstalls{
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.odex"), "/system/app/test/oat/arm/test.odex"},
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.vdex"), "/system/app/test/oat/arm/test.vdex"},
 | |
| 	}
 | |
| 
 | |
| 	if rule.Installs().String() != wantInstalls.String() {
 | |
| 		t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs())
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDexPreoptStrip(t *testing.T) {
 | |
| 	// Test that we panic if we strip in a configuration where stripping is not allowed.
 | |
| 	ctx := android.PathContextForTesting(android.TestConfig("out", nil), nil)
 | |
| 	global, module := GlobalConfigForTests(ctx), testModuleConfig(ctx)
 | |
| 
 | |
| 	global.NeverAllowStripping = true
 | |
| 	module.NoStripping = false
 | |
| 
 | |
| 	_, err := GenerateStripRule(global, module)
 | |
| 	if err == nil {
 | |
| 		t.Errorf("Expected an error when calling GenerateStripRule on a stripped module")
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDexPreoptSystemOther(t *testing.T) {
 | |
| 	ctx := android.PathContextForTesting(android.TestConfig("out", nil), nil)
 | |
| 	global, module := GlobalConfigForTests(ctx), testModuleConfig(ctx)
 | |
| 
 | |
| 	global.HasSystemOther = true
 | |
| 	global.PatternsOnSystemOther = []string{"app/%"}
 | |
| 
 | |
| 	rule, err := GenerateDexpreoptRule(ctx, global, module)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	wantInstalls := android.RuleBuilderInstalls{
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.odex"), "/system_other/app/test/oat/arm/test.odex"},
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.vdex"), "/system_other/app/test/oat/arm/test.vdex"},
 | |
| 	}
 | |
| 
 | |
| 	if rule.Installs().String() != wantInstalls.String() {
 | |
| 		t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs())
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDexPreoptProfile(t *testing.T) {
 | |
| 	ctx := android.PathContextForTesting(android.TestConfig("out", nil), nil)
 | |
| 	global, module := GlobalConfigForTests(ctx), testModuleConfig(ctx)
 | |
| 
 | |
| 	module.ProfileClassListing = android.OptionalPathForPath(android.PathForTesting("profile"))
 | |
| 
 | |
| 	rule, err := GenerateDexpreoptRule(ctx, global, module)
 | |
| 	if err != nil {
 | |
| 		t.Fatal(err)
 | |
| 	}
 | |
| 
 | |
| 	wantInstalls := android.RuleBuilderInstalls{
 | |
| 		{android.PathForOutput(ctx, "test/profile.prof"), "/system/app/test/test.apk.prof"},
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.art"), "/system/app/test/oat/arm/test.art"},
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.odex"), "/system/app/test/oat/arm/test.odex"},
 | |
| 		{android.PathForOutput(ctx, "test/oat/arm/package.vdex"), "/system/app/test/oat/arm/test.vdex"},
 | |
| 	}
 | |
| 
 | |
| 	if rule.Installs().String() != wantInstalls.String() {
 | |
| 		t.Errorf("\nwant installs:\n   %v\ngot:\n   %v", wantInstalls, rule.Installs())
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestStripDex(t *testing.T) {
 | |
| 	tests := []struct {
 | |
| 		name  string
 | |
| 		setup func(global *GlobalConfig, module *ModuleConfig)
 | |
| 		strip bool
 | |
| 	}{
 | |
| 		{
 | |
| 			name:  "default strip",
 | |
| 			setup: func(global *GlobalConfig, module *ModuleConfig) {},
 | |
| 			strip: true,
 | |
| 		},
 | |
| 		{
 | |
| 			name:  "global no stripping",
 | |
| 			setup: func(global *GlobalConfig, module *ModuleConfig) { global.DefaultNoStripping = true },
 | |
| 			strip: false,
 | |
| 		},
 | |
| 		{
 | |
| 			name:  "module no stripping",
 | |
| 			setup: func(global *GlobalConfig, module *ModuleConfig) { module.NoStripping = true },
 | |
| 			strip: false,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, test := range tests {
 | |
| 		t.Run(test.name, func(t *testing.T) {
 | |
| 
 | |
| 			ctx := android.PathContextForTesting(android.TestConfig("out", nil), nil)
 | |
| 			global, module := GlobalConfigForTests(ctx), testModuleConfig(ctx)
 | |
| 
 | |
| 			test.setup(&global, &module)
 | |
| 
 | |
| 			rule, err := GenerateStripRule(global, module)
 | |
| 			if err != nil {
 | |
| 				t.Fatal(err)
 | |
| 			}
 | |
| 
 | |
| 			if test.strip {
 | |
| 				want := `zip2zip -i out/unstripped/test.apk -o out/stripped/test.apk -x "classes*.dex"`
 | |
| 				if len(rule.Commands()) < 1 || !strings.Contains(rule.Commands()[0], want) {
 | |
| 					t.Errorf("\nwant commands[0] to have:\n   %v\ngot:\n   %v", want, rule.Commands()[0])
 | |
| 				}
 | |
| 			} else {
 | |
| 				wantCommands := []string{
 | |
| 					"cp -f out/unstripped/test.apk out/stripped/test.apk",
 | |
| 				}
 | |
| 				if !reflect.DeepEqual(rule.Commands(), wantCommands) {
 | |
| 					t.Errorf("\nwant commands:\n   %v\ngot:\n   %v", wantCommands, rule.Commands())
 | |
| 				}
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 |