Cap the number of SourceBuffers that may be added to a MediaSource.
[WebKit-https.git] / Source / WebCore / Modules / mediasource / SourceBufferList.cpp
1 /*
2  * Copyright (C) 2012 Google 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 are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "SourceBufferList.h"
33
34 #if ENABLE(MEDIA_SOURCE)
35
36 #include "Event.h"
37 #include "SourceBuffer.h"
38
39 namespace WebCore {
40
41 SourceBufferList::SourceBufferList(ScriptExecutionContext* context)
42     : m_scriptExecutionContext(context)
43     , m_lastSourceBufferId(0)
44 {
45 }
46
47 unsigned long SourceBufferList::length() const
48 {
49     return m_list.size();
50 }
51
52 SourceBuffer* SourceBufferList::item(unsigned index) const
53 {
54     if (index >= m_list.size())
55         return 0;
56     return m_list[index].get();
57 }
58
59 void SourceBufferList::add(PassRefPtr<SourceBuffer> buffer)
60 {
61     m_list.append(buffer);
62     createAndFireEvent(eventNames().webkitaddsourcebufferEvent);
63 }
64
65 bool SourceBufferList::remove(SourceBuffer* buffer)
66 {    
67     size_t index = m_list.find(buffer);
68     if (index == notFound)
69         return false;
70
71     m_list.remove(index);
72     buffer->clear();
73     createAndFireEvent(eventNames().webkitremovesourcebufferEvent);
74     return true;
75 }
76
77 void SourceBufferList::clear()
78 {
79     for (size_t i = 0; i < m_list.size(); ++i)
80         remove(m_list[i].get());
81 }
82
83 String SourceBufferList::generateUniqueId()
84 {
85     // Ensure a unique id. Until m_lastSourceBufferId wraps around (very unlikely),
86     // this loop will exit after one check. If m_lastSourceBufferId does wrap,
87     // this loop may run as many times as the size of m_list, but in most
88     // cases that should be less than 10. We may want to investigate a more
89     // efficient approach if many more SourceBuffers are allowed in the future.
90     size_t nextSourceBufferId = m_lastSourceBufferId + 1;
91
92     while (contains(nextSourceBufferId)) {
93         if (nextSourceBufferId == m_lastSourceBufferId)
94             return emptyString();
95         nextSourceBufferId++;
96     }
97     m_lastSourceBufferId = nextSourceBufferId;
98
99     return String::number(nextSourceBufferId);
100 }
101
102 bool SourceBufferList::contains(size_t id) const
103 {
104     for (size_t i = 0; i < m_list.size(); ++i) {
105         if (m_list[i]->id() == String::number(id))
106             return true;
107     }
108     return false;
109 }
110
111 void SourceBufferList::createAndFireEvent(const AtomicString& eventName)
112 {
113     RefPtr<Event> event = Event::create(eventName, false, false);
114     event->setTarget(this);
115
116     EventTarget::dispatchEvent(event);
117 }
118
119 const AtomicString& SourceBufferList::interfaceName() const
120 {
121     return eventNames().interfaceForSourceBufferList;
122 }
123
124 ScriptExecutionContext* SourceBufferList::scriptExecutionContext() const
125 {
126     return m_scriptExecutionContext;
127 }
128
129 EventTargetData* SourceBufferList::eventTargetData()
130 {
131     return &m_eventTargetData;
132 }
133
134 EventTargetData* SourceBufferList::ensureEventTargetData()
135 {
136     return &m_eventTargetData;
137 }
138
139 } // namespace WebCore
140
141 #endif