-
-
Notifications
You must be signed in to change notification settings - Fork 8.1k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
linux-pipewire: Add explicit sync support #11708
base: master
Are you sure you want to change the base?
Conversation
Please work with upstream pipewire and compositors to document the pipewire protocol and how to correctly handle explicit sync. (google doesnt bring up anything for First two commits seem ok for adding fence support at first glance. |
8bd2b03
to
951c2bb
Compare
I believe it has already been documented by upstream here: https://docs.pipewire.org/page_dma_buf.html |
Thanks. What compositor other than gnome was involved, do they have a tentative implementation this can be tested against? Are any other clients like chromium already onboard with this protocol? Basically just like when we originally implemented pipewire we dont want to include something that will only work for one compositor or risk changing dramatically when other users more important than us request changes to the protocol that break our implementation. Also, how does cross device dmabuf sharing work in this protocol, right now it seems the fences are all assumed to be on a single device if I understand the drm calls correctly (or can the fences be imported across device?). Cross device issues are already coming up a fair bit in the current pipewire dmabuf protocol unfortunately... |
KWin has an implementation that should be released with KWin 6.3: https://invent.kde.org/plasma/kwin/-/merge_requests/6204
I believe GNOME and KDE implementations should cover the majority of the Linux users. So far, the only client that I know that takes advantage of explicit sync on PipeWire is NvFBC (NVIDIA's proprietary screen recording solution). I am not sure if the explicit sync logic can really dramatically change. The only change required on the PipeWire side was the addition of SPA_META_SyncTimeline meta type. The DRM syncobj mechanism used for implementing this feature is also used by Wayland's linux-drm-syncobj-v1 protocol, which is implemented by both Mesa and NVIDIA drivers.
DRM syncobjs should be importable across devices and it does not change how dmabufs are shared. I may be able to test these changes in a multiple device configuration later. |
I did some more testing on KDE 6.3 Beta and Sway 1.10. Updated the description with the details. |
951c2bb
to
a08064f
Compare
Fixed a bug that was causing OBS to signal the wrong syncobj. This was not caught during testing because both GNOME and KDE use one syncobj per image for both acquire and release tracking with duplicated syncobj file descriptors. |
gs_sync_t *gl_egl_create_sync(EGLDisplay egl_display) | ||
{ | ||
gs_sync_t *sync = eglCreateSync(egl_display, EGL_SYNC_NATIVE_FENCE_ANDROID, NULL); | ||
glFlush(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
flush will happen once all the drawing is queued anyway, can we avoid introducing this in the middle of the frame?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think that it is avoidable with the current design of libobs. The EGL spec says:
... the next Flush() operation performed by the current client API causes a new native fence object to be created, and the EGL_SYNC_NATIVE_FENCE_FD_ANDROID attribute of the EGL native fence object is set to a file descriptor that refers to the new native fence object.
And that file descriptor is accessed in gs_sync_export_syncobj_timeline_point()
when the eglDupNativeFenceFDANDROID()
function is called.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
On the windows side we sometimes defer things until the next frame, so I don't see why that wouldnt be possible in pipewire. Instead of trying to realize a fence during drawing for every capture we could just defer returning buffers until the next frame (might need to add a tick() handle for pipewire), and add a gs_xxx function to return the previous render's fence.
It looks like the syncobj work in gnome is now in fedora 41 so I tested this a little bit on my low spec machine. The stalls from waiting for the acquire fence and creating the release fence during drawing are kind of brutal, and it seems to tickle gnome+intel causing significantly more stalls in other parts of the wsi than when disabling explicit sync during negotiation. Below is a particularly bad frame, but basically p90 cpu time is up 10x on my machine compared to implicit sync. Ill try and test with gnome+amd later, which is more of a typical machine. |
Previous issue was mostly just compositor flakiness on my machine, after doing the same things on both setups it just shows the extra time from the new waits with nothing else really being slower with the new sync/flush. Also looks fine on amd+gnome. It would be nice to have folks try it out on kde. I'm not really familiar with our pipewire code so someone else can comment on that part but LGTM from me. |
https://chromium-review.googlesource.com/c/chromium/src/+/5914502 |
This extension is needed for using explicit sync in the PipeWire plugin. Signed-off-by: Doğukan Korkmaztürk <[email protected]>
This change adds a new type named gs_sync_t and related functions to X11 and Wayland EGL backends to abstract DRM syncobjs and their various uses. Signed-off-by: Doğukan Korkmaztürk <[email protected]>
It is possible for OBS to receive incomplete frames from the compositor if the graphics drivers in use do not support implicit synchronization. The compositors usually try to workaround this issue by waiting all render operations to complete before sharing frames with OBS. However, this has performance implications for the compositors. Similarly, OBS should also make sure that all the rendering operations sourcing the received buffer are complete before returning the respective pw_buffer back to the compositor, or it would risk the image it is working on getting overwritten without some type of synchronization. This change leverages PipeWire's ability to share DRM syncobj fds to implement an explicit synchronization solution to solve the problem described above. The usage of these DRM syncobjs is negotiated with the compositor that OBS is running on. So OBS can still work even if the compositor does not support explicit synchronization. Signed-off-by: Doğukan Korkmaztürk <[email protected]>
a08064f
to
dcf9797
Compare
Thanks for the detailed testing!
Thanks for pointing this out, I didn't read the entire change but this looks like an implementation of the Wayland linux_drm_syncobj_v1 protocol. The idea is the same (using DRM syncobj timelines for synchronizing access to the shared buffers), but the domain is different. |
Description
This change leverages PipeWire's ability to share DRM syncobj fds to implement an explicit synchronization solution.
Motivation and Context
It is possible for OBS to receive incomplete frames from the compositor if the graphics drivers in use do not support implicit synchronization (e.g. NVIDIA drivers). The compositors usually try to workaround this issue by waiting all render operations to complete before sharing frames with OBS. However, this has performance implications for the compositors. Similarly, OBS should also make sure that all the rendering operations sourcing the received buffer are complete before returning the respective pw_buffer back to the compositor, or it would risk the image it is working on getting overwritten without some type of synchronization.
How Has This Been Tested?
Types of changes
Checklist: