| Index: media/formats/webm/opus_packet_builder.cc
|
| diff --git a/media/formats/webm/opus_packet_builder.cc b/media/formats/webm/opus_packet_builder.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..b2dd8dca1e1de13dfc2dbafb665a1b51ec2b74eb
|
| --- /dev/null
|
| +++ b/media/formats/webm/opus_packet_builder.cc
|
| @@ -0,0 +1,89 @@
|
| +// Copyright (c) 2015 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "base/logging.h"
|
| +#include "media/formats/webm/opus_packet_builder.h"
|
| +#include "media/formats/webm/webm_cluster_parser.h"
|
| +
|
| +namespace media {
|
| +
|
| +OpusPacket::OpusPacket(uint8_t config, uint8_t frame_count, bool is_VBR) {
|
| + DCHECK_GE(config, 0);
|
| + DCHECK_LT(config, kNumPossibleOpusConfigs);
|
| + DCHECK_GE(frame_count, kMinOpusPacketFrameCount);
|
| + DCHECK_LE(frame_count, kMaxOpusPacketFrameCount);
|
| +
|
| + duration_ms_ = frame_count *
|
| + WebMClusterParser::kOpusFrameDurationsMu[config] /
|
| + static_cast<float>(1000);
|
| +
|
| + uint8_t frame_count_code;
|
| + uint8_t frame_count_byte;
|
| +
|
| + if (frame_count == 1) {
|
| + frame_count_code = 0;
|
| + } else if (frame_count == 2) {
|
| + frame_count_code = is_VBR ? 2 : 1;
|
| + } else {
|
| + frame_count_code = 3;
|
| + frame_count_byte = (is_VBR ? 1 << 7 : 0) | frame_count;
|
| + }
|
| +
|
| + // All opus packets must have TOC byte.
|
| + uint8_t opus_toc_byte = (config << 3) | frame_count_code;
|
| + data_.push_back(opus_toc_byte);
|
| +
|
| + // For code 3 packets, the number of frames is signaled in the "frame
|
| + // count byte".
|
| + if (frame_count_code == 3) {
|
| + data_.push_back(frame_count_byte);
|
| + }
|
| +
|
| + // Packet will only conform to layout specification for the TOC byte
|
| + // and optional frame count bytes appended above. This last byte
|
| + // is purely dummy padding where frame size data or encoded data might
|
| + // otherwise start.
|
| + data_.push_back(static_cast<uint8_t>(0));
|
| +}
|
| +
|
| +OpusPacket::~OpusPacket() {
|
| +}
|
| +
|
| +const uint8_t* OpusPacket::data() const {
|
| + return &(data_[0]);
|
| +}
|
| +
|
| +int OpusPacket::size() const {
|
| + return data_.size();
|
| +}
|
| +
|
| +double OpusPacket::duration_ms() const {
|
| + return duration_ms_;
|
| +}
|
| +
|
| +ScopedVector<OpusPacket> BuildAllOpusPackets() {
|
| + ScopedVector<OpusPacket> opus_packets;
|
| +
|
| + for (int frame_count = kMinOpusPacketFrameCount;
|
| + frame_count <= kMaxOpusPacketFrameCount; frame_count++) {
|
| + for (int opus_config_num = 0; opus_config_num < kNumPossibleOpusConfigs;
|
| + opus_config_num++) {
|
| + bool is_VBR = false;
|
| + opus_packets.push_back(
|
| + new OpusPacket(opus_config_num, frame_count, is_VBR));
|
| +
|
| + if (frame_count >= 2) {
|
| + // Add another packet with VBR flag toggled. For frame counts >= 2,
|
| + // VBR triggers changes to packet framing.
|
| + is_VBR = true;
|
| + opus_packets.push_back(
|
| + new OpusPacket(opus_config_num, frame_count, is_VBR));
|
| + }
|
| + }
|
| + }
|
| +
|
| + return opus_packets.Pass();
|
| +}
|
| +
|
| +} // namespace media
|
|
|