d66ee6ac |
/*
* VDA H264 HW acceleration.
*
* copyright (c) 2011 Sebastien Zwickert
*
* This file is part of FFmpeg.
*
* FFmpeg is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* FFmpeg is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with FFmpeg; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
*/
|
1bfa349a |
#include <CoreFoundation/CFDictionary.h>
#include <CoreFoundation/CFNumber.h>
#include <CoreFoundation/CFData.h>
|
6fa2532d |
#include "vda.h" |
1bfa349a |
#include "libavutil/avutil.h" |
9df889a5 |
#include "h264dec.h" |
1bfa349a |
|
ffd7fd79 |
struct vda_buffer {
CVPixelBufferRef cv_buffer;
}; |
67afcefb |
#include "internal.h" |
11d923d4 |
#include "vda_vt_internal.h" |
31a46750 |
|
1bfa349a |
/* Decoder callback that adds the vda frame to the queue in display order. */ |
4fb1221e |
static void vda_decoder_callback(void *vda_hw_ctx,
CFDictionaryRef user_info,
OSStatus status,
uint32_t infoFlags,
CVImageBufferRef image_buffer) |
1bfa349a |
{ |
dc87ac55 |
struct vda_context *vda_ctx = vda_hw_ctx; |
1bfa349a |
|
31a0ca9e |
if (infoFlags & kVDADecodeInfo_FrameDropped)
vda_ctx->cv_buffer = NULL;
|
1bfa349a |
if (!image_buffer)
return;
if (vda_ctx->cv_pix_fmt_type != CVPixelBufferGetPixelFormatType(image_buffer))
return;
|
202c2acc |
vda_ctx->cv_buffer = CVPixelBufferRetain(image_buffer); |
1bfa349a |
}
|
11d923d4 |
static int vda_sync_decode(VTContext *ctx, struct vda_context *vda_ctx) |
1bfa349a |
{
OSStatus status;
CFDataRef coded_frame;
uint32_t flush_flags = 1 << 0; ///< kVDADecoderFlush_emitFrames
coded_frame = CFDataCreate(kCFAllocatorDefault, |
31a46750 |
ctx->bitstream,
ctx->bitstream_size); |
1bfa349a |
status = VDADecoderDecode(vda_ctx->decoder, 0, coded_frame, NULL);
if (kVDADecoderNoErr == status)
status = VDADecoderFlush(vda_ctx->decoder, flush_flags);
CFRelease(coded_frame);
return status;
} |
d66ee6ac |
|
ac3dbb4d |
|
67afcefb |
static int vda_old_h264_start_frame(AVCodecContext *avctx, |
c57a5936 |
av_unused const uint8_t *buffer,
av_unused uint32_t size) |
d66ee6ac |
{ |
11d923d4 |
VTContext *vda = avctx->internal->hwaccel_priv_data; |
d66ee6ac |
struct vda_context *vda_ctx = avctx->hwaccel_context;
if (!vda_ctx->decoder)
return -1;
|
31a46750 |
vda->bitstream_size = 0; |
d66ee6ac |
return 0;
}
|
67afcefb |
static int vda_old_h264_decode_slice(AVCodecContext *avctx, |
c57a5936 |
const uint8_t *buffer,
uint32_t size) |
d66ee6ac |
{ |
11d923d4 |
VTContext *vda = avctx->internal->hwaccel_priv_data; |
d66ee6ac |
struct vda_context *vda_ctx = avctx->hwaccel_context;
void *tmp;
if (!vda_ctx->decoder)
return -1;
|
31a46750 |
tmp = av_fast_realloc(vda->bitstream,
&vda->allocated_size,
vda->bitstream_size + size + 4); |
d66ee6ac |
if (!tmp)
return AVERROR(ENOMEM);
|
31a46750 |
vda->bitstream = tmp; |
d66ee6ac |
|
31a46750 |
AV_WB32(vda->bitstream + vda->bitstream_size, size);
memcpy(vda->bitstream + vda->bitstream_size + 4, buffer, size); |
d66ee6ac |
|
31a46750 |
vda->bitstream_size += size + 4; |
d66ee6ac |
return 0;
}
|
ffd7fd79 |
static void vda_h264_release_buffer(void *opaque, uint8_t *data)
{
struct vda_buffer *context = opaque;
CVPixelBufferRelease(context->cv_buffer);
av_free(context);
}
|
67afcefb |
static int vda_old_h264_end_frame(AVCodecContext *avctx) |
d66ee6ac |
{ |
ac3dbb4d |
H264Context *h = avctx->priv_data; |
11d923d4 |
VTContext *vda = avctx->internal->hwaccel_priv_data; |
ac3dbb4d |
struct vda_context *vda_ctx = avctx->hwaccel_context; |
a0f29460 |
AVFrame *frame = h->cur_pic_ptr->f; |
ffd7fd79 |
struct vda_buffer *context;
AVBufferRef *buffer; |
d66ee6ac |
int status;
|
31a46750 |
if (!vda_ctx->decoder || !vda->bitstream) |
d66ee6ac |
return -1;
|
31a46750 |
status = vda_sync_decode(vda, vda_ctx); |
202c2acc |
frame->data[3] = (void*)vda_ctx->cv_buffer; |
d66ee6ac |
if (status)
av_log(avctx, AV_LOG_ERROR, "Failed to decode frame (%d)\n", status);
|
ffd7fd79 |
if (!vda_ctx->use_ref_buffer || status)
return status;
context = av_mallocz(sizeof(*context));
buffer = av_buffer_create(NULL, 0, vda_h264_release_buffer, context, 0);
if (!context || !buffer) {
CVPixelBufferRelease(vda_ctx->cv_buffer);
av_free(context);
return -1;
}
context->cv_buffer = vda_ctx->cv_buffer;
frame->buf[3] = buffer;
|
d66ee6ac |
return status;
}
|
1bfa349a |
int ff_vda_create_decoder(struct vda_context *vda_ctx,
uint8_t *extradata,
int extradata_size)
{
OSStatus status;
CFNumberRef height;
CFNumberRef width;
CFNumberRef format;
CFDataRef avc_data;
CFMutableDictionaryRef config_info;
CFMutableDictionaryRef buffer_attributes;
CFMutableDictionaryRef io_surface_properties;
CFNumberRef cv_pix_fmt;
|
7427d1ca |
vda_ctx->priv_bitstream = NULL;
vda_ctx->priv_allocated_size = 0; |
1bfa349a |
|
8b63744f |
/* Each VCL NAL in the bitstream sent to the decoder |
1bfa349a |
* is preceded by a 4 bytes length header.
* Change the avcC atom header if needed, to signal headers of 4 bytes. */
if (extradata_size >= 4 && (extradata[4] & 0x03) != 0x03) {
uint8_t *rw_extradata;
if (!(rw_extradata = av_malloc(extradata_size)))
return AVERROR(ENOMEM);
memcpy(rw_extradata, extradata, extradata_size);
rw_extradata[4] |= 0x03;
avc_data = CFDataCreate(kCFAllocatorDefault, rw_extradata, extradata_size);
av_freep(&rw_extradata);
} else {
avc_data = CFDataCreate(kCFAllocatorDefault, extradata, extradata_size);
}
config_info = CFDictionaryCreateMutable(kCFAllocatorDefault,
4,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
height = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &vda_ctx->height);
width = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &vda_ctx->width);
format = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &vda_ctx->format);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_Height, height);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_Width, width);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_SourceFormat, format);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_avcCData, avc_data);
buffer_attributes = CFDictionaryCreateMutable(kCFAllocatorDefault,
2,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
io_surface_properties = CFDictionaryCreateMutable(kCFAllocatorDefault,
0,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
cv_pix_fmt = CFNumberCreate(kCFAllocatorDefault,
kCFNumberSInt32Type,
&vda_ctx->cv_pix_fmt_type);
CFDictionarySetValue(buffer_attributes,
kCVPixelBufferPixelFormatTypeKey,
cv_pix_fmt);
CFDictionarySetValue(buffer_attributes,
kCVPixelBufferIOSurfacePropertiesKey,
io_surface_properties);
status = VDADecoderCreate(config_info,
buffer_attributes, |
67afcefb |
(VDADecoderOutputCallback *)vda_decoder_callback, |
1bfa349a |
vda_ctx,
&vda_ctx->decoder);
CFRelease(height);
CFRelease(width);
CFRelease(format);
CFRelease(avc_data);
CFRelease(config_info);
CFRelease(io_surface_properties);
CFRelease(cv_pix_fmt);
CFRelease(buffer_attributes);
return status;
}
int ff_vda_destroy_decoder(struct vda_context *vda_ctx)
{
OSStatus status = kVDADecoderNoErr;
if (vda_ctx->decoder)
status = VDADecoderDestroy(vda_ctx->decoder);
return status;
}
|
67afcefb |
AVHWAccel ff_h264_vda_old_hwaccel = { |
d66ee6ac |
.name = "h264_vda",
.type = AVMEDIA_TYPE_VIDEO, |
36ef5369 |
.id = AV_CODEC_ID_H264, |
716d413c |
.pix_fmt = AV_PIX_FMT_VDA_VLD, |
67afcefb |
.start_frame = vda_old_h264_start_frame,
.decode_slice = vda_old_h264_decode_slice,
.end_frame = vda_old_h264_end_frame, |
11d923d4 |
.uninit = ff_videotoolbox_uninit,
.priv_data_size = sizeof(VTContext), |
67afcefb |
};
void ff_vda_output_callback(void *opaque,
CFDictionaryRef user_info,
OSStatus status,
uint32_t infoFlags,
CVImageBufferRef image_buffer)
{
AVCodecContext *ctx = opaque; |
11d923d4 |
VTContext *vda = ctx->internal->hwaccel_priv_data; |
67afcefb |
if (vda->frame) {
CVPixelBufferRelease(vda->frame);
vda->frame = NULL;
}
if (!image_buffer)
return;
vda->frame = CVPixelBufferRetain(image_buffer);
}
static int vda_h264_end_frame(AVCodecContext *avctx)
{
H264Context *h = avctx->priv_data; |
11d923d4 |
VTContext *vda = avctx->internal->hwaccel_priv_data; |
67afcefb |
AVVDAContext *vda_ctx = avctx->hwaccel_context; |
a0f29460 |
AVFrame *frame = h->cur_pic_ptr->f; |
67afcefb |
uint32_t flush_flags = 1 << 0; ///< kVDADecoderFlush_emitFrames
CFDataRef coded_frame;
OSStatus status;
if (!vda->bitstream_size)
return AVERROR_INVALIDDATA;
coded_frame = CFDataCreate(kCFAllocatorDefault,
vda->bitstream,
vda->bitstream_size);
status = VDADecoderDecode(vda_ctx->decoder, 0, coded_frame, NULL);
if (status == kVDADecoderNoErr)
status = VDADecoderFlush(vda_ctx->decoder, flush_flags);
CFRelease(coded_frame);
|
2cef68da |
if (!vda->frame)
return AVERROR_UNKNOWN;
|
67afcefb |
if (status != kVDADecoderNoErr) {
av_log(avctx, AV_LOG_ERROR, "Failed to decode frame (%d)\n", status);
return AVERROR_UNKNOWN;
}
|
11d923d4 |
return ff_videotoolbox_buffer_create(vda, frame); |
67afcefb |
}
int ff_vda_default_init(AVCodecContext *avctx)
{
AVVDAContext *vda_ctx = avctx->hwaccel_context;
OSStatus status = kVDADecoderNoErr;
CFNumberRef height;
CFNumberRef width;
CFNumberRef format;
CFDataRef avc_data;
CFMutableDictionaryRef config_info;
CFMutableDictionaryRef buffer_attributes;
CFMutableDictionaryRef io_surface_properties;
CFNumberRef cv_pix_fmt; |
e7c5e17d |
int32_t fmt = 'avc1', pix_fmt = vda_ctx->cv_pix_fmt_type; |
67afcefb |
// kCVPixelFormatType_420YpCbCr8Planar;
|
11d923d4 |
avc_data = ff_videotoolbox_avcc_extradata_create(avctx); |
67afcefb |
config_info = CFDictionaryCreateMutable(kCFAllocatorDefault,
4,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
height = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &avctx->height);
width = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &avctx->width);
format = CFNumberCreate(kCFAllocatorDefault, kCFNumberSInt32Type, &fmt);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_Height, height);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_Width, width);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_avcCData, avc_data);
CFDictionarySetValue(config_info, kVDADecoderConfiguration_SourceFormat, format);
buffer_attributes = CFDictionaryCreateMutable(kCFAllocatorDefault,
2,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
io_surface_properties = CFDictionaryCreateMutable(kCFAllocatorDefault,
0,
&kCFTypeDictionaryKeyCallBacks,
&kCFTypeDictionaryValueCallBacks);
cv_pix_fmt = CFNumberCreate(kCFAllocatorDefault,
kCFNumberSInt32Type,
&pix_fmt);
CFDictionarySetValue(buffer_attributes,
kCVPixelBufferPixelFormatTypeKey,
cv_pix_fmt);
CFDictionarySetValue(buffer_attributes,
kCVPixelBufferIOSurfacePropertiesKey,
io_surface_properties);
status = VDADecoderCreate(config_info,
buffer_attributes,
(VDADecoderOutputCallback *)ff_vda_output_callback,
avctx,
&vda_ctx->decoder);
CFRelease(format);
CFRelease(height);
CFRelease(width);
CFRelease(avc_data);
CFRelease(config_info);
CFRelease(cv_pix_fmt);
CFRelease(io_surface_properties);
CFRelease(buffer_attributes);
if (status != kVDADecoderNoErr) {
av_log(avctx, AV_LOG_ERROR, "Cannot initialize VDA %d\n", status);
}
switch (status) {
case kVDADecoderHardwareNotSupportedErr:
case kVDADecoderFormatNotSupportedErr:
return AVERROR(ENOSYS);
case kVDADecoderConfigurationError:
return AVERROR(EINVAL);
case kVDADecoderDecoderFailedErr:
return AVERROR_INVALIDDATA;
case kVDADecoderNoErr:
return 0;
default:
return AVERROR_UNKNOWN;
}
}
AVHWAccel ff_h264_vda_hwaccel = {
.name = "h264_vda",
.type = AVMEDIA_TYPE_VIDEO,
.id = AV_CODEC_ID_H264,
.pix_fmt = AV_PIX_FMT_VDA, |
11d923d4 |
.alloc_frame = ff_videotoolbox_alloc_frame,
.start_frame = ff_videotoolbox_h264_start_frame,
.decode_slice = ff_videotoolbox_h264_decode_slice, |
c57a5936 |
.end_frame = vda_h264_end_frame, |
11d923d4 |
.uninit = ff_videotoolbox_uninit,
.priv_data_size = sizeof(VTContext), |
d66ee6ac |
}; |