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:
parent
e3c9576716
commit
9a235e0bc9
@ -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
|
||||||
|
@ -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
|
||||||
|
@ -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
BIN
testing/testdata/matroska_doctype.mkv
vendored
Normal file
Binary file not shown.
BIN
testing/testdata/webm_doctype.webm
vendored
Normal file
BIN
testing/testdata/webm_doctype.webm
vendored
Normal file
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user