1// Copyright (C) 2021 The Android Open Source Project 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 aidl 16 17import ( 18 "android/soong/android" 19 "android/soong/rust" 20 21 "strings" 22 23 "github.com/google/blueprint" 24 "github.com/google/blueprint/proptools" 25) 26 27var ( 28 aidlRustGlueRule = pctx.StaticRule("aidlRustGlueRule", blueprint.RuleParams{ 29 Command: `${aidlRustGlueCmd} ${out} ${root} ${in} ${imports}`, 30 CommandDeps: []string{"${aidlRustGlueCmd}"}, 31 }, "root", "imports") 32) 33 34type aidlRustSourceProviderProperties struct { 35 SourceGen string `android:"path"` 36 Imports []string 37 Version string 38 AidlInterfaceName string 39} 40 41type aidlRustSourceProvider struct { 42 *rust.BaseSourceProvider 43 44 properties aidlRustSourceProviderProperties 45} 46 47var aidlRustSourceTag = struct { 48 blueprint.DependencyTag 49}{} 50 51func (sp *aidlRustSourceProvider) GenerateSource(ctx rust.ModuleContext, _ rust.PathDeps) android.Path { 52 sourceStem := proptools.String(sp.BaseSourceProvider.Properties.Source_stem) 53 topLevelOutputFile := android.PathForModuleOut(ctx, sourceStem+".rs") 54 55 aidlGenModule := ctx.GetDirectDepWithTag(sp.properties.SourceGen, aidlRustSourceTag) 56 // Find the gen directory for the source module 57 srcGenDir := aidlGenModule.(*aidlGenRule).genOutDir 58 srcPaths := aidlGenModule.(*aidlGenRule).genOutputs.Paths() 59 60 // In Rust, we import our dependency crates into `mangled`: 61 // use dependency::mangled::*; 62 // so we can use the crate:: Rust path prefix to refer to 63 // both crate-local and imported paths (from dependencies) 64 importFlags := make([]string, len(sp.properties.Imports)) 65 for i, dep := range trimVersionSuffixInList(sp.properties.Imports) { 66 importFlags[i] = "-I" + fixRustName(dep) 67 } 68 69 // In Rust, we need to do some extra post-processing: 70 // emit a top-level foo.rs that includes all the generated .rs 71 // files verbatim ("glued" together). The generated glue file 72 // replicates the AIDL package hierarchy from the input 73 // .aidl files in two ways: 74 // * 1:1 mapping in the crate::aidl namespace, used by downstream users 75 // * mangled in the crate::mangled namespace, used internally 76 // to resolve AIDL package paths between dependencies 77 ctx.Build(pctx, android.BuildParams{ 78 Rule: aidlRustGlueRule, 79 Inputs: srcPaths, 80 Output: topLevelOutputFile, 81 Args: map[string]string{ 82 "root": srcGenDir.String(), 83 "imports": strings.Join(importFlags, " "), 84 }, 85 }) 86 87 sp.BaseSourceProvider.OutputFiles = android.Paths{topLevelOutputFile} 88 return topLevelOutputFile 89} 90 91func (sp *aidlRustSourceProvider) SourceProviderProps() []interface{} { 92 return append(sp.BaseSourceProvider.SourceProviderProps(), 93 &sp.properties) 94} 95 96func (sp *aidlRustSourceProvider) SourceProviderDeps(ctx rust.DepsContext, deps rust.Deps) rust.Deps { 97 deps = sp.BaseSourceProvider.SourceProviderDeps(ctx, deps) 98 deps.Rustlibs = append(deps.Rustlibs, "libbinder_rs", "liblazy_static") 99 ai := lookupInterface(sp.properties.AidlInterfaceName, ctx.Config()) 100 for _, dep := range sp.properties.Imports { 101 deps.Rustlibs = append(deps.Rustlibs, ai.getImportWithVersion(sp.properties.Version, dep, ctx.Config())+"-"+langRust) 102 } 103 104 // Add a depencency to the source module (*-rust-source) directly via `ctx` because 105 // the source module is specific to aidlRustSourceProvider and we don't want the rust module 106 // to know about it. 107 ctx.AddDependency(ctx.Module(), aidlRustSourceTag, sp.properties.SourceGen) 108 109 return deps 110} 111 112func (sp *aidlRustSourceProvider) AndroidMk(ctx rust.AndroidMkContext, ret *android.AndroidMkEntries) { 113 ctx.SubAndroidMk(ret, sp.BaseSourceProvider) 114 ret.ExtraEntries = append(ret.ExtraEntries, 115 func(ctx android.AndroidMkExtraEntriesContext, entries *android.AndroidMkEntries) { 116 entries.SetBool("LOCAL_UNINSTALLABLE_MODULE", true) 117 }) 118} 119 120func aidlRustLibraryFactory() android.Module { 121 sourceProvider := &aidlRustSourceProvider{ 122 BaseSourceProvider: rust.NewSourceProvider(), 123 properties: aidlRustSourceProviderProperties{}, 124 } 125 126 module := rust.NewSourceProviderModule(android.HostAndDeviceSupported, sourceProvider, false) 127 return module.Init() 128} 129