Skip to content

Instantly share code, notes, and snippets.

@digitalsignalperson
Last active March 13, 2025 12:34
Show Gist options
  • Save digitalsignalperson/ef81c1afdc04d49f5ae3f6f97b2d29af to your computer and use it in GitHub Desktop.
Save digitalsignalperson/ef81c1afdc04d49f5ae3f6f97b2d29af to your computer and use it in GitHub Desktop.
xdp screen cast example, modified to pipe to ffmpeg
mkfifo /tmp/gfifo
./xdp-screen-cast-ffmpeg.py & # or run in another terminal
# Note the gstreamer pipeline
# 'pipewiresrc fd=%d path=%u ! videorate ! video/x-raw,framerate=60/1 ! videoconvert ! filesink location=/tmp/gfifo'
# I specified a 60/1 frameerate. You can do other pre-processing here, or offload it to ffmpeg.
# Encode the stream and pipe to netcat
# note: must specify correct resolution here
# we can now use h264_nvenc, mpegts, and anything else ffmpeg has
ffmpeg -f rawvideo -pixel_format bgra -video_size 2560x1600 -i /tmp/gfifo \
-c:v h264_nvenc -preset:v llhq -tune ull -zerolatency 1 -delay 0 -f mpegts - \
| nc -l -p 9001
# on remote system do
nc $ip_of_server 9001 | mpv --profile=low-latency --untimed --no-demuxer-thread -
#!/usr/bin/python3
# Source from: https://gitlab.gnome.org/-/snippets/19
import re
import signal
import dbus
from gi.repository import GLib
from dbus.mainloop.glib import DBusGMainLoop
import gi
gi.require_version('Gst', '1.0')
from gi.repository import GObject, Gst
DBusGMainLoop(set_as_default=True)
Gst.init(None)
loop = GLib.MainLoop()
bus = dbus.SessionBus()
request_iface = 'org.freedesktop.portal.Request'
screen_cast_iface = 'org.freedesktop.portal.ScreenCast'
pipeline = None
def terminate():
if pipeline is not None:
self.player.set_state(Gst.State.NULL)
loop.quit()
request_token_counter = 0
session_token_counter = 0
sender_name = re.sub(r'\.', r'_', bus.get_unique_name()[1:])
def new_request_path():
global request_token_counter
request_token_counter = request_token_counter + 1
token = 'u%d'%request_token_counter
path = '/org/freedesktop/portal/desktop/request/%s/%s'%(sender_name, token)
return (path, token)
def new_session_path():
global session_token_counter
session_token_counter = session_token_counter + 1
token = 'u%d'%session_token_counter
path = '/org/freedesktop/portal/desktop/session/%s/%s'%(sender_name, token)
return (path, token)
def screen_cast_call(method, callback, *args, options={}):
(request_path, request_token) = new_request_path()
bus.add_signal_receiver(callback,
'Response',
request_iface,
'org.freedesktop.portal.Desktop',
request_path)
options['handle_token'] = request_token
method(*(args + (options, )),
dbus_interface=screen_cast_iface)
def on_gst_message(bus, message):
type = message.type
if type == Gst.MessageType.EOS or type == Gst.MessageType.ERROR:
terminate()
def play_pipewire_stream(node_id):
empty_dict = dbus.Dictionary(signature="sv")
fd_object = portal.OpenPipeWireRemote(session, empty_dict,
dbus_interface=screen_cast_iface)
fd = fd_object.take()
pipeline = Gst.parse_launch('pipewiresrc fd=%d path=%u ! videorate ! video/x-raw,framerate=60/1 ! videoconvert ! filesink location=/tmp/gfifo' %(fd, node_id))
pipeline.set_state(Gst.State.PLAYING)
pipeline.get_bus().connect('message', on_gst_message)
def on_start_response(response, results):
if response != 0:
print("Failed to start: %s"%response)
terminate()
return
print("streams:")
for (node_id, stream_properties) in results['streams']:
print("stream {}".format(node_id))
play_pipewire_stream(node_id)
def on_select_sources_response(response, results):
if response != 0:
print("Failed to select sources: %d"%response)
terminate()
return
print("sources selected")
global session
screen_cast_call(portal.Start, on_start_response,
session, '')
def on_create_session_response(response, results):
if response != 0:
print("Failed to create session: %d"%response)
terminate()
return
global session
session = results['session_handle']
print("session %s created"%session)
screen_cast_call(portal.SelectSources, on_select_sources_response,
session,
options={ 'multiple': False,
'types': dbus.UInt32(1|2) })
portal = bus.get_object('org.freedesktop.portal.Desktop',
'/org/freedesktop/portal/desktop')
(session_path, session_token) = new_session_path()
screen_cast_call(portal.CreateSession, on_create_session_response,
options={ 'session_handle_token': session_token })
try:
loop.run()
except KeyboardInterrupt:
terminate()
@KingDuckZ
Copy link

This is not working unfortunately, it says interface org.freedesktop.portal.ScreenCast on object /org/freedesktop/portal/desktop doesn't exist (actual error not in English sorry). In fact I've been trying to understand this code and I'm completely lost. I've been looking here and here :|

@digitalsignalperson
Copy link
Author

I'm on arch linux, it might be different on another OS.

For me

./xdp-screen-cast-ffmpeg.py
session /org/freedesktop/portal/desktop/session/1_1470/u1 created
sources selected
streams:
stream 176

If you get a dbus interface error, maybe it's on a different path. Does the original snippet work for you? https://gitlab.gnome.org/-/snippets/19

You could try installing D-feet (or now D-Spy) to look in the session bus for the interface

@stellarpower
Copy link

Note - anyone without an nvidia card, change h264_nvenc to plain h264.

For anyone looking for local playback, you can omit ffmpeg and mpv and simply run ffplay -f rawvideo -pixel_format bgra -video_size 1920x1080 -i /tmp/gfifo. I get a bit of latency still, but less than encoding and decoding. Apparently there are also -fflags nobuffer, or -tune zerolatency, these caused some problems for me and haven't yet looked into other flags to tune it towards live playback of raw video.

Note to self: add a way to get the screen resolution and give this to ffplay automatically.

@digitalsignalperson
Copy link
Author

I'm not sure why this no longer works for me. Whether trying mpv or ffplay, I get the first image, and maybe a very lagged out blur of movement, but then it's all frozen.

By the way, debugging output can be enabled by adding this after the import of Gst, GObject:

from gi.repository import GObject, Gst

Gst.debug_set_active(True)
Gst.debug_set_default_threshold(4)
GObject.threads_init()

@digitalsignalperson
Copy link
Author

Also hopefully this gist is no longer needed soon. There's been work on a "pipewiregrab" patch for ffmpeg https://ffmpeg.org/pipermail/ffmpeg-devel/2024-August/thread.html#331913

@digitalsignalperson
Copy link
Author

Took a look at why this stopped working for me again... really dumb but an extra space between the mpv args seems to create havoc.

For debugging I send the stream to a file

pipeline = Gst.parse_launch('pipewiresrc fd=%d path=%u ! videorate ! video/x-raw,framerate=60/1 ! videoconvert ! filesink location=/home/andy/xdp-video.raw' %(fd, node_id))

and now I can run it with mpv with

mpv --demuxer=rawvideo --demuxer-rawvideo-w=2560 --demuxer-rawvideo-h=1440 --demuxer-rawvideo-format=bgra --demuxer-rawvideo-fps=60 --demuxer-rawvideo-mp-format=bgra --profile=low-latency --untimed --no-demuxer-thread ~/xdp-video.raw

or ffplay per stellarpower's comment above.

But when streaming with the fifo, ffplay seems pretty bad like 1 frame every 5 seconds. Streaming with mpv, better but still very laggy. Maybe related to my gpu setup will have to poke at it.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment