2014-01-08 00:15:25 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2010 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Postprocessing Decoder
|
|
|
|
// ======================
|
|
|
|
//
|
|
|
|
// This example adds postprocessing to the simple decoder loop.
|
|
|
|
//
|
|
|
|
// Initializing Postprocessing
|
|
|
|
// ---------------------------
|
|
|
|
// You must inform the codec that you might request postprocessing at
|
|
|
|
// initialization time. This is done by passing the VPX_CODEC_USE_POSTPROC
|
|
|
|
// flag to `vpx_codec_dec_init`. If the codec does not support
|
|
|
|
// postprocessing, this call will return VPX_CODEC_INCAPABLE. For
|
|
|
|
// demonstration purposes, we also fall back to default initialization if
|
|
|
|
// the codec does not provide support.
|
|
|
|
//
|
|
|
|
// Using Adaptive Postprocessing
|
|
|
|
// -----------------------------
|
|
|
|
// VP6 provides "adaptive postprocessing." It will automatically select the
|
|
|
|
// best postprocessing filter on a frame by frame basis based on the amount
|
|
|
|
// of time remaining before the user's specified deadline expires. The
|
|
|
|
// special value 0 indicates that the codec should take as long as
|
|
|
|
// necessary to provide the best quality frame. This example gives the
|
|
|
|
// codec 15ms (15000us) to return a frame. Remember that this is a soft
|
|
|
|
// deadline, and the codec may exceed it doing its regular processing. In
|
|
|
|
// these cases, no additional postprocessing will be done.
|
|
|
|
//
|
|
|
|
// Codec Specific Postprocessing Controls
|
|
|
|
// --------------------------------------
|
|
|
|
// Some codecs provide fine grained controls over their built-in
|
|
|
|
// postprocessors. VP8 is one example. The following sample code toggles
|
|
|
|
// postprocessing on and off every 15 frames.
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-01-08 00:15:25 +01:00
|
|
|
#define VPX_CODEC_DISABLE_COMPAT 1
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-01-08 00:15:25 +01:00
|
|
|
#include "vpx/vp8dx.h"
|
|
|
|
#include "vpx/vpx_decoder.h"
|
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
#include "./tools_common.h"
|
2014-02-06 03:34:46 +01:00
|
|
|
#include "./video_reader.h"
|
2014-01-28 20:27:30 +01:00
|
|
|
#include "./vpx_config.h"
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
static const char *exec_name;
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
void usage_exit() {
|
|
|
|
fprintf(stderr, "Usage: %s <infile> <outfile>\n", exec_name);
|
|
|
|
exit(EXIT_FAILURE);
|
2014-01-08 00:15:25 +01:00
|
|
|
}
|
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
int main(int argc, char **argv) {
|
|
|
|
int frame_cnt = 0;
|
2014-02-06 03:34:46 +01:00
|
|
|
FILE *outfile = NULL;
|
|
|
|
vpx_codec_ctx_t codec;
|
2014-01-28 20:27:30 +01:00
|
|
|
vpx_codec_err_t res;
|
2014-02-06 03:34:46 +01:00
|
|
|
VpxVideoReader *reader = NULL;
|
2014-02-12 06:12:23 +01:00
|
|
|
const VpxInterface *decoder = NULL;
|
2014-02-06 03:34:46 +01:00
|
|
|
const VpxVideoInfo *info = NULL;
|
2014-01-28 20:27:30 +01:00
|
|
|
|
|
|
|
exec_name = argv[0];
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
if (argc != 3)
|
2014-02-06 03:34:46 +01:00
|
|
|
die("Invalid number of arguments.");
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-02-06 03:34:46 +01:00
|
|
|
reader = vpx_video_reader_open(argv[1]);
|
|
|
|
if (!reader)
|
|
|
|
die("Failed to open %s for reading.", argv[1]);
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
if (!(outfile = fopen(argv[2], "wb")))
|
|
|
|
die("Failed to open %s for writing", argv[2]);
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-02-06 03:34:46 +01:00
|
|
|
info = vpx_video_reader_get_info(reader);
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-02-12 06:12:23 +01:00
|
|
|
decoder = get_vpx_decoder_by_fourcc(info->codec_fourcc);
|
|
|
|
if (!decoder)
|
2014-02-06 03:34:46 +01:00
|
|
|
die("Unknown input codec.");
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-02-12 06:12:23 +01:00
|
|
|
printf("Using %s\n", vpx_codec_iface_name(decoder->interface()));
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-02-12 06:12:23 +01:00
|
|
|
res = vpx_codec_dec_init(&codec, decoder->interface(), NULL,
|
|
|
|
VPX_CODEC_USE_POSTPROC);
|
|
|
|
if (res == VPX_CODEC_INCAPABLE)
|
|
|
|
die_codec(&codec, "Postproc not supported by this decoder.");
|
2014-01-28 20:27:30 +01:00
|
|
|
|
|
|
|
if (res)
|
2014-02-06 03:34:46 +01:00
|
|
|
die_codec(&codec, "Failed to initialize decoder.");
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-02-06 03:34:46 +01:00
|
|
|
while (vpx_video_reader_read_frame(reader)) {
|
2014-01-28 20:27:30 +01:00
|
|
|
vpx_codec_iter_t iter = NULL;
|
|
|
|
vpx_image_t *img = NULL;
|
|
|
|
size_t frame_size = 0;
|
2014-02-06 03:34:46 +01:00
|
|
|
const unsigned char *frame = vpx_video_reader_get_frame(reader,
|
|
|
|
&frame_size);
|
2014-01-28 20:27:30 +01:00
|
|
|
|
|
|
|
++frame_cnt;
|
|
|
|
|
|
|
|
if (frame_cnt % 30 == 1) {
|
|
|
|
vp8_postproc_cfg_t pp = {0, 0, 0};
|
|
|
|
|
|
|
|
if (vpx_codec_control(&codec, VP8_SET_POSTPROC, &pp))
|
2014-02-06 03:34:46 +01:00
|
|
|
die_codec(&codec, "Failed to turn off postproc.");
|
2014-01-28 20:27:30 +01:00
|
|
|
} else if (frame_cnt % 30 == 16) {
|
|
|
|
vp8_postproc_cfg_t pp = {VP8_DEBLOCK | VP8_DEMACROBLOCK | VP8_MFQE,
|
|
|
|
4, 0};
|
|
|
|
if (vpx_codec_control(&codec, VP8_SET_POSTPROC, &pp))
|
2014-02-06 03:34:46 +01:00
|
|
|
die_codec(&codec, "Failed to turn on postproc.");
|
2014-01-28 20:27:30 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
// Decode the frame with 15ms deadline
|
2014-02-19 20:58:28 +01:00
|
|
|
if (vpx_codec_decode(&codec, frame, (unsigned int)frame_size, NULL, 15000))
|
2014-01-28 20:27:30 +01:00
|
|
|
die_codec(&codec, "Failed to decode frame");
|
|
|
|
|
|
|
|
while ((img = vpx_codec_get_frame(&codec, &iter)) != NULL) {
|
|
|
|
vpx_img_write(img, outfile);
|
2014-01-08 00:15:25 +01:00
|
|
|
}
|
2014-01-28 20:27:30 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("Processed %d frames.\n", frame_cnt);
|
|
|
|
if (vpx_codec_destroy(&codec))
|
|
|
|
die_codec(&codec, "Failed to destroy codec");
|
|
|
|
|
|
|
|
printf("Play: ffplay -f rawvideo -pix_fmt yuv420p -s %dx%d %s\n",
|
2014-02-06 03:34:46 +01:00
|
|
|
info->frame_width, info->frame_height, argv[2]);
|
2014-01-28 20:27:30 +01:00
|
|
|
|
2014-02-06 03:34:46 +01:00
|
|
|
vpx_video_reader_close(reader);
|
2014-01-08 00:15:25 +01:00
|
|
|
|
2014-01-28 20:27:30 +01:00
|
|
|
fclose(outfile);
|
|
|
|
return EXIT_SUCCESS;
|
2014-01-08 00:15:25 +01:00
|
|
|
}
|