aya.rs 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. use std::{
  2. fs::create_dir_all,
  3. path::{Path, PathBuf},
  4. };
  5. use anyhow::{Context as _, Result};
  6. use aya_tool::bindgen;
  7. use crate::codegen::{Architecture, SysrootOptions};
  8. pub fn codegen(opts: &SysrootOptions, libbpf_dir: &Path) -> Result<()> {
  9. codegen_internal_btf_bindings(libbpf_dir)?;
  10. codegen_bindings(opts, libbpf_dir)
  11. }
  12. fn codegen_internal_btf_bindings(libbpf_dir: &Path) -> Result<()> {
  13. let dir = PathBuf::from("aya-obj");
  14. let generated = dir.join("src/generated");
  15. let mut bindgen = bindgen::user_builder()
  16. .clang_args(["-I", libbpf_dir.join("include/uapi").to_str().unwrap()])
  17. .clang_args(["-I", libbpf_dir.join("include").to_str().unwrap()])
  18. .header(libbpf_dir.join("src/libbpf_internal.h").to_str().unwrap())
  19. .constified_enum_module("bpf_core_relo_kind");
  20. let types = ["bpf_core_relo", "btf_ext_header"];
  21. for x in &types {
  22. bindgen = bindgen.allowlist_type(x);
  23. }
  24. let bindings = bindgen.generate().context("bindgen failed")?;
  25. // write the bindings, with the original helpers removed
  26. bindings.write_to_file(generated.join("btf_internal_bindings.rs"))?;
  27. Ok(())
  28. }
  29. fn codegen_bindings(opts: &SysrootOptions, libbpf_dir: &Path) -> Result<()> {
  30. let SysrootOptions {
  31. aarch64_sysroot,
  32. armv7_sysroot,
  33. loongarch64_sysroot,
  34. mips_sysroot,
  35. powerpc64_sysroot,
  36. riscv64_sysroot,
  37. s390x_sysroot,
  38. x86_64_sysroot,
  39. } = opts;
  40. let dir = PathBuf::from("aya-obj");
  41. let generated = dir.join("src/generated");
  42. create_dir_all(&generated)?;
  43. let builder = || {
  44. let mut bindgen = bindgen::user_builder()
  45. .header(dir.join("include/linux_wrapper.h").to_str().unwrap())
  46. .clang_args(["-I", libbpf_dir.join("include/uapi").to_str().unwrap()])
  47. .clang_args(["-I", libbpf_dir.join("include").to_str().unwrap()])
  48. // BPF_F_LINK is defined twice. Once in an anonymous enum
  49. // which bindgen will constify, and once via #define macro
  50. // which generates a duplicate const.
  51. .blocklist_var("BPF_F_LINK")
  52. .constified_enum("BPF_F_.*")
  53. .constified_enum("BTF_KIND_.*")
  54. .constified_enum("BTF_VAR_.*")
  55. .constified_enum("IFLA_.*")
  56. .constified_enum("TCA_.*")
  57. .constified_enum("BPF_RINGBUF_.*")
  58. // NETFILTER
  59. .constified_enum("NFPROTO_.*");
  60. let types = [
  61. // BPF
  62. "bpf_cmd",
  63. "bpf_insn",
  64. "bpf_attr",
  65. "bpf_map_type",
  66. "bpf_prog_type",
  67. "bpf_attach_type",
  68. "bpf_prog_info",
  69. "bpf_map_info",
  70. "bpf_link_info",
  71. "bpf_link_type",
  72. "bpf_btf_info",
  73. "bpf_func_id",
  74. "bpf_func_info",
  75. "bpf_line_info",
  76. "bpf_lpm_trie_key",
  77. "bpf_cpumap_val",
  78. "bpf_devmap_val",
  79. "bpf_stats_type",
  80. "bpf_perf_event_type",
  81. "bpf_task_fd_type",
  82. // BTF
  83. "btf_header",
  84. "btf_ext_info",
  85. "btf_ext_info_sec",
  86. "btf_type",
  87. "btf_enum",
  88. "btf_array",
  89. "btf_member",
  90. "btf_param",
  91. "btf_var",
  92. "btf_var_secinfo",
  93. "btf_func_linkage",
  94. "btf_decl_tag",
  95. // PERF
  96. "perf_event_attr",
  97. "perf_sw_ids",
  98. "perf_hw_id",
  99. "perf_hw_cache_id",
  100. "perf_hw_cache_op_id",
  101. "perf_hw_cache_op_result_id",
  102. "perf_event_sample_format",
  103. "perf_event_mmap_page",
  104. "perf_event_header",
  105. "perf_type_id",
  106. "perf_event_type",
  107. // NETLINK
  108. "ifinfomsg",
  109. "tcmsg",
  110. "nlmsgerr_attrs",
  111. // ITER
  112. "bpf_cgroup_iter_order",
  113. // NETFILTER
  114. "nf_inet_hooks",
  115. ];
  116. let vars = [
  117. // BPF
  118. "BPF_PSEUDO_.*",
  119. "BPF_.*",
  120. "SO_ATTACH_BPF",
  121. "SO_DETACH_BPF",
  122. // BTF
  123. "BTF_INT_.*",
  124. "BTF_KIND_.*",
  125. "BTF_VAR_.*",
  126. // PERF
  127. "PERF_FLAG_.*",
  128. "PERF_EVENT_.*",
  129. "PERF_MAX_.*",
  130. // NETLINK
  131. "NLMSG_ALIGNTO",
  132. "IFLA_XDP_FD",
  133. "TCA_KIND",
  134. "TCA_OPTIONS",
  135. "TCA_BPF_FD",
  136. "TCA_BPF_NAME",
  137. "TCA_BPF_FLAGS",
  138. "TCA_BPF_FLAG_ACT_DIRECT",
  139. "XDP_FLAGS_.*",
  140. "TC_H_MAJ_MASK",
  141. "TC_H_MIN_MASK",
  142. "TC_H_UNSPEC",
  143. "TC_H_ROOT",
  144. "TC_H_INGRESS",
  145. "TC_H_CLSACT",
  146. "TC_H_MIN_PRIORITY",
  147. "TC_H_MIN_INGRESS",
  148. "TC_H_MIN_EGRESS",
  149. // Ringbuf
  150. "BPF_RINGBUF_.*",
  151. // NETFILTER
  152. "NFPROTO_.*",
  153. ];
  154. for x in &types {
  155. bindgen = bindgen.allowlist_type(x);
  156. }
  157. for x in &vars {
  158. bindgen = bindgen.allowlist_var(x);
  159. }
  160. bindgen
  161. };
  162. for arch in Architecture::supported() {
  163. let mut bindgen = builder();
  164. // Set target triple. This will set the right flags (which you can see
  165. // running clang -target=X -E - -dM </dev/null)
  166. let target = match arch {
  167. Architecture::AArch64 => "aarch64-unknown-linux-gnu",
  168. Architecture::ARMv7 => "armv7-unknown-linux-gnu",
  169. Architecture::LoongArch64 => "loongarch64-unknown-linux-gnu",
  170. Architecture::Mips => "mips-unknown-linux-gnu",
  171. Architecture::PowerPC64 => "powerpc64le-unknown-linux-gnu",
  172. Architecture::RISCV64 => "riscv64-unknown-linux-gnu",
  173. Architecture::S390X => "s390x-unknown-linux-gnu",
  174. Architecture::X86_64 => "x86_64-unknown-linux-gnu",
  175. };
  176. bindgen = bindgen.clang_args(&["-target", target]);
  177. // Set the sysroot. This is needed to ensure that the correct arch
  178. // specific headers are imported.
  179. let sysroot = match arch {
  180. Architecture::AArch64 => aarch64_sysroot,
  181. Architecture::ARMv7 => armv7_sysroot,
  182. Architecture::LoongArch64 => loongarch64_sysroot,
  183. Architecture::Mips => mips_sysroot,
  184. Architecture::PowerPC64 => powerpc64_sysroot,
  185. Architecture::RISCV64 => riscv64_sysroot,
  186. Architecture::S390X => s390x_sysroot,
  187. Architecture::X86_64 => x86_64_sysroot,
  188. };
  189. bindgen = bindgen.clang_args(["-I", sysroot.to_str().unwrap()]);
  190. let bindings = bindgen.generate().context("bindgen failed")?;
  191. // write the bindings, with the original helpers removed
  192. bindings.write_to_file(generated.join(format!("linux_bindings_{arch}.rs")))?;
  193. }
  194. Ok(())
  195. }