Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
Q
qgroundcontrol
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Valentin Platzgummer
qgroundcontrol
Commits
d403f313
Commit
d403f313
authored
Mar 30, 2020
by
Andrew Voznytsa
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Fix race conditions with camera manager
parent
cf2628fe
Changes
4
Show whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
114 additions
and
37 deletions
+114
-37
VideoManager.cc
src/VideoManager/VideoManager.cc
+68
-28
VideoManager.h
src/VideoManager/VideoManager.h
+5
-3
GstVideoReceiver.cc
src/VideoReceiver/GstVideoReceiver.cc
+31
-6
VideoReceiver.h
src/VideoReceiver/VideoReceiver.h
+10
-0
No files found.
src/VideoManager/VideoManager.cc
View file @
d403f313
...
...
@@ -190,6 +190,7 @@ VideoManager::startVideo()
const
unsigned
timeout
=
_videoSettings
->
rtspTimeout
()
->
rawValue
().
toUInt
();
if
(
_videoReceiver
!=
nullptr
)
{
connect
(
_videoReceiver
,
&
VideoReceiver
::
onStartComplete
,
this
,
&
VideoManager
::
_onStartComplete
);
_videoReceiver
->
start
(
_videoUri
,
timeout
);
if
(
_videoSink
!=
nullptr
)
{
_videoReceiver
->
startDecoding
(
_videoSink
);
...
...
@@ -212,6 +213,10 @@ VideoManager::stopVideo()
if
(
qgcApp
()
->
runningUnitTests
())
{
return
;
}
disconnect
(
_videoReceiver
,
&
VideoReceiver
::
timeout
,
this
,
&
VideoManager
::
_restartVideo
);
disconnect
(
_videoReceiver
,
&
VideoReceiver
::
streamingChanged
,
this
,
&
VideoManager
::
_streamingChanged
);
#if defined(QGC_GST_STREAMING)
if
(
_videoReceiver
)
_videoReceiver
->
stop
();
if
(
_thermalVideoReceiver
)
_thermalVideoReceiver
->
stop
();
...
...
@@ -526,35 +531,40 @@ VideoManager::_initVideo()
}
//-----------------------------------------------------------------------------
void
bool
VideoManager
::
_updateSettings
()
{
if
(
!
_videoSettings
)
return
;
return
false
;
//-- Auto discovery
if
(
_activeVehicle
&&
_activeVehicle
->
dynamicCameras
())
{
QGCVideoStreamInfo
*
pInfo
=
_activeVehicle
->
dynamicCameras
()
->
currentStreamInstance
();
if
(
pInfo
)
{
bool
status
=
false
;
qCDebug
(
VideoManagerLog
)
<<
"Configure primary stream: "
<<
pInfo
->
uri
();
switch
(
pInfo
->
type
())
{
case
VIDEO_STREAM_TYPE_RTSP
:
_setVideoUri
(
pInfo
->
uri
());
if
(
status
=
_updateVideoUri
(
pInfo
->
uri
()))
{
_toolbox
->
settingsManager
()
->
videoSettings
()
->
videoSource
()
->
setRawValue
(
VideoSettings
::
videoSourceRTSP
);
}
break
;
case
VIDEO_STREAM_TYPE_TCP_MPEG
:
_setVideoUri
(
pInfo
->
uri
());
if
(
status
=
_updateVideoUri
(
pInfo
->
uri
()))
{
_toolbox
->
settingsManager
()
->
videoSettings
()
->
videoSource
()
->
setRawValue
(
VideoSettings
::
videoSourceTCP
);
}
break
;
case
VIDEO_STREAM_TYPE_RTPUDP
:
_setVideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
pInfo
->
uri
()));
if
(
status
=
_updateVideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
pInfo
->
uri
())))
{
_toolbox
->
settingsManager
()
->
videoSettings
()
->
videoSource
()
->
setRawValue
(
VideoSettings
::
videoSourceUDPH264
);
}
break
;
case
VIDEO_STREAM_TYPE_MPEG_TS_H264
:
_setVideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
pInfo
->
uri
()));
if
(
status
=
_updateVideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
pInfo
->
uri
())))
{
_toolbox
->
settingsManager
()
->
videoSettings
()
->
videoSource
()
->
setRawValue
(
VideoSettings
::
videoSourceMPEGTS
);
}
break
;
default:
_set
VideoUri
(
pInfo
->
uri
());
status
=
_update
VideoUri
(
pInfo
->
uri
());
break
;
}
//-- Thermal stream (if any)
...
...
@@ -564,50 +574,55 @@ VideoManager::_updateSettings()
switch
(
pTinfo
->
type
())
{
case
VIDEO_STREAM_TYPE_RTSP
:
case
VIDEO_STREAM_TYPE_TCP_MPEG
:
_
set
ThermalVideoUri
(
pTinfo
->
uri
());
_
update
ThermalVideoUri
(
pTinfo
->
uri
());
break
;
case
VIDEO_STREAM_TYPE_RTPUDP
:
_
set
ThermalVideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
pTinfo
->
uri
()));
_
update
ThermalVideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
pTinfo
->
uri
()));
break
;
case
VIDEO_STREAM_TYPE_MPEG_TS_H264
:
_
set
ThermalVideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
pTinfo
->
uri
()));
_
update
ThermalVideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
pTinfo
->
uri
()));
break
;
default:
_
set
ThermalVideoUri
(
pTinfo
->
uri
());
_
update
ThermalVideoUri
(
pTinfo
->
uri
());
break
;
}
}
return
;
return
status
;
}
}
QString
source
=
_videoSettings
->
videoSource
()
->
rawValue
().
toString
();
if
(
source
==
VideoSettings
::
videoSourceUDPH264
)
_set
VideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
return
_update
VideoUri
(
QStringLiteral
(
"udp://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
else
if
(
source
==
VideoSettings
::
videoSourceUDPH265
)
_set
VideoUri
(
QStringLiteral
(
"udp265://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
return
_update
VideoUri
(
QStringLiteral
(
"udp265://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
else
if
(
source
==
VideoSettings
::
videoSourceMPEGTS
)
_set
VideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
return
_update
VideoUri
(
QStringLiteral
(
"mpegts://0.0.0.0:%1"
).
arg
(
_videoSettings
->
udpPort
()
->
rawValue
().
toInt
()));
else
if
(
source
==
VideoSettings
::
videoSourceRTSP
)
_set
VideoUri
(
_videoSettings
->
rtspUrl
()
->
rawValue
().
toString
());
return
_update
VideoUri
(
_videoSettings
->
rtspUrl
()
->
rawValue
().
toString
());
else
if
(
source
==
VideoSettings
::
videoSourceTCP
)
_set
VideoUri
(
QStringLiteral
(
"tcp://%1"
).
arg
(
_videoSettings
->
tcpUrl
()
->
rawValue
().
toString
()));
return
_update
VideoUri
(
QStringLiteral
(
"tcp://%1"
).
arg
(
_videoSettings
->
tcpUrl
()
->
rawValue
().
toString
()));
}
void
VideoManager
::
_
set
VideoUri
(
const
QString
&
uri
)
bool
VideoManager
::
_
update
VideoUri
(
const
QString
&
uri
)
{
#if defined(QGC_GST_TAISYNC_ENABLED) && (defined(__android__) || defined(__ios__))
//-- Taisync on iOS or Android sends a raw h.264 stream
if
(
isTaisync
())
{
_videoUri
=
QString
(
"tsusb://0.0.0.0:%1"
).
arg
(
TAISYNC_VIDEO_UDP_PORT
);
return
;
uri
=
QString
(
"tsusb://0.0.0.0:%1"
).
arg
(
TAISYNC_VIDEO_UDP_PORT
);
}
#endif
if
(
uri
==
_videoUri
)
{
return
false
;
}
_videoUri
=
uri
;
return
true
;
}
void
VideoManager
::
_
set
ThermalVideoUri
(
const
QString
&
uri
)
VideoManager
::
_
update
ThermalVideoUri
(
const
QString
&
uri
)
{
#if defined(QGC_GST_TAISYNC_ENABLED) && (defined(__android__) || defined(__ios__))
//-- Taisync on iOS or Android sends a raw h.264 stream
...
...
@@ -635,6 +650,31 @@ VideoManager::_streamingChanged()
#endif
}
//-----------------------------------------------------------------------------
void
VideoManager
::
_streamChanged
()
{
if
(
_updateSettings
())
{
_restartVideo
();
}
}
//-----------------------------------------------------------------------------
void
VideoManager
::
_onStartComplete
(
bool
status
)
{
disconnect
(
_videoReceiver
,
&
VideoReceiver
::
onStartComplete
,
this
,
&
VideoManager
::
_onStartComplete
);
if
(
status
)
{
connect
(
_videoReceiver
,
&
VideoReceiver
::
timeout
,
this
,
&
VideoManager
::
_restartVideo
);
connect
(
_videoReceiver
,
&
VideoReceiver
::
streamingChanged
,
this
,
&
VideoManager
::
_streamingChanged
);
}
else
{
QTimer
::
singleShot
(
1000
,
[
this
](){
_restartVideo
();
});
}
}
//-----------------------------------------------------------------------------
void
VideoManager
::
_restartVideo
()
...
...
@@ -644,7 +684,7 @@ VideoManager::_restartVideo()
stopVideo
();
_updateSettings
();
startVideo
();
emit
aspectRatioChanged
();
//
emit aspectRatioChanged();
#endif
}
...
...
@@ -690,7 +730,7 @@ VideoManager::_setActiveVehicle(Vehicle* vehicle)
if
(
_activeVehicle
)
{
connect
(
_activeVehicle
,
&
Vehicle
::
connectionLostChanged
,
this
,
&
VideoManager
::
_connectionLostChanged
);
if
(
_activeVehicle
->
dynamicCameras
())
{
connect
(
_activeVehicle
->
dynamicCameras
(),
&
QGCCameraManager
::
streamChanged
,
this
,
&
VideoManager
::
_
restartVideo
);
connect
(
_activeVehicle
->
dynamicCameras
(),
&
QGCCameraManager
::
streamChanged
,
this
,
&
VideoManager
::
_
streamChanged
);
QGCCameraControl
*
pCamera
=
_activeVehicle
->
dynamicCameras
()
->
currentCameraInstance
();
if
(
pCamera
)
{
pCamera
->
resumeStream
();
...
...
src/VideoManager/VideoManager.h
View file @
d403f313
...
...
@@ -115,10 +115,12 @@ protected:
friend
class
FinishVideoInitialization
;
void
_initVideo
();
void
_updateSettings
();
void
_setVideoUri
(
const
QString
&
uri
);
void
_
setThermalVideoUri
(
const
QString
&
uri
);
bool
_updateSettings
();
bool
_updateVideoUri
(
const
QString
&
uri
);
void
_
updateThermalVideoUri
(
const
QString
&
uri
);
void
_cleanupOldVideos
();
void
_streamChanged
();
void
_onStartComplete
(
bool
status
);
void
_restartVideo
();
void
_streamingChanged
();
void
_recordingStarted
();
...
...
src/VideoReceiver/GstVideoReceiver.cc
View file @
d403f313
...
...
@@ -69,19 +69,26 @@ void
GstVideoReceiver
::
start
(
const
QString
&
uri
,
unsigned
timeout
)
{
if
(
_apiHandler
.
needDispatch
())
{
_apiHandler
.
dispatch
([
this
,
uri
,
timeout
]()
{
start
(
uri
,
timeout
);
QString
cachedUri
=
uri
;
_apiHandler
.
dispatch
([
this
,
cachedUri
,
timeout
]()
{
start
(
cachedUri
,
timeout
);
});
return
;
}
if
(
_pipeline
)
{
qCDebug
(
VideoReceiverLog
)
<<
"Already running!"
;
_notificationHandler
.
dispatch
([
this
](){
emit
onStartComplete
(
false
);
});
return
;
}
if
(
uri
.
isEmpty
())
{
qCDebug
(
VideoReceiverLog
)
<<
"Failed because URI is not specified"
;
_notificationHandler
.
dispatch
([
this
](){
emit
onStartComplete
(
false
);
});
return
;
}
...
...
@@ -223,9 +230,17 @@ GstVideoReceiver::start(const QString& uri, unsigned timeout)
_source
=
nullptr
;
}
}
_notificationHandler
.
dispatch
([
this
](){
emit
onStartComplete
(
false
);
});
}
else
{
GST_DEBUG_BIN_TO_DOT_FILE
(
GST_BIN
(
_pipeline
),
GST_DEBUG_GRAPH_SHOW_ALL
,
"pipeline-started"
);
qCDebug
(
VideoReceiverLog
)
<<
"Started"
;
_notificationHandler
.
dispatch
([
this
](){
emit
onStartComplete
(
true
);
});
}
}
...
...
@@ -247,6 +262,8 @@ GstVideoReceiver::stop(void)
if
((
bus
=
gst_pipeline_get_bus
(
GST_PIPELINE
(
_pipeline
)))
!=
nullptr
)
{
gst_bus_disable_sync_message_emission
(
bus
);
g_signal_handlers_disconnect_by_data
(
bus
,
this
);
gboolean
recordingValveClosed
=
TRUE
;
g_object_get
(
_recorderValve
,
"drop"
,
&
recordingValveClosed
,
nullptr
);
...
...
@@ -305,6 +322,11 @@ GstVideoReceiver::stop(void)
_notificationHandler
.
dispatch
([
this
](){
emit
streamingChanged
();
});
}
else
{
qCDebug
(
VideoReceiverLog
)
<<
"Streaming did not start"
;
_notificationHandler
.
dispatch
([
this
](){
emit
timeout
();
});
}
}
...
...
@@ -1148,18 +1170,20 @@ GstVideoReceiver::_onBusMessage(GstBus* bus, GstMessage* msg, gpointer data)
}
if
(
error
!=
nullptr
)
{
qCCritical
(
VideoReceiverLog
)
<<
error
->
message
;
qCCritical
(
VideoReceiverLog
)
<<
"GStreamer error:"
<<
error
->
message
;
g_error_free
(
error
);
error
=
nullptr
;
}
pThis
->
_apiHandler
.
dispatch
([
pThis
](){
pThis
->
stop
();
});
// pThis->_apiHandler.dispatch([pThis](){
// qCDebug(VideoReceiverLog) << "Stoppping because of error";
// pThis->stop();
// });
}
while
(
0
);
break
;
case
GST_MESSAGE_EOS
:
pThis
->
_apiHandler
.
dispatch
([
pThis
](){
qCDebug
(
VideoReceiverLog
)
<<
"Received EOS"
;
pThis
->
_handleEOS
();
});
break
;
...
...
@@ -1181,6 +1205,7 @@ GstVideoReceiver::_onBusMessage(GstBus* bus, GstMessage* msg, gpointer data)
if
(
GST_MESSAGE_TYPE
(
forward_msg
)
==
GST_MESSAGE_EOS
)
{
pThis
->
_apiHandler
.
dispatch
([
pThis
](){
qCDebug
(
VideoReceiverLog
)
<<
"Received branch EOS"
;
pThis
->
_handleEOS
();
});
}
...
...
src/VideoReceiver/VideoReceiver.h
View file @
d403f313
...
...
@@ -75,6 +75,16 @@ signals:
void
videoSizeChanged
(
void
);
void
screenshotComplete
(
void
);
// FIXME: AV: I see very big sense to convert 'bool status' into 'enum status' and clearly state what happend during operation
void
onStartComplete
(
bool
status
);
// FIXME: AV: add these signals after completing onStartComplete()
// void onStopComplete(bool status);
// void onStartDecodingComplete(bool status);
// void onStopDecodingComplete(bool status);
// void onStartRecordingComplete(bool status);
// void onStopRecordingComplete(bool status);
// void onTakeScreenshotComplete(bool status);
public
slots
:
virtual
void
start
(
const
QString
&
uri
,
unsigned
timeout
)
=
0
;
virtual
void
stop
(
void
)
=
0
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment