|  | // Copyright (C) 2021 The Android Open Source Project | 
|  | // | 
|  | // 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 apex | 
|  |  | 
|  | import ( | 
|  | "android/soong/android" | 
|  | ) | 
|  |  | 
|  | type DeapexerProperties struct { | 
|  | // List of common modules that may need access to files exported by this module. | 
|  | // | 
|  | // A common module in this sense is one that is not arch specific but uses a common variant for | 
|  | // all architectures, e.g. java. | 
|  | CommonModules []string | 
|  |  | 
|  | // List of modules that use an embedded .prof to guide optimization of the equivalent dexpreopt artifact | 
|  | // This is a subset of CommonModules | 
|  | DexpreoptProfileGuidedModules []string | 
|  |  | 
|  | // List of files exported from the .apex file by this module | 
|  | // | 
|  | // Each entry is a path from the apex root, e.g. javalib/core-libart.jar. | 
|  | ExportedFiles []string | 
|  | } | 
|  |  | 
|  | type SelectedApexProperties struct { | 
|  | // The path to the apex selected for use by this module. | 
|  | // | 
|  | // Is tagged as `android:"path"` because it will usually contain a string of the form ":<module>" | 
|  | // and is tagged as "`blueprint:"mutate"` because it is only initialized in a LoadHook not an | 
|  | // Android.bp file. | 
|  | Selected_apex *string `android:"path" blueprint:"mutated"` | 
|  | } | 
|  |  | 
|  | // deapex creates the build rules to deapex a prebuilt .apex file | 
|  | // it returns a pointer to a DeapexerInfo object | 
|  | func deapex(ctx android.ModuleContext, apexFile android.Path, deapexerProps DeapexerProperties) *android.DeapexerInfo { | 
|  | // Create and remember the directory into which the .apex file's contents will be unpacked. | 
|  | deapexerOutput := android.PathForModuleOut(ctx, "deapexer") | 
|  |  | 
|  | exports := make(map[string]android.WritablePath) | 
|  |  | 
|  | // Create mappings from apex relative path to the extracted file's path. | 
|  | exportedPaths := make(android.Paths, 0, len(exports)) | 
|  | for _, path := range deapexerProps.ExportedFiles { | 
|  | // Populate the exports that this makes available. | 
|  | extractedPath := deapexerOutput.Join(ctx, path) | 
|  | exports[path] = extractedPath | 
|  | exportedPaths = append(exportedPaths, extractedPath) | 
|  | } | 
|  |  | 
|  | // If the prebuilt_apex exports any files then create a build rule that unpacks the apex using | 
|  | // deapexer and verifies that all the required files were created. Also, make the mapping from | 
|  | // apex relative path to extracted file path available for other modules. | 
|  | if len(exports) > 0 { | 
|  | // Make the information available for other modules. | 
|  | di := android.NewDeapexerInfo(ctx.ModuleName(), exports, deapexerProps.CommonModules) | 
|  | di.AddDexpreoptProfileGuidedExportedModuleNames(deapexerProps.DexpreoptProfileGuidedModules...) | 
|  |  | 
|  | // Create a sorted list of the files that this exports. | 
|  | exportedPaths = android.SortedUniquePaths(exportedPaths) | 
|  |  | 
|  | // The apex needs to export some files so create a ninja rule to unpack the apex and check that | 
|  | // the required files are present. | 
|  | builder := android.NewRuleBuilder(pctx, ctx) | 
|  | command := builder.Command() | 
|  | command. | 
|  | Tool(android.PathForSource(ctx, "build/soong/scripts/unpack-prebuilt-apex.sh")). | 
|  | BuiltTool("deapexer"). | 
|  | BuiltTool("debugfs"). | 
|  | BuiltTool("fsck.erofs"). | 
|  | Input(apexFile). | 
|  | Text(deapexerOutput.String()) | 
|  | for _, p := range exportedPaths { | 
|  | command.Output(p.(android.WritablePath)) | 
|  | } | 
|  | builder.Build("deapexer", "deapex "+ctx.ModuleName()) | 
|  | return &di | 
|  | } | 
|  | return nil | 
|  | } |