Video sometimes flickers when playing to AppleTV
[WebKit-https.git] / Source / WebCore / platform / audio / cocoa / MediaSessionManagerCocoa.cpp
1 /*
2  * Copyright (C) 2013-2014 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "PlatformMediaSessionManager.h"
28
29 #if USE(AUDIO_SESSION)
30
31 #include "AudioSession.h"
32 #include "DeprecatedGlobalSettings.h"
33 #include "Logging.h"
34 #include <wtf/Function.h>
35
36 using namespace WebCore;
37
38 static const size_t kWebAudioBufferSize = 128;
39 static const size_t kLowPowerVideoBufferSize = 4096;
40 static const Seconds updateSessionStateDelay { 100_ms };
41
42 void PlatformMediaSessionManager::scheduleUpdateSessionState()
43 {
44     LOG(Media, "PlatformMediaSessionManager::scheduleUpdateSessionState() - types: Video(%d), Audio(%d), WebAudio(%d)", count(PlatformMediaSession::Video), count(PlatformMediaSession::Audio), count(PlatformMediaSession::WebAudio));
45
46     if (has(PlatformMediaSession::WebAudio))
47         AudioSession::sharedSession().setPreferredBufferSize(kWebAudioBufferSize);
48     // In case of audio capture, we want to grab 20 ms chunks to limit the latency so that it is not noticeable by users
49     // while having a large enough buffer so that the audio rendering remains stable, hence a computation based on sample rate.
50     else if (has(PlatformMediaSession::MediaStreamCapturingAudio))
51         AudioSession::sharedSession().setPreferredBufferSize(AudioSession::sharedSession().sampleRate() / 50);
52     else if ((has(PlatformMediaSession::Video) || has(PlatformMediaSession::Audio)) && DeprecatedGlobalSettings::lowPowerVideoAudioBufferSizeEnabled()) {
53         // FIXME: <http://webkit.org/b/116725> Figure out why enabling the code below
54         // causes media LayoutTests to fail on 10.8.
55
56         size_t bufferSize;
57         if (m_audioHardwareListener && m_audioHardwareListener->outputDeviceSupportsLowPowerMode())
58             bufferSize = kLowPowerVideoBufferSize;
59         else
60             bufferSize = kWebAudioBufferSize;
61
62         AudioSession::sharedSession().setPreferredBufferSize(bufferSize);
63     }
64
65     if (!DeprecatedGlobalSettings::shouldManageAudioSessionCategory())
66         return;
67
68     if (!m_updateStateTimer) {
69         auto updateSessionState = [this] () mutable {
70
71             bool hasWebAudioType = false;
72             bool hasAudibleAudioOrVideoMediaType = false;
73             bool hasAudioCapture = anyOfSessions([&hasWebAudioType, &hasAudibleAudioOrVideoMediaType] (PlatformMediaSession& session, size_t) mutable {
74                 auto type = session.mediaType();
75                 if (type == PlatformMediaSession::WebAudio)
76                     hasWebAudioType = true;
77                 if ((type == PlatformMediaSession::VideoAudio || type == PlatformMediaSession::Audio) && session.canProduceAudio() && session.state() == PlatformMediaSession::Playing)
78                     hasAudibleAudioOrVideoMediaType = true;
79                 return (type == PlatformMediaSession::MediaStreamCapturingAudio);
80             });
81
82             if (hasAudioCapture)
83                 AudioSession::sharedSession().setCategory(AudioSession::PlayAndRecord);
84             else if (hasAudibleAudioOrVideoMediaType)
85                 AudioSession::sharedSession().setCategory(AudioSession::MediaPlayback);
86             else if (hasWebAudioType)
87                 AudioSession::sharedSession().setCategory(AudioSession::AmbientSound);
88             else
89                 AudioSession::sharedSession().setCategory(AudioSession::None);
90         };
91
92         m_updateStateTimer = std::make_unique<DeferrableOneShotTimer>(WTFMove(updateSessionState), updateSessionStateDelay);
93     }
94
95     m_updateStateTimer->restart();
96 }
97
98 #endif // USE(AUDIO_SESSION)