The motivation behind the change is to give BPF LSM developers the ability to report accesses via the audit subsystem much like how LSMs operate today.
Series: Patch 1: Introduces bpf_audit_*() kfuncs Patch 2: Enables bpf_audit_*() kfuns Patch 3: Prepares audit helpers used for testing Patch 4: Adds self tests Documentation will be added when this becomes a versioned series. Key features: 1. Audit logs include type=AUDIT_BPF_LSM_ACCESS, BPF program ID, and comm that triggered the hook by default We wanted audit log consumers to be able to track who and what created the entry. prog-id=%d is already used for BPF LOAD/UNLOAD logs, thus is reused here for this distinction. Though, it may be better to use the tag instead to capture which _specific_ version of the program made the log, since prog-id can be reused. 2. Leverages BPF KF_AQUIRE/KF_RELEASE semantics to force use of bpf_audit_log_end(). One side effect of this decision is that the BPF documentation states that these flags allow the pointer to struct bpf_audit_context to be stored in a map, and then exchanged through bpf_kptr_xchg(). However, there's prior work with net/netfilter/nf_conntrack_bpf.c such that the struct is not exposed as a kptr to support that functionality nor is that supplying a dtor function. The verifier will not allow this use case due to not exposing the __kptr. Ideally, we don't want the pointer to be exchanged anyway because the reporting program can become ambiguous. I am sure there are other edge cases WRT to keeping the audit buffer in a strange state too that I cannot think of at this moment. 3. All bpf_audit_log_*() functions are destructive The audit subsystem allows for AUDIT_FAIL_PANIC to be set when the subsystem can detect that missing events. Further, some call paths may invoke a BUG_ON(). Therefore all the functions are marked destructive. 4. Functions are callable once per bpf_audit_context The rationale for this was to prevent abuse. Logs with repeated fields are not helpful, and may not be handled by user space audit coherently. This is in the same vein as not providing a audit_format() wrapper. Similarly, some functions such as bpf_audit_log_path() and bpf_audit_log_file() report the same information, thus can be interchangeable in use. 5. API wraps security/lsm_audit.c lsm_audit.c functions are multiplexed and not handled by BPF verifier very well, thus the wrapped functions are isolated to their sole purpose for use within hooks. Key considerations: 1. Audit field ordering AFAIK, user space audit is particular about what fields are present and their order. This patch series does not address ordering. My assumption is that the first three fields: type, prog-id, pid, comm are well known, and user space can make an assumption that other fields after those can appear in any order. If that is not acceptable, I would propose that we leverage the struct common_audit_data type order to be the order--much like how the type is used for log_once() functionality. I am open to other ideas. Signed-off-by: Frederick Lawler <[email protected]> --- Frederick Lawler (4): audit: Implement bpf_audit_log_*() wrappers audit/security: Enable audit BPF kfuncs selftests/bpf: Add audit helpers for BPF tests selftests/bpf: Add lsm_audit_kfuncs tests include/linux/lsm_audit.h | 1 + include/uapi/linux/audit.h | 1 + security/Makefile | 2 + security/lsm_audit_kfuncs.c | 306 +++++++++++ tools/testing/selftests/bpf/Makefile | 3 +- tools/testing/selftests/bpf/audit_helpers.c | 281 ++++++++++ tools/testing/selftests/bpf/audit_helpers.h | 55 ++ .../selftests/bpf/prog_tests/lsm_audit_kfuncs.c | 598 +++++++++++++++++++++ .../selftests/bpf/progs/test_lsm_audit_kfuncs.c | 263 +++++++++ 9 files changed, 1509 insertions(+), 1 deletion(-) --- base-commit: ca0f39a369c5f927c3d004e63a5a778b08a9df94 change-id: 20260105-bpf-auditd-send-message-4a883067aab8 Best regards, -- Frederick Lawler <[email protected]>

