Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

bpf: read and copy proc exe at execve for matchBinaries #1926

Merged
merged 2 commits into from
Jan 11, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions bpf/lib/bpf_task.h
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,9 @@

/* __d_path_local flags */
// #define UNRESOLVED_MOUNT_POINTS 0x01 // (deprecated)
// this error is returned by __d_path_local in the following cases:
// - the path walk did not conclude (too many dentry)
// - the path was too long to fit in the buffer
#define UNRESOLVED_PATH_COMPONENTS 0x02

#ifdef __LARGE_BPF_PROG
Expand Down
18 changes: 16 additions & 2 deletions bpf/lib/process.h
Original file line number Diff line number Diff line change
Expand Up @@ -270,6 +270,19 @@ struct msg_k8s {
char docker_id[DOCKER_ID_LENGTH];
}; // All fields aligned so no 'packed' attribute.

#define BINARY_PATH_MAX_LEN 256

struct heap_exe {
// because of verifier limitations, this has to be 2 * 256 bytes while 256
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this true with large programs? We might be able to widen this some if we feel its an actual issue.

// should be theoretically sufficient, and actually is, in unit tests.
char buf[BINARY_PATH_MAX_LEN * 2];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If we were a bit smarter this can be improved 2x waste of space. Over 256B though I don't care to do the work.

// offset points to the start of the path in the above buffer. Use offset to
// read the path in the buffer since it's written from the end.
char *off;
__u32 len;
__u32 error;
}; // All fields aligned so no 'packed' attribute.

struct msg_execve_event {
struct msg_common common;
struct msg_k8s kube;
Expand All @@ -289,10 +302,11 @@ struct msg_execve_event {
/* below fields are not part of the event, serve just as
* heap for execve programs
*/
#ifdef __LARGE_BPF_PROG
struct heap_exe exe;
#endif
}; // All fields aligned so no 'packed' attribute.

#define BINARY_PATH_MAX_LEN 256

// This structure stores the binary path that was recorded on execve.
// Technically PATH_MAX is 4096 but we limit the length we store since we have
// limits on the length of the string to compare:
Expand Down
34 changes: 34 additions & 0 deletions bpf/process/bpf_execve_event.c
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,23 @@ read_execve_shared_info(void *ctx, __u64 pid)
return secureexec;
}

#ifdef __LARGE_BPF_PROG
static inline __attribute__((always_inline)) __u32
read_exe(struct task_struct *task, struct heap_exe *exe)
{
struct file *file = BPF_CORE_READ(task, mm, exe_file);
struct path *path = __builtin_preserve_access_index(&file->f_path);

exe->len = BINARY_PATH_MAX_LEN;
exe->off = (char *)&exe->buf;
exe->off = __d_path_local(path, exe->off, (int *)&exe->len, (int *)&exe->error);
if (exe->len > 0)
exe->len = BINARY_PATH_MAX_LEN - exe->len;

return exe->len;
}
#endif

__attribute__((section("tracepoint/sys_execve"), used)) int
event_execve(struct sched_execve_args *ctx)
{
Expand Down Expand Up @@ -184,6 +201,14 @@ event_execve(struct sched_execve_args *ctx)
p->auid = get_auid();
p->uid = get_current_uid_gid();

// Reading the absolute path of the process exe for matchBinaries.
// Historically we used the filename, a potentially relative path (maybe to
// a symlink) coming from the execve tracepoint. For kernels not supporting
// large BPF prog, we still use the filename.
#ifdef __LARGE_BPF_PROG
read_exe(task, &event->exe);
#endif

p->size += read_path(ctx, event, filename);
p->size += read_args(ctx, event);
p->size += read_cwd(ctx, p);
Expand Down Expand Up @@ -265,6 +290,14 @@ execve_send(struct sched_execve_args *ctx)
// buffer can be written at clone stage with parent's info, if previous
// path is longer than current, we can have leftovers at the end.
memset(&curr->bin, 0, sizeof(curr->bin));
#ifdef __LARGE_BPF_PROG
// read from proc exe stored at execve time
if (event->exe.len <= BINARY_PATH_MAX_LEN) {
curr->bin.path_length = probe_read(curr->bin.path, event->exe.len, event->exe.off);
if (curr->bin.path_length == 0)
curr->bin.path_length = event->exe.len;
}
#else
// reuse p->args first string that contains the filename, this can't be
// above 256 in size (otherwise the complete will be send via data msg)
// which is okay because we need the 256 first bytes.
Expand All @@ -273,6 +306,7 @@ execve_send(struct sched_execve_args *ctx)
// don't include the NULL byte in the length
curr->bin.path_length--;
}
#endif
}

event->common.flags = 0;
Expand Down
Loading