'gstreamer video slows down after awhile
I use appsrc to push jpg images to create a live video. The video displays correctly for the first 5 - 10 seconds, but gradually lags behind and drops frames over time. Below is the code:
#!/usr/bin/env python3
import os
import sys
import gi
import numpy as np
gi.require_version('Gst', '1.0')
from gi.repository import Gst, GObject
import time
import random
class TestOverlay(object):
def exit(self):
self.pipeline.set_state(Gst.State.NULL)
def __init__(self):
# configure appsrc.
self.number_frames = 0
self.fps = 8
secperframe = 1 / self.fps
self.duration = secperframe * Gst.SECOND # duration of a frame in Gst seconds (nanosec?)
self.launch_string = 'appsrc name=source is-live=true format=GST_FORMAT_TIME ' \
'! jpegdec ' \
'! videoconvert ! videoscale ! video/x-raw,width=700,height=500' \
'! autovideosink'
# setup and start a pipeline.
self.pipeline = Gst.parse_launch(self.launch_string)
appsrc = self.pipeline.get_child_by_name('source')
self.pipeline.set_state(Gst.State.PLAYING)
# start loop to push data through pipeline.
cnt = 0
while True:
try:
tstart = time.perf_counter()
cnt += 1
# this function returns a block of jpeg data.
# Basically, it copies data from a mmap (memory mapped) block.
# And another app refreshes the block as a much higher rate than self.fps
data = self.getbuf()
if data is None:
print(f'+++ counter: {cnt}, warning: no data')
time.sleep(secperframe)
continue
else:
print(f'+++ counter: {cnt}, data len: {len(data)}')
buf = Gst.Buffer.new_allocate(None, len(data), None)
buf.fill(0, data)
buf.duration = self.duration
timestamp = self.number_frames * self.duration
buf.pts = buf.dts = int(timestamp)
buf.offset = timestamp
self.number_frames += 1
retval = appsrc.emit('push-buffer', buf)
if retval != Gst.FlowReturn.OK:
print(f'+++ possible error: {retval}')
# wait to make sure there are no more frames than self.fps.
tduration = time.perf_counter() - tstart
if tduration < secperframe:
remain = secperframe - tduration
time.sleep(remain)
#print(f'+++sleep: {remain}, {tduration}')
else:
print(f'+++ getbuf falls behind {tduration}')
except Exception as e:
print(f'+++ Exception: {e}')
break
self.exit()
Gst.init(None)
player = TestOverlay()
try:
loop = GObject.MainLoop()
loop.run()
except KeyboardInterrupt:
print("Exit loop due to Ctrl+c.")
player.exit()
sys.exit(0)
I suspect the frame rate may be to slow that frames are piling up in the buffer, but not sure how to set the frame rate in the lauch_string. "caps" does not seem to work:
caps=video/x-raw,format=JPEG,framerate=8/1
because JPEG is not one of the supported format. Also try to play with 'block', 'leaky-type', etc. without any lucks. Any pointers are greatly appreciated.
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|