2018-05-17 20:17:01 +02:00
|
|
|
// Copyright (C) 2018 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 bpf
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io"
|
2022-07-07 11:48:06 +02:00
|
|
|
"path/filepath"
|
2022-08-18 00:45:52 +02:00
|
|
|
"runtime"
|
2018-05-17 20:17:01 +02:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"android/soong/android"
|
2022-11-10 22:58:48 +01:00
|
|
|
"android/soong/cc"
|
2018-05-17 20:17:01 +02:00
|
|
|
|
|
|
|
"github.com/google/blueprint"
|
2021-12-03 05:09:45 +01:00
|
|
|
"github.com/google/blueprint/proptools"
|
2018-05-17 20:17:01 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
2021-02-24 19:51:54 +01:00
|
|
|
registerBpfBuildComponents(android.InitRegistrationContext)
|
2018-05-17 20:17:01 +02:00
|
|
|
pctx.Import("android/soong/cc/config")
|
2022-11-10 22:58:48 +01:00
|
|
|
pctx.StaticVariable("relPwd", cc.PwdPrefix())
|
2018-05-17 20:17:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
pctx = android.NewPackageContext("android/soong/bpf")
|
|
|
|
|
2020-01-27 20:19:44 +01:00
|
|
|
ccRule = pctx.AndroidRemoteStaticRule("ccRule", android.RemoteRuleSupports{Goma: true},
|
2018-05-17 20:17:01 +02:00
|
|
|
blueprint.RuleParams{
|
|
|
|
Depfile: "${out}.d",
|
|
|
|
Deps: blueprint.DepsGCC,
|
2022-08-18 00:45:52 +02:00
|
|
|
Command: "$relPwd $ccCmd --target=bpf -c $cFlags -MD -MF ${out}.d -o $out $in",
|
2018-05-17 20:17:01 +02:00
|
|
|
CommandDeps: []string{"$ccCmd"},
|
|
|
|
},
|
|
|
|
"ccCmd", "cFlags")
|
2021-12-03 05:09:45 +01:00
|
|
|
|
|
|
|
stripRule = pctx.AndroidStaticRule("stripRule",
|
|
|
|
blueprint.RuleParams{
|
|
|
|
Command: `$stripCmd --strip-unneeded --remove-section=.rel.BTF ` +
|
|
|
|
`--remove-section=.rel.BTF.ext --remove-section=.BTF.ext $in -o $out`,
|
|
|
|
CommandDeps: []string{"$stripCmd"},
|
|
|
|
},
|
|
|
|
"stripCmd")
|
2018-05-17 20:17:01 +02:00
|
|
|
)
|
|
|
|
|
2021-02-24 19:51:54 +01:00
|
|
|
func registerBpfBuildComponents(ctx android.RegistrationContext) {
|
|
|
|
ctx.RegisterModuleType("bpf", BpfFactory)
|
|
|
|
}
|
|
|
|
|
|
|
|
var PrepareForTestWithBpf = android.FixtureRegisterWithContext(registerBpfBuildComponents)
|
|
|
|
|
2020-09-02 10:23:38 +02:00
|
|
|
// BpfModule interface is used by the apex package to gather information from a bpf module.
|
|
|
|
type BpfModule interface {
|
|
|
|
android.Module
|
|
|
|
|
2021-11-10 15:02:57 +01:00
|
|
|
// Returns the sub install directory if the bpf module is included by apex.
|
|
|
|
SubDir() string
|
2020-09-02 10:23:38 +02:00
|
|
|
}
|
|
|
|
|
2018-05-17 20:17:01 +02:00
|
|
|
type BpfProperties struct {
|
2022-08-11 20:05:13 +02:00
|
|
|
// source paths to the files.
|
|
|
|
Srcs []string `android:"path"`
|
|
|
|
|
|
|
|
// additional cflags that should be used to build the bpf variant of
|
|
|
|
// the C/C++ module.
|
|
|
|
Cflags []string
|
|
|
|
|
|
|
|
// directories (relative to the root of the source tree) that will
|
|
|
|
// be added to the include paths using -I.
|
2018-05-17 20:17:01 +02:00
|
|
|
Include_dirs []string
|
2022-08-11 20:05:13 +02:00
|
|
|
|
|
|
|
// optional subdirectory under which this module is installed into.
|
|
|
|
Sub_dir string
|
|
|
|
|
|
|
|
// if set to true, generate BTF debug info for maps & programs.
|
|
|
|
Btf *bool
|
|
|
|
|
2019-12-12 23:23:42 +01:00
|
|
|
Vendor *bool
|
|
|
|
|
|
|
|
VendorInternal bool `blueprint:"mutated"`
|
2018-05-17 20:17:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type bpf struct {
|
|
|
|
android.ModuleBase
|
|
|
|
|
|
|
|
properties BpfProperties
|
|
|
|
|
|
|
|
objs android.Paths
|
|
|
|
}
|
|
|
|
|
2019-12-12 23:23:42 +01:00
|
|
|
var _ android.ImageInterface = (*bpf)(nil)
|
|
|
|
|
|
|
|
func (bpf *bpf) ImageMutatorBegin(ctx android.BaseModuleContext) {}
|
|
|
|
|
|
|
|
func (bpf *bpf) CoreVariantNeeded(ctx android.BaseModuleContext) bool {
|
|
|
|
return !proptools.Bool(bpf.properties.Vendor)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) RamdiskVariantNeeded(ctx android.BaseModuleContext) bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) VendorRamdiskVariantNeeded(ctx android.BaseModuleContext) bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) DebugRamdiskVariantNeeded(ctx android.BaseModuleContext) bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) RecoveryVariantNeeded(ctx android.BaseModuleContext) bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) ExtraImageVariations(ctx android.BaseModuleContext) []string {
|
|
|
|
if proptools.Bool(bpf.properties.Vendor) {
|
|
|
|
return []string{"vendor"}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-06-13 23:25:45 +02:00
|
|
|
func (bpf *bpf) SetImageVariation(ctx android.BaseModuleContext, variation string) {
|
2019-12-12 23:23:42 +01:00
|
|
|
bpf.properties.VendorInternal = variation == "vendor"
|
|
|
|
}
|
|
|
|
|
2018-05-17 20:17:01 +02:00
|
|
|
func (bpf *bpf) GenerateAndroidBuildActions(ctx android.ModuleContext) {
|
|
|
|
cflags := []string{
|
|
|
|
"-nostdlibinc",
|
2020-03-25 23:01:27 +01:00
|
|
|
|
|
|
|
// Make paths in deps files relative
|
|
|
|
"-no-canonical-prefixes",
|
|
|
|
|
2018-05-17 20:17:01 +02:00
|
|
|
"-O2",
|
|
|
|
"-isystem bionic/libc/include",
|
|
|
|
"-isystem bionic/libc/kernel/uapi",
|
|
|
|
// The architecture doesn't matter here, but asm/types.h is included by linux/types.h.
|
|
|
|
"-isystem bionic/libc/kernel/uapi/asm-arm64",
|
|
|
|
"-isystem bionic/libc/kernel/android/uapi",
|
2023-08-22 05:03:16 +02:00
|
|
|
"-I packages/modules/Connectivity/staticlibs/native/bpf_headers/include/bpf",
|
2020-02-19 00:38:36 +01:00
|
|
|
// TODO(b/149785767): only give access to specific file with AID_* constants
|
|
|
|
"-I system/core/libcutils/include",
|
2018-05-17 20:17:01 +02:00
|
|
|
"-I " + ctx.ModuleDir(),
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, dir := range android.PathsForSource(ctx, bpf.properties.Include_dirs) {
|
|
|
|
cflags = append(cflags, "-I "+dir.String())
|
|
|
|
}
|
|
|
|
|
|
|
|
cflags = append(cflags, bpf.properties.Cflags...)
|
|
|
|
|
2021-12-03 05:09:45 +01:00
|
|
|
if proptools.Bool(bpf.properties.Btf) {
|
|
|
|
cflags = append(cflags, "-g")
|
2022-08-18 00:45:52 +02:00
|
|
|
if runtime.GOOS != "darwin" {
|
|
|
|
cflags = append(cflags, "-fdebug-prefix-map=/proc/self/cwd=")
|
|
|
|
}
|
2021-12-03 05:09:45 +01:00
|
|
|
}
|
|
|
|
|
2019-03-06 07:25:09 +01:00
|
|
|
srcs := android.PathsForModuleSrc(ctx, bpf.properties.Srcs)
|
2018-05-17 20:17:01 +02:00
|
|
|
|
|
|
|
for _, src := range srcs {
|
2022-07-07 11:48:06 +02:00
|
|
|
if strings.ContainsRune(filepath.Base(src.String()), '_') {
|
|
|
|
ctx.ModuleErrorf("invalid character '_' in source name")
|
|
|
|
}
|
2021-12-03 05:09:45 +01:00
|
|
|
obj := android.ObjPathWithExt(ctx, "unstripped", src, "o")
|
2018-05-17 20:17:01 +02:00
|
|
|
|
|
|
|
ctx.Build(pctx, android.BuildParams{
|
2019-05-15 01:05:20 +02:00
|
|
|
Rule: ccRule,
|
2018-05-17 20:17:01 +02:00
|
|
|
Input: src,
|
|
|
|
Output: obj,
|
|
|
|
Args: map[string]string{
|
|
|
|
"cFlags": strings.Join(cflags, " "),
|
|
|
|
"ccCmd": "${config.ClangBin}/clang",
|
|
|
|
},
|
|
|
|
})
|
|
|
|
|
2021-12-03 05:09:45 +01:00
|
|
|
if proptools.Bool(bpf.properties.Btf) {
|
|
|
|
objStripped := android.ObjPathWithExt(ctx, "", src, "o")
|
|
|
|
ctx.Build(pctx, android.BuildParams{
|
2019-12-12 23:23:42 +01:00
|
|
|
Rule: stripRule,
|
|
|
|
Input: obj,
|
2021-12-03 05:09:45 +01:00
|
|
|
Output: objStripped,
|
|
|
|
Args: map[string]string{
|
|
|
|
"stripCmd": "${config.ClangBin}/llvm-strip",
|
|
|
|
},
|
|
|
|
})
|
|
|
|
bpf.objs = append(bpf.objs, objStripped.WithoutRel())
|
|
|
|
} else {
|
|
|
|
bpf.objs = append(bpf.objs, obj.WithoutRel())
|
|
|
|
}
|
|
|
|
|
2018-05-17 20:17:01 +02:00
|
|
|
}
|
2024-02-16 07:35:03 +01:00
|
|
|
|
|
|
|
installDir := android.PathForModuleInstall(ctx, "etc", "bpf")
|
|
|
|
if len(bpf.properties.Sub_dir) > 0 {
|
|
|
|
installDir = installDir.Join(ctx, bpf.properties.Sub_dir)
|
|
|
|
}
|
|
|
|
for _, obj := range bpf.objs {
|
|
|
|
ctx.PackageFile(installDir, obj.Base(), obj)
|
|
|
|
}
|
|
|
|
|
2023-12-14 00:19:49 +01:00
|
|
|
android.SetProvider(ctx, blueprint.SrcsFileProviderKey, blueprint.SrcsFileProviderData{SrcPaths: srcs.Strings()})
|
2024-05-22 23:36:09 +02:00
|
|
|
|
|
|
|
ctx.SetOutputFiles(bpf.objs, "")
|
2018-05-17 20:17:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (bpf *bpf) AndroidMk() android.AndroidMkData {
|
|
|
|
return android.AndroidMkData{
|
|
|
|
Custom: func(w io.Writer, name, prefix, moduleDir string, data android.AndroidMkData) {
|
|
|
|
var names []string
|
|
|
|
fmt.Fprintln(w)
|
|
|
|
fmt.Fprintln(w, "LOCAL_PATH :=", moduleDir)
|
|
|
|
fmt.Fprintln(w)
|
2019-12-12 23:23:42 +01:00
|
|
|
var localModulePath string
|
|
|
|
if bpf.properties.VendorInternal {
|
|
|
|
localModulePath = "LOCAL_MODULE_PATH := $(TARGET_OUT_VENDOR_ETC)/bpf"
|
|
|
|
} else {
|
|
|
|
localModulePath = "LOCAL_MODULE_PATH := $(TARGET_OUT_ETC)/bpf"
|
|
|
|
}
|
2021-11-10 15:02:57 +01:00
|
|
|
if len(bpf.properties.Sub_dir) > 0 {
|
|
|
|
localModulePath += "/" + bpf.properties.Sub_dir
|
|
|
|
}
|
2018-05-17 20:17:01 +02:00
|
|
|
for _, obj := range bpf.objs {
|
|
|
|
objName := name + "_" + obj.Base()
|
|
|
|
names = append(names, objName)
|
2022-12-01 19:49:23 +01:00
|
|
|
fmt.Fprintln(w, "include $(CLEAR_VARS)", " # bpf.bpf.obj")
|
2018-05-17 20:17:01 +02:00
|
|
|
fmt.Fprintln(w, "LOCAL_MODULE := ", objName)
|
|
|
|
fmt.Fprintln(w, "LOCAL_PREBUILT_MODULE_FILE :=", obj.String())
|
|
|
|
fmt.Fprintln(w, "LOCAL_MODULE_STEM :=", obj.Base())
|
|
|
|
fmt.Fprintln(w, "LOCAL_MODULE_CLASS := ETC")
|
2021-11-10 15:02:57 +01:00
|
|
|
fmt.Fprintln(w, localModulePath)
|
2024-01-11 00:42:36 +01:00
|
|
|
// AconfigUpdateAndroidMkData may have added elements to Extra. Process them here.
|
|
|
|
for _, extra := range data.Extra {
|
|
|
|
extra(w, nil)
|
|
|
|
}
|
2018-05-17 20:17:01 +02:00
|
|
|
fmt.Fprintln(w, "include $(BUILD_PREBUILT)")
|
|
|
|
fmt.Fprintln(w)
|
|
|
|
}
|
2022-12-01 19:49:23 +01:00
|
|
|
fmt.Fprintln(w, "include $(CLEAR_VARS)", " # bpf.bpf")
|
2018-05-17 20:17:01 +02:00
|
|
|
fmt.Fprintln(w, "LOCAL_MODULE := ", name)
|
2022-12-08 19:41:33 +01:00
|
|
|
android.AndroidMkEmitAssignList(w, "LOCAL_REQUIRED_MODULES", names)
|
2018-05-17 20:17:01 +02:00
|
|
|
fmt.Fprintln(w, "include $(BUILD_PHONY_PACKAGE)")
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-10 15:02:57 +01:00
|
|
|
func (bpf *bpf) SubDir() string {
|
|
|
|
return bpf.properties.Sub_dir
|
|
|
|
}
|
|
|
|
|
2020-09-02 10:23:38 +02:00
|
|
|
func BpfFactory() android.Module {
|
2018-05-17 20:17:01 +02:00
|
|
|
module := &bpf{}
|
|
|
|
|
|
|
|
module.AddProperties(&module.properties)
|
|
|
|
|
|
|
|
android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
|
|
|
|
return module
|
|
|
|
}
|