Add "btf" option that generates BTF debug info to support easier map inspection. This is accomplished by passing the "-g" flag to clang when compiling the BPF program. The "-g" option also generates a number of DWARF debug sections which are not necessary for loading BTF information, so strip these to avoid increasing file size unnecessarily. bpfloader currently only supports BTF info for maps, not programs, so we also strip the .BTF.ext section containing program BTF info. Bug: 203823368 Test: libbpf_load_test Test: verify time_in_state.o includes .BTF section iff "btf: true" is set Test: verify time_in_state.o still loads if BTF is enabled Change-Id: Ica25b253bace59d04130b0210350188399889bbe Signed-off-by: Connor O'Brien <connoro@google.com>
208 lines
5.8 KiB
Go
208 lines
5.8 KiB
Go
// 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"
|
|
"strings"
|
|
|
|
"android/soong/android"
|
|
_ "android/soong/cc/config"
|
|
|
|
"github.com/google/blueprint"
|
|
"github.com/google/blueprint/proptools"
|
|
)
|
|
|
|
func init() {
|
|
registerBpfBuildComponents(android.InitRegistrationContext)
|
|
pctx.Import("android/soong/cc/config")
|
|
}
|
|
|
|
var (
|
|
pctx = android.NewPackageContext("android/soong/bpf")
|
|
|
|
ccRule = pctx.AndroidRemoteStaticRule("ccRule", android.RemoteRuleSupports{Goma: true},
|
|
blueprint.RuleParams{
|
|
Depfile: "${out}.d",
|
|
Deps: blueprint.DepsGCC,
|
|
Command: "$ccCmd --target=bpf -c $cFlags -MD -MF ${out}.d -o $out $in",
|
|
CommandDeps: []string{"$ccCmd"},
|
|
},
|
|
"ccCmd", "cFlags")
|
|
|
|
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")
|
|
)
|
|
|
|
func registerBpfBuildComponents(ctx android.RegistrationContext) {
|
|
ctx.RegisterModuleType("bpf", BpfFactory)
|
|
}
|
|
|
|
var PrepareForTestWithBpf = android.FixtureRegisterWithContext(registerBpfBuildComponents)
|
|
|
|
// BpfModule interface is used by the apex package to gather information from a bpf module.
|
|
type BpfModule interface {
|
|
android.Module
|
|
|
|
OutputFiles(tag string) (android.Paths, error)
|
|
|
|
// Returns the sub install directory if the bpf module is included by apex.
|
|
SubDir() string
|
|
}
|
|
|
|
type BpfProperties struct {
|
|
Srcs []string `android:"path"`
|
|
Cflags []string
|
|
Include_dirs []string
|
|
Sub_dir string
|
|
// If set to true, generate BTF debug info for maps & programs
|
|
Btf *bool
|
|
}
|
|
|
|
type bpf struct {
|
|
android.ModuleBase
|
|
|
|
properties BpfProperties
|
|
|
|
objs android.Paths
|
|
}
|
|
|
|
func (bpf *bpf) GenerateAndroidBuildActions(ctx android.ModuleContext) {
|
|
cflags := []string{
|
|
"-nostdlibinc",
|
|
|
|
// Make paths in deps files relative
|
|
"-no-canonical-prefixes",
|
|
|
|
"-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",
|
|
"-I frameworks/libs/net/common/native/bpf_headers/include/bpf",
|
|
// TODO(b/149785767): only give access to specific file with AID_* constants
|
|
"-I system/core/libcutils/include",
|
|
"-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...)
|
|
|
|
if proptools.Bool(bpf.properties.Btf) {
|
|
cflags = append(cflags, "-g")
|
|
}
|
|
|
|
srcs := android.PathsForModuleSrc(ctx, bpf.properties.Srcs)
|
|
|
|
for _, src := range srcs {
|
|
obj := android.ObjPathWithExt(ctx, "unstripped", src, "o")
|
|
|
|
ctx.Build(pctx, android.BuildParams{
|
|
Rule: ccRule,
|
|
Input: src,
|
|
Output: obj,
|
|
Args: map[string]string{
|
|
"cFlags": strings.Join(cflags, " "),
|
|
"ccCmd": "${config.ClangBin}/clang",
|
|
},
|
|
})
|
|
|
|
if proptools.Bool(bpf.properties.Btf) {
|
|
objStripped := android.ObjPathWithExt(ctx, "", src, "o")
|
|
ctx.Build(pctx, android.BuildParams{
|
|
Rule: stripRule,
|
|
Input: obj,
|
|
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())
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
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)
|
|
localModulePath := "LOCAL_MODULE_PATH := $(TARGET_OUT_ETC)/bpf"
|
|
if len(bpf.properties.Sub_dir) > 0 {
|
|
localModulePath += "/" + bpf.properties.Sub_dir
|
|
}
|
|
for _, obj := range bpf.objs {
|
|
objName := name + "_" + obj.Base()
|
|
names = append(names, objName)
|
|
fmt.Fprintln(w, "include $(CLEAR_VARS)")
|
|
fmt.Fprintln(w, "LOCAL_MODULE := ", objName)
|
|
data.Entries.WriteLicenseVariables(w)
|
|
fmt.Fprintln(w, "LOCAL_PREBUILT_MODULE_FILE :=", obj.String())
|
|
fmt.Fprintln(w, "LOCAL_MODULE_STEM :=", obj.Base())
|
|
fmt.Fprintln(w, "LOCAL_MODULE_CLASS := ETC")
|
|
fmt.Fprintln(w, localModulePath)
|
|
fmt.Fprintln(w, "include $(BUILD_PREBUILT)")
|
|
fmt.Fprintln(w)
|
|
}
|
|
fmt.Fprintln(w, "include $(CLEAR_VARS)")
|
|
fmt.Fprintln(w, "LOCAL_MODULE := ", name)
|
|
data.Entries.WriteLicenseVariables(w)
|
|
fmt.Fprintln(w, "LOCAL_REQUIRED_MODULES :=", strings.Join(names, " "))
|
|
fmt.Fprintln(w, "include $(BUILD_PHONY_PACKAGE)")
|
|
},
|
|
}
|
|
}
|
|
|
|
// Implements OutputFileFileProducer interface so that the obj output can be used in the data property
|
|
// of other modules.
|
|
func (bpf *bpf) OutputFiles(tag string) (android.Paths, error) {
|
|
switch tag {
|
|
case "":
|
|
return bpf.objs, nil
|
|
default:
|
|
return nil, fmt.Errorf("unsupported module reference tag %q", tag)
|
|
}
|
|
}
|
|
|
|
func (bpf *bpf) SubDir() string {
|
|
return bpf.properties.Sub_dir
|
|
}
|
|
|
|
var _ android.OutputFileProducer = (*bpf)(nil)
|
|
|
|
func BpfFactory() android.Module {
|
|
module := &bpf{}
|
|
|
|
module.AddProperties(&module.properties)
|
|
|
|
android.InitAndroidArchModule(module, android.DeviceSupported, android.MultilibCommon)
|
|
return module
|
|
}
|