2015-12-17 05:54:42 +01:00
|
|
|
// Copyright (c) 2015 The WebM project authors. All Rights Reserved.
|
|
|
|
//
|
|
|
|
// Use of this source code is governed by a BSD-style license
|
|
|
|
// that can be found in the LICENSE file in the root of the source
|
|
|
|
// tree. An additional intellectual property rights grant can be found
|
|
|
|
// in the file PATENTS. All contributing project authors may
|
|
|
|
// be found in the AUTHORS file in the root of the source tree.
|
2016-03-07 20:21:23 +01:00
|
|
|
#include "common/libwebm_util.h"
|
2015-12-17 05:54:42 +01:00
|
|
|
|
2018-01-31 23:24:38 +01:00
|
|
|
#include <cassert>
|
2015-12-17 05:54:42 +01:00
|
|
|
#include <cstdint>
|
|
|
|
#include <cstdio>
|
2018-01-31 23:24:38 +01:00
|
|
|
#include <limits>
|
2015-12-17 05:54:42 +01:00
|
|
|
|
|
|
|
namespace libwebm {
|
|
|
|
|
|
|
|
std::int64_t NanosecondsTo90KhzTicks(std::int64_t nanoseconds) {
|
|
|
|
const double pts_seconds = nanoseconds / kNanosecondsPerSecond;
|
2016-03-30 19:55:33 +02:00
|
|
|
return static_cast<std::int64_t>(pts_seconds * 90000);
|
2015-12-17 05:54:42 +01:00
|
|
|
}
|
|
|
|
|
2016-07-22 20:09:26 +02:00
|
|
|
std::int64_t Khz90TicksToNanoseconds(std::int64_t ticks) {
|
|
|
|
const double seconds = ticks / 90000.0;
|
|
|
|
return static_cast<std::int64_t>(seconds * kNanosecondsPerSecond);
|
|
|
|
}
|
|
|
|
|
2015-12-17 05:54:42 +01:00
|
|
|
bool ParseVP9SuperFrameIndex(const std::uint8_t* frame,
|
2018-01-31 23:24:38 +01:00
|
|
|
std::size_t frame_length, Ranges* frame_ranges,
|
|
|
|
bool* error) {
|
|
|
|
if (frame == nullptr || frame_length == 0 || frame_ranges == nullptr ||
|
|
|
|
error == nullptr) {
|
2015-12-17 05:54:42 +01:00
|
|
|
return false;
|
2018-01-31 23:24:38 +01:00
|
|
|
}
|
2015-12-17 05:54:42 +01:00
|
|
|
|
|
|
|
bool parse_ok = false;
|
2015-12-17 17:38:43 +01:00
|
|
|
const std::uint8_t marker = frame[frame_length - 1];
|
2015-12-17 05:54:42 +01:00
|
|
|
const std::uint32_t kHasSuperFrameIndexMask = 0xe0;
|
|
|
|
const std::uint32_t kSuperFrameMarker = 0xc0;
|
2015-12-17 17:38:43 +01:00
|
|
|
const std::uint32_t kLengthFieldSizeMask = 0x3;
|
2015-12-17 05:54:42 +01:00
|
|
|
|
|
|
|
if ((marker & kHasSuperFrameIndexMask) == kSuperFrameMarker) {
|
|
|
|
const std::uint32_t kFrameCountMask = 0x7;
|
|
|
|
const int num_frames = (marker & kFrameCountMask) + 1;
|
2015-12-17 17:38:43 +01:00
|
|
|
const int length_field_size = ((marker >> 3) & kLengthFieldSizeMask) + 1;
|
2015-12-17 05:54:42 +01:00
|
|
|
const std::size_t index_length = 2 + length_field_size * num_frames;
|
|
|
|
|
2015-12-17 17:38:43 +01:00
|
|
|
if (frame_length < index_length) {
|
2018-01-31 23:24:38 +01:00
|
|
|
std::fprintf(stderr,
|
|
|
|
"VP9ParseSuperFrameIndex: Invalid superframe index size.\n");
|
|
|
|
*error = true;
|
2015-12-17 17:38:43 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Consume the super frame index. Note: it's at the end of the super frame.
|
|
|
|
const std::size_t length = frame_length - index_length;
|
2015-12-17 05:54:42 +01:00
|
|
|
|
2015-12-17 17:38:43 +01:00
|
|
|
if (length >= index_length &&
|
|
|
|
frame[frame_length - index_length] == marker) {
|
2015-12-17 05:54:42 +01:00
|
|
|
// Found a valid superframe index.
|
2015-12-17 17:38:43 +01:00
|
|
|
const std::uint8_t* byte = frame + length + 1;
|
2015-12-17 05:54:42 +01:00
|
|
|
|
2015-12-17 17:38:43 +01:00
|
|
|
std::size_t frame_offset = 0;
|
2018-01-31 23:24:38 +01:00
|
|
|
|
2015-12-17 05:54:42 +01:00
|
|
|
for (int i = 0; i < num_frames; ++i) {
|
|
|
|
std::uint32_t child_frame_length = 0;
|
|
|
|
|
|
|
|
for (int j = 0; j < length_field_size; ++j) {
|
|
|
|
child_frame_length |= (*byte++) << (j * 8);
|
|
|
|
}
|
|
|
|
|
2018-01-31 23:24:38 +01:00
|
|
|
if (length - frame_offset < child_frame_length) {
|
|
|
|
std::fprintf(stderr,
|
|
|
|
"ParseVP9SuperFrameIndex: Invalid superframe, sub frame "
|
|
|
|
"larger than entire frame.\n");
|
|
|
|
*error = true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-12-17 05:54:42 +01:00
|
|
|
frame_ranges->push_back(Range(frame_offset, child_frame_length));
|
|
|
|
frame_offset += child_frame_length;
|
|
|
|
}
|
|
|
|
|
2016-02-25 18:47:01 +01:00
|
|
|
if (static_cast<int>(frame_ranges->size()) != num_frames) {
|
2016-07-22 20:03:57 +02:00
|
|
|
std::fprintf(stderr, "VP9Parse: superframe index parse failed.\n");
|
2018-01-31 23:24:38 +01:00
|
|
|
*error = true;
|
2015-12-17 05:54:42 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
parse_ok = true;
|
|
|
|
} else {
|
2016-07-22 20:03:57 +02:00
|
|
|
std::fprintf(stderr, "VP9Parse: Invalid superframe index.\n");
|
2018-01-31 23:24:38 +01:00
|
|
|
*error = true;
|
2015-12-17 05:54:42 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return parse_ok;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool WriteUint8(std::uint8_t val, std::FILE* fileptr) {
|
|
|
|
if (fileptr == nullptr)
|
|
|
|
return false;
|
|
|
|
return (std::fputc(val, fileptr) == val);
|
|
|
|
}
|
|
|
|
|
2016-08-04 22:54:09 +02:00
|
|
|
std::uint16_t ReadUint16(const std::uint8_t* buf) {
|
|
|
|
if (buf == nullptr)
|
|
|
|
return 0;
|
|
|
|
return ((buf[0] << 8) | buf[1]);
|
|
|
|
}
|
|
|
|
|
2015-12-17 05:54:42 +01:00
|
|
|
} // namespace libwebm
|