Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc-2-2
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc-2-2
Commits
841eb240
Commit
841eb240
authored
Nov 21, 2010
by
Rémi Denis-Courmont
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Qt4: do not crash if the video window is released after the interface
This should fix #3359.
parent
6088b41d
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
50 additions
and
26 deletions
+50
-26
modules/gui/qt4/main_interface.cpp
modules/gui/qt4/main_interface.cpp
+7
-3
modules/gui/qt4/qt4.cpp
modules/gui/qt4/qt4.cpp
+43
-23
No files found.
modules/gui/qt4/main_interface.cpp
View file @
841eb240
...
@@ -267,6 +267,7 @@ MainInterface::~MainInterface()
...
@@ -267,6 +267,7 @@ MainInterface::~MainInterface()
if
(
stackCentralOldWidget
==
videoWidget
)
if
(
stackCentralOldWidget
==
videoWidget
)
showTab
(
bgWidget
);
showTab
(
bgWidget
);
if
(
videoWidget
)
releaseVideoSlot
();
releaseVideoSlot
();
#ifdef WIN32
#ifdef WIN32
...
@@ -601,7 +602,10 @@ void MainInterface::releaseVideo( void )
...
@@ -601,7 +602,10 @@ void MainInterface::releaseVideo( void )
/* Function that is CONNECTED to the previous emit */
/* Function that is CONNECTED to the previous emit */
void
MainInterface
::
releaseVideoSlot
(
void
)
void
MainInterface
::
releaseVideoSlot
(
void
)
{
{
if
(
videoWidget
)
/* This function is called when the embedded video window is destroyed,
* or in the rare case that the embedded window is still here but the
* Qt4 interface exits. */
assert
(
videoWidget
);
videoWidget
->
release
();
videoWidget
->
release
();
setVideoOnTop
(
false
);
setVideoOnTop
(
false
);
setVideoFullScreen
(
false
);
setVideoFullScreen
(
false
);
...
...
modules/gui/qt4/qt4.cpp
View file @
841eb240
...
@@ -28,6 +28,7 @@
...
@@ -28,6 +28,7 @@
#include <QApplication>
#include <QApplication>
#include <QDate>
#include <QDate>
#include <QMutex>
#include "qt4.hpp"
#include "qt4.hpp"
...
@@ -271,11 +272,9 @@ static vlc_sem_t ready;
...
@@ -271,11 +272,9 @@ static vlc_sem_t ready;
#ifdef Q_WS_X11
#ifdef Q_WS_X11
static
char
*
x11_display
=
NULL
;
static
char
*
x11_display
=
NULL
;
#endif
#endif
static
struct
static
QMutex
lock
;
{
static
bool
busy
=
false
;
vlc_mutex_t
lock
;
static
bool
active
=
false
;
bool
busy
;
}
one
=
{
VLC_STATIC_MUTEX
,
false
};
/*****************************************************************************
/*****************************************************************************
* Module callbacks
* Module callbacks
...
@@ -321,11 +320,7 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
...
@@ -321,11 +320,7 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
char
*
display
=
NULL
;
char
*
display
=
NULL
;
#endif
#endif
bool
busy
;
QMutexLocker
locker
(
&
lock
);
vlc_mutex_lock
(
&
one
.
lock
);
busy
=
one
.
busy
;
one
.
busy
=
true
;
vlc_mutex_unlock
(
&
one
.
lock
);
if
(
busy
)
if
(
busy
)
{
{
msg_Err
(
p_this
,
"cannot start Qt4 multiple times"
);
msg_Err
(
p_this
,
"cannot start Qt4 multiple times"
);
...
@@ -352,9 +347,6 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
...
@@ -352,9 +347,6 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
{
{
delete
p_sys
;
delete
p_sys
;
free
(
display
);
free
(
display
);
vlc_mutex_lock
(
&
one
.
lock
);
one
.
busy
=
false
;
vlc_mutex_unlock
(
&
one
.
lock
);
return
VLC_ENOMEM
;
return
VLC_ENOMEM
;
}
}
#endif
#endif
...
@@ -364,6 +356,7 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
...
@@ -364,6 +356,7 @@ static int Open( vlc_object_t *p_this, bool isDialogProvider )
* an embedded video window. */
* an embedded video window. */
vlc_sem_wait
(
&
ready
);
vlc_sem_wait
(
&
ready
);
vlc_sem_destroy
(
&
ready
);
vlc_sem_destroy
(
&
ready
);
busy
=
active
=
true
;
#ifndef Q_WS_MAC
#ifndef Q_WS_MAC
if
(
!
isDialogProvider
)
if
(
!
isDialogProvider
)
...
@@ -416,8 +409,9 @@ static void Close( vlc_object_t *p_this )
...
@@ -416,8 +409,9 @@ static void Close( vlc_object_t *p_this )
#endif
#endif
delete
p_sys
;
delete
p_sys
;
vlc_mutex_locker
locker
(
&
one
.
lock
);
QMutexLocker
locker
(
&
lock
);
one
.
busy
=
false
;
assert
(
busy
);
busy
=
false
;
}
}
static
void
*
Thread
(
void
*
obj
)
static
void
*
Thread
(
void
*
obj
)
...
@@ -487,15 +481,17 @@ static void *Thread( void *obj )
...
@@ -487,15 +481,17 @@ static void *Thread( void *obj )
/* Create the normal interface in non-DP mode */
/* Create the normal interface in non-DP mode */
if
(
!
p_intf
->
p_sys
->
b_isDialogProvider
)
if
(
!
p_intf
->
p_sys
->
b_isDialogProvider
)
{
p_mi
=
new
MainInterface
(
p_intf
);
p_mi
=
new
MainInterface
(
p_intf
);
p_intf
->
p_sys
->
p_mi
=
p_mi
;
}
else
else
p_mi
=
NULL
;
p_mi
=
NULL
;
p_intf
->
p_sys
->
p_mi
=
p_mi
;
/* Explain how to show a dialog :D */
/* Explain how to show a dialog :D */
p_intf
->
pf_show_dialog
=
ShowDialog
;
p_intf
->
pf_show_dialog
=
ShowDialog
;
/* */
/*
Tell the main LibVLC thread we are ready
*/
vlc_sem_post
(
&
ready
);
vlc_sem_post
(
&
ready
);
#ifdef Q_WS_MAC
#ifdef Q_WS_MAC
...
@@ -525,9 +521,8 @@ static void *Thread( void *obj )
...
@@ -525,9 +521,8 @@ static void *Thread( void *obj )
msg_Dbg
(
p_intf
,
"QApp exec() finished"
);
msg_Dbg
(
p_intf
,
"QApp exec() finished"
);
if
(
p_mi
!=
NULL
)
if
(
p_mi
!=
NULL
)
{
{
#warning BUG!
QMutexLocker
locker
(
&
lock
);
/* FIXME: the video window may still be registerd at this point */
active
=
false
;
/* See LP#448082 as an example. */
p_intf
->
p_sys
->
p_mi
=
NULL
;
p_intf
->
p_sys
->
p_mi
=
NULL
;
/* Destroy first the main interface because it is connected to some
/* Destroy first the main interface because it is connected to some
...
@@ -602,6 +597,10 @@ static int WindowOpen( vlc_object_t *p_obj )
...
@@ -602,6 +597,10 @@ static int WindowOpen( vlc_object_t *p_obj )
return
VLC_EGENERIC
;
return
VLC_EGENERIC
;
}
}
QMutexLocker
locker
(
&
lock
);
if
(
unlikely
(
!
active
))
return
VLC_EGENERIC
;
MainInterface
*
p_mi
=
p_intf
->
p_sys
->
p_mi
;
MainInterface
*
p_mi
=
p_intf
->
p_sys
->
p_mi
;
msg_Dbg
(
p_obj
,
"requesting video..."
);
msg_Dbg
(
p_obj
,
"requesting video..."
);
...
@@ -637,6 +636,13 @@ static int WindowOpen( vlc_object_t *p_obj )
...
@@ -637,6 +636,13 @@ static int WindowOpen( vlc_object_t *p_obj )
static
int
WindowControl
(
vout_window_t
*
p_wnd
,
int
i_query
,
va_list
args
)
static
int
WindowControl
(
vout_window_t
*
p_wnd
,
int
i_query
,
va_list
args
)
{
{
MainInterface
*
p_mi
=
(
MainInterface
*
)
p_wnd
->
sys
;
MainInterface
*
p_mi
=
(
MainInterface
*
)
p_wnd
->
sys
;
QMutexLocker
locker
(
&
lock
);
if
(
unlikely
(
!
active
))
{
msg_Warn
(
p_wnd
,
"video already released before control"
);
return
VLC_EGENERIC
;
}
return
p_mi
->
controlVideo
(
i_query
,
args
);
return
p_mi
->
controlVideo
(
i_query
,
args
);
}
}
...
@@ -644,8 +650,22 @@ static void WindowClose( vlc_object_t *p_obj )
...
@@ -644,8 +650,22 @@ static void WindowClose( vlc_object_t *p_obj )
{
{
vout_window_t
*
p_wnd
=
(
vout_window_t
*
)
p_obj
;
vout_window_t
*
p_wnd
=
(
vout_window_t
*
)
p_obj
;
MainInterface
*
p_mi
=
(
MainInterface
*
)
p_wnd
->
sys
;
MainInterface
*
p_mi
=
(
MainInterface
*
)
p_wnd
->
sys
;
QMutexLocker
locker
(
&
lock
);
msg_Dbg
(
p_obj
,
"releasing video..."
);
/* Normally, the interface terminates after the video. In the contrary, the
* Qt4 main loop is gone, so we cannot send any event to the user interface
* widgets. Ideally, we would keep the Qt4 main loop running until after
* the video window is released. But it is far simpler to just have the Qt4
* thread destroy the window early, and to turn this function into a stub.
*
* That assumes the video output will behave sanely if it window is
* destroyed asynchronously.
* XCB and Xlib-XCB are fine with that. Plain Xlib wouldn't, */
if
(
unlikely
(
!
active
))
{
msg_Warn
(
p_obj
,
"video already released"
);
return
;
}
msg_Dbg
(
p_obj
,
"releasing video..."
);
p_mi
->
releaseVideo
();
p_mi
->
releaseVideo
();
}
}
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