Commit 418ce915 authored by Marton Balint's avatar Marton Balint

avdevice/decklink_enc: convert AVFMT_RAWPICTURE to AV_CODEC_ID_WRAPPED_AVFRAME

Signed-off-by: 's avatarMarton Balint <cus@passwd.hu>
parent 48f8ad32
...@@ -28,6 +28,7 @@ extern "C" { ...@@ -28,6 +28,7 @@ extern "C" {
#include "libavformat/avformat.h" #include "libavformat/avformat.h"
#include "libavformat/internal.h" #include "libavformat/internal.h"
#include "libavutil/imgutils.h" #include "libavutil/imgutils.h"
#include "libavutil/atomic.h"
} }
#include "decklink_common.h" #include "decklink_common.h"
...@@ -38,33 +39,43 @@ extern "C" { ...@@ -38,33 +39,43 @@ extern "C" {
class decklink_frame : public IDeckLinkVideoFrame class decklink_frame : public IDeckLinkVideoFrame
{ {
public: public:
decklink_frame(struct decklink_ctx *ctx, AVFrame *avframe, long width, decklink_frame(struct decklink_ctx *ctx, AVFrame *avframe) :
long height, void *buffer) : _ctx(ctx), _avframe(avframe), _refs(1) { }
_ctx(ctx), _avframe(avframe), _width(width),
_height(height), _buffer(buffer), _refs(0) { } virtual long STDMETHODCALLTYPE GetWidth (void) { return _avframe->width; }
virtual long STDMETHODCALLTYPE GetHeight (void) { return _avframe->height; }
virtual long STDMETHODCALLTYPE GetWidth (void) { return _width; } virtual long STDMETHODCALLTYPE GetRowBytes (void) { return _avframe->linesize[0] < 0 ? -_avframe->linesize[0] : _avframe->linesize[0]; }
virtual long STDMETHODCALLTYPE GetHeight (void) { return _height; }
virtual long STDMETHODCALLTYPE GetRowBytes (void) { return _width<<1; }
virtual BMDPixelFormat STDMETHODCALLTYPE GetPixelFormat(void) { return bmdFormat8BitYUV; } virtual BMDPixelFormat STDMETHODCALLTYPE GetPixelFormat(void) { return bmdFormat8BitYUV; }
virtual BMDFrameFlags STDMETHODCALLTYPE GetFlags (void) { return bmdVideoOutputFlagDefault; } virtual BMDFrameFlags STDMETHODCALLTYPE GetFlags (void) { return _avframe->linesize[0] < 0 ? bmdFrameFlagFlipVertical : bmdFrameFlagDefault; }
virtual HRESULT STDMETHODCALLTYPE GetBytes (void **buffer) { *buffer = _buffer; return S_OK; } virtual HRESULT STDMETHODCALLTYPE GetBytes (void **buffer)
{
if (_avframe->linesize[0] < 0)
*buffer = (void *)(_avframe->data[0] + _avframe->linesize[0] * (_avframe->height - 1));
else
*buffer = (void *)(_avframe->data[0]);
return S_OK;
}
virtual HRESULT STDMETHODCALLTYPE GetTimecode (BMDTimecodeFormat format, IDeckLinkTimecode **timecode) { return S_FALSE; } virtual HRESULT STDMETHODCALLTYPE GetTimecode (BMDTimecodeFormat format, IDeckLinkTimecode **timecode) { return S_FALSE; }
virtual HRESULT STDMETHODCALLTYPE GetAncillaryData(IDeckLinkVideoFrameAncillary **ancillary) { return S_FALSE; } virtual HRESULT STDMETHODCALLTYPE GetAncillaryData(IDeckLinkVideoFrameAncillary **ancillary) { return S_FALSE; }
virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID iid, LPVOID *ppv) { return E_NOINTERFACE; } virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID iid, LPVOID *ppv) { return E_NOINTERFACE; }
virtual ULONG STDMETHODCALLTYPE AddRef(void) { return ++_refs; } virtual ULONG STDMETHODCALLTYPE AddRef(void) { return avpriv_atomic_int_add_and_fetch(&_refs, 1); }
virtual ULONG STDMETHODCALLTYPE Release(void) { if (!--_refs) {delete this; return 0;} return _refs; } virtual ULONG STDMETHODCALLTYPE Release(void)
{
int ret = avpriv_atomic_int_add_and_fetch(&_refs, -1);
if (!ret) {
av_frame_free(&_avframe);
delete this;
}
return ret;
}
struct decklink_ctx *_ctx; struct decklink_ctx *_ctx;
AVFrame *_avframe; AVFrame *_avframe;
private: private:
long _width; volatile int _refs;
long _height;
void *_buffer;
int _refs;
}; };
class decklink_output_callback : public IDeckLinkVideoOutputCallback class decklink_output_callback : public IDeckLinkVideoOutputCallback
...@@ -76,7 +87,7 @@ public: ...@@ -76,7 +87,7 @@ public:
struct decklink_ctx *ctx = frame->_ctx; struct decklink_ctx *ctx = frame->_ctx;
AVFrame *avframe = frame->_avframe; AVFrame *avframe = frame->_avframe;
av_frame_free(&avframe); av_frame_unref(avframe);
sem_post(&ctx->semaphore); sem_post(&ctx->semaphore);
...@@ -209,41 +220,27 @@ static int decklink_write_video_packet(AVFormatContext *avctx, AVPacket *pkt) ...@@ -209,41 +220,27 @@ static int decklink_write_video_packet(AVFormatContext *avctx, AVPacket *pkt)
{ {
struct decklink_cctx *cctx = (struct decklink_cctx *)avctx->priv_data; struct decklink_cctx *cctx = (struct decklink_cctx *)avctx->priv_data;
struct decklink_ctx *ctx = (struct decklink_ctx *)cctx->ctx; struct decklink_ctx *ctx = (struct decklink_ctx *)cctx->ctx;
AVPicture *avpicture = (AVPicture *) pkt->data; AVFrame *avframe, *tmp = (AVFrame *)pkt->data;
AVFrame *avframe, *tmp;
decklink_frame *frame; decklink_frame *frame;
buffercount_type buffered; buffercount_type buffered;
HRESULT hr; HRESULT hr;
/* HACK while av_uncoded_frame() isn't implemented */ if (tmp->format != AV_PIX_FMT_UYVY422 ||
int ret; tmp->width != ctx->bmd_width ||
tmp->height != ctx->bmd_height) {
tmp = av_frame_alloc(); av_log(avctx, AV_LOG_ERROR, "Got a frame with invalid pixel format or dimension.\n");
if (!tmp) return AVERROR(EINVAL);
return AVERROR(ENOMEM);
tmp->format = AV_PIX_FMT_UYVY422;
tmp->width = ctx->bmd_width;
tmp->height = ctx->bmd_height;
ret = av_frame_get_buffer(tmp, 32);
if (ret < 0) {
av_frame_free(&tmp);
return ret;
} }
av_image_copy(tmp->data, tmp->linesize, (const uint8_t **) avpicture->data,
avpicture->linesize, (AVPixelFormat) tmp->format, tmp->width,
tmp->height);
avframe = av_frame_clone(tmp); avframe = av_frame_clone(tmp);
av_frame_free(&tmp);
if (!avframe) { if (!avframe) {
av_log(avctx, AV_LOG_ERROR, "Could not clone video frame.\n"); av_log(avctx, AV_LOG_ERROR, "Could not clone video frame.\n");
return AVERROR(EIO); return AVERROR(EIO);
} }
/* end HACK */
frame = new decklink_frame(ctx, avframe, ctx->bmd_width, ctx->bmd_height, frame = new decklink_frame(ctx, avframe);
(void *) avframe->data[0]);
if (!frame) { if (!frame) {
av_log(avctx, AV_LOG_ERROR, "Could not create new frame.\n"); av_log(avctx, AV_LOG_ERROR, "Could not create new frame.\n");
av_frame_free(&avframe);
return AVERROR(EIO); return AVERROR(EIO);
} }
...@@ -254,10 +251,11 @@ static int decklink_write_video_packet(AVFormatContext *avctx, AVPacket *pkt) ...@@ -254,10 +251,11 @@ static int decklink_write_video_packet(AVFormatContext *avctx, AVPacket *pkt)
hr = ctx->dlo->ScheduleVideoFrame((struct IDeckLinkVideoFrame *) frame, hr = ctx->dlo->ScheduleVideoFrame((struct IDeckLinkVideoFrame *) frame,
pkt->pts * ctx->bmd_tb_num, pkt->pts * ctx->bmd_tb_num,
ctx->bmd_tb_num, ctx->bmd_tb_den); ctx->bmd_tb_num, ctx->bmd_tb_den);
/* Pass ownership to DeckLink, or release on failure */
frame->Release();
if (hr != S_OK) { if (hr != S_OK) {
av_log(avctx, AV_LOG_ERROR, "Could not schedule video frame." av_log(avctx, AV_LOG_ERROR, "Could not schedule video frame."
" error %08x.\n", (uint32_t) hr); " error %08x.\n", (uint32_t) hr);
frame->Release();
return AVERROR(EIO); return AVERROR(EIO);
} }
......
...@@ -46,9 +46,9 @@ AVOutputFormat ff_decklink_muxer = { ...@@ -46,9 +46,9 @@ AVOutputFormat ff_decklink_muxer = {
.name = "decklink", .name = "decklink",
.long_name = NULL_IF_CONFIG_SMALL("Blackmagic DeckLink output"), .long_name = NULL_IF_CONFIG_SMALL("Blackmagic DeckLink output"),
.audio_codec = AV_CODEC_ID_PCM_S16LE, .audio_codec = AV_CODEC_ID_PCM_S16LE,
.video_codec = AV_CODEC_ID_RAWVIDEO, .video_codec = AV_CODEC_ID_WRAPPED_AVFRAME,
.subtitle_codec = AV_CODEC_ID_NONE, .subtitle_codec = AV_CODEC_ID_NONE,
.flags = AVFMT_NOFILE | AVFMT_RAWPICTURE, .flags = AVFMT_NOFILE,
.priv_class = &decklink_muxer_class, .priv_class = &decklink_muxer_class,
.priv_data_size = sizeof(struct decklink_cctx), .priv_data_size = sizeof(struct decklink_cctx),
.write_header = ff_decklink_write_header, .write_header = ff_decklink_write_header,
......
...@@ -28,8 +28,8 @@ ...@@ -28,8 +28,8 @@
#include "libavutil/version.h" #include "libavutil/version.h"
#define LIBAVDEVICE_VERSION_MAJOR 57 #define LIBAVDEVICE_VERSION_MAJOR 57
#define LIBAVDEVICE_VERSION_MINOR 2 #define LIBAVDEVICE_VERSION_MINOR 3
#define LIBAVDEVICE_VERSION_MICRO 101 #define LIBAVDEVICE_VERSION_MICRO 100
#define LIBAVDEVICE_VERSION_INT AV_VERSION_INT(LIBAVDEVICE_VERSION_MAJOR, \ #define LIBAVDEVICE_VERSION_INT AV_VERSION_INT(LIBAVDEVICE_VERSION_MAJOR, \
LIBAVDEVICE_VERSION_MINOR, \ LIBAVDEVICE_VERSION_MINOR, \
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment