mkvmuxer: Set doctype to matroska when muxing non-WebM codecs.

Also, add some constants for WebVTT codec ID strings so they
won't cause doctype to incorrectly change to matroska.

Change-Id: I4740a3e45b28a22e462601b9ce051aa01817dace
This commit is contained in:
Tom Finegan 2016-04-27 13:50:20 -07:00
parent e3c9576716
commit 9a235e0bc9
5 changed files with 85 additions and 6 deletions

View File

@ -16,6 +16,7 @@
#include <ctime> #include <ctime>
#include <memory> #include <memory>
#include <new> #include <new>
#include <string>
#include <vector> #include <vector>
#include "common/webmids.h" #include "common/webmids.h"
@ -29,6 +30,10 @@ const float MasteringMetadata::kValueNotPresent = FLT_MAX;
const uint64_t Colour::kValueNotPresent = UINT64_MAX; const uint64_t Colour::kValueNotPresent = UINT64_MAX;
namespace { namespace {
const char kDocTypeWebm[] = "webm";
const char kDocTypeMatroska[] = "matroska";
// Deallocate the string designated by |dst|, and then copy the |src| // Deallocate the string designated by |dst|, and then copy the |src|
// string to |dst|. The caller owns both the |src| string and the // string to |dst|. The caller owns both the |src| string and the
// |dst| copy (hence the caller is responsible for eventually // |dst| copy (hence the caller is responsible for eventually
@ -80,7 +85,8 @@ IMkvWriter::IMkvWriter() {}
IMkvWriter::~IMkvWriter() {} IMkvWriter::~IMkvWriter() {}
bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version) { bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version,
const char* const doc_type) {
// Level 0 // Level 0
uint64_t size = uint64_t size =
EbmlElementSize(libwebm::kMkvEBMLVersion, static_cast<uint64>(1)); EbmlElementSize(libwebm::kMkvEBMLVersion, static_cast<uint64>(1));
@ -88,7 +94,7 @@ bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version) {
size += EbmlElementSize(libwebm::kMkvEBMLMaxIDLength, static_cast<uint64>(4)); size += EbmlElementSize(libwebm::kMkvEBMLMaxIDLength, static_cast<uint64>(4));
size += size +=
EbmlElementSize(libwebm::kMkvEBMLMaxSizeLength, static_cast<uint64>(8)); EbmlElementSize(libwebm::kMkvEBMLMaxSizeLength, static_cast<uint64>(8));
size += EbmlElementSize(libwebm::kMkvDocType, "webm"); size += EbmlElementSize(libwebm::kMkvDocType, doc_type);
size += EbmlElementSize(libwebm::kMkvDocTypeVersion, size += EbmlElementSize(libwebm::kMkvDocTypeVersion,
static_cast<uint64>(doc_type_version)); static_cast<uint64>(doc_type_version));
size += size +=
@ -112,7 +118,7 @@ bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version) {
static_cast<uint64>(8))) { static_cast<uint64>(8))) {
return false; return false;
} }
if (!WriteEbmlElement(writer, libwebm::kMkvDocType, "webm")) if (!WriteEbmlElement(writer, libwebm::kMkvDocType, doc_type))
return false; return false;
if (!WriteEbmlElement(writer, libwebm::kMkvDocTypeVersion, if (!WriteEbmlElement(writer, libwebm::kMkvDocTypeVersion,
static_cast<uint64>(doc_type_version))) { static_cast<uint64>(doc_type_version))) {
@ -126,6 +132,10 @@ bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version) {
return true; return true;
} }
bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version) {
return WriteEbmlHeader(writer, doc_type_version, kDocTypeWebm);
}
bool WriteEbmlHeader(IMkvWriter* writer) { bool WriteEbmlHeader(IMkvWriter* writer) {
return WriteEbmlHeader(writer, mkvmuxer::Segment::kDefaultDocTypeVersion); return WriteEbmlHeader(writer, mkvmuxer::Segment::kDefaultDocTypeVersion);
} }
@ -1483,6 +1493,10 @@ const char Tracks::kVorbisCodecId[] = "A_VORBIS";
const char Tracks::kVp8CodecId[] = "V_VP8"; const char Tracks::kVp8CodecId[] = "V_VP8";
const char Tracks::kVp9CodecId[] = "V_VP9"; const char Tracks::kVp9CodecId[] = "V_VP9";
const char Tracks::kVp10CodecId[] = "V_VP10"; const char Tracks::kVp10CodecId[] = "V_VP10";
const char Tracks::kWebVttCaptionsId[] = "D_WEBVTT/CAPTIONS";
const char Tracks::kWebVttDescriptionsId[] = "D_WEBVTT/DESCRIPTIONS";
const char Tracks::kWebVttMetadataId[] = "D_WEBVTT/METADATA";
const char Tracks::kWebVttSubtitlesId[] = "D_WEBVTT/SUBTITLES";
Tracks::Tracks() Tracks::Tracks()
: track_entries_(NULL), track_entries_size_(0), wrote_tracks_(false) {} : track_entries_(NULL), track_entries_size_(0), wrote_tracks_(false) {}
@ -3038,7 +3052,9 @@ bool Segment::Finalize() {
if (writer_header_->Position(0)) if (writer_header_->Position(0))
return false; return false;
if (!WriteEbmlHeader(writer_header_, doc_type_version_)) const char* const doc_type =
DocTypeIsWebm() ? kDocTypeWebm : kDocTypeMatroska;
if (!WriteEbmlHeader(writer_header_, doc_type_version_, doc_type))
return false; return false;
if (writer_header_->Position() != ebml_header_size_) if (writer_header_->Position() != ebml_header_size_)
return false; return false;
@ -3389,8 +3405,9 @@ Track* Segment::GetTrackByNumber(uint64_t track_number) const {
bool Segment::WriteSegmentHeader() { bool Segment::WriteSegmentHeader() {
UpdateDocTypeVersion(); UpdateDocTypeVersion();
// TODO(fgalligan): Support more than one segment. const char* const doc_type =
if (!WriteEbmlHeader(writer_header_, doc_type_version_)) DocTypeIsWebm() ? kDocTypeWebm : kDocTypeMatroska;
if (!WriteEbmlHeader(writer_header_, doc_type_version_, doc_type))
return false; return false;
doc_type_version_written_ = doc_type_version_; doc_type_version_written_ = doc_type_version_;
ebml_header_size_ = static_cast<int32_t>(writer_header_->Position()); ebml_header_size_ = static_cast<int32_t>(writer_header_->Position());
@ -3863,4 +3880,35 @@ bool Segment::WriteFramesLessThan(uint64_t timestamp) {
return true; return true;
} }
bool Segment::DocTypeIsWebm() const {
const int kNumCodecIds = 9;
// TODO(vigneshv): Tweak .clang-format.
const char* kWebmCodecIds[kNumCodecIds] = {
Tracks::kOpusCodecId, Tracks::kVorbisCodecId,
Tracks::kVp8CodecId, Tracks::kVp9CodecId,
Tracks::kVp10CodecId, Tracks::kWebVttCaptionsId,
Tracks::kWebVttDescriptionsId, Tracks::kWebVttMetadataId,
Tracks::kWebVttSubtitlesId};
const int num_tracks = static_cast<int>(tracks_.track_entries_size());
for (int track_index = 0; track_index < num_tracks; ++track_index) {
const Track* const track = tracks_.GetTrackByIndex(track_index);
const std::string codec_id = track->codec_id();
bool id_is_webm = false;
for (int id_index = 0; id_index < kNumCodecIds; ++id_index) {
if (codec_id == kWebmCodecIds[id_index]) {
id_is_webm = true;
break;
}
}
if (!id_is_webm)
return false;
}
return true;
}
} // namespace mkvmuxer } // namespace mkvmuxer

View File

@ -64,6 +64,12 @@ class IMkvWriter {
LIBWEBM_DISALLOW_COPY_AND_ASSIGN(IMkvWriter); LIBWEBM_DISALLOW_COPY_AND_ASSIGN(IMkvWriter);
}; };
// Writes out the EBML header for a WebM file, but allows caller to specify
// DocType. This function must be called before any other libwebm writing
// functions are called.
bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version,
const char* const doc_type);
// Writes out the EBML header for a WebM file. This function must be called // Writes out the EBML header for a WebM file. This function must be called
// before any other libwebm writing functions are called. // before any other libwebm writing functions are called.
bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version); bool WriteEbmlHeader(IMkvWriter* writer, uint64_t doc_type_version);
@ -670,6 +676,10 @@ class Tracks {
static const char kVp8CodecId[]; static const char kVp8CodecId[];
static const char kVp9CodecId[]; static const char kVp9CodecId[];
static const char kVp10CodecId[]; static const char kVp10CodecId[];
static const char kWebVttCaptionsId[];
static const char kWebVttDescriptionsId[];
static const char kWebVttMetadataId[];
static const char kWebVttSubtitlesId[];
Tracks(); Tracks();
~Tracks(); ~Tracks();
@ -1483,6 +1493,9 @@ class Segment {
bool output_cues() const { return output_cues_; } bool output_cues() const { return output_cues_; }
const SegmentInfo* segment_info() const { return &segment_info_; } const SegmentInfo* segment_info() const { return &segment_info_; }
// Returns true when codec IDs are valid for WebM.
bool DocTypeIsWebm() const;
private: private:
// Checks if header information has been output and initialized. If not it // Checks if header information has been output and initialized. If not it
// will output the Segment element and initialize the SeekHead elment and // will output the Segment element and initialize the SeekHead elment and

View File

@ -723,6 +723,24 @@ TEST_F(MuxerTest, UseFixedSizeClusterTimecode) {
filename_)); filename_));
} }
TEST_F(MuxerTest, DocTypeWebm) {
EXPECT_TRUE(SegmentInit(false, false, false));
AddVideoTrack();
Track* const vid_track = segment_.GetTrackByNumber(kVideoTrackNumber);
vid_track->set_codec_id(kVP9CodecId);
AddDummyFrameAndFinalize(kVideoTrackNumber);
EXPECT_TRUE(CompareFiles(GetTestFilePath("webm_doctype.webm"), filename_));
}
TEST_F(MuxerTest, DocTypeMatroska) {
EXPECT_TRUE(SegmentInit(false, false, false));
AddVideoTrack();
Track* const vid_track = segment_.GetTrackByNumber(kVideoTrackNumber);
vid_track->set_codec_id("V_SOMETHING_NOT_IN_WEBM");
AddDummyFrameAndFinalize(kVideoTrackNumber);
EXPECT_TRUE(CompareFiles(GetTestFilePath("matroska_doctype.mkv"), filename_));
}
} // namespace test } // namespace test
int main(int argc, char* argv[]) { int main(int argc, char* argv[]) {

BIN
testing/testdata/matroska_doctype.mkv vendored Normal file

Binary file not shown.

BIN
testing/testdata/webm_doctype.webm vendored Normal file

Binary file not shown.