filepath.Join("foo", "") returns a newly allocated copy of "foo",
while filepath.Join("foo") does not.  Strip out any empty path
components before calling filepath.Join.
Test: TestValidatePath
Change-Id: Ib47dbcd9d6463809acfe260dfd9af87ea280b4de
		
	
		
			
				
	
	
		
			1608 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			1608 lines
		
	
	
		
			40 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2015 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 android
 | |
| 
 | |
| import (
 | |
| 	"errors"
 | |
| 	"fmt"
 | |
| 	"reflect"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/google/blueprint/proptools"
 | |
| )
 | |
| 
 | |
| type strsTestCase struct {
 | |
| 	in  []string
 | |
| 	out string
 | |
| 	err []error
 | |
| }
 | |
| 
 | |
| var commonValidatePathTestCases = []strsTestCase{
 | |
| 	{
 | |
| 		in:  []string{""},
 | |
| 		out: "",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"", ""},
 | |
| 		out: "",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a", ""},
 | |
| 		out: "a",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"", "a"},
 | |
| 		out: "a",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"", "a", ""},
 | |
| 		out: "a",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a/b"},
 | |
| 		out: "a/b",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a/b", "c"},
 | |
| 		out: "a/b/c",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a/.."},
 | |
| 		out: ".",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"."},
 | |
| 		out: ".",
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{".."},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ..")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"../a"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ../a")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"b/../../a"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ../a")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"/a"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: /a")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a", "../b"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ../b")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a", "b/../../c"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ../c")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"a", "./.."},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path is outside directory: ..")},
 | |
| 	},
 | |
| }
 | |
| 
 | |
| var validateSafePathTestCases = append(commonValidatePathTestCases, []strsTestCase{
 | |
| 	{
 | |
| 		in:  []string{"$host/../$a"},
 | |
| 		out: "$a",
 | |
| 	},
 | |
| }...)
 | |
| 
 | |
| var validatePathTestCases = append(commonValidatePathTestCases, []strsTestCase{
 | |
| 	{
 | |
| 		in:  []string{"$host/../$a"},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path contains invalid character($): $host/../$a")},
 | |
| 	},
 | |
| 	{
 | |
| 		in:  []string{"$host/.."},
 | |
| 		out: "",
 | |
| 		err: []error{errors.New("Path contains invalid character($): $host/..")},
 | |
| 	},
 | |
| }...)
 | |
| 
 | |
| func TestValidateSafePath(t *testing.T) {
 | |
| 	for _, testCase := range validateSafePathTestCases {
 | |
| 		t.Run(strings.Join(testCase.in, ","), func(t *testing.T) {
 | |
| 			ctx := &configErrorWrapper{}
 | |
| 			out, err := validateSafePath(testCase.in...)
 | |
| 			if err != nil {
 | |
| 				reportPathError(ctx, err)
 | |
| 			}
 | |
| 			check(t, "validateSafePath", p(testCase.in), out, ctx.errors, testCase.out, testCase.err)
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestValidatePath(t *testing.T) {
 | |
| 	for _, testCase := range validatePathTestCases {
 | |
| 		t.Run(strings.Join(testCase.in, ","), func(t *testing.T) {
 | |
| 			ctx := &configErrorWrapper{}
 | |
| 			out, err := validatePath(testCase.in...)
 | |
| 			if err != nil {
 | |
| 				reportPathError(ctx, err)
 | |
| 			}
 | |
| 			check(t, "validatePath", p(testCase.in), out, ctx.errors, testCase.out, testCase.err)
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestOptionalPath(t *testing.T) {
 | |
| 	var path OptionalPath
 | |
| 	checkInvalidOptionalPath(t, path, "unknown")
 | |
| 
 | |
| 	path = OptionalPathForPath(nil)
 | |
| 	checkInvalidOptionalPath(t, path, "unknown")
 | |
| 
 | |
| 	path = InvalidOptionalPath("foo")
 | |
| 	checkInvalidOptionalPath(t, path, "foo")
 | |
| 
 | |
| 	path = InvalidOptionalPath("")
 | |
| 	checkInvalidOptionalPath(t, path, "unknown")
 | |
| 
 | |
| 	path = OptionalPathForPath(PathForTesting("path"))
 | |
| 	checkValidOptionalPath(t, path, "path")
 | |
| }
 | |
| 
 | |
| func checkInvalidOptionalPath(t *testing.T, path OptionalPath, expectedInvalidReason string) {
 | |
| 	t.Helper()
 | |
| 	if path.Valid() {
 | |
| 		t.Errorf("Invalid OptionalPath should not be valid")
 | |
| 	}
 | |
| 	if path.InvalidReason() != expectedInvalidReason {
 | |
| 		t.Errorf("Wrong invalid reason: expected %q, got %q", expectedInvalidReason, path.InvalidReason())
 | |
| 	}
 | |
| 	if path.String() != "" {
 | |
| 		t.Errorf("Invalid OptionalPath String() should return \"\", not %q", path.String())
 | |
| 	}
 | |
| 	paths := path.AsPaths()
 | |
| 	if len(paths) != 0 {
 | |
| 		t.Errorf("Invalid OptionalPath AsPaths() should return empty Paths, not %q", paths)
 | |
| 	}
 | |
| 	defer func() {
 | |
| 		if r := recover(); r == nil {
 | |
| 			t.Errorf("Expected a panic when calling Path() on an uninitialized OptionalPath")
 | |
| 		}
 | |
| 	}()
 | |
| 	path.Path()
 | |
| }
 | |
| 
 | |
| func checkValidOptionalPath(t *testing.T, path OptionalPath, expectedString string) {
 | |
| 	t.Helper()
 | |
| 	if !path.Valid() {
 | |
| 		t.Errorf("Initialized OptionalPath should not be invalid")
 | |
| 	}
 | |
| 	if path.InvalidReason() != "" {
 | |
| 		t.Errorf("Initialized OptionalPath should not have an invalid reason, got: %q", path.InvalidReason())
 | |
| 	}
 | |
| 	if path.String() != expectedString {
 | |
| 		t.Errorf("Initialized OptionalPath String() should return %q, not %q", expectedString, path.String())
 | |
| 	}
 | |
| 	paths := path.AsPaths()
 | |
| 	if len(paths) != 1 {
 | |
| 		t.Errorf("Initialized OptionalPath AsPaths() should return Paths with length 1, not %q", paths)
 | |
| 	}
 | |
| 	path.Path()
 | |
| }
 | |
| 
 | |
| func check(t *testing.T, testType, testString string,
 | |
| 	got interface{}, err []error,
 | |
| 	expected interface{}, expectedErr []error) {
 | |
| 	t.Helper()
 | |
| 
 | |
| 	printedTestCase := false
 | |
| 	e := func(s string, expected, got interface{}) {
 | |
| 		t.Helper()
 | |
| 		if !printedTestCase {
 | |
| 			t.Errorf("test case %s: %s", testType, testString)
 | |
| 			printedTestCase = true
 | |
| 		}
 | |
| 		t.Errorf("incorrect %s", s)
 | |
| 		t.Errorf("  expected: %s", p(expected))
 | |
| 		t.Errorf("       got: %s", p(got))
 | |
| 	}
 | |
| 
 | |
| 	if !reflect.DeepEqual(expectedErr, err) {
 | |
| 		e("errors:", expectedErr, err)
 | |
| 	}
 | |
| 
 | |
| 	if !reflect.DeepEqual(expected, got) {
 | |
| 		e("output:", expected, got)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func p(in interface{}) string {
 | |
| 	if v, ok := in.([]interface{}); ok {
 | |
| 		s := make([]string, len(v))
 | |
| 		for i := range v {
 | |
| 			s[i] = fmt.Sprintf("%#v", v[i])
 | |
| 		}
 | |
| 		return "[" + strings.Join(s, ", ") + "]"
 | |
| 	} else {
 | |
| 		return fmt.Sprintf("%#v", in)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func pathTestConfig(buildDir string) Config {
 | |
| 	return TestConfig(buildDir, nil, "", nil)
 | |
| }
 | |
| 
 | |
| func TestPathForModuleInstall(t *testing.T) {
 | |
| 	testConfig := pathTestConfig("")
 | |
| 
 | |
| 	hostTarget := Target{Os: Linux, Arch: Arch{ArchType: X86}}
 | |
| 	deviceTarget := Target{Os: Android, Arch: Arch{ArchType: Arm64}}
 | |
| 
 | |
| 	testCases := []struct {
 | |
| 		name         string
 | |
| 		ctx          *testModuleInstallPathContext
 | |
| 		in           []string
 | |
| 		out          string
 | |
| 		partitionDir string
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "host binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     hostTarget.Os,
 | |
| 					target: hostTarget,
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "host/linux-x86/bin/my_test",
 | |
| 			partitionDir: "host/linux-x86",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "system binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/system/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/system",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "vendor binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: socSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/vendor/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/vendor",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "odm binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: deviceSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/odm/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/odm",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "product binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: productSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/product/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/product",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "system_ext binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: systemExtSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/system_ext/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/system_ext",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "root binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRoot: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/root/my_test",
 | |
| 			partitionDir: "target/product/test_device/root",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "recovery binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRecovery: true,
 | |
| 			},
 | |
| 			in:           []string{"bin/my_test"},
 | |
| 			out:          "target/product/test_device/recovery/root/system/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/recovery/root/system",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "recovery root binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRecovery: true,
 | |
| 				inRoot:     true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/recovery/root/my_test",
 | |
| 			partitionDir: "target/product/test_device/recovery/root",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "ramdisk binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRamdisk: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/ramdisk/system/my_test",
 | |
| 			partitionDir: "target/product/test_device/ramdisk/system",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "ramdisk root binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRamdisk: true,
 | |
| 				inRoot:    true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/ramdisk/my_test",
 | |
| 			partitionDir: "target/product/test_device/ramdisk",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "vendor_ramdisk binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inVendorRamdisk: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/vendor_ramdisk/system/my_test",
 | |
| 			partitionDir: "target/product/test_device/vendor_ramdisk/system",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "vendor_ramdisk root binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inVendorRamdisk: true,
 | |
| 				inRoot:          true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/vendor_ramdisk/my_test",
 | |
| 			partitionDir: "target/product/test_device/vendor_ramdisk",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "debug_ramdisk binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inDebugRamdisk: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/debug_ramdisk/my_test",
 | |
| 			partitionDir: "target/product/test_device/debug_ramdisk",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "system native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inData: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "vendor native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: socSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "odm native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: deviceSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "product native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: productSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "system_ext native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: systemExtSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "sanitized system binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/system/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/system",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized vendor binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: socSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/vendor/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/vendor",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized odm binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: deviceSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/odm/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/odm",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized product binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: productSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/product/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/product",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "sanitized system_ext binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: systemExtSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"bin", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/system_ext/bin/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/system_ext",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "sanitized system native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inData:         true,
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized vendor native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: socSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData:         true,
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized odm native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: deviceSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData:         true,
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized product native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: productSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData:         true,
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/data",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "sanitized system_ext native test binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 					earlyModuleContext: earlyModuleContext{
 | |
| 						kind: systemExtSpecificModule,
 | |
| 					},
 | |
| 				},
 | |
| 				inData:         true,
 | |
| 				inSanitizerDir: true,
 | |
| 			},
 | |
| 			in:           []string{"nativetest", "my_test"},
 | |
| 			out:          "target/product/test_device/data/asan/data/nativetest/my_test",
 | |
| 			partitionDir: "target/product/test_device/data/asan/data",
 | |
| 		}, {
 | |
| 			name: "device testcases",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inTestcases: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test", "my_test_bin"},
 | |
| 			out:          "target/product/test_device/testcases/my_test/my_test_bin",
 | |
| 			partitionDir: "target/product/test_device/testcases",
 | |
| 		}, {
 | |
| 			name: "host testcases",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     hostTarget.Os,
 | |
| 					target: hostTarget,
 | |
| 				},
 | |
| 				inTestcases: true,
 | |
| 			},
 | |
| 			in:           []string{"my_test", "my_test_bin"},
 | |
| 			out:          "host/linux-x86/testcases/my_test/my_test_bin",
 | |
| 			partitionDir: "host/linux-x86/testcases",
 | |
| 		}, {
 | |
| 			name: "forced host testcases",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inTestcases: true,
 | |
| 				forceOS:     &Linux,
 | |
| 				forceArch:   &X86,
 | |
| 			},
 | |
| 			in:           []string{"my_test", "my_test_bin"},
 | |
| 			out:          "host/linux-x86/testcases/my_test/my_test_bin",
 | |
| 			partitionDir: "host/linux-x86/testcases",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tc := range testCases {
 | |
| 		t.Run(tc.name, func(t *testing.T) {
 | |
| 			tc.ctx.baseModuleContext.config = testConfig
 | |
| 			output := PathForModuleInstall(tc.ctx, tc.in...)
 | |
| 			if output.basePath.path != tc.out {
 | |
| 				t.Errorf("unexpected path:\n got: %q\nwant: %q\n",
 | |
| 					output.basePath.path,
 | |
| 					tc.out)
 | |
| 			}
 | |
| 			if output.partitionDir != tc.partitionDir {
 | |
| 				t.Errorf("unexpected partitionDir:\n got: %q\nwant: %q\n",
 | |
| 					output.partitionDir, tc.partitionDir)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestPathForModuleInstallRecoveryAsBoot(t *testing.T) {
 | |
| 	testConfig := pathTestConfig("")
 | |
| 	testConfig.TestProductVariables.BoardUsesRecoveryAsBoot = proptools.BoolPtr(true)
 | |
| 	testConfig.TestProductVariables.BoardMoveRecoveryResourcesToVendorBoot = proptools.BoolPtr(true)
 | |
| 	deviceTarget := Target{Os: Android, Arch: Arch{ArchType: Arm64}}
 | |
| 
 | |
| 	testCases := []struct {
 | |
| 		name         string
 | |
| 		ctx          *testModuleInstallPathContext
 | |
| 		in           []string
 | |
| 		out          string
 | |
| 		partitionDir string
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "ramdisk binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inRamdisk: true,
 | |
| 				inRoot:    true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/recovery/root/first_stage_ramdisk/my_test",
 | |
| 			partitionDir: "target/product/test_device/recovery/root/first_stage_ramdisk",
 | |
| 		},
 | |
| 
 | |
| 		{
 | |
| 			name: "vendor_ramdisk binary",
 | |
| 			ctx: &testModuleInstallPathContext{
 | |
| 				baseModuleContext: baseModuleContext{
 | |
| 					os:     deviceTarget.Os,
 | |
| 					target: deviceTarget,
 | |
| 				},
 | |
| 				inVendorRamdisk: true,
 | |
| 				inRoot:          true,
 | |
| 			},
 | |
| 			in:           []string{"my_test"},
 | |
| 			out:          "target/product/test_device/vendor_ramdisk/first_stage_ramdisk/my_test",
 | |
| 			partitionDir: "target/product/test_device/vendor_ramdisk/first_stage_ramdisk",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, tc := range testCases {
 | |
| 		t.Run(tc.name, func(t *testing.T) {
 | |
| 			tc.ctx.baseModuleContext.config = testConfig
 | |
| 			output := PathForModuleInstall(tc.ctx, tc.in...)
 | |
| 			if output.basePath.path != tc.out {
 | |
| 				t.Errorf("unexpected path:\n got: %q\nwant: %q\n",
 | |
| 					output.basePath.path,
 | |
| 					tc.out)
 | |
| 			}
 | |
| 			if output.partitionDir != tc.partitionDir {
 | |
| 				t.Errorf("unexpected partitionDir:\n got: %q\nwant: %q\n",
 | |
| 					output.partitionDir, tc.partitionDir)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestBaseDirForInstallPath(t *testing.T) {
 | |
| 	testConfig := pathTestConfig("")
 | |
| 	deviceTarget := Target{Os: Android, Arch: Arch{ArchType: Arm64}}
 | |
| 
 | |
| 	ctx := &testModuleInstallPathContext{
 | |
| 		baseModuleContext: baseModuleContext{
 | |
| 			os:     deviceTarget.Os,
 | |
| 			target: deviceTarget,
 | |
| 		},
 | |
| 	}
 | |
| 	ctx.baseModuleContext.config = testConfig
 | |
| 
 | |
| 	actual := PathForModuleInstall(ctx, "foo", "bar")
 | |
| 	expectedBaseDir := "target/product/test_device/system"
 | |
| 	if actual.partitionDir != expectedBaseDir {
 | |
| 		t.Errorf("unexpected partitionDir:\n got: %q\nwant: %q\n", actual.partitionDir, expectedBaseDir)
 | |
| 	}
 | |
| 	expectedRelPath := "foo/bar"
 | |
| 	if actual.Rel() != expectedRelPath {
 | |
| 		t.Errorf("unexpected Rel():\n got: %q\nwant: %q\n", actual.Rel(), expectedRelPath)
 | |
| 	}
 | |
| 
 | |
| 	actualAfterJoin := actual.Join(ctx, "baz")
 | |
| 	// partitionDir is preserved even after joining
 | |
| 	if actualAfterJoin.partitionDir != expectedBaseDir {
 | |
| 		t.Errorf("unexpected partitionDir after joining:\n got: %q\nwant: %q\n", actualAfterJoin.partitionDir, expectedBaseDir)
 | |
| 	}
 | |
| 	// Rel() is updated though
 | |
| 	expectedRelAfterJoin := "baz"
 | |
| 	if actualAfterJoin.Rel() != expectedRelAfterJoin {
 | |
| 		t.Errorf("unexpected Rel() after joining:\n got: %q\nwant: %q\n", actualAfterJoin.Rel(), expectedRelAfterJoin)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestDirectorySortedPaths(t *testing.T) {
 | |
| 	config := TestConfig("out", nil, "", map[string][]byte{
 | |
| 		"Android.bp": nil,
 | |
| 		"a.txt":      nil,
 | |
| 		"a/txt":      nil,
 | |
| 		"a/b/c":      nil,
 | |
| 		"a/b/d":      nil,
 | |
| 		"b":          nil,
 | |
| 		"b/b.txt":    nil,
 | |
| 		"a/a.txt":    nil,
 | |
| 	})
 | |
| 
 | |
| 	ctx := PathContextForTesting(config)
 | |
| 
 | |
| 	makePaths := func() Paths {
 | |
| 		return Paths{
 | |
| 			PathForSource(ctx, "a.txt"),
 | |
| 			PathForSource(ctx, "a/txt"),
 | |
| 			PathForSource(ctx, "a/b/c"),
 | |
| 			PathForSource(ctx, "a/b/d"),
 | |
| 			PathForSource(ctx, "b"),
 | |
| 			PathForSource(ctx, "b/b.txt"),
 | |
| 			PathForSource(ctx, "a/a.txt"),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	expected := []string{
 | |
| 		"a.txt",
 | |
| 		"a/a.txt",
 | |
| 		"a/b/c",
 | |
| 		"a/b/d",
 | |
| 		"a/txt",
 | |
| 		"b",
 | |
| 		"b/b.txt",
 | |
| 	}
 | |
| 
 | |
| 	paths := makePaths()
 | |
| 	reversePaths := ReversePaths(paths)
 | |
| 
 | |
| 	sortedPaths := PathsToDirectorySortedPaths(paths)
 | |
| 	reverseSortedPaths := PathsToDirectorySortedPaths(reversePaths)
 | |
| 
 | |
| 	if !reflect.DeepEqual(Paths(sortedPaths).Strings(), expected) {
 | |
| 		t.Fatalf("sorted paths:\n %#v\n != \n %#v", paths.Strings(), expected)
 | |
| 	}
 | |
| 
 | |
| 	if !reflect.DeepEqual(Paths(reverseSortedPaths).Strings(), expected) {
 | |
| 		t.Fatalf("sorted reversed paths:\n %#v\n !=\n %#v", reversePaths.Strings(), expected)
 | |
| 	}
 | |
| 
 | |
| 	expectedA := []string{
 | |
| 		"a/a.txt",
 | |
| 		"a/b/c",
 | |
| 		"a/b/d",
 | |
| 		"a/txt",
 | |
| 	}
 | |
| 
 | |
| 	inA := sortedPaths.PathsInDirectory("a")
 | |
| 	if !reflect.DeepEqual(inA.Strings(), expectedA) {
 | |
| 		t.Errorf("FilesInDirectory(a):\n %#v\n != \n %#v", inA.Strings(), expectedA)
 | |
| 	}
 | |
| 
 | |
| 	expectedA_B := []string{
 | |
| 		"a/b/c",
 | |
| 		"a/b/d",
 | |
| 	}
 | |
| 
 | |
| 	inA_B := sortedPaths.PathsInDirectory("a/b")
 | |
| 	if !reflect.DeepEqual(inA_B.Strings(), expectedA_B) {
 | |
| 		t.Errorf("FilesInDirectory(a/b):\n %#v\n != \n %#v", inA_B.Strings(), expectedA_B)
 | |
| 	}
 | |
| 
 | |
| 	expectedB := []string{
 | |
| 		"b/b.txt",
 | |
| 	}
 | |
| 
 | |
| 	inB := sortedPaths.PathsInDirectory("b")
 | |
| 	if !reflect.DeepEqual(inB.Strings(), expectedB) {
 | |
| 		t.Errorf("FilesInDirectory(b):\n %#v\n != \n %#v", inA.Strings(), expectedA)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestMaybeRel(t *testing.T) {
 | |
| 	testCases := []struct {
 | |
| 		name   string
 | |
| 		base   string
 | |
| 		target string
 | |
| 		out    string
 | |
| 		isRel  bool
 | |
| 	}{
 | |
| 		{
 | |
| 			name:   "normal",
 | |
| 			base:   "a/b/c",
 | |
| 			target: "a/b/c/d",
 | |
| 			out:    "d",
 | |
| 			isRel:  true,
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "parent",
 | |
| 			base:   "a/b/c/d",
 | |
| 			target: "a/b/c",
 | |
| 			isRel:  false,
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "not relative",
 | |
| 			base:   "a/b",
 | |
| 			target: "c/d",
 | |
| 			isRel:  false,
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "abs1",
 | |
| 			base:   "/a",
 | |
| 			target: "a",
 | |
| 			isRel:  false,
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "abs2",
 | |
| 			base:   "a",
 | |
| 			target: "/a",
 | |
| 			isRel:  false,
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	for _, testCase := range testCases {
 | |
| 		t.Run(testCase.name, func(t *testing.T) {
 | |
| 			ctx := &configErrorWrapper{}
 | |
| 			out, isRel := MaybeRel(ctx, testCase.base, testCase.target)
 | |
| 			if len(ctx.errors) > 0 {
 | |
| 				t.Errorf("MaybeRel(..., %s, %s) reported unexpected errors %v",
 | |
| 					testCase.base, testCase.target, ctx.errors)
 | |
| 			}
 | |
| 			if isRel != testCase.isRel || out != testCase.out {
 | |
| 				t.Errorf("MaybeRel(..., %s, %s) want %v, %v got %v, %v",
 | |
| 					testCase.base, testCase.target, testCase.out, testCase.isRel, out, isRel)
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestPathForSource(t *testing.T) {
 | |
| 	testCases := []struct {
 | |
| 		name     string
 | |
| 		buildDir string
 | |
| 		src      string
 | |
| 		err      string
 | |
| 	}{
 | |
| 		{
 | |
| 			name:     "normal",
 | |
| 			buildDir: "out",
 | |
| 			src:      "a/b/c",
 | |
| 		},
 | |
| 		{
 | |
| 			name:     "abs",
 | |
| 			buildDir: "out",
 | |
| 			src:      "/a/b/c",
 | |
| 			err:      "is outside directory",
 | |
| 		},
 | |
| 		{
 | |
| 			name:     "in out dir",
 | |
| 			buildDir: "out",
 | |
| 			src:      "out/soong/a/b/c",
 | |
| 			err:      "is in output",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	funcs := []struct {
 | |
| 		name string
 | |
| 		f    func(ctx PathContext, pathComponents ...string) (SourcePath, error)
 | |
| 	}{
 | |
| 		{"pathForSource", pathForSource},
 | |
| 		{"safePathForSource", safePathForSource},
 | |
| 	}
 | |
| 
 | |
| 	for _, f := range funcs {
 | |
| 		t.Run(f.name, func(t *testing.T) {
 | |
| 			for _, test := range testCases {
 | |
| 				t.Run(test.name, func(t *testing.T) {
 | |
| 					testConfig := pathTestConfig(test.buildDir)
 | |
| 					ctx := &configErrorWrapper{config: testConfig}
 | |
| 					_, err := f.f(ctx, test.src)
 | |
| 					if len(ctx.errors) > 0 {
 | |
| 						t.Fatalf("unexpected errors %v", ctx.errors)
 | |
| 					}
 | |
| 					if err != nil {
 | |
| 						if test.err == "" {
 | |
| 							t.Fatalf("unexpected error %q", err.Error())
 | |
| 						} else if !strings.Contains(err.Error(), test.err) {
 | |
| 							t.Fatalf("incorrect error, want substring %q got %q", test.err, err.Error())
 | |
| 						}
 | |
| 					} else {
 | |
| 						if test.err != "" {
 | |
| 							t.Fatalf("missing error %q", test.err)
 | |
| 						}
 | |
| 					}
 | |
| 				})
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type pathForModuleSrcTestModule struct {
 | |
| 	ModuleBase
 | |
| 	props struct {
 | |
| 		Srcs         []string `android:"path"`
 | |
| 		Exclude_srcs []string `android:"path"`
 | |
| 
 | |
| 		Src *string `android:"path"`
 | |
| 
 | |
| 		Module_handles_missing_deps bool
 | |
| 	}
 | |
| 
 | |
| 	src string
 | |
| 	rel string
 | |
| 
 | |
| 	srcs []string
 | |
| 	rels []string
 | |
| 
 | |
| 	missingDeps []string
 | |
| }
 | |
| 
 | |
| func pathForModuleSrcTestModuleFactory() Module {
 | |
| 	module := &pathForModuleSrcTestModule{}
 | |
| 	module.AddProperties(&module.props)
 | |
| 	InitAndroidModule(module)
 | |
| 	return module
 | |
| }
 | |
| 
 | |
| func (p *pathForModuleSrcTestModule) GenerateAndroidBuildActions(ctx ModuleContext) {
 | |
| 	var srcs Paths
 | |
| 	if p.props.Module_handles_missing_deps {
 | |
| 		srcs, p.missingDeps = PathsAndMissingDepsForModuleSrcExcludes(ctx, p.props.Srcs, p.props.Exclude_srcs)
 | |
| 	} else {
 | |
| 		srcs = PathsForModuleSrcExcludes(ctx, p.props.Srcs, p.props.Exclude_srcs)
 | |
| 	}
 | |
| 	p.srcs = srcs.Strings()
 | |
| 
 | |
| 	for _, src := range srcs {
 | |
| 		p.rels = append(p.rels, src.Rel())
 | |
| 	}
 | |
| 
 | |
| 	if p.props.Src != nil {
 | |
| 		src := PathForModuleSrc(ctx, *p.props.Src)
 | |
| 		if src != nil {
 | |
| 			p.src = src.String()
 | |
| 			p.rel = src.Rel()
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if !p.props.Module_handles_missing_deps {
 | |
| 		p.missingDeps = ctx.GetMissingDependencies()
 | |
| 	}
 | |
| 
 | |
| 	ctx.Build(pctx, BuildParams{
 | |
| 		Rule:   Touch,
 | |
| 		Output: PathForModuleOut(ctx, "output"),
 | |
| 	})
 | |
| }
 | |
| 
 | |
| type pathForModuleSrcOutputFileProviderModule struct {
 | |
| 	ModuleBase
 | |
| 	props struct {
 | |
| 		Outs   []string
 | |
| 		Tagged []string
 | |
| 	}
 | |
| 
 | |
| 	outs   Paths
 | |
| 	tagged Paths
 | |
| }
 | |
| 
 | |
| func pathForModuleSrcOutputFileProviderModuleFactory() Module {
 | |
| 	module := &pathForModuleSrcOutputFileProviderModule{}
 | |
| 	module.AddProperties(&module.props)
 | |
| 	InitAndroidModule(module)
 | |
| 	return module
 | |
| }
 | |
| 
 | |
| func (p *pathForModuleSrcOutputFileProviderModule) GenerateAndroidBuildActions(ctx ModuleContext) {
 | |
| 	for _, out := range p.props.Outs {
 | |
| 		p.outs = append(p.outs, PathForModuleOut(ctx, out))
 | |
| 	}
 | |
| 
 | |
| 	for _, tagged := range p.props.Tagged {
 | |
| 		p.tagged = append(p.tagged, PathForModuleOut(ctx, tagged))
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (p *pathForModuleSrcOutputFileProviderModule) OutputFiles(tag string) (Paths, error) {
 | |
| 	switch tag {
 | |
| 	case "":
 | |
| 		return p.outs, nil
 | |
| 	case ".tagged":
 | |
| 		return p.tagged, nil
 | |
| 	default:
 | |
| 		return nil, fmt.Errorf("unsupported tag %q", tag)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type pathForModuleSrcTestCase struct {
 | |
| 	name string
 | |
| 	bp   string
 | |
| 	srcs []string
 | |
| 	rels []string
 | |
| 	src  string
 | |
| 	rel  string
 | |
| 
 | |
| 	// Make test specific preparations to the test fixture.
 | |
| 	preparer FixturePreparer
 | |
| 
 | |
| 	// A test specific error handler.
 | |
| 	errorHandler FixtureErrorHandler
 | |
| }
 | |
| 
 | |
| func testPathForModuleSrc(t *testing.T, tests []pathForModuleSrcTestCase) {
 | |
| 	for _, test := range tests {
 | |
| 		t.Run(test.name, func(t *testing.T) {
 | |
| 			fgBp := `
 | |
| 				filegroup {
 | |
| 					name: "a",
 | |
| 					srcs: ["src/a"],
 | |
| 				}
 | |
| 			`
 | |
| 
 | |
| 			ofpBp := `
 | |
| 				output_file_provider {
 | |
| 					name: "b",
 | |
| 					outs: ["gen/b"],
 | |
| 					tagged: ["gen/c"],
 | |
| 				}
 | |
| 			`
 | |
| 
 | |
| 			mockFS := MockFS{
 | |
| 				"fg/Android.bp":     []byte(fgBp),
 | |
| 				"foo/Android.bp":    []byte(test.bp),
 | |
| 				"ofp/Android.bp":    []byte(ofpBp),
 | |
| 				"fg/src/a":          nil,
 | |
| 				"foo/src/b":         nil,
 | |
| 				"foo/src/c":         nil,
 | |
| 				"foo/src/d":         nil,
 | |
| 				"foo/src/e/e":       nil,
 | |
| 				"foo/src_special/$": nil,
 | |
| 			}
 | |
| 
 | |
| 			errorHandler := test.errorHandler
 | |
| 			if errorHandler == nil {
 | |
| 				errorHandler = FixtureExpectsNoErrors
 | |
| 			}
 | |
| 
 | |
| 			result := GroupFixturePreparers(
 | |
| 				FixtureRegisterWithContext(func(ctx RegistrationContext) {
 | |
| 					ctx.RegisterModuleType("test", pathForModuleSrcTestModuleFactory)
 | |
| 					ctx.RegisterModuleType("output_file_provider", pathForModuleSrcOutputFileProviderModuleFactory)
 | |
| 				}),
 | |
| 				PrepareForTestWithFilegroup,
 | |
| 				PrepareForTestWithNamespace,
 | |
| 				mockFS.AddToFixture(),
 | |
| 				OptionalFixturePreparer(test.preparer),
 | |
| 			).
 | |
| 				ExtendWithErrorHandler(errorHandler).
 | |
| 				RunTest(t)
 | |
| 
 | |
| 			m := result.ModuleForTests("foo", "").Module().(*pathForModuleSrcTestModule)
 | |
| 
 | |
| 			AssertStringPathsRelativeToTopEquals(t, "srcs", result.Config, test.srcs, m.srcs)
 | |
| 			AssertStringPathsRelativeToTopEquals(t, "rels", result.Config, test.rels, m.rels)
 | |
| 			AssertStringPathRelativeToTopEquals(t, "src", result.Config, test.src, m.src)
 | |
| 			AssertStringPathRelativeToTopEquals(t, "rel", result.Config, test.rel, m.rel)
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestPathsForModuleSrc(t *testing.T) {
 | |
| 	tests := []pathForModuleSrcTestCase{
 | |
| 		{
 | |
| 			name: "path",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: ["src/b"],
 | |
| 			}`,
 | |
| 			srcs: []string{"foo/src/b"},
 | |
| 			rels: []string{"src/b"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "glob",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: [
 | |
| 					"src/*",
 | |
| 					"src/e/*",
 | |
| 				],
 | |
| 			}`,
 | |
| 			srcs: []string{"foo/src/b", "foo/src/c", "foo/src/d", "foo/src/e/e"},
 | |
| 			rels: []string{"src/b", "src/c", "src/d", "src/e/e"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "recursive glob",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: ["src/**/*"],
 | |
| 			}`,
 | |
| 			srcs: []string{"foo/src/b", "foo/src/c", "foo/src/d", "foo/src/e/e"},
 | |
| 			rels: []string{"src/b", "src/c", "src/d", "src/e/e"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "filegroup",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: [":a"],
 | |
| 			}`,
 | |
| 			srcs: []string{"fg/src/a"},
 | |
| 			rels: []string{"src/a"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: [":b"],
 | |
| 			}`,
 | |
| 			srcs: []string{"out/soong/.intermediates/ofp/b/gen/b"},
 | |
| 			rels: []string{"gen/b"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider tagged",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: [":b{.tagged}"],
 | |
| 			}`,
 | |
| 			srcs: []string{"out/soong/.intermediates/ofp/b/gen/c"},
 | |
| 			rels: []string{"gen/c"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider with exclude",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: [":b", ":c"],
 | |
| 				exclude_srcs: [":c"]
 | |
| 			}
 | |
| 			output_file_provider {
 | |
| 				name: "c",
 | |
| 				outs: ["gen/c"],
 | |
| 			}`,
 | |
| 			srcs: []string{"out/soong/.intermediates/ofp/b/gen/b"},
 | |
| 			rels: []string{"gen/b"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "special characters glob",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				srcs: ["src_special/*"],
 | |
| 			}`,
 | |
| 			srcs: []string{"foo/src_special/$"},
 | |
| 			rels: []string{"src_special/$"},
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	testPathForModuleSrc(t, tests)
 | |
| }
 | |
| 
 | |
| func TestPathForModuleSrc(t *testing.T) {
 | |
| 	tests := []pathForModuleSrcTestCase{
 | |
| 		{
 | |
| 			name: "path",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "src/b",
 | |
| 			}`,
 | |
| 			src: "foo/src/b",
 | |
| 			rel: "src/b",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "glob",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "src/e/*",
 | |
| 			}`,
 | |
| 			src: "foo/src/e/e",
 | |
| 			rel: "src/e/e",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "filegroup",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: ":a",
 | |
| 			}`,
 | |
| 			src: "fg/src/a",
 | |
| 			rel: "src/a",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: ":b",
 | |
| 			}`,
 | |
| 			src: "out/soong/.intermediates/ofp/b/gen/b",
 | |
| 			rel: "gen/b",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider tagged",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: ":b{.tagged}",
 | |
| 			}`,
 | |
| 			src: "out/soong/.intermediates/ofp/b/gen/c",
 | |
| 			rel: "gen/c",
 | |
| 		},
 | |
| 		{
 | |
| 			name: "special characters glob",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "src_special/*",
 | |
| 			}`,
 | |
| 			src: "foo/src_special/$",
 | |
| 			rel: "src_special/$",
 | |
| 		},
 | |
| 		{
 | |
| 			// This test makes sure that an unqualified module name cannot contain characters that make
 | |
| 			// it appear as a qualified module name.
 | |
| 			name: "output file provider, invalid fully qualified name",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "://other:b",
 | |
| 				srcs: ["://other:c"],
 | |
| 			}`,
 | |
| 			preparer: FixtureAddTextFile("other/Android.bp", `
 | |
| 				soong_namespace {}
 | |
| 
 | |
| 				output_file_provider {
 | |
| 					name: "b",
 | |
| 					outs: ["gen/b"],
 | |
| 				}
 | |
| 
 | |
| 				output_file_provider {
 | |
| 					name: "c",
 | |
| 					outs: ["gen/c"],
 | |
| 				}
 | |
| 			`),
 | |
| 			src:  "foo/:/other:b",
 | |
| 			rel:  ":/other:b",
 | |
| 			srcs: []string{"foo/:/other:c"},
 | |
| 			rels: []string{":/other:c"},
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider, missing fully qualified name",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "//other:b",
 | |
| 				srcs: ["//other:c"],
 | |
| 			}`,
 | |
| 			errorHandler: FixtureExpectsAllErrorsToMatchAPattern([]string{
 | |
| 				`"foo" depends on undefined module "//other:b"`,
 | |
| 				`"foo" depends on undefined module "//other:c"`,
 | |
| 			}),
 | |
| 		},
 | |
| 		{
 | |
| 			name: "output file provider, fully qualified name",
 | |
| 			bp: `
 | |
| 			test {
 | |
| 				name: "foo",
 | |
| 				src: "//other:b",
 | |
| 				srcs: ["//other:c"],
 | |
| 			}`,
 | |
| 			src:  "out/soong/.intermediates/other/b/gen/b",
 | |
| 			rel:  "gen/b",
 | |
| 			srcs: []string{"out/soong/.intermediates/other/c/gen/c"},
 | |
| 			rels: []string{"gen/c"},
 | |
| 			preparer: FixtureAddTextFile("other/Android.bp", `
 | |
| 				soong_namespace {}
 | |
| 
 | |
| 				output_file_provider {
 | |
| 					name: "b",
 | |
| 					outs: ["gen/b"],
 | |
| 				}
 | |
| 
 | |
| 				output_file_provider {
 | |
| 					name: "c",
 | |
| 					outs: ["gen/c"],
 | |
| 				}
 | |
| 			`),
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	testPathForModuleSrc(t, tests)
 | |
| }
 | |
| 
 | |
| func TestPathsForModuleSrc_AllowMissingDependencies(t *testing.T) {
 | |
| 	bp := `
 | |
| 		test {
 | |
| 			name: "foo",
 | |
| 			srcs: [":a"],
 | |
| 			exclude_srcs: [":b"],
 | |
| 			src: ":c",
 | |
| 		}
 | |
| 
 | |
| 		test {
 | |
| 			name: "bar",
 | |
| 			srcs: [":d"],
 | |
| 			exclude_srcs: [":e"],
 | |
| 			module_handles_missing_deps: true,
 | |
| 		}
 | |
| 	`
 | |
| 
 | |
| 	result := GroupFixturePreparers(
 | |
| 		PrepareForTestWithAllowMissingDependencies,
 | |
| 		FixtureRegisterWithContext(func(ctx RegistrationContext) {
 | |
| 			ctx.RegisterModuleType("test", pathForModuleSrcTestModuleFactory)
 | |
| 		}),
 | |
| 		FixtureWithRootAndroidBp(bp),
 | |
| 	).RunTest(t)
 | |
| 
 | |
| 	foo := result.ModuleForTests("foo", "").Module().(*pathForModuleSrcTestModule)
 | |
| 
 | |
| 	AssertArrayString(t, "foo missing deps", []string{"a", "b", "c"}, foo.missingDeps)
 | |
| 	AssertArrayString(t, "foo srcs", []string{}, foo.srcs)
 | |
| 	AssertStringEquals(t, "foo src", "", foo.src)
 | |
| 
 | |
| 	bar := result.ModuleForTests("bar", "").Module().(*pathForModuleSrcTestModule)
 | |
| 
 | |
| 	AssertArrayString(t, "bar missing deps", []string{"d", "e"}, bar.missingDeps)
 | |
| 	AssertArrayString(t, "bar srcs", []string{}, bar.srcs)
 | |
| }
 | |
| 
 | |
| func TestPathRelativeToTop(t *testing.T) {
 | |
| 	testConfig := pathTestConfig("/tmp/build/top")
 | |
| 	deviceTarget := Target{Os: Android, Arch: Arch{ArchType: Arm64}}
 | |
| 
 | |
| 	ctx := &testModuleInstallPathContext{
 | |
| 		baseModuleContext: baseModuleContext{
 | |
| 			os:     deviceTarget.Os,
 | |
| 			target: deviceTarget,
 | |
| 		},
 | |
| 	}
 | |
| 	ctx.baseModuleContext.config = testConfig
 | |
| 
 | |
| 	t.Run("install for soong", func(t *testing.T) {
 | |
| 		p := PathForModuleInstall(ctx, "install/path")
 | |
| 		AssertPathRelativeToTopEquals(t, "install path for soong", "out/soong/target/product/test_device/system/install/path", p)
 | |
| 	})
 | |
| 	t.Run("install for make", func(t *testing.T) {
 | |
| 		p := PathForModuleInstall(ctx, "install/path")
 | |
| 		p.makePath = true
 | |
| 		AssertPathRelativeToTopEquals(t, "install path for make", "out/target/product/test_device/system/install/path", p)
 | |
| 	})
 | |
| 	t.Run("output", func(t *testing.T) {
 | |
| 		p := PathForOutput(ctx, "output/path")
 | |
| 		AssertPathRelativeToTopEquals(t, "output path", "out/soong/output/path", p)
 | |
| 	})
 | |
| 	t.Run("source", func(t *testing.T) {
 | |
| 		p := PathForSource(ctx, "source/path")
 | |
| 		AssertPathRelativeToTopEquals(t, "source path", "source/path", p)
 | |
| 	})
 | |
| 	t.Run("mixture", func(t *testing.T) {
 | |
| 		paths := Paths{
 | |
| 			PathForModuleInstall(ctx, "install/path"),
 | |
| 			PathForOutput(ctx, "output/path"),
 | |
| 			PathForSource(ctx, "source/path"),
 | |
| 		}
 | |
| 
 | |
| 		expected := []string{
 | |
| 			"out/soong/target/product/test_device/system/install/path",
 | |
| 			"out/soong/output/path",
 | |
| 			"source/path",
 | |
| 		}
 | |
| 		AssertPathsRelativeToTopEquals(t, "mixture", expected, paths)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func ExampleOutputPath_ReplaceExtension() {
 | |
| 	ctx := &configErrorWrapper{
 | |
| 		config: TestConfig("out", nil, "", nil),
 | |
| 	}
 | |
| 	p := PathForOutput(ctx, "system/framework").Join(ctx, "boot.art")
 | |
| 	p2 := p.ReplaceExtension(ctx, "oat")
 | |
| 	fmt.Println(p, p2)
 | |
| 	fmt.Println(p.Rel(), p2.Rel())
 | |
| 
 | |
| 	// Output:
 | |
| 	// out/soong/system/framework/boot.art out/soong/system/framework/boot.oat
 | |
| 	// boot.art boot.oat
 | |
| }
 | |
| 
 | |
| func ExampleOutputPath_InSameDir() {
 | |
| 	ctx := &configErrorWrapper{
 | |
| 		config: TestConfig("out", nil, "", nil),
 | |
| 	}
 | |
| 	p := PathForOutput(ctx, "system/framework").Join(ctx, "boot.art")
 | |
| 	p2 := p.InSameDir(ctx, "oat", "arm", "boot.vdex")
 | |
| 	fmt.Println(p, p2)
 | |
| 	fmt.Println(p.Rel(), p2.Rel())
 | |
| 
 | |
| 	// Output:
 | |
| 	// out/soong/system/framework/boot.art out/soong/system/framework/oat/arm/boot.vdex
 | |
| 	// boot.art oat/arm/boot.vdex
 | |
| }
 | |
| 
 | |
| func BenchmarkFirstUniquePaths(b *testing.B) {
 | |
| 	implementations := []struct {
 | |
| 		name string
 | |
| 		f    func(Paths) Paths
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "list",
 | |
| 			f:    firstUniquePathsList,
 | |
| 		},
 | |
| 		{
 | |
| 			name: "map",
 | |
| 			f:    firstUniquePathsMap,
 | |
| 		},
 | |
| 	}
 | |
| 	const maxSize = 1024
 | |
| 	uniquePaths := make(Paths, maxSize)
 | |
| 	for i := range uniquePaths {
 | |
| 		uniquePaths[i] = PathForTesting(strconv.Itoa(i))
 | |
| 	}
 | |
| 	samePath := make(Paths, maxSize)
 | |
| 	for i := range samePath {
 | |
| 		samePath[i] = uniquePaths[0]
 | |
| 	}
 | |
| 
 | |
| 	f := func(b *testing.B, imp func(Paths) Paths, paths Paths) {
 | |
| 		for i := 0; i < b.N; i++ {
 | |
| 			b.ReportAllocs()
 | |
| 			paths = append(Paths(nil), paths...)
 | |
| 			imp(paths)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for n := 1; n <= maxSize; n <<= 1 {
 | |
| 		b.Run(strconv.Itoa(n), func(b *testing.B) {
 | |
| 			for _, implementation := range implementations {
 | |
| 				b.Run(implementation.name, func(b *testing.B) {
 | |
| 					b.Run("same", func(b *testing.B) {
 | |
| 						f(b, implementation.f, samePath[:n])
 | |
| 					})
 | |
| 					b.Run("unique", func(b *testing.B) {
 | |
| 						f(b, implementation.f, uniquePaths[:n])
 | |
| 					})
 | |
| 				})
 | |
| 			}
 | |
| 		})
 | |
| 	}
 | |
| }
 |