Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
M
mpd
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Registry
Registry
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Иван Мажукин
mpd
Commits
b1d75672
Commit
b1d75672
authored
Dec 02, 2020
by
Shen-Ta Hsieh
Committed by
Max Kellermann
Mar 04, 2021
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
win32: Add ComWorker to run all COM function on same thread
parent
5103eb30
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
160 additions
and
0 deletions
+160
-0
meson.build
meson.build
+1
-0
ComWorker.cxx
src/win32/ComWorker.cxx
+49
-0
ComWorker.hxx
src/win32/ComWorker.hxx
+110
-0
No files found.
meson.build
View file @
b1d75672
...
...
@@ -322,6 +322,7 @@ sources = [
if is_windows
sources += [
'src/win32/Win32Main.cxx',
'src/win32/ComWorker.cxx',
]
endif
...
...
src/win32/ComWorker.cxx
0 → 100644
View file @
b1d75672
/*
* Copyright 2020 The Music Player Daemon Project
* http://www.musicpd.org
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "ComWorker.hxx"
#include "Log.hxx"
#include "thread/Name.hxx"
#include "util/Domain.hxx"
#include "win32/Com.hxx"
namespace
{
static
constexpr
Domain
com_worker_domain
(
"com_worker"
);
}
Mutex
COMWorker
::
mutex
;
unsigned
int
COMWorker
::
reference_count
=
0
;
std
::
optional
<
COMWorker
::
COMWorkerThread
>
COMWorker
::
thread
;
void
COMWorker
::
COMWorkerThread
::
Work
()
noexcept
{
FormatDebug
(
com_worker_domain
,
"Working thread started"
);
SetThreadName
(
"COM Worker"
);
COM
com
{
true
};
while
(
true
)
{
if
(
!
running_flag
.
test_and_set
())
{
FormatDebug
(
com_worker_domain
,
"Working thread ended"
);
return
;
}
while
(
!
spsc_buffer
.
empty
())
{
std
::
function
<
void
()
>
function
;
spsc_buffer
.
pop
(
function
);
function
();
}
event
.
Wait
(
200
);
}
}
src/win32/ComWorker.hxx
0 → 100644
View file @
b1d75672
/*
* Copyright 2020 The Music Player Daemon Project
* http://www.musicpd.org
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifndef MPD_WIN32_COM_WORKER_HXX
#define MPD_WIN32_COM_WORKER_HXX
#include <boost/lockfree/spsc_queue.hpp>
#include <condition_variable>
#include <mutex>
#include <optional>
#include "thread/Future.hxx"
#include "thread/Mutex.hxx"
#include "thread/Thread.hxx"
#include "win32/WinEvent.hxx"
#include <objbase.h>
#include <windows.h>
// Worker thread for all COM operation
class
COMWorker
{
private
:
class
COMWorkerThread
:
public
Thread
{
public
:
COMWorkerThread
()
:
Thread
{
BIND_THIS_METHOD
(
Work
)}
{}
private
:
friend
class
COMWorker
;
void
Work
()
noexcept
;
void
Finish
()
noexcept
{
running_flag
.
clear
();
event
.
Set
();
}
void
Push
(
const
std
::
function
<
void
()
>
&
function
)
{
spsc_buffer
.
push
(
function
);
event
.
Set
();
}
boost
::
lockfree
::
spsc_queue
<
std
::
function
<
void
()
>>
spsc_buffer
{
32
};
std
::
atomic_flag
running_flag
=
true
;
WinEvent
event
{};
};
public
:
static
void
Aquire
()
{
std
::
unique_lock
locker
(
mutex
);
if
(
reference_count
==
0
)
{
thread
.
emplace
();
thread
->
Start
();
}
++
reference_count
;
}
static
void
Release
()
noexcept
{
std
::
unique_lock
locker
(
mutex
);
--
reference_count
;
if
(
reference_count
==
0
)
{
thread
->
Finish
();
thread
->
Join
();
thread
.
reset
();
}
}
template
<
typename
Function
,
typename
...
Args
>
static
auto
Async
(
Function
&&
function
,
Args
&&
...
args
)
{
using
R
=
std
::
invoke_result_t
<
std
::
decay_t
<
Function
>
,
std
::
decay_t
<
Args
>
...
>
;
auto
promise
=
std
::
make_shared
<
Promise
<
R
>>
();
auto
future
=
promise
->
get_future
();
thread
->
Push
([
function
=
std
::
forward
<
Function
>
(
function
),
args
=
std
::
make_tuple
(
std
::
forward
<
Args
>
(
args
)...),
promise
=
std
::
move
(
promise
)]()
mutable
{
try
{
if
constexpr
(
std
::
is_void_v
<
R
>
)
{
std
::
apply
(
std
::
forward
<
Function
>
(
function
),
std
::
move
(
args
));
promise
->
set_value
();
}
else
{
promise
->
set_value
(
std
::
apply
(
std
::
forward
<
Function
>
(
function
),
std
::
move
(
args
)));
}
}
catch
(...)
{
promise
->
set_exception
(
std
::
current_exception
());
}
});
return
future
;
}
private
:
static
Mutex
mutex
;
static
unsigned
int
reference_count
;
static
std
::
optional
<
COMWorkerThread
>
thread
;
};
#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