зеркало из https://github.com/mozilla/gecko-dev.git
165 строки
4.7 KiB
C++
165 строки
4.7 KiB
C++
/*
|
|
* Copyright (c) 2017, Alliance for Open Media. All rights reserved
|
|
*
|
|
* This source code is subject to the terms of the BSD 2 Clause License and
|
|
* the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
|
|
* was not distributed with this source code in the LICENSE file, you can
|
|
* obtain it at www.aomedia.org/license/software. If the Alliance for Open
|
|
* Media Patent License 1.0 was not distributed with this source code in the
|
|
* PATENTS file, you can obtain it at www.aomedia.org/license/patent.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
#include <memory>
|
|
#include <string>
|
|
|
|
#include "config/aom_config.h"
|
|
|
|
#include "common/ivfdec.h"
|
|
#include "common/obudec.h"
|
|
#include "common/tools_common.h"
|
|
#include "common/webmdec.h"
|
|
#include "tools/obu_parser.h"
|
|
|
|
namespace {
|
|
|
|
const size_t kInitialBufferSize = 100 * 1024;
|
|
|
|
struct InputContext {
|
|
InputContext() = default;
|
|
~InputContext() { free(unit_buffer); }
|
|
|
|
void Init() {
|
|
memset(avx_ctx, 0, sizeof(*avx_ctx));
|
|
memset(obu_ctx, 0, sizeof(*obu_ctx));
|
|
obu_ctx->avx_ctx = avx_ctx;
|
|
#if CONFIG_WEBM_IO
|
|
memset(webm_ctx, 0, sizeof(*webm_ctx));
|
|
#endif
|
|
}
|
|
|
|
AvxInputContext *avx_ctx = nullptr;
|
|
ObuDecInputContext *obu_ctx = nullptr;
|
|
#if CONFIG_WEBM_IO
|
|
WebmInputContext *webm_ctx = nullptr;
|
|
#endif
|
|
uint8_t *unit_buffer = nullptr;
|
|
size_t unit_buffer_size = 0;
|
|
};
|
|
|
|
void PrintUsage() {
|
|
printf("Libaom OBU dump.\nUsage: dump_obu <input_file>\n");
|
|
}
|
|
|
|
VideoFileType GetFileType(InputContext *ctx) {
|
|
if (file_is_ivf(ctx->avx_ctx)) return FILE_TYPE_IVF;
|
|
if (file_is_obu(ctx->obu_ctx)) return FILE_TYPE_OBU;
|
|
#if CONFIG_WEBM_IO
|
|
if (file_is_webm(ctx->webm_ctx, ctx->avx_ctx)) return FILE_TYPE_WEBM;
|
|
#endif
|
|
return FILE_TYPE_RAW;
|
|
}
|
|
|
|
bool ReadTemporalUnit(InputContext *ctx, size_t *unit_size) {
|
|
const VideoFileType file_type = ctx->avx_ctx->file_type;
|
|
switch (file_type) {
|
|
case FILE_TYPE_IVF: {
|
|
if (ivf_read_frame(ctx->avx_ctx->file, &ctx->unit_buffer, unit_size,
|
|
&ctx->unit_buffer_size, NULL)) {
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
case FILE_TYPE_OBU: {
|
|
if (obudec_read_temporal_unit(ctx->obu_ctx, &ctx->unit_buffer, unit_size,
|
|
&ctx->unit_buffer_size)) {
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
#if CONFIG_WEBM_IO
|
|
case FILE_TYPE_WEBM: {
|
|
if (webm_read_frame(ctx->webm_ctx, &ctx->unit_buffer, unit_size,
|
|
&ctx->unit_buffer_size)) {
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
#endif
|
|
default:
|
|
// TODO(tomfinegan): Abuse FILE_TYPE_RAW for AV1/OBU elementary streams?
|
|
fprintf(stderr, "Error: Unsupported file type.\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
int main(int argc, const char *argv[]) {
|
|
// TODO(tomfinegan): Could do with some params for verbosity.
|
|
if (argc < 2) {
|
|
PrintUsage();
|
|
return EXIT_SUCCESS;
|
|
}
|
|
|
|
const std::string filename = argv[1];
|
|
|
|
using FilePtr = std::unique_ptr<FILE, decltype(&fclose)>;
|
|
FilePtr input_file(fopen(filename.c_str(), "rb"), &fclose);
|
|
if (input_file.get() == nullptr) {
|
|
input_file.release();
|
|
fprintf(stderr, "Error: Cannot open input file.\n");
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
AvxInputContext avx_ctx;
|
|
InputContext input_ctx;
|
|
input_ctx.avx_ctx = &avx_ctx;
|
|
ObuDecInputContext obu_ctx;
|
|
input_ctx.obu_ctx = &obu_ctx;
|
|
#if CONFIG_WEBM_IO
|
|
WebmInputContext webm_ctx;
|
|
input_ctx.webm_ctx = &webm_ctx;
|
|
#endif
|
|
|
|
input_ctx.Init();
|
|
avx_ctx.file = input_file.get();
|
|
avx_ctx.file_type = GetFileType(&input_ctx);
|
|
|
|
// Note: the reader utilities will realloc the buffer using realloc() etc.
|
|
// Can't have nice things like unique_ptr wrappers with that type of
|
|
// behavior underneath the function calls.
|
|
input_ctx.unit_buffer =
|
|
reinterpret_cast<uint8_t *>(calloc(kInitialBufferSize, 1));
|
|
if (!input_ctx.unit_buffer) {
|
|
fprintf(stderr, "Error: No memory, can't alloc input buffer.\n");
|
|
return EXIT_FAILURE;
|
|
}
|
|
input_ctx.unit_buffer_size = kInitialBufferSize;
|
|
|
|
size_t unit_size = 0;
|
|
int unit_number = 0;
|
|
int64_t obu_overhead_bytes_total = 0;
|
|
while (ReadTemporalUnit(&input_ctx, &unit_size)) {
|
|
printf("Temporal unit %d\n", unit_number);
|
|
|
|
int obu_overhead_current_unit = 0;
|
|
if (!aom_tools::DumpObu(input_ctx.unit_buffer, static_cast<int>(unit_size),
|
|
&obu_overhead_current_unit)) {
|
|
fprintf(stderr, "Error: Temporal Unit parse failed on unit number %d.\n",
|
|
unit_number);
|
|
return EXIT_FAILURE;
|
|
}
|
|
printf(" OBU overhead: %d\n", obu_overhead_current_unit);
|
|
++unit_number;
|
|
obu_overhead_bytes_total += obu_overhead_current_unit;
|
|
}
|
|
|
|
printf("File total OBU overhead: %" PRId64 "\n", obu_overhead_bytes_total);
|
|
return EXIT_SUCCESS;
|
|
}
|