1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright 2015 The WebRTC Project Authors. All rights reserved.
3*d9f75844SAndroid Build Coastguard Worker *
4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker */
10*d9f75844SAndroid Build Coastguard Worker
11*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/bit_buffer.h"
12*d9f75844SAndroid Build Coastguard Worker
13*d9f75844SAndroid Build Coastguard Worker #include <limits>
14*d9f75844SAndroid Build Coastguard Worker
15*d9f75844SAndroid Build Coastguard Worker #include "api/array_view.h"
16*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/arraysize.h"
17*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/bitstream_reader.h"
18*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/byte_buffer.h"
19*d9f75844SAndroid Build Coastguard Worker #include "test/gmock.h"
20*d9f75844SAndroid Build Coastguard Worker #include "test/gtest.h"
21*d9f75844SAndroid Build Coastguard Worker
22*d9f75844SAndroid Build Coastguard Worker namespace rtc {
23*d9f75844SAndroid Build Coastguard Worker
24*d9f75844SAndroid Build Coastguard Worker using ::testing::ElementsAre;
25*d9f75844SAndroid Build Coastguard Worker using ::webrtc::BitstreamReader;
26*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,ConsumeBits)27*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, ConsumeBits) {
28*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[64] = {0};
29*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 32);
30*d9f75844SAndroid Build Coastguard Worker uint64_t total_bits = 32 * 8;
31*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
32*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(3));
33*d9f75844SAndroid Build Coastguard Worker total_bits -= 3;
34*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
35*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(3));
36*d9f75844SAndroid Build Coastguard Worker total_bits -= 3;
37*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
38*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(15));
39*d9f75844SAndroid Build Coastguard Worker total_bits -= 15;
40*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
41*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(37));
42*d9f75844SAndroid Build Coastguard Worker total_bits -= 37;
43*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
44*d9f75844SAndroid Build Coastguard Worker
45*d9f75844SAndroid Build Coastguard Worker EXPECT_FALSE(buffer.ConsumeBits(32 * 8));
46*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(total_bits, buffer.RemainingBitCount());
47*d9f75844SAndroid Build Coastguard Worker }
48*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterDeathTest,SetOffsetValues)49*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterDeathTest, SetOffsetValues) {
50*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[4] = {0};
51*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 4);
52*d9f75844SAndroid Build Coastguard Worker
53*d9f75844SAndroid Build Coastguard Worker size_t byte_offset, bit_offset;
54*d9f75844SAndroid Build Coastguard Worker // Bit offsets are [0,7].
55*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.Seek(0, 0));
56*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.Seek(0, 7));
57*d9f75844SAndroid Build Coastguard Worker buffer.GetCurrentOffset(&byte_offset, &bit_offset);
58*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0u, byte_offset);
59*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(7u, bit_offset);
60*d9f75844SAndroid Build Coastguard Worker EXPECT_FALSE(buffer.Seek(0, 8));
61*d9f75844SAndroid Build Coastguard Worker buffer.GetCurrentOffset(&byte_offset, &bit_offset);
62*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0u, byte_offset);
63*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(7u, bit_offset);
64*d9f75844SAndroid Build Coastguard Worker // Byte offsets are [0,length]. At byte offset length, the bit offset must be
65*d9f75844SAndroid Build Coastguard Worker // 0.
66*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.Seek(0, 0));
67*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.Seek(2, 4));
68*d9f75844SAndroid Build Coastguard Worker buffer.GetCurrentOffset(&byte_offset, &bit_offset);
69*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(2u, byte_offset);
70*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(4u, bit_offset);
71*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.Seek(4, 0));
72*d9f75844SAndroid Build Coastguard Worker EXPECT_FALSE(buffer.Seek(5, 0));
73*d9f75844SAndroid Build Coastguard Worker buffer.GetCurrentOffset(&byte_offset, &bit_offset);
74*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(4u, byte_offset);
75*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0u, bit_offset);
76*d9f75844SAndroid Build Coastguard Worker EXPECT_FALSE(buffer.Seek(4, 1));
77*d9f75844SAndroid Build Coastguard Worker
78*d9f75844SAndroid Build Coastguard Worker // Disable death test on Android because it relies on fork() and doesn't play
79*d9f75844SAndroid Build Coastguard Worker // nicely.
80*d9f75844SAndroid Build Coastguard Worker #if GTEST_HAS_DEATH_TEST
81*d9f75844SAndroid Build Coastguard Worker #if !defined(WEBRTC_ANDROID)
82*d9f75844SAndroid Build Coastguard Worker // Passing a null out parameter is death.
83*d9f75844SAndroid Build Coastguard Worker EXPECT_DEATH(buffer.GetCurrentOffset(&byte_offset, nullptr), "");
84*d9f75844SAndroid Build Coastguard Worker #endif
85*d9f75844SAndroid Build Coastguard Worker #endif
86*d9f75844SAndroid Build Coastguard Worker }
87*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,WriteNonSymmetricSameNumberOfBitsWhenNumValuesPowerOf2)88*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest,
89*d9f75844SAndroid Build Coastguard Worker WriteNonSymmetricSameNumberOfBitsWhenNumValuesPowerOf2) {
90*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[2] = {};
91*d9f75844SAndroid Build Coastguard Worker BitBufferWriter writer(bytes, 2);
92*d9f75844SAndroid Build Coastguard Worker
93*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 16u);
94*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(0xf, /*num_values=*/1 << 4));
95*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 12u);
96*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(0x3, /*num_values=*/1 << 4));
97*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 8u);
98*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(0xa, /*num_values=*/1 << 4));
99*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 4u);
100*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(0x0, /*num_values=*/1 << 4));
101*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 0u);
102*d9f75844SAndroid Build Coastguard Worker
103*d9f75844SAndroid Build Coastguard Worker EXPECT_THAT(bytes, ElementsAre(0xf3, 0xa0));
104*d9f75844SAndroid Build Coastguard Worker }
105*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,NonSymmetricReadsMatchesWrites)106*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, NonSymmetricReadsMatchesWrites) {
107*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[2] = {};
108*d9f75844SAndroid Build Coastguard Worker BitBufferWriter writer(bytes, 2);
109*d9f75844SAndroid Build Coastguard Worker
110*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(BitBufferWriter::SizeNonSymmetricBits(/*val=*/1, /*num_values=*/6),
111*d9f75844SAndroid Build Coastguard Worker 2u);
112*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(BitBufferWriter::SizeNonSymmetricBits(/*val=*/2, /*num_values=*/6),
113*d9f75844SAndroid Build Coastguard Worker 3u);
114*d9f75844SAndroid Build Coastguard Worker // Values [0, 1] can fit into two bit.
115*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 16u);
116*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/0, /*num_values=*/6));
117*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 14u);
118*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/1, /*num_values=*/6));
119*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 12u);
120*d9f75844SAndroid Build Coastguard Worker // Values [2, 5] require 3 bits.
121*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/2, /*num_values=*/6));
122*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 9u);
123*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/3, /*num_values=*/6));
124*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 6u);
125*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/4, /*num_values=*/6));
126*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 3u);
127*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(/*val=*/5, /*num_values=*/6));
128*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 0u);
129*d9f75844SAndroid Build Coastguard Worker
130*d9f75844SAndroid Build Coastguard Worker // Bit values are
131*d9f75844SAndroid Build Coastguard Worker // 00.01.100.101.110.111 = 00011001|01110111 = 0x19|77
132*d9f75844SAndroid Build Coastguard Worker EXPECT_THAT(bytes, ElementsAre(0x19, 0x77));
133*d9f75844SAndroid Build Coastguard Worker
134*d9f75844SAndroid Build Coastguard Worker BitstreamReader reader(bytes);
135*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 0u);
136*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 1u);
137*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 2u);
138*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 3u);
139*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 4u);
140*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadNonSymmetric(/*num_values=*/6), 5u);
141*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(reader.Ok());
142*d9f75844SAndroid Build Coastguard Worker }
143*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,WriteNonSymmetricOnlyValueConsumesNoBits)144*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, WriteNonSymmetricOnlyValueConsumesNoBits) {
145*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[2] = {};
146*d9f75844SAndroid Build Coastguard Worker BitBufferWriter writer(bytes, 2);
147*d9f75844SAndroid Build Coastguard Worker ASSERT_EQ(writer.RemainingBitCount(), 16u);
148*d9f75844SAndroid Build Coastguard Worker
149*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(writer.WriteNonSymmetric(0, /*num_values=*/1));
150*d9f75844SAndroid Build Coastguard Worker
151*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(writer.RemainingBitCount(), 16u);
152*d9f75844SAndroid Build Coastguard Worker }
153*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,SymmetricReadWrite)154*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, SymmetricReadWrite) {
155*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[16] = {0};
156*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 4);
157*d9f75844SAndroid Build Coastguard Worker
158*d9f75844SAndroid Build Coastguard Worker // Write some bit data at various sizes.
159*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x2u, 3));
160*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x1u, 2));
161*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x53u, 7));
162*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x0u, 2));
163*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x1u, 1));
164*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0x1ABCDu, 17));
165*d9f75844SAndroid Build Coastguard Worker // That should be all that fits in the buffer.
166*d9f75844SAndroid Build Coastguard Worker EXPECT_FALSE(buffer.WriteBits(1, 1));
167*d9f75844SAndroid Build Coastguard Worker
168*d9f75844SAndroid Build Coastguard Worker BitstreamReader reader(rtc::MakeArrayView(bytes, 4));
169*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(3), 0x2u);
170*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(2), 0x1u);
171*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(7), 0x53u);
172*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(2), 0x0u);
173*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(1), 0x1u);
174*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.ReadBits(17), 0x1ABCDu);
175*d9f75844SAndroid Build Coastguard Worker // And there should be nothing left.
176*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.RemainingBitCount(), 0);
177*d9f75844SAndroid Build Coastguard Worker }
178*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,SymmetricBytesMisaligned)179*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, SymmetricBytesMisaligned) {
180*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[16] = {0};
181*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 16);
182*d9f75844SAndroid Build Coastguard Worker
183*d9f75844SAndroid Build Coastguard Worker // Offset 3, to get things misaligned.
184*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(3));
185*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteUInt8(0x12u));
186*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteUInt16(0x3456u));
187*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteUInt32(0x789ABCDEu));
188*d9f75844SAndroid Build Coastguard Worker
189*d9f75844SAndroid Build Coastguard Worker BitstreamReader reader(bytes);
190*d9f75844SAndroid Build Coastguard Worker reader.ConsumeBits(3);
191*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.Read<uint8_t>(), 0x12u);
192*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.Read<uint16_t>(), 0x3456u);
193*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(reader.Read<uint32_t>(), 0x789ABCDEu);
194*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(reader.Ok());
195*d9f75844SAndroid Build Coastguard Worker }
196*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,SymmetricGolomb)197*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, SymmetricGolomb) {
198*d9f75844SAndroid Build Coastguard Worker char test_string[] = "my precious";
199*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[64] = {0};
200*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 64);
201*d9f75844SAndroid Build Coastguard Worker for (size_t i = 0; i < arraysize(test_string); ++i) {
202*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteExponentialGolomb(test_string[i]));
203*d9f75844SAndroid Build Coastguard Worker }
204*d9f75844SAndroid Build Coastguard Worker BitstreamReader reader(bytes);
205*d9f75844SAndroid Build Coastguard Worker for (size_t i = 0; i < arraysize(test_string); ++i) {
206*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(int64_t{reader.ReadExponentialGolomb()}, int64_t{test_string[i]});
207*d9f75844SAndroid Build Coastguard Worker }
208*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(reader.Ok());
209*d9f75844SAndroid Build Coastguard Worker }
210*d9f75844SAndroid Build Coastguard Worker
TEST(BitBufferWriterTest,WriteClearsBits)211*d9f75844SAndroid Build Coastguard Worker TEST(BitBufferWriterTest, WriteClearsBits) {
212*d9f75844SAndroid Build Coastguard Worker uint8_t bytes[] = {0xFF, 0xFF};
213*d9f75844SAndroid Build Coastguard Worker BitBufferWriter buffer(bytes, 2);
214*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.ConsumeBits(3));
215*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0, 1));
216*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0xEFu, bytes[0]);
217*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0, 3));
218*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0xE1u, bytes[0]);
219*d9f75844SAndroid Build Coastguard Worker EXPECT_TRUE(buffer.WriteBits(0, 2));
220*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0xE0u, bytes[0]);
221*d9f75844SAndroid Build Coastguard Worker EXPECT_EQ(0x7F, bytes[1]);
222*d9f75844SAndroid Build Coastguard Worker }
223*d9f75844SAndroid Build Coastguard Worker
224*d9f75844SAndroid Build Coastguard Worker } // namespace rtc
225