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
d007a98f
Commit
d007a98f
authored
Aug 14, 2008
by
Rémi Denis-Courmont
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
vlc_clone, vlc_join: untangle objects and threads
parent
079a3e8f
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
177 additions
and
109 deletions
+177
-109
include/vlc_threads.h
include/vlc_threads.h
+10
-1
src/libvlccore.sym
src/libvlccore.sym
+2
-0
src/misc/threads.c
src/misc/threads.c
+165
-108
No files found.
include/vlc_threads.h
View file @
d007a98f
...
...
@@ -122,7 +122,13 @@ typedef pthread_cond_t vlc_cond_t;
typedef
pthread_key_t
vlc_threadvar_t
;
#elif defined( WIN32 ) || defined( UNDER_CE )
typedef
HANDLE
vlc_thread_t
;
typedef
struct
{
HANDLE
handle
;
void
*
(
*
entry
)
(
void
*
);
void
*
data
;
}
*
vlc_thread_t
;
typedef
HANDLE
vlc_mutex_t
;
typedef
HANDLE
vlc_cond_t
;
typedef
DWORD
vlc_threadvar_t
;
...
...
@@ -170,6 +176,9 @@ VLC_EXPORT( int, __vlc_thread_create, ( vlc_object_t *, const char *, int, cons
VLC_EXPORT
(
int
,
__vlc_thread_set_priority
,
(
vlc_object_t
*
,
const
char
*
,
int
,
int
)
);
VLC_EXPORT
(
void
,
__vlc_thread_join
,
(
vlc_object_t
*
,
const
char
*
,
int
)
);
VLC_EXPORT
(
int
,
vlc_clone
,
(
vlc_thread_t
*
,
void
*
(
*
)
(
void
*
),
void
*
,
int
)
);
VLC_EXPORT
(
int
,
vlc_join
,
(
vlc_thread_t
,
void
**
)
);
#define vlc_thread_ready vlc_object_signal
/*****************************************************************************
...
...
src/libvlccore.sym
View file @
d007a98f
...
...
@@ -416,6 +416,7 @@ vlc_b64_decode_binary_to_buffer
vlc_b64_encode
vlc_b64_encode_binary
VLC_Changeset
vlc_clone
VLC_CompileBy
VLC_CompileDomain
VLC_CompileHost
...
...
@@ -447,6 +448,7 @@ vlc_iconv_close
vlc_iconv_open
vlc_inet_ntop
vlc_inet_pton
vlc_join
__vlc_list_children
__vlc_list_find
vlc_list_release
...
...
src/misc/threads.c
View file @
d007a98f
/*****************************************************************************
* threads.c : threads implementation for the VideoLAN client
*****************************************************************************
* Copyright (C) 1999-200
7
the VideoLAN team
* Copyright (C) 1999-200
8
the VideoLAN team
* $Id$
*
* Authors: Jean-Marc Dressler <polux@via.ecp.fr>
...
...
@@ -438,59 +438,29 @@ void vlc_threadvar_delete (vlc_threadvar_t *p_tls)
#endif
}
struct
vlc_thread_boot
{
void
*
(
*
entry
)
(
vlc_object_t
*
);
vlc_object_t
*
object
;
};
#if defined (LIBVLC_USE_PTHREAD)
# define THREAD_RTYPE void *
# define THREAD_RVAL NULL
#elif defined (WIN32)
# define THREAD_RTYPE __stdcall unsigned
# define THREAD_RVAL 0
#endif
static
THREAD_RTYPE
thread_entry
(
void
*
data
)
static
unsigned
__stdcall
vlc_entry
(
void
*
data
)
{
vlc_object_t
*
obj
=
((
struct
vlc_thread_boot
*
)
data
)
->
object
;
void
*
(
*
func
)
(
vlc_object_t
*
)
=
((
struct
vlc_thread_boot
*
)
data
)
->
entry
;
free
(
data
);
#ifndef NDEBUG
vlc_threadvar_set
(
&
thread_object_key
,
obj
);
#endif
msg_Dbg
(
obj
,
"thread started"
);
func
(
obj
);
msg_Dbg
(
obj
,
"thread ended"
);
return
THREAD_RVAL
;
vlc_thread_t
self
=
data
;
self
->
data
=
self
->
entry
(
self
->
data
);
return
0
;
}
#endif
/*****************************************************************************
* vlc_thread_create: create a thread, inner version
*****************************************************************************
* Note that i_priority is only taken into account on platforms supporting
* userland real-time priority threads.
*****************************************************************************/
int
__vlc_thread_create
(
vlc_object_t
*
p_this
,
const
char
*
psz_file
,
int
i_line
,
const
char
*
psz_name
,
void
*
(
*
func
)
(
vlc_object_t
*
),
int
i_priority
,
bool
b_wait
)
/**
* Creates and starts new thread.
*
* @param p_handle [OUT] pointer to write the handle of the created thread to
* @param entry entry point for the thread
* @param data data parameter given to the entry point
* @param priority thread priority value
* @return 0 on success, a standard error code on error.
*/
int
vlc_clone
(
vlc_thread_t
*
p_handle
,
void
*
(
*
entry
)
(
void
*
),
void
*
data
,
int
priority
)
{
int
i_ret
;
vlc_object_internals_t
*
p_priv
=
vlc_internals
(
p_this
);
struct
vlc_thread_boot
*
boot
=
malloc
(
sizeof
(
*
boot
));
if
(
boot
==
NULL
)
return
errno
;
boot
->
entry
=
func
;
boot
->
object
=
p_this
;
vlc_object_lock
(
p_this
);
/* Make sure we don't re-create a thread if the object has already one */
assert
(
!
p_priv
->
b_thread
);
int
ret
;
#if defined( LIBVLC_USE_PTHREAD )
pthread_attr_t
attr
;
...
...
@@ -506,7 +476,9 @@ int __vlc_thread_create( vlc_object_t *p_this, const char * psz_file, int i_line
* where it fails to handle EINTR (bug reports welcome). Some underlying
* libraries might also not handle EINTR properly.
*/
sigset_t
set
,
oldset
;
sigset_t
oldset
;
{
sigset_t
set
;
sigemptyset
(
&
set
);
sigdelset
(
&
set
,
SIGHUP
);
sigaddset
(
&
set
,
SIGINT
);
...
...
@@ -515,27 +487,21 @@ int __vlc_thread_create( vlc_object_t *p_this, const char * psz_file, int i_line
sigaddset
(
&
set
,
SIGPIPE
);
/* We don't want this one, really! */
pthread_sigmask
(
SIG_BLOCK
,
&
set
,
&
oldset
);
#ifndef __APPLE__
if
(
config_GetInt
(
p_this
,
"rt-priority"
)
>
0
)
#endif
}
{
struct
sched_param
p
=
{
.
sched_priority
=
i_
priority
,
};
struct
sched_param
sp
=
{
.
sched_priority
=
priority
,
};
int
policy
;
/* Hack to avoid error msg */
if
(
config_GetType
(
p_this
,
"rt-offset"
)
)
p
.
sched_priority
+=
config_GetInt
(
p_this
,
"rt-offset"
);
if
(
p
.
sched_priority
<=
0
)
p
.
sched_priority
+=
sched_get_priority_max
(
policy
=
SCHED_OTHER
);
if
(
sp
.
sched_priority
<=
0
)
sp
.
sched_priority
+=
sched_get_priority_max
(
policy
=
SCHED_OTHER
);
else
p
.
sched_priority
+=
sched_get_priority_min
(
policy
=
SCHED_RR
);
s
p
.
sched_priority
+=
sched_get_priority_min
(
policy
=
SCHED_RR
);
pthread_attr_setschedpolicy
(
&
attr
,
policy
);
pthread_attr_setschedparam
(
&
attr
,
&
p
);
pthread_attr_setschedparam
(
&
attr
,
&
s
p
);
}
i_ret
=
pthread_create
(
&
p_priv
->
thread_id
,
&
attr
,
thread_entry
,
boot
);
ret
=
pthread_create
(
p_handle
,
&
attr
,
entry
,
data
);
pthread_sigmask
(
SIG_SETMASK
,
&
oldset
,
NULL
);
pthread_attr_destroy
(
&
attr
);
...
...
@@ -544,34 +510,144 @@ int __vlc_thread_create( vlc_object_t *p_this, const char * psz_file, int i_line
* function instead of CreateThread, otherwise you'll end up with
* memory leaks and the signal functions not working (see Microsoft
* Knowledge Base, article 104641) */
HANDLE
hThread
;
vlc_thread_t
th
=
malloc
(
sizeof
(
*
p_handle
));
if
(
th
==
NULL
)
return
ENOMEM
;
th
->
data
=
data
;
th
->
entry
=
entry
;
#if defined( UNDER_CE )
HANDLE
hThread
=
CreateThread
(
NULL
,
0
,
thread_entry
,
(
LPVOID
)
boot
,
CREATE_SUSPENDED
,
NULL
);
hThread
=
CreateThread
(
NULL
,
0
,
vlc_entry
,
th
,
CREATE_SUSPENDED
,
NULL
);
#else
HANDLE
hThread
=
(
HANDLE
)(
uintptr_t
)
_beginthreadex
(
NULL
,
0
,
thread_entry
,
boot
,
CREATE_SUSPENDED
,
NULL
);
hThread
=
(
HANDLE
)(
uintptr_t
)
_beginthreadex
(
NULL
,
0
,
vlc_entry
,
th
,
CREATE_SUSPENDED
,
NULL
);
#endif
if
(
hThread
)
if
(
hThread
)
{
p_priv
->
thread_id
=
hThread
;
ResumeThread
(
hThread
);
i_ret
=
0
;
if
(
i_priority
&&
!
SetThreadPriority
(
hThread
,
i_priority
)
)
{
msg_Warn
(
p_this
,
"couldn't set a faster priority"
);
i_priority
=
0
;
}
th
->
handle
=
hThread
;
if
(
priority
)
SetThreadPriority
(
hThread
,
priority
);
ret
=
0
;
}
else
i_ret
=
errno
;
{
ret
=
errno
;
free
(
th
);
th
=
NULL
;
}
*
p_handle
=
th
;
#elif defined( HAVE_KERNEL_SCHEDULER_H )
p_priv
->
thread_id
=
spawn_thread
(
(
thread_func
)
thread_entry
,
psz_name
,
i_priority
,
p_data
);
i_ret
=
resume_thread
(
p_priv
->
thread_id
);
*
p_handle
=
spawn_thread
(
entry
,
psz_name
,
priority
,
data
);
ret
=
resume_thread
(
*
p_handle
);
#endif
return
ret
;
}
/**
* Waits for a thread to complete (if needed), and destroys it.
* @param handle thread handle
* @param p_result [OUT] pointer to write the thread return value or NULL
* @return 0 on success, a standard error code otherwise.
*/
int
vlc_join
(
vlc_thread_t
handle
,
void
**
result
)
{
#if defined( LIBVLC_USE_PTHREAD )
return
pthread_join
(
handle
,
result
);
#elif defined( UNDER_CE ) || defined( WIN32 )
HANDLE
hThread
;
/*
** object will close its thread handle when destroyed, duplicate it here
** to be on the safe side
*/
if
(
!
DuplicateHandle
(
GetCurrentProcess
(),
handle
->
handle
,
GetCurrentProcess
(),
&
hThread
,
0
,
FALSE
,
DUPLICATE_SAME_ACCESS
))
return
GetLastError
();
/* FIXME: errno */
WaitForSingleObject
(
hThread
,
INFINITE
);
CloseHandle
(
hThread
);
if
(
result
)
*
result
=
handle
->
data
;
free
(
handle
);
return
0
;
#elif defined( HAVE_KERNEL_SCHEDULER_H )
int32_t
exit_value
;
ret
=
(
B_OK
==
wait_for_thread
(
p_priv
->
thread_id
,
&
exit_value
));
if
(
!
ret
&&
result
)
*
result
=
(
void
*
)
exit_value
;
return
ret
;
#endif
}
struct
vlc_thread_boot
{
void
*
(
*
entry
)
(
vlc_object_t
*
);
vlc_object_t
*
object
;
};
static
void
*
thread_entry
(
void
*
data
)
{
vlc_object_t
*
obj
=
((
struct
vlc_thread_boot
*
)
data
)
->
object
;
void
*
(
*
func
)
(
vlc_object_t
*
)
=
((
struct
vlc_thread_boot
*
)
data
)
->
entry
;
free
(
data
);
#ifndef NDEBUG
vlc_threadvar_set
(
&
thread_object_key
,
obj
);
#endif
msg_Dbg
(
obj
,
"thread started"
);
func
(
obj
);
msg_Dbg
(
obj
,
"thread ended"
);
return
NULL
;
}
/*****************************************************************************
* vlc_thread_create: create a thread, inner version
*****************************************************************************
* Note that i_priority is only taken into account on platforms supporting
* userland real-time priority threads.
*****************************************************************************/
int
__vlc_thread_create
(
vlc_object_t
*
p_this
,
const
char
*
psz_file
,
int
i_line
,
const
char
*
psz_name
,
void
*
(
*
func
)
(
vlc_object_t
*
),
int
i_priority
,
bool
b_wait
)
{
int
i_ret
;
vlc_object_internals_t
*
p_priv
=
vlc_internals
(
p_this
);
struct
vlc_thread_boot
*
boot
=
malloc
(
sizeof
(
*
boot
));
if
(
boot
==
NULL
)
return
errno
;
boot
->
entry
=
func
;
boot
->
object
=
p_this
;
vlc_object_lock
(
p_this
);
/* Make sure we don't re-create a thread if the object has already one */
assert
(
!
p_priv
->
b_thread
);
#if defined( LIBVLC_USE_PTHREAD )
#ifndef __APPLE__
if
(
config_GetInt
(
p_this
,
"rt-priority"
)
>
0
)
#endif
{
/* Hack to avoid error msg */
if
(
config_GetType
(
p_this
,
"rt-offset"
)
)
i_priority
+=
config_GetInt
(
p_this
,
"rt-offset"
);
}
#endif
i_ret
=
vlc_clone
(
&
p_priv
->
thread_id
,
thread_entry
,
boot
,
i_priority
);
if
(
i_ret
==
0
)
{
if
(
b_wait
)
...
...
@@ -645,7 +721,7 @@ int __vlc_thread_set_priority( vlc_object_t *p_this, const char * psz_file,
#elif defined( WIN32 ) || defined( UNDER_CE )
VLC_UNUSED
(
psz_file
);
VLC_UNUSED
(
i_line
);
if
(
!
SetThreadPriority
(
p_priv
->
thread_id
,
i_priority
)
)
if
(
!
SetThreadPriority
(
p_priv
->
thread_id
->
handle
,
i_priority
)
)
{
msg_Warn
(
p_this
,
"couldn't set a faster priority"
);
return
1
;
...
...
@@ -673,22 +749,15 @@ void __vlc_thread_join( vlc_object_t *p_this, const char * psz_file, int i_line
i_ret
=
pthread_detach
(
p_priv
->
thread_id
);
}
else
i_ret
=
pthread
_join
(
p_priv
->
thread_id
,
NULL
);
i_ret
=
vlc
_join
(
p_priv
->
thread_id
,
NULL
);
#elif defined( UNDER_CE ) || defined( WIN32 )
HMODULE
hmodule
;
BOOL
(
WINAPI
*
OurGetThreadTimes
)(
HANDLE
,
FILETIME
*
,
FILETIME
*
,
FILETIME
*
,
FILETIME
*
);
HANDLE
hThread
;
FILETIME
create_ft
,
exit_ft
,
kernel_ft
,
user_ft
;
int64_t
real_time
,
kernel_time
,
user_time
;
HANDLE
hThread
;
/*
** object will close its thread handle when destroyed, duplicate it here
** to be on the safe side
*/
if
(
!
DuplicateHandle
(
GetCurrentProcess
(),
p_priv
->
thread_id
,
p_priv
->
thread_id
->
handle
,
GetCurrentProcess
(),
&
hThread
,
0
,
...
...
@@ -700,20 +769,9 @@ void __vlc_thread_join( vlc_object_t *p_this, const char * psz_file, int i_line
goto
error
;
}
WaitForSingleObject
(
hThread
,
INFINITE
);
vlc_join
(
p_priv
->
thread_id
,
NULL
);
#if defined( UNDER_CE )
hmodule
=
GetModuleHandle
(
_T
(
"COREDLL"
)
);
#else
hmodule
=
GetModuleHandle
(
_T
(
"KERNEL32"
)
);
#endif
OurGetThreadTimes
=
(
BOOL
(
WINAPI
*
)(
HANDLE
,
FILETIME
*
,
FILETIME
*
,
FILETIME
*
,
FILETIME
*
))
GetProcAddress
(
hmodule
,
_T
(
"GetThreadTimes"
)
);
if
(
OurGetThreadTimes
&&
OurGetThreadTimes
(
hThread
,
&
create_ft
,
&
exit_ft
,
&
kernel_ft
,
&
user_ft
)
)
if
(
GetThreadTimes
(
hThread
,
&
create_ft
,
&
exit_ft
,
&
kernel_ft
,
&
user_ft
)
)
{
real_time
=
((((
int64_t
)
exit_ft
.
dwHighDateTime
)
<<
32
)
|
exit_ft
.
dwLowDateTime
)
-
...
...
@@ -740,9 +798,8 @@ void __vlc_thread_join( vlc_object_t *p_this, const char * psz_file, int i_line
CloseHandle
(
hThread
);
error:
#elif defined( HAVE_KERNEL_SCHEDULER_H )
int32_t
exit_value
;
i_ret
=
(
B_OK
==
wait_for_thread
(
p_priv
->
thread_id
,
&
exit_value
));
#else
i_ret
=
vlc_join
(
p_priv
->
thread_id
,
NULL
);
#endif
...
...
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