Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
W
wine-winehq
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
wine
wine-winehq
Commits
63ce6219
Commit
63ce6219
authored
Nov 02, 2015
by
Maarten Lankhorst
Committed by
Alexandre Julliard
Nov 03, 2015
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
winepulse: Add session support.
Signed-off-by:
Andrew Eikum
<
aeikum@codeweavers.com
>
Signed-off-by:
Alexandre Julliard
<
julliard@winehq.org
>
parent
94d88e00
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
848 additions
and
1 deletion
+848
-1
mmdevdrv.c
dlls/winepulse.drv/mmdevdrv.c
+848
-1
No files found.
dlls/winepulse.drv/mmdevdrv.c
View file @
63ce6219
...
...
@@ -76,6 +76,7 @@ static pa_mainloop *pulse_ml;
static
HANDLE
pulse_thread
;
static
pthread_mutex_t
pulse_lock
;
static
pthread_cond_t
pulse_cond
=
PTHREAD_COND_INITIALIZER
;
static
struct
list
g_sessions
=
LIST_INIT
(
g_sessions
);
/* Mixer format + period times */
static
WAVEFORMATEXTENSIBLE
pulse_fmt
[
2
];
...
...
@@ -117,6 +118,31 @@ BOOL WINAPI DllMain(HINSTANCE dll, DWORD reason, void *reserved)
typedef
struct
ACImpl
ACImpl
;
typedef
struct
_AudioSession
{
GUID
guid
;
struct
list
clients
;
IMMDevice
*
device
;
float
master_vol
;
UINT32
channel_count
;
float
*
channel_vols
;
BOOL
mute
;
struct
list
entry
;
}
AudioSession
;
typedef
struct
_AudioSessionWrapper
{
IAudioSessionControl2
IAudioSessionControl2_iface
;
IChannelAudioVolume
IChannelAudioVolume_iface
;
ISimpleAudioVolume
ISimpleAudioVolume_iface
;
LONG
ref
;
ACImpl
*
client
;
AudioSession
*
session
;
}
AudioSessionWrapper
;
typedef
struct
_ACPacket
{
struct
list
entry
;
UINT64
qpcpos
;
...
...
@@ -150,6 +176,8 @@ struct ACImpl {
INT64
clock_lastpos
,
clock_written
;
AudioSession
*
session
;
AudioSessionWrapper
*
session_wrapper
;
struct
list
packet_free_head
;
struct
list
packet_filled_head
;
};
...
...
@@ -159,10 +187,15 @@ static const WCHAR defaultW[] = {'P','u','l','s','e','a','u','d','i','o',0};
static
const
IAudioClientVtbl
AudioClient_Vtbl
;
static
const
IAudioRenderClientVtbl
AudioRenderClient_Vtbl
;
static
const
IAudioCaptureClientVtbl
AudioCaptureClient_Vtbl
;
static
const
IAudioSessionControl2Vtbl
AudioSessionControl2_Vtbl
;
static
const
ISimpleAudioVolumeVtbl
SimpleAudioVolume_Vtbl
;
static
const
IChannelAudioVolumeVtbl
ChannelAudioVolume_Vtbl
;
static
const
IAudioClockVtbl
AudioClock_Vtbl
;
static
const
IAudioClock2Vtbl
AudioClock2_Vtbl
;
static
const
IAudioStreamVolumeVtbl
AudioStreamVolume_Vtbl
;
static
AudioSessionWrapper
*
AudioSessionWrapper_Create
(
ACImpl
*
client
);
static
inline
ACImpl
*
impl_from_IAudioClient
(
IAudioClient
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
ACImpl
,
IAudioClient_iface
);
...
...
@@ -178,6 +211,21 @@ static inline ACImpl *impl_from_IAudioCaptureClient(IAudioCaptureClient *iface)
return
CONTAINING_RECORD
(
iface
,
ACImpl
,
IAudioCaptureClient_iface
);
}
static
inline
AudioSessionWrapper
*
impl_from_IAudioSessionControl2
(
IAudioSessionControl2
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
AudioSessionWrapper
,
IAudioSessionControl2_iface
);
}
static
inline
AudioSessionWrapper
*
impl_from_ISimpleAudioVolume
(
ISimpleAudioVolume
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
AudioSessionWrapper
,
ISimpleAudioVolume_iface
);
}
static
inline
AudioSessionWrapper
*
impl_from_IChannelAudioVolume
(
IChannelAudioVolume
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
AudioSessionWrapper
,
IChannelAudioVolume_iface
);
}
static
inline
ACImpl
*
impl_from_IAudioClock
(
IAudioClock
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
ACImpl
,
IAudioClock_iface
);
...
...
@@ -953,6 +1001,85 @@ static DWORD get_channel_mask(unsigned int channels)
return
0
;
}
static
void
session_init_vols
(
AudioSession
*
session
,
UINT
channels
)
{
if
(
session
->
channel_count
<
channels
)
{
UINT
i
;
if
(
session
->
channel_vols
)
session
->
channel_vols
=
HeapReAlloc
(
GetProcessHeap
(),
0
,
session
->
channel_vols
,
sizeof
(
float
)
*
channels
);
else
session
->
channel_vols
=
HeapAlloc
(
GetProcessHeap
(),
0
,
sizeof
(
float
)
*
channels
);
if
(
!
session
->
channel_vols
)
return
;
for
(
i
=
session
->
channel_count
;
i
<
channels
;
++
i
)
session
->
channel_vols
[
i
]
=
1
.
f
;
session
->
channel_count
=
channels
;
}
}
static
AudioSession
*
create_session
(
const
GUID
*
guid
,
IMMDevice
*
device
,
UINT
num_channels
)
{
AudioSession
*
ret
;
ret
=
HeapAlloc
(
GetProcessHeap
(),
HEAP_ZERO_MEMORY
,
sizeof
(
AudioSession
));
if
(
!
ret
)
return
NULL
;
memcpy
(
&
ret
->
guid
,
guid
,
sizeof
(
GUID
));
ret
->
device
=
device
;
list_init
(
&
ret
->
clients
);
list_add_head
(
&
g_sessions
,
&
ret
->
entry
);
session_init_vols
(
ret
,
num_channels
);
ret
->
master_vol
=
1
.
f
;
return
ret
;
}
/* if channels == 0, then this will return or create a session with
* matching dataflow and GUID. otherwise, channels must also match */
static
HRESULT
get_audio_session
(
const
GUID
*
sessionguid
,
IMMDevice
*
device
,
UINT
channels
,
AudioSession
**
out
)
{
AudioSession
*
session
;
if
(
!
sessionguid
||
IsEqualGUID
(
sessionguid
,
&
GUID_NULL
))
{
*
out
=
create_session
(
&
GUID_NULL
,
device
,
channels
);
if
(
!*
out
)
return
E_OUTOFMEMORY
;
return
S_OK
;
}
*
out
=
NULL
;
LIST_FOR_EACH_ENTRY
(
session
,
&
g_sessions
,
AudioSession
,
entry
)
{
if
(
session
->
device
==
device
&&
IsEqualGUID
(
sessionguid
,
&
session
->
guid
))
{
session_init_vols
(
session
,
channels
);
*
out
=
session
;
break
;
}
}
if
(
!*
out
)
{
*
out
=
create_session
(
sessionguid
,
device
,
channels
);
if
(
!*
out
)
return
E_OUTOFMEMORY
;
}
return
S_OK
;
}
static
HRESULT
pulse_spec_from_waveformat
(
ACImpl
*
This
,
const
WAVEFORMATEX
*
fmt
)
{
pa_channel_map_init
(
&
This
->
map
);
...
...
@@ -1180,6 +1307,10 @@ static HRESULT WINAPI AudioClient_Initialize(IAudioClient *iface,
assert
(
!
capture_packets
||
data
-
This
->
bufsize_bytes
==
This
->
tmp_buffer
);
}
}
if
(
SUCCEEDED
(
hr
))
hr
=
get_audio_session
(
sessionguid
,
This
->
parent
,
fmt
->
nChannels
,
&
This
->
session
);
if
(
SUCCEEDED
(
hr
))
list_add_tail
(
&
This
->
session
->
clients
,
&
This
->
entry
);
exit:
if
(
FAILED
(
hr
))
{
...
...
@@ -1609,6 +1740,20 @@ static HRESULT WINAPI AudioClient_GetService(IAudioClient *iface, REFIID riid,
*
ppv
=
&
This
->
IAudioClock_iface
;
}
else
if
(
IsEqualIID
(
riid
,
&
IID_IAudioStreamVolume
))
{
*
ppv
=
&
This
->
IAudioStreamVolume_iface
;
}
else
if
(
IsEqualIID
(
riid
,
&
IID_IAudioSessionControl
)
||
IsEqualIID
(
riid
,
&
IID_IChannelAudioVolume
)
||
IsEqualIID
(
riid
,
&
IID_ISimpleAudioVolume
))
{
if
(
!
This
->
session_wrapper
)
{
This
->
session_wrapper
=
AudioSessionWrapper_Create
(
This
);
if
(
!
This
->
session_wrapper
)
return
E_OUTOFMEMORY
;
}
if
(
IsEqualIID
(
riid
,
&
IID_IAudioSessionControl
))
*
ppv
=
&
This
->
session_wrapper
->
IAudioSessionControl2_iface
;
else
if
(
IsEqualIID
(
riid
,
&
IID_IChannelAudioVolume
))
*
ppv
=
&
This
->
session_wrapper
->
IChannelAudioVolume_iface
;
else
if
(
IsEqualIID
(
riid
,
&
IID_ISimpleAudioVolume
))
*
ppv
=
&
This
->
session_wrapper
->
ISimpleAudioVolume_iface
;
}
if
(
*
ppv
)
{
...
...
@@ -2230,9 +2375,711 @@ static const IAudioStreamVolumeVtbl AudioStreamVolume_Vtbl =
AudioStreamVolume_GetAllVolumes
};
static
AudioSessionWrapper
*
AudioSessionWrapper_Create
(
ACImpl
*
client
)
{
AudioSessionWrapper
*
ret
;
ret
=
HeapAlloc
(
GetProcessHeap
(),
HEAP_ZERO_MEMORY
,
sizeof
(
AudioSessionWrapper
));
if
(
!
ret
)
return
NULL
;
ret
->
IAudioSessionControl2_iface
.
lpVtbl
=
&
AudioSessionControl2_Vtbl
;
ret
->
ISimpleAudioVolume_iface
.
lpVtbl
=
&
SimpleAudioVolume_Vtbl
;
ret
->
IChannelAudioVolume_iface
.
lpVtbl
=
&
ChannelAudioVolume_Vtbl
;
ret
->
ref
=
!
client
;
ret
->
client
=
client
;
if
(
client
)
{
ret
->
session
=
client
->
session
;
AudioClient_AddRef
(
&
client
->
IAudioClient_iface
);
}
return
ret
;
}
static
HRESULT
WINAPI
AudioSessionControl_QueryInterface
(
IAudioSessionControl2
*
iface
,
REFIID
riid
,
void
**
ppv
)
{
TRACE
(
"(%p)->(%s, %p)
\n
"
,
iface
,
debugstr_guid
(
riid
),
ppv
);
if
(
!
ppv
)
return
E_POINTER
;
*
ppv
=
NULL
;
if
(
IsEqualIID
(
riid
,
&
IID_IUnknown
)
||
IsEqualIID
(
riid
,
&
IID_IAudioSessionControl
)
||
IsEqualIID
(
riid
,
&
IID_IAudioSessionControl2
))
*
ppv
=
iface
;
if
(
*
ppv
)
{
IUnknown_AddRef
((
IUnknown
*
)
*
ppv
);
return
S_OK
;
}
WARN
(
"Unknown interface %s
\n
"
,
debugstr_guid
(
riid
));
return
E_NOINTERFACE
;
}
static
ULONG
WINAPI
AudioSessionControl_AddRef
(
IAudioSessionControl2
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
ULONG
ref
;
ref
=
InterlockedIncrement
(
&
This
->
ref
);
TRACE
(
"(%p) Refcount now %u
\n
"
,
This
,
ref
);
return
ref
;
}
static
ULONG
WINAPI
AudioSessionControl_Release
(
IAudioSessionControl2
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
ULONG
ref
;
ref
=
InterlockedDecrement
(
&
This
->
ref
);
TRACE
(
"(%p) Refcount now %u
\n
"
,
This
,
ref
);
if
(
!
ref
)
{
if
(
This
->
client
)
{
This
->
client
->
session_wrapper
=
NULL
;
AudioClient_Release
(
&
This
->
client
->
IAudioClient_iface
);
}
HeapFree
(
GetProcessHeap
(),
0
,
This
);
}
return
ref
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetState
(
IAudioSessionControl2
*
iface
,
AudioSessionState
*
state
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
ACImpl
*
client
;
TRACE
(
"(%p)->(%p)
\n
"
,
This
,
state
);
if
(
!
state
)
return
NULL_PTR_ERR
;
pthread_mutex_lock
(
&
pulse_lock
);
if
(
list_empty
(
&
This
->
session
->
clients
))
{
*
state
=
AudioSessionStateExpired
;
goto
out
;
}
LIST_FOR_EACH_ENTRY
(
client
,
&
This
->
session
->
clients
,
ACImpl
,
entry
)
{
if
(
client
->
started
)
{
*
state
=
AudioSessionStateActive
;
goto
out
;
}
}
*
state
=
AudioSessionStateInactive
;
out:
pthread_mutex_unlock
(
&
pulse_lock
);
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetDisplayName
(
IAudioSessionControl2
*
iface
,
WCHAR
**
name
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
name
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_SetDisplayName
(
IAudioSessionControl2
*
iface
,
const
WCHAR
*
name
,
const
GUID
*
session
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p, %s) - stub
\n
"
,
This
,
name
,
debugstr_guid
(
session
));
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetIconPath
(
IAudioSessionControl2
*
iface
,
WCHAR
**
path
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
path
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_SetIconPath
(
IAudioSessionControl2
*
iface
,
const
WCHAR
*
path
,
const
GUID
*
session
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p, %s) - stub
\n
"
,
This
,
path
,
debugstr_guid
(
session
));
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetGroupingParam
(
IAudioSessionControl2
*
iface
,
GUID
*
group
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
group
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_SetGroupingParam
(
IAudioSessionControl2
*
iface
,
const
GUID
*
group
,
const
GUID
*
session
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%s, %s) - stub
\n
"
,
This
,
debugstr_guid
(
group
),
debugstr_guid
(
session
));
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_RegisterAudioSessionNotification
(
IAudioSessionControl2
*
iface
,
IAudioSessionEvents
*
events
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
events
);
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionControl_UnregisterAudioSessionNotification
(
IAudioSessionControl2
*
iface
,
IAudioSessionEvents
*
events
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
events
);
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetSessionIdentifier
(
IAudioSessionControl2
*
iface
,
WCHAR
**
id
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
id
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetSessionInstanceIdentifier
(
IAudioSessionControl2
*
iface
,
WCHAR
**
id
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
id
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionControl_GetProcessId
(
IAudioSessionControl2
*
iface
,
DWORD
*
pid
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
TRACE
(
"(%p)->(%p)
\n
"
,
This
,
pid
);
if
(
!
pid
)
return
E_POINTER
;
*
pid
=
GetCurrentProcessId
();
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionControl_IsSystemSoundsSession
(
IAudioSessionControl2
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
TRACE
(
"(%p)
\n
"
,
This
);
return
S_FALSE
;
}
static
HRESULT
WINAPI
AudioSessionControl_SetDuckingPreference
(
IAudioSessionControl2
*
iface
,
BOOL
optout
)
{
AudioSessionWrapper
*
This
=
impl_from_IAudioSessionControl2
(
iface
);
TRACE
(
"(%p)->(%d)
\n
"
,
This
,
optout
);
return
S_OK
;
}
static
const
IAudioSessionControl2Vtbl
AudioSessionControl2_Vtbl
=
{
AudioSessionControl_QueryInterface
,
AudioSessionControl_AddRef
,
AudioSessionControl_Release
,
AudioSessionControl_GetState
,
AudioSessionControl_GetDisplayName
,
AudioSessionControl_SetDisplayName
,
AudioSessionControl_GetIconPath
,
AudioSessionControl_SetIconPath
,
AudioSessionControl_GetGroupingParam
,
AudioSessionControl_SetGroupingParam
,
AudioSessionControl_RegisterAudioSessionNotification
,
AudioSessionControl_UnregisterAudioSessionNotification
,
AudioSessionControl_GetSessionIdentifier
,
AudioSessionControl_GetSessionInstanceIdentifier
,
AudioSessionControl_GetProcessId
,
AudioSessionControl_IsSystemSoundsSession
,
AudioSessionControl_SetDuckingPreference
};
typedef
struct
_SessionMgr
{
IAudioSessionManager2
IAudioSessionManager2_iface
;
LONG
ref
;
IMMDevice
*
device
;
}
SessionMgr
;
static
HRESULT
WINAPI
AudioSessionManager_QueryInterface
(
IAudioSessionManager2
*
iface
,
REFIID
riid
,
void
**
ppv
)
{
TRACE
(
"(%p)->(%s, %p)
\n
"
,
iface
,
debugstr_guid
(
riid
),
ppv
);
if
(
!
ppv
)
return
E_POINTER
;
*
ppv
=
NULL
;
if
(
IsEqualIID
(
riid
,
&
IID_IUnknown
)
||
IsEqualIID
(
riid
,
&
IID_IAudioSessionManager
)
||
IsEqualIID
(
riid
,
&
IID_IAudioSessionManager2
))
*
ppv
=
iface
;
if
(
*
ppv
)
{
IUnknown_AddRef
((
IUnknown
*
)
*
ppv
);
return
S_OK
;
}
WARN
(
"Unknown interface %s
\n
"
,
debugstr_guid
(
riid
));
return
E_NOINTERFACE
;
}
static
inline
SessionMgr
*
impl_from_IAudioSessionManager2
(
IAudioSessionManager2
*
iface
)
{
return
CONTAINING_RECORD
(
iface
,
SessionMgr
,
IAudioSessionManager2_iface
);
}
static
ULONG
WINAPI
AudioSessionManager_AddRef
(
IAudioSessionManager2
*
iface
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
ULONG
ref
;
ref
=
InterlockedIncrement
(
&
This
->
ref
);
TRACE
(
"(%p) Refcount now %u
\n
"
,
This
,
ref
);
return
ref
;
}
static
ULONG
WINAPI
AudioSessionManager_Release
(
IAudioSessionManager2
*
iface
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
ULONG
ref
;
ref
=
InterlockedDecrement
(
&
This
->
ref
);
TRACE
(
"(%p) Refcount now %u
\n
"
,
This
,
ref
);
if
(
!
ref
)
HeapFree
(
GetProcessHeap
(),
0
,
This
);
return
ref
;
}
static
HRESULT
WINAPI
AudioSessionManager_GetAudioSessionControl
(
IAudioSessionManager2
*
iface
,
const
GUID
*
session_guid
,
DWORD
flags
,
IAudioSessionControl
**
out
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
AudioSession
*
session
;
AudioSessionWrapper
*
wrapper
;
HRESULT
hr
;
TRACE
(
"(%p)->(%s, %x, %p)
\n
"
,
This
,
debugstr_guid
(
session_guid
),
flags
,
out
);
hr
=
get_audio_session
(
session_guid
,
This
->
device
,
0
,
&
session
);
if
(
FAILED
(
hr
))
return
hr
;
wrapper
=
AudioSessionWrapper_Create
(
NULL
);
if
(
!
wrapper
)
return
E_OUTOFMEMORY
;
wrapper
->
session
=
session
;
*
out
=
(
IAudioSessionControl
*
)
&
wrapper
->
IAudioSessionControl2_iface
;
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionManager_GetSimpleAudioVolume
(
IAudioSessionManager2
*
iface
,
const
GUID
*
session_guid
,
DWORD
flags
,
ISimpleAudioVolume
**
out
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
AudioSession
*
session
;
AudioSessionWrapper
*
wrapper
;
HRESULT
hr
;
TRACE
(
"(%p)->(%s, %x, %p)
\n
"
,
This
,
debugstr_guid
(
session_guid
),
flags
,
out
);
hr
=
get_audio_session
(
session_guid
,
This
->
device
,
0
,
&
session
);
if
(
FAILED
(
hr
))
return
hr
;
wrapper
=
AudioSessionWrapper_Create
(
NULL
);
if
(
!
wrapper
)
return
E_OUTOFMEMORY
;
wrapper
->
session
=
session
;
*
out
=
&
wrapper
->
ISimpleAudioVolume_iface
;
return
S_OK
;
}
static
HRESULT
WINAPI
AudioSessionManager_GetSessionEnumerator
(
IAudioSessionManager2
*
iface
,
IAudioSessionEnumerator
**
out
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
out
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionManager_RegisterSessionNotification
(
IAudioSessionManager2
*
iface
,
IAudioSessionNotification
*
notification
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
notification
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionManager_UnregisterSessionNotification
(
IAudioSessionManager2
*
iface
,
IAudioSessionNotification
*
notification
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
notification
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionManager_RegisterDuckNotification
(
IAudioSessionManager2
*
iface
,
const
WCHAR
*
session_id
,
IAudioVolumeDuckNotification
*
notification
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
notification
);
return
E_NOTIMPL
;
}
static
HRESULT
WINAPI
AudioSessionManager_UnregisterDuckNotification
(
IAudioSessionManager2
*
iface
,
IAudioVolumeDuckNotification
*
notification
)
{
SessionMgr
*
This
=
impl_from_IAudioSessionManager2
(
iface
);
FIXME
(
"(%p)->(%p) - stub
\n
"
,
This
,
notification
);
return
E_NOTIMPL
;
}
static
const
IAudioSessionManager2Vtbl
AudioSessionManager2_Vtbl
=
{
AudioSessionManager_QueryInterface
,
AudioSessionManager_AddRef
,
AudioSessionManager_Release
,
AudioSessionManager_GetAudioSessionControl
,
AudioSessionManager_GetSimpleAudioVolume
,
AudioSessionManager_GetSessionEnumerator
,
AudioSessionManager_RegisterSessionNotification
,
AudioSessionManager_UnregisterSessionNotification
,
AudioSessionManager_RegisterDuckNotification
,
AudioSessionManager_UnregisterDuckNotification
};
static
HRESULT
WINAPI
SimpleAudioVolume_QueryInterface
(
ISimpleAudioVolume
*
iface
,
REFIID
riid
,
void
**
ppv
)
{
TRACE
(
"(%p)->(%s, %p)
\n
"
,
iface
,
debugstr_guid
(
riid
),
ppv
);
if
(
!
ppv
)
return
E_POINTER
;
*
ppv
=
NULL
;
if
(
IsEqualIID
(
riid
,
&
IID_IUnknown
)
||
IsEqualIID
(
riid
,
&
IID_ISimpleAudioVolume
))
*
ppv
=
iface
;
if
(
*
ppv
)
{
IUnknown_AddRef
((
IUnknown
*
)
*
ppv
);
return
S_OK
;
}
WARN
(
"Unknown interface %s
\n
"
,
debugstr_guid
(
riid
));
return
E_NOINTERFACE
;
}
static
ULONG
WINAPI
SimpleAudioVolume_AddRef
(
ISimpleAudioVolume
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
return
AudioSessionControl_AddRef
(
&
This
->
IAudioSessionControl2_iface
);
}
static
ULONG
WINAPI
SimpleAudioVolume_Release
(
ISimpleAudioVolume
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
return
AudioSessionControl_Release
(
&
This
->
IAudioSessionControl2_iface
);
}
static
HRESULT
WINAPI
SimpleAudioVolume_SetMasterVolume
(
ISimpleAudioVolume
*
iface
,
float
level
,
const
GUID
*
context
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%f, %s)
\n
"
,
session
,
level
,
wine_dbgstr_guid
(
context
));
if
(
level
<
0
.
f
||
level
>
1
.
f
)
return
E_INVALIDARG
;
if
(
context
)
FIXME
(
"Notifications not supported yet
\n
"
);
TRACE
(
"Pulseaudio does not support session volume control
\n
"
);
pthread_mutex_lock
(
&
pulse_lock
);
session
->
master_vol
=
level
;
pthread_mutex_unlock
(
&
pulse_lock
);
return
S_OK
;
}
static
HRESULT
WINAPI
SimpleAudioVolume_GetMasterVolume
(
ISimpleAudioVolume
*
iface
,
float
*
level
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%p)
\n
"
,
session
,
level
);
if
(
!
level
)
return
NULL_PTR_ERR
;
*
level
=
session
->
master_vol
;
return
S_OK
;
}
static
HRESULT
WINAPI
SimpleAudioVolume_SetMute
(
ISimpleAudioVolume
*
iface
,
BOOL
mute
,
const
GUID
*
context
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%u, %p)
\n
"
,
session
,
mute
,
context
);
if
(
context
)
FIXME
(
"Notifications not supported yet
\n
"
);
session
->
mute
=
mute
;
return
S_OK
;
}
static
HRESULT
WINAPI
SimpleAudioVolume_GetMute
(
ISimpleAudioVolume
*
iface
,
BOOL
*
mute
)
{
AudioSessionWrapper
*
This
=
impl_from_ISimpleAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%p)
\n
"
,
session
,
mute
);
if
(
!
mute
)
return
NULL_PTR_ERR
;
*
mute
=
session
->
mute
;
return
S_OK
;
}
static
const
ISimpleAudioVolumeVtbl
SimpleAudioVolume_Vtbl
=
{
SimpleAudioVolume_QueryInterface
,
SimpleAudioVolume_AddRef
,
SimpleAudioVolume_Release
,
SimpleAudioVolume_SetMasterVolume
,
SimpleAudioVolume_GetMasterVolume
,
SimpleAudioVolume_SetMute
,
SimpleAudioVolume_GetMute
};
static
HRESULT
WINAPI
ChannelAudioVolume_QueryInterface
(
IChannelAudioVolume
*
iface
,
REFIID
riid
,
void
**
ppv
)
{
TRACE
(
"(%p)->(%s, %p)
\n
"
,
iface
,
debugstr_guid
(
riid
),
ppv
);
if
(
!
ppv
)
return
E_POINTER
;
*
ppv
=
NULL
;
if
(
IsEqualIID
(
riid
,
&
IID_IUnknown
)
||
IsEqualIID
(
riid
,
&
IID_IChannelAudioVolume
))
*
ppv
=
iface
;
if
(
*
ppv
)
{
IUnknown_AddRef
((
IUnknown
*
)
*
ppv
);
return
S_OK
;
}
WARN
(
"Unknown interface %s
\n
"
,
debugstr_guid
(
riid
));
return
E_NOINTERFACE
;
}
static
ULONG
WINAPI
ChannelAudioVolume_AddRef
(
IChannelAudioVolume
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
return
AudioSessionControl_AddRef
(
&
This
->
IAudioSessionControl2_iface
);
}
static
ULONG
WINAPI
ChannelAudioVolume_Release
(
IChannelAudioVolume
*
iface
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
return
AudioSessionControl_Release
(
&
This
->
IAudioSessionControl2_iface
);
}
static
HRESULT
WINAPI
ChannelAudioVolume_GetChannelCount
(
IChannelAudioVolume
*
iface
,
UINT32
*
out
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%p)
\n
"
,
session
,
out
);
if
(
!
out
)
return
NULL_PTR_ERR
;
*
out
=
session
->
channel_count
;
return
S_OK
;
}
static
HRESULT
WINAPI
ChannelAudioVolume_SetChannelVolume
(
IChannelAudioVolume
*
iface
,
UINT32
index
,
float
level
,
const
GUID
*
context
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%d, %f, %s)
\n
"
,
session
,
index
,
level
,
wine_dbgstr_guid
(
context
));
if
(
level
<
0
.
f
||
level
>
1
.
f
)
return
E_INVALIDARG
;
if
(
index
>=
session
->
channel_count
)
return
E_INVALIDARG
;
if
(
context
)
FIXME
(
"Notifications not supported yet
\n
"
);
TRACE
(
"Pulseaudio does not support session volume control
\n
"
);
pthread_mutex_lock
(
&
pulse_lock
);
session
->
channel_vols
[
index
]
=
level
;
pthread_mutex_unlock
(
&
pulse_lock
);
return
S_OK
;
}
static
HRESULT
WINAPI
ChannelAudioVolume_GetChannelVolume
(
IChannelAudioVolume
*
iface
,
UINT32
index
,
float
*
level
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
TRACE
(
"(%p)->(%d, %p)
\n
"
,
session
,
index
,
level
);
if
(
!
level
)
return
NULL_PTR_ERR
;
if
(
index
>=
session
->
channel_count
)
return
E_INVALIDARG
;
*
level
=
session
->
channel_vols
[
index
];
return
S_OK
;
}
static
HRESULT
WINAPI
ChannelAudioVolume_SetAllVolumes
(
IChannelAudioVolume
*
iface
,
UINT32
count
,
const
float
*
levels
,
const
GUID
*
context
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
int
i
;
TRACE
(
"(%p)->(%d, %p, %s)
\n
"
,
session
,
count
,
levels
,
wine_dbgstr_guid
(
context
));
if
(
!
levels
)
return
NULL_PTR_ERR
;
if
(
count
!=
session
->
channel_count
)
return
E_INVALIDARG
;
if
(
context
)
FIXME
(
"Notifications not supported yet
\n
"
);
TRACE
(
"Pulseaudio does not support session volume control
\n
"
);
pthread_mutex_lock
(
&
pulse_lock
);
for
(
i
=
0
;
i
<
count
;
++
i
)
session
->
channel_vols
[
i
]
=
levels
[
i
];
pthread_mutex_unlock
(
&
pulse_lock
);
return
S_OK
;
}
static
HRESULT
WINAPI
ChannelAudioVolume_GetAllVolumes
(
IChannelAudioVolume
*
iface
,
UINT32
count
,
float
*
levels
)
{
AudioSessionWrapper
*
This
=
impl_from_IChannelAudioVolume
(
iface
);
AudioSession
*
session
=
This
->
session
;
int
i
;
TRACE
(
"(%p)->(%d, %p)
\n
"
,
session
,
count
,
levels
);
if
(
!
levels
)
return
NULL_PTR_ERR
;
if
(
count
!=
session
->
channel_count
)
return
E_INVALIDARG
;
for
(
i
=
0
;
i
<
count
;
++
i
)
levels
[
i
]
=
session
->
channel_vols
[
i
];
return
S_OK
;
}
static
const
IChannelAudioVolumeVtbl
ChannelAudioVolume_Vtbl
=
{
ChannelAudioVolume_QueryInterface
,
ChannelAudioVolume_AddRef
,
ChannelAudioVolume_Release
,
ChannelAudioVolume_GetChannelCount
,
ChannelAudioVolume_SetChannelVolume
,
ChannelAudioVolume_GetChannelVolume
,
ChannelAudioVolume_SetAllVolumes
,
ChannelAudioVolume_GetAllVolumes
};
HRESULT
WINAPI
AUDDRV_GetAudioSessionManager
(
IMMDevice
*
device
,
IAudioSessionManager2
**
out
)
{
SessionMgr
*
This
=
HeapAlloc
(
GetProcessHeap
(),
HEAP_ZERO_MEMORY
,
sizeof
(
SessionMgr
));
*
out
=
NULL
;
return
E_NOTIMPL
;
if
(
!
This
)
return
E_OUTOFMEMORY
;
This
->
IAudioSessionManager2_iface
.
lpVtbl
=
&
AudioSessionManager2_Vtbl
;
This
->
device
=
device
;
This
->
ref
=
1
;
*
out
=
&
This
->
IAudioSessionManager2_iface
;
return
S_OK
;
}
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