Skip to content
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

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

dkorkmazturk
Copy link

@dkorkmazturk dkorkmazturk commented Jan 7, 2025

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?

  • Recorded the screen using the "Screen Capture (PipeWire)" option as the source on a Wayland session on both GNOME 47.3 and KDE 6.3 Beta, which support explicit sync for screencasting. Verified that explicit sync paths were in use by attaching a debugger to OBS.
  • Recorded the screen using the "Screen Capture (PipeWire)" option as the source on Sway 1.10 to make sure that this change does not introduce regressions for the compositors that do not support PipeWire explicit sync.

Types of changes

  • Bug fix (non-breaking change which fixes an issue)
  • Performance enhancement (non-breaking change which improves efficiency)

Checklist:

  • My code has been run through clang-format.
  • I have read the contributing document.
  • My code is not on the master branch.
  • The code has been tested.
  • All commit messages are properly formatted and commits squashed where appropriate.
  • I have included updates to all appropriate documentation.

@tytan652 tytan652 added the Linux Affects Linux label Jan 7, 2025
@kkartaltepe
Copy link
Collaborator

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 SPA_META_SyncTimeline)

First two commits seem ok for adding fence support at first glance.

@dkorkmazturk
Copy link
Author

I believe it has already been documented by upstream here: https://docs.pipewire.org/page_dma_buf.html

@kkartaltepe
Copy link
Collaborator

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...

@dkorkmazturk
Copy link
Author

dkorkmazturk commented Jan 8, 2025

What compositor other than gnome was involved, do they have a tentative implementation this can be tested against?

KWin has an implementation that should be released with KWin 6.3: https://invent.kde.org/plasma/kwin/-/merge_requests/6204

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.

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.

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...

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.

@dkorkmazturk
Copy link
Author

I did some more testing on KDE 6.3 Beta and Sway 1.10. Updated the description with the details.

@dkorkmazturk
Copy link
Author

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.

@WizardCM WizardCM added the Bug Fix Non-breaking change which fixes an issue label Jan 11, 2025
gs_sync_t *gl_egl_create_sync(EGLDisplay egl_display)
{
gs_sync_t *sync = eglCreateSync(egl_display, EGL_SYNC_NATIVE_FENCE_ANDROID, NULL);
glFlush();
Copy link
Collaborator

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?

Copy link
Author

@dkorkmazturk dkorkmazturk Jan 13, 2025

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.

Copy link
Collaborator

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.

@kkartaltepe
Copy link
Collaborator

kkartaltepe commented Jan 18, 2025

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.

screen_20250117232956

Ill try and test with gnome+amd later, which is more of a typical machine.

@kkartaltepe
Copy link
Collaborator

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.

@eszlari
Copy link
Contributor

eszlari commented Jan 20, 2025

Are any other clients like chromium already onboard with this protocol?

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]>
@dkorkmazturk
Copy link
Author

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.

Thanks for the detailed testing!

https://chromium-review.googlesource.com/c/chromium/src/+/5914502

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.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug Fix Non-breaking change which fixes an issue Linux Affects Linux
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants