Browse Source

player,bench: add queue+nearest-neighbour before videoscale to prevent pipeline stalls

main
Matteo Benedetto 1 week ago
parent
commit
65665f4cff
  1. 52
      src/r36s_dlna_browser/player/gstreamer_backend.py
  2. 18
      tests/benchmark_nv12_decode.py

52
src/r36s_dlna_browser/player/gstreamer_backend.py

@ -428,24 +428,42 @@ class GStreamerBackend(PlayerBackend):
sink.set_property("caps", self._gst.Caps.from_string("video/x-raw,format=BGRA"))
return sink
# Hardware decode (NV12): insert a videoscale element before the appsink
# so mppvideodec can decode at full resolution in HW, but Python only
# receives a frame scaled to the display size (default 640x480).
# This cuts the memmove from 3.1 MB (1080p) to ~460 KB (640x480) per frame
# — a 6.7× reduction in CPU copy cost.
# Hardware decode (NV12): insert a queue → videoscale → capsfilter chain
# inside a GstBin before the appsink so playbin accepts it as a single
# video-sink element.
#
# queue — decouples mppvideodec from the scale thread so the HW
# decoder is never stalled waiting for SW scaling to finish.
# leaky=2 (downstream) drops the oldest queued frame when
# full, ensuring Python always receives the latest frame.
#
# videoscale(method=nearest) — scales 1920×1080 → 640×480 using
# nearest-neighbour interpolation (fastest SW method).
# Python receives 460 KB per frame instead of 3.1 MB,
# cutting memmove cost from ~32 ms to ~1 ms (30× reduction).
#
# capsfilter — enforces the target resolution and NV12 format so
# GStreamer's autoplugging can insert any needed conversion.
app_w, app_h = self._viewport[0], self._viewport[1]
scale_w, scale_h = (app_w or 640), (app_h or 480)
log.info("NV12 appsink: inserting videoscale → %dx%d before appsink", scale_w, scale_h)
scale = self._gst.ElementFactory.make("videoscale", "vscale")
capsfilter = self._gst.ElementFactory.make("capsfilter", "vcaps")
if scale is None or capsfilter is None:
# videoscale not available — fall back to unscaled NV12
log.warning("videoscale element unavailable; using unscaled NV12 appsink")
log.info("NV12 appsink: queue → videoscale(nearest) → %dx%d before appsink", scale_w, scale_h)
queue = self._gst.ElementFactory.make("queue", "vqueue")
scale = self._gst.ElementFactory.make("videoscale", "vscale")
capsfilter = self._gst.ElementFactory.make("capsfilter", "vcaps")
if queue is None or scale is None or capsfilter is None:
# Core elements unavailable — fall back to unscaled NV12.
log.warning("queue/videoscale/capsfilter unavailable; using unscaled NV12 appsink")
sink.set_property("caps", self._gst.Caps.from_string(
"video/x-raw,format=NV12;video/x-raw,format=BGRA"))
return sink
# nearest-neighbour: reads only the needed source pixels (strided),
# much cheaper than bilinear which reads all adjacent pixels.
scale.set_property("method", 0)
# drop oldest buffered frame when queue is full — keep the latest.
queue.set_property("max-size-buffers", 4)
queue.set_property("leaky", 2)
capsfilter.set_property(
"caps",
self._gst.Caps.from_string(
@ -453,17 +471,19 @@ class GStreamerBackend(PlayerBackend):
),
)
# Wire scale → capsfilter → appsink inside a bin so playbin accepts it
# as a single video-sink element.
# Wire queue → scale → capsfilter → appsink inside a bin.
bin_ = self._gst.Bin.new("vscale-bin")
bin_.add(queue)
bin_.add(scale)
bin_.add(capsfilter)
bin_.add(sink)
queue.link(scale)
scale.link(capsfilter)
capsfilter.link(sink)
# Expose the scale element's sink pad as the bin's ghost sink pad.
sink_pad = scale.get_static_pad("sink")
# Expose the queue element's sink pad as the bin's ghost sink pad
# so playbin can push decoded frames into the bin.
sink_pad = queue.get_static_pad("sink")
ghost = self._gst.GhostPad.new("sink", sink_pad)
ghost.set_active(True)
bin_.add_pad(ghost)

18
tests/benchmark_nv12_decode.py

@ -114,9 +114,17 @@ appsink.set_property("drop", True)
video_sink = appsink
if hw_decoders and "--noscale" not in sys.argv:
queue_el = Gst.ElementFactory.make("queue", "vqueue")
scale_el = Gst.ElementFactory.make("videoscale", "vscale")
cfilt_el = Gst.ElementFactory.make("capsfilter", "vcaps")
if scale_el is not None and cfilt_el is not None:
if scale_el is not None and cfilt_el is not None and queue_el is not None:
# nearest-neighbour: reads ~1/7 of source pixels (strided), far cheaper
# than bilinear which reads all adjacent pixels for each output sample.
scale_el.set_property("method", 0)
# queue decouples mppvideodec from the scale thread so the decoder is
# never stalled waiting for software scaling to finish.
queue_el.set_property("max-size-buffers", 4)
queue_el.set_property("leaky", 2) # drop oldest when full → keep latest
cfilt_el.set_property(
"caps",
Gst.Caps.from_string(
@ -124,19 +132,21 @@ if hw_decoders and "--noscale" not in sys.argv:
),
)
bin_ = Gst.Bin.new("vscale-bin")
bin_.add(queue_el)
bin_.add(scale_el)
bin_.add(cfilt_el)
bin_.add(appsink)
queue_el.link(scale_el)
scale_el.link(cfilt_el)
cfilt_el.link(appsink)
sink_pad = scale_el.get_static_pad("sink")
sink_pad = queue_el.get_static_pad("sink")
ghost = Gst.GhostPad.new("sink", sink_pad)
ghost.set_active(True)
bin_.add_pad(ghost)
video_sink = bin_
print(f"[scale] videoscale → {SCALE_W}×{SCALE_H} NV12 (mirrors app _create_appsink)")
print(f"[scale] queue → videoscale(nearest){SCALE_W}×{SCALE_H} NV12")
else:
print("[scale] videoscale element unavailable — falling back to unscaled NV12")
print("[scale] queue/videoscale element unavailable — falling back to unscaled NV12")
appsink.set_property(
"caps", Gst.Caps.from_string("video/x-raw,format=NV12;video/x-raw,format=BGRA")
)

Loading…
Cancel
Save