Pavel Begunkov <asml.silence@xxxxxxxxx> [23. April 2021]: > Yeah, absolutely. I don't see much profit in registering them > dynamically, so for now they will be needed to be loaded and attached > in advance. Or can be done in a more dynamic fashion, doesn't really > matter. > > btw, bpf splits compilation and attach steps, adds some flexibility. So, I'm currently working on rebasing your work onto the tag 'for-5.13/io_uring-2021-04-27'. So if you already have some branch on this, just let me know to save the work. > Should look similar to the userspace, fill a 64B chunk of memory, > where the exact program is specified by an index, the same that is > used during attach/registration When looking at the current implementation, when can only perform the attachment once and there is no "append eBPF". While this is probably OK for code, for eBPF maps, we will need some kind of append eBPF map. > and context fd is just another field in the SQE. On the space -- it > depends. Some opcodes pass more info than others, and even for those we > yet have 16 bytes unused. For bpf I don't expect passing much in SQE, so > it should be ok. So besides an eBPF-Progam ID, we would also pass an ID for an eEBF map in the SQE. One thought that came to my mind: Why do we have to register the eBPF programs and maps? We could also just pass the FDs for those objects in the SQE. As long as there is no other state, it could be the userspaces choice to either attach it or pass it every time. For other FDs we already support both modes, right? >> - My proposed serialization promise > > It can be an optional feature, but 1) it may become a bottleneck at > some point, 2) users use several rings, e.g. per-thread, so they > might need to reimplement serialisation anyway. If we make it possible to pass some FD to an synchronization object (e.g. semaphore), this might do the trick to support both modes at the interface. >> - Exposing synchronization primitives to the eBPF program. I don't think >> that we can argue for semaphores in an eBPF program. > > I remember a discussion about sleep-able bpf, we need to look what has > happened with it. But surely this would hurt a lot as we would have to manage not only eBPF programs, but also eBPF processes. While this is surely possible, I don't know if it is really suitable for a high-performance interface like io_uring. But, don't know about the state. > >> With the serialization promise, we at least avoid the need to >> synchronize callbacks with callbacks. However, synchronization between >> user space and callback is still a problem. > > Need to look up up-to-date BPF capabilities, but can also be spinlocks, > for both: bpf-userspace sync, and between bpf > https://lwn.net/ml/netdev/20190116050830.1881316-1-ast@xxxxxxxxxx/ Using Spinlocks between kernel and userspace just feels wrong, very wrong. But it might be an alternate route to synchronization > With a bit of work nothing forbids to make them userspace visible, > just next step to the idea. In the end I want to have no difference > between CQs, and everyone can reap from anywhere, and it's up to > user to use/synchronise properly. I like the notion of orthogonality with this route. Perhaps, we don't need to have user-invisible CQs but it can be enough to address the CQ of another uring in my SQE as the sink for the resulting CQE. Downside with that idea would be that the user has to setup another ring with SQ and CQ, but only the CQ is used. > [...] > CQ is specified by index in SQE, in each SQE. So either as you say, or > just specify index of the main CQ in that previous linked request in > the first place. >From looking at the code: This is not yet the case, or? >> How do I indicate at the first SQE into which CQ the result should be >> written? > Yes, adds a bit of complexity, but without it you can only get last CQE, > 1) it's not flexible enough and shoots off different potential scenarios > > 2) not performance efficient -- overhead on running a bpf request after > each I/O request can be too large. > > 3) does require mandatory linking if you want to get result. Without it > we can submit a single BPF request and let it running multiple times, > e.g. waiting for on CQ, but linking would much limit options > > 4) bodgy from the implementation perspective When taking a step back, this is nearly a io_uring_enter(minwait=N)-SQE with an attached eBPF callback, or? At that point, we are nearly full circle. >> Are we able to encode all of that into a single SQE that also holds an >> eBPF function pointer and (potenitally) an pointer to a context map? > > yes, but can be just a separate linked request... So, let's make a little collection about the (potential) information that our poor SQE has to hold. Thereby, FDs should be registrable and addressible by an index. - FD to eBPF program - FD to eBPF map - FD to synchronization object during the execution - FD to foreign CQ for waiting on N CQEs That are a lot of references to other object for which we would have to extend the registration interface. > Right. And it should know what it's doing anyway in most cases. All > more complex dispatching / state machines can be pretty well > implemented via context. You convinced me that an eBPF map as a context is the more canonical way of doing it by achieving the same degree of flexibility. > I believe there was something for accessing userspace memory, we > need to look it up. Either way, from a researcher perspective, we can just allow it and look how it can performs. chris -- Dr.-Ing. Christian Dietrich Operating System Group (E-EXK4) Technische Universität Hamburg Am Schwarzenberg-Campus 3 (E) 21073 Hamburg eMail: christian.dietrich@xxxxxxx Tel: +49 40 42878 2188 WWW: https://osg.tuhh.de/