build.rs 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. use std::{
  2. env,
  3. ffi::OsString,
  4. fs,
  5. path::PathBuf,
  6. process::{Child, Command, Output, Stdio},
  7. };
  8. use aya_build::cargo_metadata::{Metadata, MetadataCommand, Package, Target, TargetKind};
  9. use xtask::{exec, AYA_BUILD_INTEGRATION_BPF, LIBBPF_DIR};
  10. /// This file, along with the xtask crate, allows analysis tools such as `cargo check`, `cargo
  11. /// clippy`, and even `cargo build` to work as users expect. Prior to this file's existence, this
  12. /// crate's undeclared dependency on artifacts from `integration-ebpf` would cause build (and `cargo check`,
  13. /// and `cargo clippy`) failures until the user ran certain other commands in the workspace. Conversely,
  14. /// those same tools (e.g. cargo test --no-run) would produce stale results if run naively because
  15. /// they'd make use of artifacts from a previous build of `integration-ebpf`.
  16. ///
  17. /// Note that this solution is imperfect: in particular it has to balance correctness with
  18. /// performance; an environment variable is used to replace true builds of `integration-ebpf` with
  19. /// stubs to preserve the property that code generation and linking (in `integration-ebpf`) do not
  20. /// occur on metadata-only actions such as `cargo check` or `cargo clippy` of this crate. This means
  21. /// that naively attempting to `cargo test --no-run` this crate will produce binaries that fail at
  22. /// runtime because the stubs are inadequate for actually running the tests.
  23. fn main() {
  24. println!("cargo:rerun-if-env-changed={}", AYA_BUILD_INTEGRATION_BPF);
  25. // TODO(https://github.com/rust-lang/cargo/issues/4001): generalize this and move it to
  26. // aya-build if we can determine that we're in a check build.
  27. let build_integration_bpf = env::var(AYA_BUILD_INTEGRATION_BPF)
  28. .as_deref()
  29. .map(str::parse)
  30. .map(Result::unwrap)
  31. .unwrap_or_default();
  32. let Metadata { packages, .. } = MetadataCommand::new().no_deps().exec().unwrap();
  33. let integration_ebpf_package = packages
  34. .into_iter()
  35. .find(|Package { name, .. }| name == "integration-ebpf")
  36. .unwrap();
  37. let manifest_dir = env::var_os("CARGO_MANIFEST_DIR").unwrap();
  38. let manifest_dir = PathBuf::from(manifest_dir);
  39. let out_dir = env::var_os("OUT_DIR").unwrap();
  40. let out_dir = PathBuf::from(out_dir);
  41. const C_BPF: &[(&str, bool)] = &[
  42. ("ext.bpf.c", false),
  43. ("iter.bpf.c", true),
  44. ("main.bpf.c", false),
  45. ("multimap-btf.bpf.c", false),
  46. ("reloc.bpf.c", true),
  47. ("text_64_64_reloc.c", false),
  48. ("variables_reloc.bpf.c", false),
  49. ];
  50. if build_integration_bpf {
  51. let endian = env::var_os("CARGO_CFG_TARGET_ENDIAN").unwrap();
  52. let target = if endian == "big" {
  53. "bpfeb"
  54. } else if endian == "little" {
  55. "bpfel"
  56. } else {
  57. panic!("unsupported endian={:?}", endian)
  58. };
  59. let libbpf_dir = manifest_dir
  60. .parent()
  61. .unwrap()
  62. .parent()
  63. .unwrap()
  64. .join(LIBBPF_DIR);
  65. println!("cargo:rerun-if-changed={}", libbpf_dir.to_str().unwrap());
  66. let libbpf_headers_dir = out_dir.join("libbpf_headers");
  67. let mut includedir = OsString::new();
  68. includedir.push("INCLUDEDIR=");
  69. includedir.push(&libbpf_headers_dir);
  70. exec(
  71. Command::new("make")
  72. .arg("-C")
  73. .arg(libbpf_dir.join("src"))
  74. .arg(includedir)
  75. .arg("install_headers"),
  76. )
  77. .unwrap();
  78. let bpf_dir = manifest_dir.join("bpf");
  79. let mut target_arch = OsString::new();
  80. target_arch.push("-D__TARGET_ARCH_");
  81. let arch = env::var_os("CARGO_CFG_TARGET_ARCH").unwrap();
  82. if arch == "x86_64" {
  83. target_arch.push("x86");
  84. } else if arch == "aarch64" {
  85. target_arch.push("arm64");
  86. } else {
  87. target_arch.push(&arch);
  88. };
  89. // NB: libbpf's documentation suggests that vmlinux.h be generated by running `bpftool btf
  90. // dump file /sys/kernel/btf/vmlinux format c`; this allows CO-RE to work.
  91. //
  92. // However in our tests we do not make use of kernel data structures, and so any vmlinux.h
  93. // which defines the constants we need (e.g. `__u8`, `__u64`, `BPF_MAP_TYPE_ARRAY`,
  94. // `BPF_ANY`, `XDP_PASS`, `XDP_DROP`, etc.) will suffice. Since we already have a libbpf
  95. // submodule which happens to include such a file, we use it.
  96. let libbpf_vmlinux_dir = libbpf_dir.join(".github/actions/build-selftests");
  97. let clang = || {
  98. let mut cmd = Command::new("clang");
  99. cmd.arg("-nostdlibinc")
  100. .arg("-I")
  101. .arg(&libbpf_headers_dir)
  102. .arg("-I")
  103. .arg(&libbpf_vmlinux_dir)
  104. .args(["-g", "-O2", "-target", target, "-c"])
  105. .arg(&target_arch);
  106. cmd
  107. };
  108. for (src, build_btf) in C_BPF {
  109. let dst = out_dir.join(src).with_extension("o");
  110. let src = bpf_dir.join(src);
  111. println!("cargo:rerun-if-changed={}", src.to_str().unwrap());
  112. exec(clang().arg(&src).arg("-o").arg(&dst)).unwrap();
  113. if *build_btf {
  114. let mut cmd = clang();
  115. let mut child = cmd
  116. .arg("-DTARGET")
  117. .arg(&src)
  118. .args(["-o", "-"])
  119. .stdout(Stdio::piped())
  120. .spawn()
  121. .unwrap_or_else(|err| panic!("failed to spawn {cmd:?}: {err}"));
  122. let Child { stdout, .. } = &mut child;
  123. let stdout = stdout.take().unwrap();
  124. let dst = dst.with_extension("target.o");
  125. let mut output = OsString::new();
  126. output.push(".BTF=");
  127. output.push(dst);
  128. exec(
  129. // NB: objcopy doesn't support reading from stdin, so we have to use llvm-objcopy.
  130. Command::new("llvm-objcopy")
  131. .arg("--dump-section")
  132. .arg(output)
  133. .arg("-")
  134. .stdin(stdout),
  135. )
  136. .unwrap();
  137. let output = child
  138. .wait_with_output()
  139. .unwrap_or_else(|err| panic!("failed to wait for {cmd:?}: {err}"));
  140. let Output { status, .. } = &output;
  141. assert_eq!(status.code(), Some(0), "{cmd:?} failed: {output:?}");
  142. }
  143. }
  144. aya_build::build_ebpf([integration_ebpf_package]).unwrap();
  145. } else {
  146. for (src, build_btf) in C_BPF {
  147. let dst = out_dir.join(src).with_extension("o");
  148. fs::write(&dst, []).unwrap_or_else(|err| panic!("failed to create {dst:?}: {err}"));
  149. if *build_btf {
  150. let dst = dst.with_extension("target.o");
  151. fs::write(&dst, []).unwrap_or_else(|err| panic!("failed to create {dst:?}: {err}"));
  152. }
  153. }
  154. let Package { targets, .. } = integration_ebpf_package;
  155. for Target { name, kind, .. } in targets {
  156. if *kind != [TargetKind::Bin] {
  157. continue;
  158. }
  159. let dst = out_dir.join(name);
  160. fs::write(&dst, []).unwrap_or_else(|err| panic!("failed to create {dst:?}: {err}"));
  161. }
  162. }
  163. }