22 #include <DeckLinkAPI.h>
45 static uint8_t calc_parity_and_line_offset(
int line)
47 uint8_t ret = (line < 313) << 5;
48 if (line >= 7 && line <= 22)
50 if (line >= 320 && line <= 335)
57 vbi_bit_slicer slicer;
59 vbi_bit_slicer_init(&slicer, 720, 13500000, 6937500, 6937500, 0x00aaaae4, 0xffff, 18, 6, 42 * 8, VBI_MODULATION_NRZ_MSB, VBI_PIXFMT_UYVY);
61 if (vbi_bit_slice(&slicer, src, tgt + 4) ==
FALSE)
66 tgt[2] = calc_parity_and_line_offset(line);
107 unsigned long long size;
193 virtual HRESULT STDMETHODCALLTYPE
VideoInputFormatChanged(BMDVideoInputFormatChangedEvents, IDeckLinkDisplayMode*, BMDDetectedVideoInputFormatFlags);
244 IDeckLinkAudioInputPacket *audioFrame,
250 BMDTimeValue bmd_pts;
251 BMDTimeValue bmd_duration;
256 res = audioFrame->GetPacketTime(&bmd_pts, time_base.
den);
260 res = videoFrame->GetStreamTime(&bmd_pts, &bmd_duration, time_base.
den);
264 res = videoFrame->GetHardwareReferenceTimestamp(time_base.
den, &bmd_pts, &bmd_duration);
271 pts = bmd_pts / time_base.
num;
282 IDeckLinkVideoInputFrame *videoFrame, IDeckLinkAudioInputPacket *audioFrame)
285 void *audioFrameBytes;
286 BMDTimeValue frameTime;
287 BMDTimeValue frameDuration;
288 int64_t wallclock = 0;
301 "Frame received (#%lu) - Valid (%liB) - QSize %fMB\n",
303 videoFrame->GetRowBytes() * videoFrame->GetHeight(),
304 (double)qsize / 1024 / 1024);
307 videoFrame->GetBytes(&frameBytes);
308 videoFrame->GetStreamTime(&frameTime, &frameDuration,
311 if (videoFrame->GetFlags() & bmdFrameHasNoInputSource) {
312 if (
ctx->
draw_bars && videoFrame->GetPixelFormat() == bmdFormat8BitYUV) {
314 0xEA80EA80, 0xD292D210, 0xA910A9A5, 0x90229035,
315 0x6ADD6ACA, 0x51EF515A, 0x286D28EF, 0x10801080 };
316 int width = videoFrame->GetWidth();
317 int height = videoFrame->GetHeight();
318 unsigned *p = (
unsigned *)frameBytes;
320 for (
int y = 0; y <
height; y++) {
321 for (
int x = 0; x <
width; x += 2)
322 *p++ = bars[(x * 8) /
width];
347 pkt.
size = videoFrame->GetRowBytes() *
348 videoFrame->GetHeight();
352 if (!
no_video &&
ctx->
teletext_lines && videoFrame->GetPixelFormat() == bmdFormat8BitYUV && videoFrame->GetWidth() == 720) {
353 IDeckLinkVideoFrameAncillary *vanc;
358 if (videoFrame->GetAncillaryData(&vanc) ==
S_OK) {
360 int64_t line_mask = 1;
363 for (i = 6; i < 336; i++, line_mask <<= 1) {
365 if ((
ctx->
teletext_lines & line_mask) && vanc->GetBufferForVerticalBlankingLine(i, (
void**)&buf) ==
S_OK) {
366 if (teletext_data_unit_from_vbi_data(i, buf, txt_buf) >= 0)
373 if (txt_buf - txt_buf0 > 1) {
374 int stuffing_units = (4 - ((45 + txt_buf - txt_buf0) / 46) % 4) % 4;
375 while (stuffing_units--) {
376 memset(txt_buf, 0xff, 46);
384 txt_pkt.
data = txt_buf0;
385 txt_pkt.
size = txt_buf - txt_buf0;
402 BMDTimeValue audio_pts;
407 audioFrame->GetBytes(&audioFrameBytes);
426 BMDVideoInputFormatChangedEvents events, IDeckLinkDisplayMode *
mode,
427 BMDDetectedVideoInputFormatFlags)
438 ctx->dli->SetCallback(
ctx->input_callback);
439 return ctx->dli->StartStreams();
449 if (
ctx->capture_started) {
450 ctx->dli->StopStreams();
451 ctx->dli->DisableVideoInput();
452 ctx->dli->DisableAudioInput();
492 if (
ctx->teletext_lines) {
493 av_log(avctx,
AV_LOG_ERROR,
"Libzvbi support is needed for capturing teletext, please recompile FFmpeg.\n");
510 if (
ctx->list_devices) {
516 tmp=strchr (fname,
'@');
518 mode_num = atoi (
tmp+1);
527 if (
ctx->dl->QueryInterface(IID_IDeckLinkInput, (
void **) &
ctx->dli) !=
S_OK) {
535 if (
ctx->list_formats) {
558 st->codecpar->sample_rate = bmdAudioSampleRate48kHz;
570 st->codecpar->width =
ctx->bmd_width;
571 st->codecpar->height =
ctx->bmd_height;
573 st->time_base.den =
ctx->bmd_tb_den;
574 st->time_base.num =
ctx->bmd_tb_num;
579 st->codecpar->codec_tag =
MKTAG(
'V',
'2',
'1',
'0');
580 st->codecpar->bit_rate =
av_rescale(
ctx->bmd_width *
ctx->bmd_height * 64, st->time_base.den, st->time_base.num * 3);
584 st->codecpar->codec_tag =
MKTAG(
'U',
'Y',
'V',
'Y');
585 st->codecpar->bit_rate =
av_rescale(
ctx->bmd_width *
ctx->bmd_height * 16, st->time_base.den, st->time_base.num);
592 if (
ctx->teletext_lines) {
600 st->time_base.den =
ctx->bmd_tb_den;
601 st->time_base.num =
ctx->bmd_tb_num;
604 ctx->teletext_st = st;
608 result =
ctx->dli->EnableAudioInput(bmdAudioSampleRate48kHz, bmdAudioSampleType16bitInteger,
ctx->audio_st->codecpar->channels);
610 if (result !=
S_OK) {
616 result =
ctx->dli->EnableVideoInput(
ctx->bmd_mode,
617 cctx->
v210 ? bmdFormat10BitYUV : bmdFormat8BitYUV,
618 bmdVideoInputFlagDefault);
620 if (result !=
S_OK) {
648 if (frame && (
ctx->bmd_field_dominance == bmdUpperFieldFirst ||
ctx->bmd_field_dominance == bmdLowerFieldFirst)) {
650 if (
ctx->bmd_field_dominance == bmdUpperFieldFirst) {
packed YUV 4:2:2, 16bpp, Cb Y0 Cr Y1
static av_always_inline int pthread_mutex_destroy(pthread_mutex_t *mutex)
This structure describes decoded (raw) audio or video data.
static av_always_inline int pthread_cond_wait(pthread_cond_t *cond, pthread_mutex_t *mutex)
#define AV_LOG_WARNING
Something somehow does not look correct.
int index
stream index in AVFormatContext
Convenience header that includes libavutil's core.
static void avpacket_queue_init(AVFormatContext *avctx, AVPacketQueue *q)
int ff_decklink_init_device(AVFormatContext *avctx, const char *name)
void * av_mallocz(size_t size)
Allocate a memory block with alignment suitable for all memory accesses (including vectors if availab...
static av_always_inline int pthread_cond_destroy(pthread_cond_t *cond)
DecklinkPtsSource audio_pts_source
static const BMDVideoConnection decklink_video_connection_map[]
DecklinkPtsSource audio_pts_source
int64_t duration
Duration of this packet in AVStream->time_base units, 0 if unknown.
static const BMDAudioConnection decklink_audio_connection_map[]
AVStream * avformat_new_stream(AVFormatContext *s, const AVCodec *c)
Add a new stream to a media file.
#define AV_LOG_VERBOSE
Detailed information.
static av_always_inline int pthread_cond_signal(pthread_cond_t *cond)
int interlaced_frame
The content of the picture is interlaced.
int ff_decklink_set_format(AVFormatContext *avctx, int width, int height, int tb_num, int tb_den, decklink_direction_t direction, int num)
#define AV_PKT_FLAG_KEY
The packet contains a keyframe.
int64_t av_rescale_q(int64_t a, AVRational bq, AVRational cq)
Rescale a 64-bit integer by 2 rational numbers.
static void avpacket_queue_flush(AVPacketQueue *q)
#define AV_LOG_ERROR
Something went wrong and cannot losslessly be recovered.
attribute_deprecated int av_dup_packet(AVPacket *pkt)
#define AV_LOG_DEBUG
Stuff which is only useful for libav* developers.
av_cold int ff_decklink_read_close(AVFormatContext *avctx)
int flags
A combination of AV_PKT_FLAG values.
char filename[1024]
input or output filename
int64_t av_rescale(int64_t a, int64_t b, int64_t c)
Rescale a 64-bit integer with rounding to nearest.
static av_always_inline int pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr)
int ff_decklink_list_formats(AVFormatContext *avctx, decklink_direction_t direction)
static int avpacket_queue_put(AVPacketQueue *q, AVPacket *pkt)
#define AVERROR_EXIT
Immediate exit was requested; the called function should not be restarted.
#define FF_ARRAY_ELEMS(a)
#define AV_TIME_BASE_Q
Internal time base represented as fractional value.
int ff_decklink_list_devices(AVFormatContext *avctx)
static AVRational av_make_q(int num, int den)
Create an AVRational.
DecklinkPtsSource video_pts_source
void av_packet_unref(AVPacket *pkt)
Wipe the packet.
static HRESULT decklink_start_input(AVFormatContext *avctx)
static int avpacket_queue_get(AVPacketQueue *q, AVPacket *pkt, int block)
Rational number (pair of numerator and denominator).
static int64_t pts
Global timestamp for the audio frames.
int64_t av_gettime_relative(void)
Get the current time in microseconds since some unspecified starting point.
void ff_decklink_cleanup(AVFormatContext *avctx)
static int64_t get_pkt_pts(IDeckLinkVideoInputFrame *videoFrame, IDeckLinkAudioInputPacket *audioFrame, int64_t wallclock, DecklinkPtsSource pts_src, AVRational time_base, int64_t *initial_pts)
static void avpacket_queue_end(AVPacketQueue *q)
struct AVPacketList * next
common internal and external API header
static unsigned long long avpacket_queue_size(AVPacketQueue *q)
int ff_decklink_read_packet(AVFormatContext *avctx, AVPacket *pkt)
static av_always_inline int pthread_cond_init(pthread_cond_t *cond, const pthread_condattr_t *attr)
void av_init_packet(AVPacket *pkt)
Initialize optional fields of a packet with default values.
int top_field_first
If the content is interlaced, is top field displayed first.
static av_always_inline int pthread_mutex_unlock(pthread_mutex_t *mutex)
void * priv_data
Format private data.
av_cold int ff_decklink_read_header(AVFormatContext *avctx)
int64_t dts
Decompression timestamp in AVStream->time_base units; the time at which the packet is decompressed...
DecklinkPtsSource video_pts_source
static av_always_inline int pthread_mutex_lock(pthread_mutex_t *mutex)
AVCodecParameters * codecpar
AVRational time_base
This is the fundamental unit of time (in seconds) in terms of which frame timestamps are represented...
#define MKTAG(a, b, c, d)
This structure stores compressed data.
int64_t pts
Presentation timestamp in AVStream->time_base units; the time at which the decompressed packet will b...
#define AV_NOPTS_VALUE
Undefined timestamp value.