OLD | NEW |
| (Empty) |
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "media/webm/cluster_builder.h" | |
6 #include "media/webm/webm_constants.h" | |
7 #include "media/webm/webm_parser.h" | |
8 #include "testing/gmock/include/gmock/gmock.h" | |
9 #include "testing/gtest/include/gtest/gtest.h" | |
10 | |
11 using ::testing::InSequence; | |
12 using ::testing::Return; | |
13 using ::testing::StrictMock; | |
14 using ::testing::_; | |
15 | |
16 namespace media { | |
17 | |
18 class MockWebMParserClient : public WebMParserClient { | |
19 public: | |
20 virtual ~MockWebMParserClient() {} | |
21 | |
22 // WebMParserClient methods. | |
23 MOCK_METHOD1(OnListStart, bool(int)); | |
24 MOCK_METHOD1(OnListEnd, bool(int)); | |
25 MOCK_METHOD2(OnUInt, bool(int, int64)); | |
26 MOCK_METHOD2(OnFloat, bool(int, double)); | |
27 MOCK_METHOD3(OnBinary, bool(int, const uint8*, int)); | |
28 MOCK_METHOD2(OnString, bool(int, const std::string&)); | |
29 MOCK_METHOD5(OnSimpleBlock, bool(int, int, int, const uint8*, int)); | |
30 }; | |
31 | |
32 class WebMParserTest : public testing::Test { | |
33 protected: | |
34 StrictMock<MockWebMParserClient> client_; | |
35 }; | |
36 | |
37 struct SimpleBlockInfo { | |
38 int track_num; | |
39 int timestamp; | |
40 }; | |
41 | |
42 static void AddSimpleBlock(ClusterBuilder* cb, int track_num, | |
43 int64 timecode) { | |
44 uint8 data[] = { 0x00 }; | |
45 cb->AddSimpleBlock(track_num, timecode, 0, data, sizeof(data)); | |
46 } | |
47 | |
48 static Cluster* CreateCluster(int timecode, | |
49 const SimpleBlockInfo* block_info, | |
50 int block_count) { | |
51 ClusterBuilder cb; | |
52 cb.SetClusterTimecode(0); | |
53 | |
54 for (int i = 0; i < block_count; i++) | |
55 AddSimpleBlock(&cb, block_info[i].track_num, block_info[i].timestamp); | |
56 | |
57 return cb.Finish(); | |
58 } | |
59 | |
60 static void CreateClusterExpectations(int timecode, | |
61 const SimpleBlockInfo* block_info, | |
62 int block_count, | |
63 MockWebMParserClient* client) { | |
64 | |
65 InSequence s; | |
66 EXPECT_CALL(*client, OnListStart(kWebMIdCluster)).WillOnce(Return(true)); | |
67 EXPECT_CALL(*client, OnUInt(kWebMIdTimecode, 0)).WillOnce(Return(true)); | |
68 | |
69 for (int i = 0; i < block_count; i++) { | |
70 EXPECT_CALL(*client, OnSimpleBlock(block_info[i].track_num, | |
71 block_info[i].timestamp, | |
72 _, _, _)) | |
73 .WillOnce(Return(true)); | |
74 } | |
75 | |
76 EXPECT_CALL(*client, OnListEnd(kWebMIdCluster)).WillOnce(Return(true)); | |
77 } | |
78 | |
79 TEST_F(WebMParserTest, EmptyCluster) { | |
80 const uint8 kEmptyCluster[] = { | |
81 0x1F, 0x43, 0xB6, 0x75, 0x80 // CLUSTER (size = 0) | |
82 }; | |
83 int size = sizeof(kEmptyCluster); | |
84 | |
85 InSequence s; | |
86 EXPECT_CALL(client_, OnListStart(kWebMIdCluster)).WillOnce(Return(true)); | |
87 EXPECT_CALL(client_, OnListEnd(kWebMIdCluster)).WillOnce(Return(true)); | |
88 | |
89 WebMListParser parser(kWebMIdCluster); | |
90 int result = parser.Parse(kEmptyCluster, size, &client_); | |
91 EXPECT_EQ(size, result); | |
92 EXPECT_TRUE(parser.IsParsingComplete()); | |
93 } | |
94 | |
95 TEST_F(WebMParserTest, EmptyClusterInSegment) { | |
96 const uint8 kBuffer[] = { | |
97 0x18, 0x53, 0x80, 0x67, 0x85, // SEGMENT (size = 5) | |
98 0x1F, 0x43, 0xB6, 0x75, 0x80, // CLUSTER (size = 0) | |
99 }; | |
100 int size = sizeof(kBuffer); | |
101 | |
102 InSequence s; | |
103 EXPECT_CALL(client_, OnListStart(kWebMIdSegment)).WillOnce(Return(true)); | |
104 EXPECT_CALL(client_, OnListStart(kWebMIdCluster)).WillOnce(Return(true)); | |
105 EXPECT_CALL(client_, OnListEnd(kWebMIdCluster)).WillOnce(Return(true)); | |
106 EXPECT_CALL(client_, OnListEnd(kWebMIdSegment)).WillOnce(Return(true)); | |
107 | |
108 WebMListParser parser(kWebMIdSegment); | |
109 int result = parser.Parse(kBuffer, size, &client_); | |
110 EXPECT_EQ(size, result); | |
111 EXPECT_TRUE(parser.IsParsingComplete()); | |
112 } | |
113 | |
114 // Test the case where a non-list child element has a size | |
115 // that is beyond the end of the parent. | |
116 TEST_F(WebMParserTest, ChildNonListLargerThanParent) { | |
117 const uint8 kBuffer[] = { | |
118 0x1F, 0x43, 0xB6, 0x75, 0x81, // CLUSTER (size = 1) | |
119 0xE7, 0x81, 0x01, // Timecode (size=1, value=1) | |
120 }; | |
121 int size = sizeof(kBuffer); | |
122 | |
123 InSequence s; | |
124 EXPECT_CALL(client_, OnListStart(kWebMIdCluster)).WillOnce(Return(true)); | |
125 | |
126 WebMListParser parser(kWebMIdCluster); | |
127 int result = parser.Parse(kBuffer, size, &client_); | |
128 EXPECT_EQ(-1, result); | |
129 EXPECT_FALSE(parser.IsParsingComplete()); | |
130 } | |
131 | |
132 // Test the case where a list child element has a size | |
133 // that is beyond the end of the parent. | |
134 TEST_F(WebMParserTest, ChildListLargerThanParent) { | |
135 const uint8 kBuffer[] = { | |
136 0x18, 0x53, 0x80, 0x67, 0x85, // SEGMENT (size = 5) | |
137 0x1F, 0x43, 0xB6, 0x75, 0x81, 0x11 // CLUSTER (size = 1) | |
138 }; | |
139 int size = sizeof(kBuffer); | |
140 | |
141 InSequence s; | |
142 EXPECT_CALL(client_, OnListStart(kWebMIdSegment)).WillOnce(Return(true)); | |
143 | |
144 WebMListParser parser(kWebMIdSegment); | |
145 int result = parser.Parse(kBuffer, size, &client_); | |
146 EXPECT_EQ(-1, result); | |
147 EXPECT_FALSE(parser.IsParsingComplete()); | |
148 } | |
149 | |
150 // Expecting to parse a Cluster, but get a Segment. | |
151 TEST_F(WebMParserTest, ListIdDoesNotMatch) { | |
152 const uint8 kBuffer[] = { | |
153 0x18, 0x53, 0x80, 0x67, 0x80, // SEGMENT (size = 0) | |
154 }; | |
155 int size = sizeof(kBuffer); | |
156 | |
157 WebMListParser parser(kWebMIdCluster); | |
158 int result = parser.Parse(kBuffer, size, &client_); | |
159 EXPECT_EQ(-1, result); | |
160 EXPECT_FALSE(parser.IsParsingComplete()); | |
161 } | |
162 | |
163 TEST_F(WebMParserTest, InvalidElementInList) { | |
164 const uint8 kBuffer[] = { | |
165 0x18, 0x53, 0x80, 0x67, 0x82, // SEGMENT (size = 2) | |
166 0xAE, 0x80, // TrackEntry (size = 0) | |
167 }; | |
168 int size = sizeof(kBuffer); | |
169 | |
170 InSequence s; | |
171 EXPECT_CALL(client_, OnListStart(kWebMIdSegment)).WillOnce(Return(true)); | |
172 | |
173 WebMListParser parser(kWebMIdSegment); | |
174 int result = parser.Parse(kBuffer, size, &client_); | |
175 EXPECT_EQ(-1, result); | |
176 EXPECT_FALSE(parser.IsParsingComplete()); | |
177 } | |
178 | |
179 TEST_F(WebMParserTest, VoidAndCRC32InList) { | |
180 const uint8 kBuffer[] = { | |
181 0x18, 0x53, 0x80, 0x67, 0x99, // SEGMENT (size = 25) | |
182 0xEC, 0x83, 0x00, 0x00, 0x00, // Void (size = 3) | |
183 0xBF, 0x83, 0x00, 0x00, 0x00, // CRC32 (size = 3) | |
184 0x1F, 0x43, 0xB6, 0x75, 0x8A, // CLUSTER (size = 10) | |
185 0xEC, 0x83, 0x00, 0x00, 0x00, // Void (size = 3) | |
186 0xBF, 0x83, 0x00, 0x00, 0x00, // CRC32 (size = 3) | |
187 }; | |
188 int size = sizeof(kBuffer); | |
189 | |
190 InSequence s; | |
191 EXPECT_CALL(client_, OnListStart(kWebMIdSegment)).WillOnce(Return(true)); | |
192 EXPECT_CALL(client_, OnListStart(kWebMIdCluster)).WillOnce(Return(true)); | |
193 EXPECT_CALL(client_, OnListEnd(kWebMIdCluster)).WillOnce(Return(true)); | |
194 EXPECT_CALL(client_, OnListEnd(kWebMIdSegment)).WillOnce(Return(true)); | |
195 | |
196 WebMListParser parser(kWebMIdSegment); | |
197 int result = parser.Parse(kBuffer, size, &client_); | |
198 EXPECT_EQ(size, result); | |
199 EXPECT_TRUE(parser.IsParsingComplete()); | |
200 } | |
201 | |
202 | |
203 TEST_F(WebMParserTest, ParseListElementWithSingleCall) { | |
204 const SimpleBlockInfo kBlockInfo[] = { | |
205 { 0, 1 }, | |
206 { 1, 2 }, | |
207 { 0, 3 }, | |
208 { 0, 4 }, | |
209 { 1, 4 }, | |
210 }; | |
211 int block_count = arraysize(kBlockInfo); | |
212 | |
213 scoped_ptr<Cluster> cluster(CreateCluster(0, kBlockInfo, block_count)); | |
214 CreateClusterExpectations(0, kBlockInfo, block_count, &client_); | |
215 | |
216 WebMListParser parser(kWebMIdCluster); | |
217 int result = parser.Parse(cluster->data(), cluster->size(), &client_); | |
218 EXPECT_EQ(cluster->size(), result); | |
219 EXPECT_TRUE(parser.IsParsingComplete()); | |
220 } | |
221 | |
222 TEST_F(WebMParserTest, ParseListElementWithMultipleCalls) { | |
223 const SimpleBlockInfo kBlockInfo[] = { | |
224 { 0, 1 }, | |
225 { 1, 2 }, | |
226 { 0, 3 }, | |
227 { 0, 4 }, | |
228 { 1, 4 }, | |
229 }; | |
230 int block_count = arraysize(kBlockInfo); | |
231 | |
232 scoped_ptr<Cluster> cluster(CreateCluster(0, kBlockInfo, block_count)); | |
233 CreateClusterExpectations(0, kBlockInfo, block_count, &client_); | |
234 | |
235 const uint8* data = cluster->data(); | |
236 int size = cluster->size(); | |
237 int default_parse_size = 3; | |
238 WebMListParser parser(kWebMIdCluster); | |
239 int parse_size = std::min(default_parse_size, size); | |
240 | |
241 while (size > 0) { | |
242 int result = parser.Parse(data, parse_size, &client_); | |
243 EXPECT_GE(result, 0); | |
244 EXPECT_LE(result, parse_size); | |
245 | |
246 if (result == 0) { | |
247 // The parser needs more data so increase the parse_size a little. | |
248 EXPECT_FALSE(parser.IsParsingComplete()); | |
249 parse_size += default_parse_size; | |
250 parse_size = std::min(parse_size, size); | |
251 continue; | |
252 } | |
253 | |
254 parse_size = default_parse_size; | |
255 | |
256 data += result; | |
257 size -= result; | |
258 | |
259 EXPECT_EQ((size == 0), parser.IsParsingComplete()); | |
260 } | |
261 EXPECT_TRUE(parser.IsParsingComplete()); | |
262 } | |
263 | |
264 } // namespace media | |
OLD | NEW |