在 2024/8/13 1:57, Andrii Nakryiko 写道: > On Mon, Aug 12, 2024 at 5:05 AM Liao, Chang <liaochang1@xxxxxxxxxx> wrote: >> >> >> >> 在 2024/8/10 2:40, Andrii Nakryiko 写道: >>> On Fri, Aug 9, 2024 at 11:34 AM Andrii Nakryiko >>> <andrii.nakryiko@xxxxxxxxx> wrote: >>>> >>>> On Fri, Aug 9, 2024 at 12:16 AM Liao, Chang <liaochang1@xxxxxxxxxx> wrote: >>>>> >>>>> >>>>> >>>>> 在 2024/8/9 2:26, Andrii Nakryiko 写道: >>>>>> On Thu, Aug 8, 2024 at 1:45 AM Liao, Chang <liaochang1@xxxxxxxxxx> wrote: >>>>>>> >>>>>>> Hi Andrii and Oleg. >>>>>>> >>>>>>> This patch sent by me two weeks ago also aim to optimize the performance of uprobe >>>>>>> on arm64. I notice recent discussions on the performance and scalability of uprobes >>>>>>> within the mailing list. Considering this interest, I've added you and other relevant >>>>>>> maintainers to the CC list for broader visibility and potential collaboration. >>>>>>> >>>>>> >>>>>> Hi Liao, >>>>>> >>>>>> As you can see there is an active work to improve uprobes, that >>>>>> changes lifetime management of uprobes, removes a bunch of locks taken >>>>>> in the uprobe/uretprobe hot path, etc. It would be nice if you can >>>>>> hold off a bit with your changes until all that lands. And then >>>>>> re-benchmark, as costs might shift. >>>>> >>>>> Andrii, I'm trying to integrate your lockless changes into the upstream >>>>> next-20240806 kernel tree. And I ran into some conflicts. please let me >>>>> know which kernel you're currently working on. >>>>> >>>> >>>> My patches are based on tip/perf/core. But I also just pushed all the >>>> changes I have accumulated (including patches I haven't sent for >>>> review just yet), plus your patches for sighand lock removed applied >>>> on top into [0]. So you can take a look and use that as a base for >>>> now. Keep in mind, a bunch of those patches might still change, but >>>> this should give you the best currently achievable performance with >>>> uprobes/uretprobes. E.g., I'm getting something like below on x86-64 >>>> (note somewhat linear scalability with number of CPU cores, with >>>> per-CPU performance *slowly* declining): >>>> >>>> uprobe-nop ( 1 cpus): 3.565 ± 0.004M/s ( 3.565M/s/cpu) >>>> uprobe-nop ( 2 cpus): 6.742 ± 0.009M/s ( 3.371M/s/cpu) >>>> uprobe-nop ( 3 cpus): 10.029 ± 0.056M/s ( 3.343M/s/cpu) >>>> uprobe-nop ( 4 cpus): 13.118 ± 0.014M/s ( 3.279M/s/cpu) >>>> uprobe-nop ( 5 cpus): 16.360 ± 0.011M/s ( 3.272M/s/cpu) >>>> uprobe-nop ( 6 cpus): 19.650 ± 0.045M/s ( 3.275M/s/cpu) >>>> uprobe-nop ( 7 cpus): 22.926 ± 0.010M/s ( 3.275M/s/cpu) >>>> uprobe-nop ( 8 cpus): 24.707 ± 0.025M/s ( 3.088M/s/cpu) >>>> uprobe-nop (10 cpus): 30.842 ± 0.018M/s ( 3.084M/s/cpu) >>>> uprobe-nop (12 cpus): 33.623 ± 0.037M/s ( 2.802M/s/cpu) >>>> uprobe-nop (14 cpus): 39.199 ± 0.009M/s ( 2.800M/s/cpu) >>>> uprobe-nop (16 cpus): 41.698 ± 0.018M/s ( 2.606M/s/cpu) >>>> uprobe-nop (24 cpus): 65.078 ± 0.018M/s ( 2.712M/s/cpu) >>>> uprobe-nop (32 cpus): 84.580 ± 0.017M/s ( 2.643M/s/cpu) >>>> uprobe-nop (40 cpus): 101.992 ± 0.268M/s ( 2.550M/s/cpu) >>>> uprobe-nop (48 cpus): 101.032 ± 1.428M/s ( 2.105M/s/cpu) >>>> uprobe-nop (56 cpus): 110.986 ± 0.736M/s ( 1.982M/s/cpu) >>>> uprobe-nop (64 cpus): 124.145 ± 0.110M/s ( 1.940M/s/cpu) >>>> uprobe-nop (72 cpus): 134.940 ± 0.200M/s ( 1.874M/s/cpu) >>>> uprobe-nop (80 cpus): 143.918 ± 0.235M/s ( 1.799M/s/cpu) >>>> >>>> uretprobe-nop ( 1 cpus): 1.987 ± 0.001M/s ( 1.987M/s/cpu) >>>> uretprobe-nop ( 2 cpus): 3.766 ± 0.003M/s ( 1.883M/s/cpu) >>>> uretprobe-nop ( 3 cpus): 5.638 ± 0.002M/s ( 1.879M/s/cpu) >>>> uretprobe-nop ( 4 cpus): 7.275 ± 0.003M/s ( 1.819M/s/cpu) >>>> uretprobe-nop ( 5 cpus): 9.124 ± 0.004M/s ( 1.825M/s/cpu) >>>> uretprobe-nop ( 6 cpus): 10.818 ± 0.007M/s ( 1.803M/s/cpu) >>>> uretprobe-nop ( 7 cpus): 12.721 ± 0.014M/s ( 1.817M/s/cpu) >>>> uretprobe-nop ( 8 cpus): 13.639 ± 0.007M/s ( 1.705M/s/cpu) >>>> uretprobe-nop (10 cpus): 17.023 ± 0.009M/s ( 1.702M/s/cpu) >>>> uretprobe-nop (12 cpus): 18.576 ± 0.014M/s ( 1.548M/s/cpu) >>>> uretprobe-nop (14 cpus): 21.660 ± 0.004M/s ( 1.547M/s/cpu) >>>> uretprobe-nop (16 cpus): 22.922 ± 0.013M/s ( 1.433M/s/cpu) >>>> uretprobe-nop (24 cpus): 34.756 ± 0.069M/s ( 1.448M/s/cpu) >>>> uretprobe-nop (32 cpus): 44.869 ± 0.153M/s ( 1.402M/s/cpu) >>>> uretprobe-nop (40 cpus): 53.397 ± 0.220M/s ( 1.335M/s/cpu) >>>> uretprobe-nop (48 cpus): 48.903 ± 2.277M/s ( 1.019M/s/cpu) >>>> uretprobe-nop (56 cpus): 42.144 ± 1.206M/s ( 0.753M/s/cpu) >>>> uretprobe-nop (64 cpus): 42.656 ± 1.104M/s ( 0.666M/s/cpu) >>>> uretprobe-nop (72 cpus): 46.299 ± 1.443M/s ( 0.643M/s/cpu) >>>> uretprobe-nop (80 cpus): 46.469 ± 0.808M/s ( 0.581M/s/cpu) >>>> >>>> uprobe-ret ( 1 cpus): 1.219 ± 0.008M/s ( 1.219M/s/cpu) >>>> uprobe-ret ( 2 cpus): 1.862 ± 0.008M/s ( 0.931M/s/cpu) >>>> uprobe-ret ( 3 cpus): 2.874 ± 0.005M/s ( 0.958M/s/cpu) >>>> uprobe-ret ( 4 cpus): 3.512 ± 0.002M/s ( 0.878M/s/cpu) >>>> uprobe-ret ( 5 cpus): 3.549 ± 0.001M/s ( 0.710M/s/cpu) >>>> uprobe-ret ( 6 cpus): 3.425 ± 0.003M/s ( 0.571M/s/cpu) >>>> uprobe-ret ( 7 cpus): 3.551 ± 0.009M/s ( 0.507M/s/cpu) >>>> uprobe-ret ( 8 cpus): 3.050 ± 0.002M/s ( 0.381M/s/cpu) >>>> uprobe-ret (10 cpus): 2.706 ± 0.002M/s ( 0.271M/s/cpu) >>>> uprobe-ret (12 cpus): 2.588 ± 0.003M/s ( 0.216M/s/cpu) >>>> uprobe-ret (14 cpus): 2.589 ± 0.003M/s ( 0.185M/s/cpu) >>>> uprobe-ret (16 cpus): 2.575 ± 0.001M/s ( 0.161M/s/cpu) >>>> uprobe-ret (24 cpus): 1.808 ± 0.011M/s ( 0.075M/s/cpu) >>>> uprobe-ret (32 cpus): 1.853 ± 0.001M/s ( 0.058M/s/cpu) >>>> uprobe-ret (40 cpus): 1.952 ± 0.002M/s ( 0.049M/s/cpu) >>>> uprobe-ret (48 cpus): 2.075 ± 0.007M/s ( 0.043M/s/cpu) >>>> uprobe-ret (56 cpus): 2.441 ± 0.004M/s ( 0.044M/s/cpu) >>>> uprobe-ret (64 cpus): 1.880 ± 0.012M/s ( 0.029M/s/cpu) >>>> uprobe-ret (72 cpus): 0.962 ± 0.002M/s ( 0.013M/s/cpu) >>>> uprobe-ret (80 cpus): 1.040 ± 0.011M/s ( 0.013M/s/cpu) >>>> >>>> uretprobe-ret ( 1 cpus): 0.981 ± 0.000M/s ( 0.981M/s/cpu) >>>> uretprobe-ret ( 2 cpus): 1.421 ± 0.001M/s ( 0.711M/s/cpu) >>>> uretprobe-ret ( 3 cpus): 2.050 ± 0.003M/s ( 0.683M/s/cpu) >>>> uretprobe-ret ( 4 cpus): 2.596 ± 0.002M/s ( 0.649M/s/cpu) >>>> uretprobe-ret ( 5 cpus): 3.105 ± 0.003M/s ( 0.621M/s/cpu) >>>> uretprobe-ret ( 6 cpus): 3.886 ± 0.002M/s ( 0.648M/s/cpu) >>>> uretprobe-ret ( 7 cpus): 3.016 ± 0.001M/s ( 0.431M/s/cpu) >>>> uretprobe-ret ( 8 cpus): 2.903 ± 0.000M/s ( 0.363M/s/cpu) >>>> uretprobe-ret (10 cpus): 2.755 ± 0.001M/s ( 0.276M/s/cpu) >>>> uretprobe-ret (12 cpus): 2.400 ± 0.001M/s ( 0.200M/s/cpu) >>>> uretprobe-ret (14 cpus): 3.972 ± 0.001M/s ( 0.284M/s/cpu) >>>> uretprobe-ret (16 cpus): 3.940 ± 0.003M/s ( 0.246M/s/cpu) >>>> uretprobe-ret (24 cpus): 3.002 ± 0.003M/s ( 0.125M/s/cpu) >>>> uretprobe-ret (32 cpus): 3.018 ± 0.003M/s ( 0.094M/s/cpu) >>>> uretprobe-ret (40 cpus): 1.846 ± 0.000M/s ( 0.046M/s/cpu) >>>> uretprobe-ret (48 cpus): 2.487 ± 0.004M/s ( 0.052M/s/cpu) >>>> uretprobe-ret (56 cpus): 2.470 ± 0.006M/s ( 0.044M/s/cpu) >>>> uretprobe-ret (64 cpus): 2.027 ± 0.014M/s ( 0.032M/s/cpu) >>>> uretprobe-ret (72 cpus): 1.108 ± 0.011M/s ( 0.015M/s/cpu) >>>> uretprobe-ret (80 cpus): 0.982 ± 0.005M/s ( 0.012M/s/cpu) >>>> >>>> >>>> -ret variants (single-stepping case for x86-64) still suck, but they >>>> suck 2x less now with your patches :) Clearly more work ahead for >>>> those, though. >>>> >>> >>> Quick profiling shows that it's mostly xol_take_insn_slot() and >>> xol_free_insn_slot(), now. So it seems like your planned work might >>> help here. >> >> Andrii, I'm glad we've reached a similar result, The profiling result on >> my machine reveals that about 80% cycles spent on the atomic operations >> on area->bitmap and area->slot_count. I guess the atomic access leads to >> the intensive cacheline bouncing bewteen CPUs. >> >> In the passed weekend, I have been working on another patch that optimizes >> the xol_take_insn_slot() and xol_free_inns_slot() for better scalability. >> This involves delaying the freeing of xol insn slots to reduce the times >> of atomic operations and cacheline bouncing. Additionally, per-task refcounts >> and an RCU-style management of linked-list of allocated insn slots. In short >> summary, this patch try to replace coarse-grained atomic variables with >> finer-grained ones, aiming to elimiate the expensive atomic instructions >> in the hot path. If you or others have bandwidth and interest, I'd welcome >> a brainstorming session on this topic. > > I'm happy to help, but I still feel like it's best to concentrate on > landing all the other pending things for uprobe, and then switch to > optimizing the xol case. > > We have: > - RCU protection and avoiding refcounting for uprobes (I'll be > sending latest revision soon); > - SRCU+timeout for uretprobe and single-step (pending the above > landing first); > - removing shared nhit counter increment in trace_uprobe (I've sent > patches last week, see [0]); > - lockless VMA -> inode -> uprobe look up (also pending for #1 to > land, and some more benchmarking for mm_lock_seq changes from Suren, > see [1]); > - and, of course, your work to remove sighand lock. > > So as you can see, there is plenty to discuss and land already, I just > don't want to spread the efforts too thin. But if you can help improve > the benchmark for ARM64, that would be a great parallel effort setting > us up for further work nicely. Thanks! Agree. Let's prioritize landing your exising patches. I'll build upon you works for the further uprobe optimizatoin for Arm64. Thanks. > > [0] https://lore.kernel.org/bpf/20240809192357.4061484-1-andrii@xxxxxxxxxx/ > [1] https://lore.kernel.org/linux-mm/CAEf4BzaocU-CQsFZ=s5gDM6XQ0Foss_HroFsPUesBn=qgJCprg@xxxxxxxxxxxxxx/ > >> >> Thanks. >> >>> >>>> >>>> [0] https://github.com/anakryiko/linux/commits/uprobes-lockless-cumulative/ >>>> >>>> >>>>> Thanks. >>>>> >>>>>> >>>>>> But also see some remarks below. >>>>>> >>>>>>> Thanks. >>>>>>> > > [...] -- BR Liao, Chang