OLD | NEW |
---|---|
1 /* | 1 /* |
2 * Copyright (C) 2010, Google Inc. All rights reserved. | 2 * Copyright (C) 2010, Google Inc. All rights reserved. |
3 * | 3 * |
4 * Redistribution and use in source and binary forms, with or without | 4 * Redistribution and use in source and binary forms, with or without |
5 * modification, are permitted provided that the following conditions | 5 * modification, are permitted provided that the following conditions |
6 * are met: | 6 * are met: |
7 * 1. Redistributions of source code must retain the above copyright | 7 * 1. Redistributions of source code must retain the above copyright |
8 * notice, this list of conditions and the following disclaimer. | 8 * notice, this list of conditions and the following disclaimer. |
9 * 2. Redistributions in binary form must reproduce the above copyright | 9 * 2. Redistributions in binary form must reproduce the above copyright |
10 * notice, this list of conditions and the following disclaimer in the | 10 * notice, this list of conditions and the following disclaimer in the |
(...skipping 14 matching lines...) Expand all Loading... | |
25 #include "modules/webaudio/AbstractAudioContext.h" | 25 #include "modules/webaudio/AbstractAudioContext.h" |
26 #include "bindings/core/v8/Dictionary.h" | 26 #include "bindings/core/v8/Dictionary.h" |
27 #include "bindings/core/v8/ExceptionMessages.h" | 27 #include "bindings/core/v8/ExceptionMessages.h" |
28 #include "bindings/core/v8/ExceptionState.h" | 28 #include "bindings/core/v8/ExceptionState.h" |
29 #include "bindings/core/v8/ScriptPromiseResolver.h" | 29 #include "bindings/core/v8/ScriptPromiseResolver.h" |
30 #include "bindings/core/v8/ScriptState.h" | 30 #include "bindings/core/v8/ScriptState.h" |
31 #include "core/dom/DOMException.h" | 31 #include "core/dom/DOMException.h" |
32 #include "core/dom/Document.h" | 32 #include "core/dom/Document.h" |
33 #include "core/dom/ExceptionCode.h" | 33 #include "core/dom/ExceptionCode.h" |
34 #include "core/dom/ExecutionContextTask.h" | 34 #include "core/dom/ExecutionContextTask.h" |
35 #include "core/frame/Settings.h" | |
35 #include "core/html/HTMLMediaElement.h" | 36 #include "core/html/HTMLMediaElement.h" |
36 #include "modules/mediastream/MediaStream.h" | 37 #include "modules/mediastream/MediaStream.h" |
37 #include "modules/webaudio/AnalyserNode.h" | 38 #include "modules/webaudio/AnalyserNode.h" |
38 #include "modules/webaudio/AudioBuffer.h" | 39 #include "modules/webaudio/AudioBuffer.h" |
39 #include "modules/webaudio/AudioBufferCallback.h" | 40 #include "modules/webaudio/AudioBufferCallback.h" |
40 #include "modules/webaudio/AudioBufferSourceNode.h" | 41 #include "modules/webaudio/AudioBufferSourceNode.h" |
41 #include "modules/webaudio/AudioContext.h" | 42 #include "modules/webaudio/AudioContext.h" |
42 #include "modules/webaudio/AudioListener.h" | 43 #include "modules/webaudio/AudioListener.h" |
43 #include "modules/webaudio/AudioNodeInput.h" | 44 #include "modules/webaudio/AudioNodeInput.h" |
44 #include "modules/webaudio/AudioNodeOutput.h" | 45 #include "modules/webaudio/AudioNodeOutput.h" |
(...skipping 12 matching lines...) Expand all Loading... | |
57 #include "modules/webaudio/OfflineAudioCompletionEvent.h" | 58 #include "modules/webaudio/OfflineAudioCompletionEvent.h" |
58 #include "modules/webaudio/OfflineAudioContext.h" | 59 #include "modules/webaudio/OfflineAudioContext.h" |
59 #include "modules/webaudio/OfflineAudioDestinationNode.h" | 60 #include "modules/webaudio/OfflineAudioDestinationNode.h" |
60 #include "modules/webaudio/OscillatorNode.h" | 61 #include "modules/webaudio/OscillatorNode.h" |
61 #include "modules/webaudio/PannerNode.h" | 62 #include "modules/webaudio/PannerNode.h" |
62 #include "modules/webaudio/PeriodicWave.h" | 63 #include "modules/webaudio/PeriodicWave.h" |
63 #include "modules/webaudio/PeriodicWaveConstraints.h" | 64 #include "modules/webaudio/PeriodicWaveConstraints.h" |
64 #include "modules/webaudio/ScriptProcessorNode.h" | 65 #include "modules/webaudio/ScriptProcessorNode.h" |
65 #include "modules/webaudio/StereoPannerNode.h" | 66 #include "modules/webaudio/StereoPannerNode.h" |
66 #include "modules/webaudio/WaveShaperNode.h" | 67 #include "modules/webaudio/WaveShaperNode.h" |
68 #include "platform/Histogram.h" | |
67 #include "platform/ThreadSafeFunctional.h" | 69 #include "platform/ThreadSafeFunctional.h" |
70 #include "platform/UserGestureIndicator.h" | |
68 #include "platform/audio/IIRFilter.h" | 71 #include "platform/audio/IIRFilter.h" |
69 #include "public/platform/Platform.h" | 72 #include "public/platform/Platform.h" |
70 #include "wtf/text/WTFString.h" | 73 #include "wtf/text/WTFString.h" |
71 | 74 |
72 namespace blink { | 75 namespace blink { |
73 | 76 |
77 namespace { | |
78 | |
79 enum UserGestureRecord { | |
80 UserGestureRequiredAndAvailable = 0, | |
81 UserGestureRequiredAndNotAvailable, | |
82 UserGestureNotRequiredAndAvailable, | |
83 UserGestureNotRequiredAndNotAvailable, | |
84 UserGestureRecordMax | |
85 }; | |
86 | |
87 } // anonymous namespace | |
88 | |
74 AbstractAudioContext* AbstractAudioContext::create(Document& document, Exception State& exceptionState) | 89 AbstractAudioContext* AbstractAudioContext::create(Document& document, Exception State& exceptionState) |
75 { | 90 { |
76 return AudioContext::create(document, exceptionState); | 91 return AudioContext::create(document, exceptionState); |
77 } | 92 } |
78 | 93 |
79 // FIXME(dominicc): Devolve these constructors to AudioContext | 94 // FIXME(dominicc): Devolve these constructors to AudioContext |
80 // and OfflineAudioContext respectively. | 95 // and OfflineAudioContext respectively. |
81 | 96 |
82 // Constructor for rendering to the audio hardware. | 97 // Constructor for rendering to the audio hardware. |
83 AbstractAudioContext::AbstractAudioContext(Document* document) | 98 AbstractAudioContext::AbstractAudioContext(Document* document) |
84 : ActiveScriptWrappable(this) | 99 : ActiveScriptWrappable(this) |
85 , ActiveDOMObject(document) | 100 , ActiveDOMObject(document) |
86 , m_destinationNode(nullptr) | 101 , m_destinationNode(nullptr) |
87 , m_isCleared(false) | 102 , m_isCleared(false) |
88 , m_isResolvingResumePromises(false) | 103 , m_isResolvingResumePromises(false) |
104 , m_userGestureRequired(false) | |
89 , m_connectionCount(0) | 105 , m_connectionCount(0) |
90 , m_deferredTaskHandler(DeferredTaskHandler::create()) | 106 , m_deferredTaskHandler(DeferredTaskHandler::create()) |
91 , m_contextState(Suspended) | 107 , m_contextState(Suspended) |
92 , m_closedContextSampleRate(-1) | 108 , m_closedContextSampleRate(-1) |
93 , m_periodicWaveSine(nullptr) | 109 , m_periodicWaveSine(nullptr) |
94 , m_periodicWaveSquare(nullptr) | 110 , m_periodicWaveSquare(nullptr) |
95 , m_periodicWaveSawtooth(nullptr) | 111 , m_periodicWaveSawtooth(nullptr) |
96 , m_periodicWaveTriangle(nullptr) | 112 , m_periodicWaveTriangle(nullptr) |
97 { | 113 { |
114 // TODO(mlamouri): we might want to use other ways of checking for this but | |
115 // in order to record metrics, re-using the HTMLMediaElement setting is | |
116 // probably the simplest solution. | |
117 if (document->settings() && document->settings()->mediaPlaybackRequiresUserG esture()) | |
118 m_userGestureRequired = true; | |
119 | |
98 m_destinationNode = DefaultAudioDestinationNode::create(this); | 120 m_destinationNode = DefaultAudioDestinationNode::create(this); |
99 | 121 |
100 initialize(); | 122 initialize(); |
101 } | 123 } |
102 | 124 |
103 // Constructor for offline (non-realtime) rendering. | 125 // Constructor for offline (non-realtime) rendering. |
104 AbstractAudioContext::AbstractAudioContext(Document* document, unsigned numberOf Channels, size_t numberOfFrames, float sampleRate) | 126 AbstractAudioContext::AbstractAudioContext(Document* document, unsigned numberOf Channels, size_t numberOfFrames, float sampleRate) |
105 : ActiveScriptWrappable(this) | 127 : ActiveScriptWrappable(this) |
106 , ActiveDOMObject(document) | 128 , ActiveDOMObject(document) |
107 , m_destinationNode(nullptr) | 129 , m_destinationNode(nullptr) |
108 , m_isCleared(false) | 130 , m_isCleared(false) |
109 , m_isResolvingResumePromises(false) | 131 , m_isResolvingResumePromises(false) |
132 , m_userGestureRequired(false) | |
110 , m_connectionCount(0) | 133 , m_connectionCount(0) |
111 , m_deferredTaskHandler(DeferredTaskHandler::create()) | 134 , m_deferredTaskHandler(DeferredTaskHandler::create()) |
112 , m_contextState(Suspended) | 135 , m_contextState(Suspended) |
113 , m_closedContextSampleRate(-1) | 136 , m_closedContextSampleRate(-1) |
114 , m_periodicWaveSine(nullptr) | 137 , m_periodicWaveSine(nullptr) |
115 , m_periodicWaveSquare(nullptr) | 138 , m_periodicWaveSquare(nullptr) |
116 , m_periodicWaveSawtooth(nullptr) | 139 , m_periodicWaveSawtooth(nullptr) |
117 , m_periodicWaveTriangle(nullptr) | 140 , m_periodicWaveTriangle(nullptr) |
118 { | 141 { |
142 // TODO(mlamouri): we might want to use other ways of checking for this but | |
143 // in order to record metrics, re-using the HTMLMediaElement setting is | |
144 // probably the simplest solution. | |
145 if (document->settings() && document->settings()->mediaPlaybackRequiresUserG esture()) | |
146 m_userGestureRequired = true; | |
Raymond Toy
2016/06/06 15:10:08
I don't think this is needed or wanted for an offl
mlamouri (slow - plz ping)
2016/06/06 15:14:45
My understanding is that Safari iOS doesn't make a
| |
119 } | 147 } |
120 | 148 |
121 AbstractAudioContext::~AbstractAudioContext() | 149 AbstractAudioContext::~AbstractAudioContext() |
122 { | 150 { |
123 deferredTaskHandler().contextWillBeDestroyed(); | 151 deferredTaskHandler().contextWillBeDestroyed(); |
124 // AudioNodes keep a reference to their context, so there should be no way t o be in the destructor if there are still AudioNodes around. | 152 // AudioNodes keep a reference to their context, so there should be no way t o be in the destructor if there are still AudioNodes around. |
125 ASSERT(!isDestinationInitialized()); | 153 ASSERT(!isDestinationInitialized()); |
126 ASSERT(!m_activeSourceNodes.size()); | 154 ASSERT(!m_activeSourceNodes.size()); |
127 ASSERT(!m_finishedSourceHandlers.size()); | 155 ASSERT(!m_finishedSourceHandlers.size()); |
128 ASSERT(!m_isResolvingResumePromises); | 156 ASSERT(!m_isResolvingResumePromises); |
(...skipping 337 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
466 // Initialize the table if necessary | 494 // Initialize the table if necessary |
467 if (!m_periodicWaveTriangle) | 495 if (!m_periodicWaveTriangle) |
468 m_periodicWaveTriangle = PeriodicWave::createTriangle(sampleRate()); | 496 m_periodicWaveTriangle = PeriodicWave::createTriangle(sampleRate()); |
469 return m_periodicWaveTriangle; | 497 return m_periodicWaveTriangle; |
470 default: | 498 default: |
471 ASSERT_NOT_REACHED(); | 499 ASSERT_NOT_REACHED(); |
472 return nullptr; | 500 return nullptr; |
473 } | 501 } |
474 } | 502 } |
475 | 503 |
504 void AbstractAudioContext::recordUserGestureState() | |
505 { | |
506 DEFINE_STATIC_LOCAL(EnumerationHistogram, userGestureHistogram, ("WebAudio.U serGesture", UserGestureRecordMax)); | |
507 | |
508 if (!m_userGestureRequired) { | |
509 if (UserGestureIndicator::processingUserGesture()) | |
510 userGestureHistogram.count(UserGestureNotRequiredAndAvailable); | |
511 else | |
512 userGestureHistogram.count(UserGestureNotRequiredAndNotAvailable); | |
513 return; | |
514 } | |
515 if (!UserGestureIndicator::processingUserGesture()) { | |
516 userGestureHistogram.count(UserGestureRequiredAndNotAvailable); | |
517 return; | |
518 } | |
519 userGestureHistogram.count(UserGestureRequiredAndAvailable); | |
520 m_userGestureRequired = false; | |
521 } | |
522 | |
476 String AbstractAudioContext::state() const | 523 String AbstractAudioContext::state() const |
477 { | 524 { |
478 // These strings had better match the strings for AudioContextState in Audio Context.idl. | 525 // These strings had better match the strings for AudioContextState in Audio Context.idl. |
479 switch (m_contextState) { | 526 switch (m_contextState) { |
480 case Suspended: | 527 case Suspended: |
481 return "suspended"; | 528 return "suspended"; |
482 case Running: | 529 case Running: |
483 return "running"; | 530 return "running"; |
484 case Closed: | 531 case Closed: |
485 return "closed"; | 532 return "closed"; |
(...skipping 224 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
710 { | 757 { |
711 return ActiveDOMObject::getExecutionContext(); | 758 return ActiveDOMObject::getExecutionContext(); |
712 } | 759 } |
713 | 760 |
714 void AbstractAudioContext::startRendering() | 761 void AbstractAudioContext::startRendering() |
715 { | 762 { |
716 // This is called for both online and offline contexts. | 763 // This is called for both online and offline contexts. |
717 ASSERT(isMainThread()); | 764 ASSERT(isMainThread()); |
718 ASSERT(m_destinationNode); | 765 ASSERT(m_destinationNode); |
719 | 766 |
767 recordUserGestureState(); | |
768 | |
720 if (m_contextState == Suspended) { | 769 if (m_contextState == Suspended) { |
721 destination()->audioDestinationHandler().startRendering(); | 770 destination()->audioDestinationHandler().startRendering(); |
722 setContextState(Running); | 771 setContextState(Running); |
723 } | 772 } |
724 } | 773 } |
725 | 774 |
726 DEFINE_TRACE(AbstractAudioContext) | 775 DEFINE_TRACE(AbstractAudioContext) |
727 { | 776 { |
728 visitor->trace(m_destinationNode); | 777 visitor->trace(m_destinationNode); |
729 visitor->trace(m_listener); | 778 visitor->trace(m_listener); |
(...skipping 11 matching lines...) Expand all Loading... | |
741 | 790 |
742 SecurityOrigin* AbstractAudioContext::getSecurityOrigin() const | 791 SecurityOrigin* AbstractAudioContext::getSecurityOrigin() const |
743 { | 792 { |
744 if (getExecutionContext()) | 793 if (getExecutionContext()) |
745 return getExecutionContext()->getSecurityOrigin(); | 794 return getExecutionContext()->getSecurityOrigin(); |
746 | 795 |
747 return nullptr; | 796 return nullptr; |
748 } | 797 } |
749 | 798 |
750 } // namespace blink | 799 } // namespace blink |
OLD | NEW |