Implement BitWriter
Change-Id: I5b7dfc2bc07227c2fe05f6de8857042bcab944c4
This commit is contained in:
parent
b5a8185543
commit
867244645a
|
@ -0,0 +1,42 @@
|
||||||
|
// Copyright 2017 Google Inc. All rights reserved.
|
||||||
|
//
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file or at
|
||||||
|
// https://developers.google.com/open-source/licenses/bsd
|
||||||
|
|
||||||
|
#include "packager/media/base/bit_writer.h"
|
||||||
|
|
||||||
|
namespace shaka {
|
||||||
|
namespace media {
|
||||||
|
|
||||||
|
BitWriter::BitWriter(std::vector<uint8_t>* storage)
|
||||||
|
: storage_(storage), initial_storage_size_(storage_->size()) {}
|
||||||
|
|
||||||
|
void BitWriter::WriteBits(uint32_t bits, size_t number_of_bits) {
|
||||||
|
DCHECK_NE(number_of_bits, 0u);
|
||||||
|
DCHECK_LE(number_of_bits, 32u);
|
||||||
|
DCHECK_LT(bits, 1ULL << number_of_bits);
|
||||||
|
|
||||||
|
num_bits_ += number_of_bits;
|
||||||
|
DCHECK_LE(num_bits_, 64);
|
||||||
|
bits_ |= static_cast<uint64_t>(bits) << (64 - num_bits_);
|
||||||
|
|
||||||
|
while (num_bits_ >= 8) {
|
||||||
|
storage_->push_back(bits_ >> 56);
|
||||||
|
bits_ <<= 8;
|
||||||
|
num_bits_ -= 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void BitWriter::Flush() {
|
||||||
|
while (num_bits_ > 0) {
|
||||||
|
storage_->push_back(bits_ >> 56);
|
||||||
|
bits_ <<= 8;
|
||||||
|
num_bits_ -= 8;
|
||||||
|
}
|
||||||
|
bits_ = 0;
|
||||||
|
num_bits_ = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace media
|
||||||
|
} // namespace shaka
|
|
@ -0,0 +1,60 @@
|
||||||
|
// Copyright 2017 Google Inc. All rights reserved.
|
||||||
|
//
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file or at
|
||||||
|
// https://developers.google.com/open-source/licenses/bsd
|
||||||
|
|
||||||
|
#ifndef PACKAGER_MEDIA_BASE_BIT_WRITER_H_
|
||||||
|
#define PACKAGER_MEDIA_BASE_BIT_WRITER_H_
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "packager/base/logging.h"
|
||||||
|
|
||||||
|
namespace shaka {
|
||||||
|
namespace media {
|
||||||
|
|
||||||
|
class BitWriter {
|
||||||
|
public:
|
||||||
|
/// Constructor a BitWriter instance which writes to the provided storage.
|
||||||
|
/// @param storage points to vector this BitWriter writes to. Cannot be
|
||||||
|
/// nullptr.
|
||||||
|
explicit BitWriter(std::vector<uint8_t>* storage);
|
||||||
|
~BitWriter() = default;
|
||||||
|
|
||||||
|
/// Appends the sequence 'bits' of length 'number_of_bits' <= 32.
|
||||||
|
/// Note that 'bits' should contain at most 'number_of_bits' bits, i.e.
|
||||||
|
/// bits should be less than 1 << number_of_bits.
|
||||||
|
/// @param bits is the data to write.
|
||||||
|
/// @param number_of_bits is the number of LSB to write, capped at 32. Cannot
|
||||||
|
/// be zero.
|
||||||
|
void WriteBits(uint32_t bits, size_t number_of_bits);
|
||||||
|
|
||||||
|
/// Write pending bits, and align bitstream with extra zero bits.
|
||||||
|
void Flush();
|
||||||
|
|
||||||
|
/// @return last written position, in bits.
|
||||||
|
size_t BitPos() const { return BytePos() * 8 + num_bits_; }
|
||||||
|
|
||||||
|
/// @return last written position, in bytes.
|
||||||
|
size_t BytePos() const { return storage_->size() - initial_storage_size_; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
BitWriter(const BitWriter&) = delete;
|
||||||
|
BitWriter& operator=(const BitWriter&) = delete;
|
||||||
|
|
||||||
|
// Accumulator for unwritten bits.
|
||||||
|
uint64_t bits_ = 0;
|
||||||
|
// Number of unwritten bits.
|
||||||
|
int num_bits_ = 0;
|
||||||
|
// Buffer contains the written bits.
|
||||||
|
std::vector<uint8_t>* const storage_ = nullptr;
|
||||||
|
const size_t initial_storage_size_ = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace media
|
||||||
|
} // namespace shaka
|
||||||
|
|
||||||
|
#endif // PACKAGER_MEDIA_BASE_BIT_WRITER_H_
|
|
@ -0,0 +1,62 @@
|
||||||
|
// Copyright 2017 Google Inc. All rights reserved.
|
||||||
|
//
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file or at
|
||||||
|
// https://developers.google.com/open-source/licenses/bsd
|
||||||
|
|
||||||
|
#include "packager/media/base/bit_writer.h"
|
||||||
|
|
||||||
|
#include <gmock/gmock.h>
|
||||||
|
#include <gtest/gtest.h>
|
||||||
|
|
||||||
|
using testing::ElementsAreArray;
|
||||||
|
|
||||||
|
namespace shaka {
|
||||||
|
namespace media {
|
||||||
|
|
||||||
|
TEST(BitWriterTest, Simple) {
|
||||||
|
std::vector<uint8_t> storage;
|
||||||
|
BitWriter writer(&storage);
|
||||||
|
writer.WriteBits(1, 1);
|
||||||
|
EXPECT_EQ(1u, writer.BitPos());
|
||||||
|
EXPECT_EQ(0u, writer.BytePos());
|
||||||
|
writer.Flush();
|
||||||
|
// Bits are byte-aligned after flush.
|
||||||
|
EXPECT_EQ(8u, writer.BitPos());
|
||||||
|
EXPECT_EQ(1u, writer.BytePos());
|
||||||
|
|
||||||
|
EXPECT_THAT(storage, ElementsAreArray({0x80}));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(BitWriterTest, Test) {
|
||||||
|
std::vector<uint8_t> storage;
|
||||||
|
BitWriter writer(&storage);
|
||||||
|
writer.WriteBits(0, 1);
|
||||||
|
EXPECT_EQ(1u, writer.BitPos());
|
||||||
|
EXPECT_EQ(0u, writer.BytePos());
|
||||||
|
writer.WriteBits(0xab, 8);
|
||||||
|
EXPECT_EQ(9u, writer.BitPos());
|
||||||
|
EXPECT_EQ(1u, writer.BytePos());
|
||||||
|
writer.WriteBits(0x34, 6);
|
||||||
|
EXPECT_EQ(15u, writer.BitPos());
|
||||||
|
EXPECT_EQ(1u, writer.BytePos());
|
||||||
|
writer.WriteBits(0x55995599, 32);
|
||||||
|
EXPECT_EQ(47u, writer.BitPos());
|
||||||
|
EXPECT_EQ(5u, writer.BytePos());
|
||||||
|
writer.WriteBits(1, 1);
|
||||||
|
EXPECT_EQ(48u, writer.BitPos());
|
||||||
|
EXPECT_EQ(6u, writer.BytePos());
|
||||||
|
writer.WriteBits(0x13, 21);
|
||||||
|
EXPECT_EQ(69u, writer.BitPos());
|
||||||
|
EXPECT_EQ(8u, writer.BytePos());
|
||||||
|
writer.Flush();
|
||||||
|
// Bits are byte-aligned after flush.
|
||||||
|
EXPECT_EQ(72u, writer.BitPos());
|
||||||
|
EXPECT_EQ(9u, writer.BytePos());
|
||||||
|
|
||||||
|
EXPECT_THAT(storage, ElementsAreArray({0x55, 0xe8, 0xab, 0x32, 0xab, 0x33,
|
||||||
|
0x00, 0x00, 0x98}));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace media
|
||||||
|
} // namespace shaka
|
|
@ -27,6 +27,8 @@
|
||||||
'audio_timestamp_helper.h',
|
'audio_timestamp_helper.h',
|
||||||
'bit_reader.cc',
|
'bit_reader.cc',
|
||||||
'bit_reader.h',
|
'bit_reader.h',
|
||||||
|
'bit_writer.cc',
|
||||||
|
'bit_writer.h',
|
||||||
'buffer_reader.cc',
|
'buffer_reader.cc',
|
||||||
'buffer_reader.h',
|
'buffer_reader.h',
|
||||||
'buffer_writer.cc',
|
'buffer_writer.cc',
|
||||||
|
@ -138,6 +140,7 @@
|
||||||
'aes_pattern_cryptor_unittest.cc',
|
'aes_pattern_cryptor_unittest.cc',
|
||||||
'audio_timestamp_helper_unittest.cc',
|
'audio_timestamp_helper_unittest.cc',
|
||||||
'bit_reader_unittest.cc',
|
'bit_reader_unittest.cc',
|
||||||
|
'bit_writer_unittest.cc',
|
||||||
'buffer_writer_unittest.cc',
|
'buffer_writer_unittest.cc',
|
||||||
'closure_thread_unittest.cc',
|
'closure_thread_unittest.cc',
|
||||||
'container_names_unittest.cc',
|
'container_names_unittest.cc',
|
||||||
|
|
Loading…
Reference in New Issue