WebKit Bugzilla
New
Browse
Log In
×
Sign in with GitHub
or
Remember my login
Create Account
·
Forgot Password
Forgotten password account recovery
RESOLVED FIXED
220471
[HANG] 496ms to 1360ms in WebCore::AVAudioSessionCaptureDeviceManager::refreshAudioCaptureDevices()
https://bugs.webkit.org/show_bug.cgi?id=220471
Summary
[HANG] 496ms to 1360ms in WebCore::AVAudioSessionCaptureDeviceManager::refres...
Jer Noble
Reported
2021-01-08 11:35:31 PST
[HANG] 496ms to 1360ms in WebCore::AVAudioSessionCaptureDeviceManager::refreshAudioCaptureDevices()
Attachments
Patch
(26.82 KB, patch)
2021-01-08 11:54 PST
,
Jer Noble
no flags
Details
Formatted Diff
Diff
Patch
(28.66 KB, patch)
2021-01-12 23:47 PST
,
Jer Noble
youennf
: review+
Details
Formatted Diff
Diff
Patch for landing
(28.61 KB, patch)
2021-01-13 14:00 PST
,
Jer Noble
no flags
Details
Formatted Diff
Diff
Show Obsolete
(1)
View All
Add attachment
proposed patch, testcase, etc.
Jer Noble
Comment 1
2021-01-08 11:46:00 PST
<
rdar://72349642
>
Jer Noble
Comment 2
2021-01-08 11:54:22 PST
Created
attachment 417285
[details]
Patch
Peng Liu
Comment 3
2021-01-08 14:56:55 PST
Comment on
attachment 417285
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=417285&action=review
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:146 > + unsigned m_resultsRemaining { 1 };
Maybe my understanding is wrong. I feel the initial value here needs to be 0?
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:150 > + auto accumulator = CaptureDeviceAccumulator::create([completion = WTFMove(completion)] (Vector<CaptureDevice> devices) mutable {
Nit. The space between square brackets and parentheses can be removed.
> Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:153 > + getCaptureDevices([] (Vector<CaptureDevice>) { });
Ditto.
> Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:183 > + dispatch_async(m_dispatchQueue, makeBlockPtr([this, completion = WTFMove(completion)] () mutable {
Ditto.
Jer Noble
Comment 4
2021-01-08 15:34:23 PST
(In reply to Peng Liu from
comment #3
)
> Comment on
attachment 417285
[details]
> Patch > > View in context: >
https://bugs.webkit.org/attachment.cgi?id=417285&action=review
> > > Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:146 > > + unsigned m_resultsRemaining { 1 }; > > Maybe my understanding is wrong. I feel the initial value here needs to be 0?
Nope. We need to handle the case where all the callbacks are called synchronously. If we start at zero, the master callback will be called if the first callback is called before the second one is added.
youenn fablet
Comment 5
2021-01-11 08:31:00 PST
Comment on
attachment 417285
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=417285&action=review
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:115 > +void RealtimeMediaSourceCenter::getMediaStreamDevices(CompletionHandler<void(Vector<CaptureDevice>)>&& completion)
Should probably be CompletionHandler<void(Vector<CaptureDevice>&&)>&&. Here and below
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:117 > + class CaptureDeviceAccumulator : public RefCounted<CaptureDeviceAccumulator> {
Can we use CallbackAggregator instead?
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:136 > + m_completion(m_results);
WTFMove()
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:140 > + CaptureDeviceAccumulator(CompletionHandler<void(Vector<CaptureDevice>)>&& completion)
explicit
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:145 > + CompletionHandler<void(Vector<CaptureDevice>)> m_completion;
s/m_completion/m_completionHandler/
>> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:150 >> + auto accumulator = CaptureDeviceAccumulator::create([completion = WTFMove(completion)] (Vector<CaptureDevice> devices) mutable { > > Nit. The space between square brackets and parentheses can be removed.
s/(Vector<CaptureDevice> devices)/(auto&& devices)
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:154 > + completion(devices);
WTFMove(devices)
> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.h:69 > + WEBCORE_EXPORT void getMediaStreamDevices(CompletionHandler<void(Vector<CaptureDevice>)>&&);
&&
>> Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:153 >> + getCaptureDevices([] (Vector<CaptureDevice>) { }); > > Ditto.
auto
> Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:185 > + callOnWebThreadOrDispatchAsyncOnMainThread(makeBlockPtr([this, completion = WTFMove(completion), newAudioDevices = WTFMove(newAudioDevices)] () mutable {
Why not simply callOnMainThread? This code is not expected to work in WebKit 1 anyway. It might be safer also to isolate copy newAudioDevices. Something like newAudioDevices = crossThreadCopy(WTFMove(newAudioDevices)) after adding a CaptureDevice::isolatedCopy().
> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:662 > + RealtimeMediaSourceCenter::singleton().getMediaStreamDevices([this, weakThis = makeWeakPtr(this), revealIdsAndLabels, completion = WTFMove(completion)] (Vector<CaptureDevice> devices) mutable {
auto&&
> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:664 > + int microphoneCount = 0;
unsigned maybe
> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:743 > + computeFilteredDeviceList(revealIdsAndLabels, [completionHandler = WTFMove(completionHandler), deviceIDHashSalt] (Vector<CaptureDevice>&& devices) mutable {
deviceIDHashSalt = WTFMove(deviceIDHashSalt)
> Source/WebKit/UIProcess/UserMediaProcessManager.cpp:222 > + WebCore::RealtimeMediaSourceCenter::singleton().getMediaStreamDevices([this, shouldNotify] (Vector<WebCore::CaptureDevice>&& newDevices) mutable {
auto&&
Jer Noble
Comment 6
2021-01-11 10:09:11 PST
(In reply to youenn fablet from
comment #5
)
> Comment on
attachment 417285
[details]
> Patch > > View in context: >
https://bugs.webkit.org/attachment.cgi?id=417285&action=review
> > > Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:115 > > +void RealtimeMediaSourceCenter::getMediaStreamDevices(CompletionHandler<void(Vector<CaptureDevice>)>&& completion) > > Should probably be CompletionHandler<void(Vector<CaptureDevice>&&)>&&. > Here and below
Sure.
> > Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:117 > > + class CaptureDeviceAccumulator : public RefCounted<CaptureDeviceAccumulator> { > > Can we use CallbackAggregator instead?
No, despite its name, CallbackAggregator doesn't aggregate the results passed into the callbacks.
Jer Noble
Comment 7
2021-01-11 12:55:03 PST
(In reply to youenn fablet from
comment #5
)
> Comment on
attachment 417285
[details]
> > Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:185 > > + callOnWebThreadOrDispatchAsyncOnMainThread(makeBlockPtr([this, completion = WTFMove(completion), newAudioDevices = WTFMove(newAudioDevices)] () mutable { > > Why not simply callOnMainThread? This code is not expected to work in WebKit > 1 anyway.
I'm not looking to make any significant architectural changes here; the existing code calls to the WebThread, so I'm going to keep that code in place.
youenn fablet
Comment 8
2021-01-11 12:57:06 PST
Comment on
attachment 417285
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=417285&action=review
>>> Source/WebCore/platform/mediastream/RealtimeMediaSourceCenter.cpp:117 >>> + class CaptureDeviceAccumulator : public RefCounted<CaptureDeviceAccumulator> { >> >> Can we use CallbackAggregator instead? > > No, despite its name, CallbackAggregator doesn't aggregate the results passed into the callbacks.
That is a bit sad we do not have a pattern for it. In any case, I think we should stick with the idea of calling the completion handler when CaptureDeviceAccumulator gets destroyed. That should remove the need for m_resultsRemaining.
Jer Noble
Comment 9
2021-01-12 23:47:28 PST
Created
attachment 417512
[details]
Patch
youenn fablet
Comment 10
2021-01-13 01:34:08 PST
Comment on
attachment 417512
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=417512&action=review
> Source/WebCore/platform/mediastream/CaptureDeviceManager.h:37 > virtual const Vector<CaptureDevice>& captureDevices() = 0;
Can we make captureDevices() private or protected at least?
> Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:184 > + auto newAudioDevices = retrieveAudioSessionCaptureDevices();
This is fine as is now but could easily break in the future if we decide to ref/unref some strings there. I would tend to isolateCopy just in case. Or we could make retrieveAudioSessionCaptureDevices a free function and pass it the audio session.
> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:662 > + RealtimeMediaSourceCenter::singleton().getMediaStreamDevices([this, weakThis = makeWeakPtr(this), revealIdsAndLabels, completion = WTFMove(completion)] (auto&& devices) mutable {
We probably need a weakThis check here.
> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:743 > + computeFilteredDeviceList(revealIdsAndLabels, [completionHandler = WTFMove(completionHandler), deviceIDHashSalt = WTFMove(deviceIDHashSalt)] (Vector<CaptureDevice>&& devices) mutable {
auto&&?
> Source/WebKit/UIProcess/UserMediaProcessManager.cpp:222 > + WebCore::RealtimeMediaSourceCenter::singleton().getMediaStreamDevices([this, shouldNotify] (Vector<WebCore::CaptureDevice>&& newDevices) mutable {
auto&&?
> LayoutTests/ChangeLog:12 > + does), then the test turns that into a video capture constraint, which fails.
I wonder why we have this change of behavior. It seems the audiooutput was filtered out previously and is included now. This seems like a progression to me. As a side bonus, we could make MockAudioCaptureDeviceManager implement getCaptureDevices and return the list of devices asynchronously to validate this code path. That will make it closer to the real manager and might catch future bugs.
Jer Noble
Comment 11
2021-01-13 09:50:06 PST
Comment on
attachment 417512
[details]
Patch View in context:
https://bugs.webkit.org/attachment.cgi?id=417512&action=review
>> Source/WebCore/platform/mediastream/CaptureDeviceManager.h:37 >> virtual const Vector<CaptureDevice>& captureDevices() = 0; > > Can we make captureDevices() private or protected at least?
Private, probably not. Protected, maybe. I'll see.
>> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:662 >> + RealtimeMediaSourceCenter::singleton().getMediaStreamDevices([this, weakThis = makeWeakPtr(this), revealIdsAndLabels, completion = WTFMove(completion)] (auto&& devices) mutable { > > We probably need a weakThis check here.
Ok.
>> Source/WebKit/UIProcess/UserMediaPermissionRequestManagerProxy.cpp:743 >> + computeFilteredDeviceList(revealIdsAndLabels, [completionHandler = WTFMove(completionHandler), deviceIDHashSalt = WTFMove(deviceIDHashSalt)] (Vector<CaptureDevice>&& devices) mutable { > > auto&&?
Sure.
Jer Noble
Comment 12
2021-01-13 13:18:58 PST
(In reply to Jer Noble from
comment #11
)
> Comment on
attachment 417512
[details]
> Patch > > View in context: >
https://bugs.webkit.org/attachment.cgi?id=417512&action=review
> > >> Source/WebCore/platform/mediastream/CaptureDeviceManager.h:37 > >> virtual const Vector<CaptureDevice>& captureDevices() = 0; > > > > Can we make captureDevices() private or protected at least? > > Private, probably not. Protected, maybe. I'll see.
Nope: ``` ./platform/mediastream/RealtimeMediaSourceCenter.cpp:211:79: error: 'captureDevices' is a protected member of 'WebCore::CaptureDeviceManager' ``` I think we'll need to go even further to purge all the synchronous calls from the rest of WebKit before we can do this.
Jer Noble
Comment 13
2021-01-13 13:57:44 PST
(In reply to youenn fablet from
comment #10
)
> Comment on
attachment 417512
[details]
> Patch > > View in context: >
https://bugs.webkit.org/attachment.cgi?id=417512&action=review
> > > Source/WebCore/platform/mediastream/CaptureDeviceManager.h:37 > > virtual const Vector<CaptureDevice>& captureDevices() = 0; > > Can we make captureDevices() private or protected at least? > > > Source/WebCore/platform/mediastream/ios/AVAudioSessionCaptureDeviceManager.mm:184 > > + auto newAudioDevices = retrieveAudioSessionCaptureDevices(); > > This is fine as is now but could easily break in the future if we decide to > ref/unref some strings there. > I would tend to isolateCopy just in case.
``` usr/local/include/wtf/Vector.h:1610:38: No member named 'isolatedCopy' in 'WebCore::AVAudioSessionCaptureDevice' ``` We'd need to implement an isolatedCopy() method on the capture device first; we can do that as a follow up.
Jer Noble
Comment 14
2021-01-13 14:00:16 PST
Created
attachment 417560
[details]
Patch for landing
EWS
Comment 15
2021-01-13 16:01:42 PST
Committed
r271471
: <
https://trac.webkit.org/changeset/271471
> All reviewed patches have been landed. Closing bug and clearing flags on
attachment 417560
[details]
.
youenn fablet
Comment 16
2021-01-15 02:02:18 PST
Comment on
attachment 417560
[details]
Patch for landing View in context:
https://bugs.webkit.org/attachment.cgi?id=417560&action=review
> LayoutTests/ChangeLog:4 > +
https://bugs.webkit.org/show_bug.cgi?id=220493
The title and bug number is wrong here.
Note
You need to
log in
before you can comment on or make changes to this bug.
Top of Page
Format For Printing
XML
Clone This Bug