libvpxenc: add psnr support

Signed-off-by: Michael Niedermayer <michaelni@gmx.at>
This commit is contained in:
Pascal Massimino 2013-02-28 14:14:46 -08:00 committed by Michael Niedermayer
parent b56e029bdc
commit 3c3a8c1489

View File

@ -48,6 +48,9 @@ struct FrameListData {
unsigned long duration; /**< duration to show frame unsigned long duration; /**< duration to show frame
(in timebase units) */ (in timebase units) */
uint32_t flags; /**< flags for this frame */ uint32_t flags; /**< flags for this frame */
uint64_t sse[4];
int have_sse; /**< true if we have pending sse[] */
uint64_t frame_number;
struct FrameListData *next; struct FrameListData *next;
}; };
@ -57,6 +60,9 @@ typedef struct VP8EncoderContext {
struct vpx_image rawimg; struct vpx_image rawimg;
struct vpx_fixed_buf twopass_stats; struct vpx_fixed_buf twopass_stats;
int deadline; //i.e., RT/GOOD/BEST int deadline; //i.e., RT/GOOD/BEST
uint64_t sse[4];
int have_sse; /**< true if we have pending sse[] */
uint64_t frame_number;
struct FrameListData *coded_frame_list; struct FrameListData *coded_frame_list;
int cpu_used; int cpu_used;
@ -232,6 +238,7 @@ static av_cold int vpx_init(AVCodecContext *avctx,
{ {
VP8Context *ctx = avctx->priv_data; VP8Context *ctx = avctx->priv_data;
struct vpx_codec_enc_cfg enccfg; struct vpx_codec_enc_cfg enccfg;
vpx_codec_flags_t flags = (avctx->flags & CODEC_FLAG_PSNR) ? VPX_CODEC_USE_PSNR : 0;
int res; int res;
av_log(avctx, AV_LOG_INFO, "%s\n", vpx_codec_version_str()); av_log(avctx, AV_LOG_INFO, "%s\n", vpx_codec_version_str());
@ -364,7 +371,7 @@ static av_cold int vpx_init(AVCodecContext *avctx,
dump_enc_cfg(avctx, &enccfg); dump_enc_cfg(avctx, &enccfg);
/* Construct Encoder Context */ /* Construct Encoder Context */
res = vpx_codec_enc_init(&ctx->encoder, iface, &enccfg, 0); res = vpx_codec_enc_init(&ctx->encoder, iface, &enccfg, flags);
if (res != VPX_CODEC_OK) { if (res != VPX_CODEC_OK) {
log_encoder_error(avctx, "Failed to initialize encoder"); log_encoder_error(avctx, "Failed to initialize encoder");
return AVERROR(EINVAL); return AVERROR(EINVAL);
@ -397,6 +404,9 @@ static av_cold int vpx_init(AVCodecContext *avctx,
vpx_img_wrap(&ctx->rawimg, VPX_IMG_FMT_I420, avctx->width, avctx->height, 1, vpx_img_wrap(&ctx->rawimg, VPX_IMG_FMT_I420, avctx->width, avctx->height, 1,
(unsigned char*)1); (unsigned char*)1);
ctx->have_sse = 0;
ctx->frame_number = 0;
avctx->coded_frame = avcodec_alloc_frame(); avctx->coded_frame = avcodec_alloc_frame();
if (!avctx->coded_frame) { if (!avctx->coded_frame) {
av_log(avctx, AV_LOG_ERROR, "Error allocating coded frame\n"); av_log(avctx, AV_LOG_ERROR, "Error allocating coded frame\n");
@ -407,13 +417,30 @@ static av_cold int vpx_init(AVCodecContext *avctx,
} }
static inline void cx_pktcpy(struct FrameListData *dst, static inline void cx_pktcpy(struct FrameListData *dst,
const struct vpx_codec_cx_pkt *src) const struct vpx_codec_cx_pkt *src,
VP8Context *ctx)
{ {
dst->pts = src->data.frame.pts; dst->pts = src->data.frame.pts;
dst->duration = src->data.frame.duration; dst->duration = src->data.frame.duration;
dst->flags = src->data.frame.flags; dst->flags = src->data.frame.flags;
dst->sz = src->data.frame.sz; dst->sz = src->data.frame.sz;
dst->buf = src->data.frame.buf; dst->buf = src->data.frame.buf;
dst->have_sse = 0;
/* For alt-ref frame, don't store PSNR or increment frame_number */
if (!(dst->flags & VPX_FRAME_IS_INVISIBLE)) {
dst->frame_number = ++ctx->frame_number;
dst->have_sse = ctx->have_sse;
if (ctx->have_sse) {
/* associate last-seen SSE to the frame. */
/* Transfers ownership from ctx to dst. */
/* WARNING! This makes the assumption that PSNR_PKT comes
just before the frame it refers to! */
memcpy(dst->sse, ctx->sse, sizeof(dst->sse));
ctx->have_sse = 0;
}
} else {
dst->frame_number = -1; /* sanity marker */
}
} }
/** /**
@ -438,6 +465,19 @@ static int storeframe(AVCodecContext *avctx, struct FrameListData *cx_frame,
pkt->flags |= AV_PKT_FLAG_KEY; pkt->flags |= AV_PKT_FLAG_KEY;
} else } else
coded_frame->pict_type = AV_PICTURE_TYPE_P; coded_frame->pict_type = AV_PICTURE_TYPE_P;
if (cx_frame->have_sse) {
int i;
/* Beware of the Y/U/V/all order! */
coded_frame->error[0] = cx_frame->sse[1];
coded_frame->error[1] = cx_frame->sse[2];
coded_frame->error[2] = cx_frame->sse[3];
coded_frame->error[3] = 0; // alpha
for (i = 0; i < 4; ++i) {
avctx->error[i] += coded_frame->error[i];
}
cx_frame->have_sse = 0;
}
} else { } else {
return ret; return ret;
} }
@ -481,7 +521,7 @@ static int queue_frames(AVCodecContext *avctx, AVPacket *pkt_out,
/* avoid storing the frame when the list is empty and we haven't yet /* avoid storing the frame when the list is empty and we haven't yet
provided a frame for output */ provided a frame for output */
av_assert0(!ctx->coded_frame_list); av_assert0(!ctx->coded_frame_list);
cx_pktcpy(&cx_frame, pkt); cx_pktcpy(&cx_frame, pkt, ctx);
size = storeframe(avctx, &cx_frame, pkt_out, coded_frame); size = storeframe(avctx, &cx_frame, pkt_out, coded_frame);
if (size < 0) if (size < 0)
return size; return size;
@ -494,7 +534,7 @@ static int queue_frames(AVCodecContext *avctx, AVPacket *pkt_out,
"Frame queue element alloc failed\n"); "Frame queue element alloc failed\n");
return AVERROR(ENOMEM); return AVERROR(ENOMEM);
} }
cx_pktcpy(cx_frame, pkt); cx_pktcpy(cx_frame, pkt, ctx);
cx_frame->buf = av_malloc(cx_frame->sz); cx_frame->buf = av_malloc(cx_frame->sz);
if (!cx_frame->buf) { if (!cx_frame->buf) {
@ -521,7 +561,14 @@ static int queue_frames(AVCodecContext *avctx, AVPacket *pkt_out,
stats->sz += pkt->data.twopass_stats.sz; stats->sz += pkt->data.twopass_stats.sz;
break; break;
} }
case VPX_CODEC_PSNR_PKT: //FIXME add support for CODEC_FLAG_PSNR case VPX_CODEC_PSNR_PKT:
av_assert0(!ctx->have_sse);
ctx->sse[0] = pkt->data.psnr.sse[0];
ctx->sse[1] = pkt->data.psnr.sse[1];
ctx->sse[2] = pkt->data.psnr.sse[2];
ctx->sse[3] = pkt->data.psnr.sse[3];
ctx->have_sse = 1;
break;
case VPX_CODEC_CUSTOM_PKT: case VPX_CODEC_CUSTOM_PKT:
//ignore unsupported/unrecognized packet types //ignore unsupported/unrecognized packet types
break; break;