On Thu, Mar 30, 2023 at 10:12 AM Lorenz Bauer <lmb@xxxxxxxxxxxxx> wrote: > > On Wed, Mar 29, 2023 at 12:56 AM Andrii Nakryiko <andrii@xxxxxxxxxx> wrote: > > I like the approach! My main problem is always following the various > offset calculations. I'm still not convinced I understand them fully. > > For example, there is a bunch of accounting for the implicit 0 that is > written out to user space. Wouldn't it be easier to not terminate the > buffer during normal operation, and instead only do that in > vlog_finalize()? So I'm preserving the original behavior, but I also think the original behavior makes sense, as it tries to keep valid string contents at all times. At least I don't really see much problem with it, do you? > > Another example is start_pos in bpf_verifier_log. In a way, the > verifier log is a bit peculiar as ring buffers go, since there is no > reader we need to account for. It seems to me like we don't need to > track start_pos at all? We can deduce that truncation has occurred if > end_pos >= total_len. Similarly, the number of useful bytes is > min(end_pos, total_len). We might need to track the total required > size later (haven't had time to look at your other patch set, sorry), > but then we could do that independent of the ring buffer > implementation. Hm... start_pos definitely is necessary due to bpf_vlog_reset() at least. Similarly, end_pos can go back on log reset. But second patch set simplifies this further, as we keep track of maximum log content size we ever reach, and that could be straightforwardly compared to log->len_total. > > In my mind we could end up with something like this: > > void bpf_verifier_vlog(...) { > ... > if (log->level & BPF_LOG_FIXED) > n = min(n, bpf_vlog_available()); // TODO: Need to signal ENOSPC > somehow, maybe just a bool? > > bpf_vlog_append(..., n) > } > > bpf_vlog_append() would only deal with writing into the buffer, no > BPF_LOG_FIXED specific code. Null termination would happen in > bpf_vlog_finalize. I'm not following. bpf_vlog_append() would have to distinguish between BPF_LOG_FIXED and rotating mode, because in rotating mode you have to wrap around the physical end of the buffer. > > Some more thoughts below: > > > #define BPF_LOG_LEVEL1 1 > > #define BPF_LOG_LEVEL2 2 > > #define BPF_LOG_STATS 4 > > +#define BPF_LOG_FIXED 8 > > Nit: how about calling this BPF_LOG_TRUNCATE_TAIL instead? FIXED only > makes sense with the context that we implement this using a (rotating) > ring buffer. It's more verbose, so BPF_LOG_FIXED seems more in line with existing constants names. But note that this is not part of UAPI, user-space won't see/have "BPF_LOG_FIXED" constant. > > > --- a/kernel/bpf/log.c > > +++ b/kernel/bpf/log.c > > @@ -8,6 +8,7 @@ > > #include <linux/types.h> > > #include <linux/bpf.h> > > #include <linux/bpf_verifier.h> > > +#include <linux/math64.h> > > > > bool bpf_verifier_log_attr_valid(const struct bpf_verifier_log *log) > > { > > @@ -32,23 +33,199 @@ void bpf_verifier_vlog(struct bpf_verifier_log *log, const char *fmt, > > return; > > } > > > > - n = min(log->len_total - log->len_used - 1, n); > > - log->kbuf[n] = '\0'; > > - if (!copy_to_user(log->ubuf + log->len_used, log->kbuf, n + 1)) > > - log->len_used += n; > > - else > > - log->ubuf = NULL; > > + if (log->level & BPF_LOG_FIXED) { > > + n = min(log->len_total - bpf_log_used(log) - 1, n); > > Noob question, doesn't log->len_total - bpf_log_used(log) - 1 > underflow if the log is full? Maybe worth turning this into > bpf_vlog_available() as well. It's a valid question. The answer is that currently bpf_verifier_vlog() assumes log is not completely full, because all the callers first check bpf_verifier_log_needed(), which returns false if the log buffer is full. I'm removing this assumption (and the whole "is log full" check) in the follow up patch set. > > > + log->kbuf[n] = '\0'; > > + n += 1; > > I personally find the original style of doing copy_to_user(..., n+1) > instead of adding and subtracting from n easier to read. Still prefer > to terminate in finalize(). > This makes the rest of the math (especially for rotating mode) simpler with no extra +1/-1. See how many uses of n are below, and imagine that we'll have to do +1 there. Conceptually, n is "how much stuff we write to user-space", so it makes sense if we write a zero-terminating byte. > > + > > + if (copy_to_user(log->ubuf + log->end_pos, log->kbuf, n)) > > + goto fail; > > + > > + log->end_pos += n - 1; /* don't count terminating '\0' */ > > + } else { > > + u64 new_end, new_start, cur_pos; > > + u32 buf_start, buf_end, new_n; > > + > > + log->kbuf[n] = '\0'; > > + n += 1; > > + > > + new_end = log->end_pos + n; > > + if (new_end - log->start_pos >= log->len_total) > > + new_start = new_end - log->len_total; > > + else > > + new_start = log->start_pos; > > + new_n = min(n, log->len_total); > > + cur_pos = new_end - new_n; > > + > > + div_u64_rem(cur_pos, log->len_total, &buf_start); > > + div_u64_rem(new_end, log->len_total, &buf_end); > > + /* new_end and buf_end are exclusive indices, so if buf_end is > > + * exactly zero, then it actually points right to the end of > > + * ubuf and there is no wrap around > > + */ > > + if (buf_end == 0) > > + buf_end = log->len_total; > > + > > + /* if buf_start > buf_end, we wrapped around; > > + * if buf_start == buf_end, then we fill ubuf completely; we > > + * can't have buf_start == buf_end to mean that there is > > + * nothing to write, because we always write at least > > + * something, even if terminal '\0' > > + */ > > + if (buf_start < buf_end) { > > + /* message fits within contiguous chunk of ubuf */ > > + if (copy_to_user(log->ubuf + buf_start, > > + log->kbuf + n - new_n, > > + buf_end - buf_start)) > > + goto fail; > > + } else { > > + /* message wraps around the end of ubuf, copy in two chunks */ > > + if (copy_to_user(log->ubuf + buf_start, > > + log->kbuf + n - new_n, > > + log->len_total - buf_start)) > > + goto fail; > > + if (copy_to_user(log->ubuf, > > + log->kbuf + n - buf_end, > > + buf_end)) > > + goto fail; > > + } > > + > > + log->start_pos = new_start; > > + log->end_pos = new_end - 1; /* don't count terminating '\0' */ > > + } > > + > > + return; > > +fail: > > + log->ubuf = NULL; > > } > > > > -void bpf_vlog_reset(struct bpf_verifier_log *log, u32 new_pos) > > +void bpf_vlog_reset(struct bpf_verifier_log *log, u64 new_pos) > > { > > char zero = 0; > > + u32 pos; > > > > if (!bpf_verifier_log_needed(log)) > > return; > > > > - log->len_used = new_pos; > > - if (put_user(zero, log->ubuf + new_pos)) > > + /* if position to which we reset is beyond current log window, > > + * then we didn't preserve any useful content and should adjust > > + * start_pos to end up with an empty log (start_pos == end_pos) > > + */ > > + log->end_pos = new_pos; > > Is it valid for new_pos to be > log->end_pos? If not it would be nice > to bail out here (maybe with a WARN_ON_ONCE?). Then this function > could become bpf_vlog_truncate. new_pos should always be <= end_pos, unless there is some bug somewhere else. It's unlikely we'll have a bug with this, because verifier code just remembers previous end_pos temporarily and resets to it, it doesn't do any match on that, I think. I can add this extra check, but this feels like it falls into the "unnecessary defensive code" bucket, so I expect a push back for that. > > > + if (log->end_pos < log->start_pos) > > + log->start_pos = log->end_pos; > > + div_u64_rem(new_pos, log->len_total, &pos); > > + if (put_user(zero, log->ubuf + pos)) > > + log->ubuf = NULL; > > +} > > + > > +static void bpf_vlog_reverse_kbuf(char *buf, int len) > > This isn't really kbuf specific, how about just reverse_buf? kbuf as opposed to ubuf. Kernel-space manipulations, which don't require copy_from_user. I wanted to emphasize this distinction and keep symmetry with bpf_vlog_reverse_ubuf(). > > > +{ > > + int i, j; > > + > > + for (i = 0, j = len - 1; i < j; i++, j--) > > + swap(buf[i], buf[j]); > > +} > > + > > +static int bpf_vlog_reverse_ubuf(struct bpf_verifier_log *log, int start, int end) > > Oh boy, I spent a while trying to understand this, then trying to come > up with something simpler. Neither of which I was very successful with > :o) yeah, it's conceptually very simple, but because of kernel-user boundary and buffering, we need to do it two symmetrical chunks at a time > > > +{ > > + /* we split log->kbuf into two equal parts for both ends of array */ > > + int n = sizeof(log->kbuf) / 2, nn; > > + char *lbuf = log->kbuf, *rbuf = log->kbuf + n; > > + > > + /* Read ubuf's section [start, end) two chunks at a time, from left > > + * and right side; within each chunk, swap all the bytes; after that > > + * reverse the order of lbuf and rbuf and write result back to ubuf. > > + * This way we'll end up with swapped contents of specified > > + * [start, end) ubuf segment. > > + */ > > + while (end - start > 1) { > > + nn = min(n, (end - start ) / 2); > > + > > + if (copy_from_user(lbuf, log->ubuf + start, nn)) > > + return -EFAULT; > > + if (copy_from_user(rbuf, log->ubuf + end - nn, nn)) > > + return -EFAULT; > > + > > + bpf_vlog_reverse_kbuf(lbuf, nn); > > + bpf_vlog_reverse_kbuf(rbuf, nn); > > + > > + /* we write lbuf to the right end of ubuf, while rbuf to the > > + * left one to end up with properly reversed overall ubuf > > + */ > > + if (copy_to_user(log->ubuf + start, rbuf, nn)) > > + return -EFAULT; > > + if (copy_to_user(log->ubuf + end - nn, lbuf, nn)) > > + return -EFAULT; > > + > > + start += nn; > > + end -= nn; > > + } > > + > > + return 0; > > +} > > + > > +bool bpf_vlog_truncated(const struct bpf_verifier_log *log) > > +{ > > + if (log->level & BPF_LOG_FIXED) > > + return bpf_log_used(log) >= log->len_total - 1; > > Can this ever return true? In verifier_vlog we avoid writing more than > total - bpf_log_used. Seems like your new test case covers this, so > I'm a bit lost... I think so. bpf_log_used() returns buffer contents excluding terminating zero, so it can only go up to log->len_total - 1, but if it is as log->len_total - 1, then we treat it as overflow. I mention this in the second patch set, but this approach doesn't distinguish log buffer filled up to exactly N bytes, vs log_buffer that was > N bytes long, but got truncated. I actually change and simplify this logic in second patch set and make it distinguish these two situations. This was important to make sure that log_size_actual can be used as is exactly, otherwise users would have to supply at least log_size_actual + 1. So, basically, here, I'm preserving existing behavior, and patch set 2 changes this bit as part of overall fixed mode overhaul. > > > + else > > + return log->start_pos > 0; > > +} > > + > > +void bpf_vlog_finalize(struct bpf_verifier_log *log) > > +{ > > + u32 sublen; > > Nit: "pivot" might be more appropriate? div_u64_rem(log->start_pos, log->len_total, &sublen); sublen = log->len_total - sublen; pivot implies it's an index of an element, and then causes questions about inclusive/exclusive, right? While length is length, so seems less ambiguous? > > > + int err; > > + > > + if (!log || !log->level || !log->ubuf) > > + return; > > + if ((log->level & BPF_LOG_FIXED) || log->level == BPF_LOG_KERNEL) > > + return; > > + > > + /* If we never truncated log, there is nothing to move around. */ > > + if (log->start_pos == 0) > > + return; > > + > > + /* Otherwise we need to rotate log contents to make it start from the > > + * buffer beginning and be a continuous zero-terminated string. Note > > + * that if log->start_pos != 0 then we definitely filled up entire log > > + * buffer with no gaps, and we just need to shift buffer contents to > > + * the left by (log->start_pos % log->len_total) bytes. > > + * > > + * Unfortunately, user buffer could be huge and we don't want to > > + * allocate temporary kernel memory of the same size just to shift > > + * contents in a straightforward fashion. Instead, we'll be clever and > > + * do in-place array rotation. This is a leetcode-style problem, which > > + * could be solved by three rotations. > > It took me a while to understand this, but the problem isn't in place > rotation, right? It's that we can't directly access ubuf so we have to > use kbuf as a window. Hm.. no, it is the rotation in place. Even if it was in the kernel buffer and we wanted to rotate this without creating a second large copy of the buffer, we'd have to do this double rotation. Kernel vs user space adds complications for sure, but it is hidden inside bpf_vlog_reverse_ubuf(), and as far as this overall algorithm goes, doesn't matter. > > > + * > > + * Let's say we have log buffer that has to be shifted left by 7 bytes > > + * (spaces and vertical bar is just for demonstrative purposes): > > + * E F G H I J K | A B C D > > + * > > + * First, we reverse entire array: > > + * D C B A | K J I H G F E > > + * > > + * Then we rotate first 4 bytes (DCBA) and separately last 7 bytes > > + * (KJIHGFE), resulting in a properly rotated array: > > + * A B C D | E F G H I J K > > + * > > + * We'll utilize log->kbuf to read user memory chunk by chunk, swap > > + * bytes, and write them back. Doing it byte-by-byte would be > > + * unnecessarily inefficient. Altogether we are going to read and > > + * write each byte twice. > > But four copies in total? So each rotation reads each byte once and writes each byte once. So two copies. And then the entire buffer is rotated twice (three rotating steps, but overall contents is rotated twice), so two reads and two writes for each byte, 4 memory copies altogether. Would you like me to clarify this some more?