/* -LICENSE-START- ** Copyright (c) 2009 Blackmagic Design ** ** Permission is hereby granted, free of charge, to any person or organization ** obtaining a copy of the software and accompanying documentation covered by ** this license (the "Software") to use, reproduce, display, distribute, ** execute, and transmit the Software, and to prepare derivative works of the ** Software, and to permit third-parties to whom the Software is furnished to ** do so, all subject to the following: ** ** The copyright notices in the Software and this entire statement, including ** the above license grant, this restriction and the following disclaimer, ** must be included in all copies of the Software, in whole or in part, and ** all derivative works of the Software, unless such copies or derivative ** works are solely in the form of machine-executable object code generated by ** a source language processor. ** ** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR ** IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, ** FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT ** SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE ** FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, ** ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER ** DEALINGS IN THE SOFTWARE. ** -LICENSE-END- */ #include #include #include #include #include #include "gstdecklinksrc.h" #include "capture.h" #define GST_CAT_DEFAULT gst_decklink_src_debug_category static BMDTimecodeFormat g_timecodeFormat = (BMDTimecodeFormat) 0; DeckLinkCaptureDelegate::DeckLinkCaptureDelegate ():priv (NULL), m_refCount (0) { g_mutex_init (&m_mutex); } DeckLinkCaptureDelegate::~DeckLinkCaptureDelegate () { g_mutex_clear (&m_mutex); } ULONG DeckLinkCaptureDelegate::AddRef (void) { g_mutex_lock (&m_mutex); m_refCount++; g_mutex_unlock (&m_mutex); return (ULONG) m_refCount; } ULONG DeckLinkCaptureDelegate::Release (void) { g_mutex_lock (&m_mutex); m_refCount--; g_mutex_unlock (&m_mutex); if (m_refCount == 0) { delete this; return 0; } return (ULONG) m_refCount; } HRESULT DeckLinkCaptureDelegate::VideoInputFrameArrived (IDeckLinkVideoInputFrame * videoFrame, IDeckLinkAudioInputPacket * audioFrame) { GstDecklinkSrc *decklinksrc; const char *timecodeString = NULL; g_return_val_if_fail (priv != NULL, S_OK); g_return_val_if_fail (GST_IS_DECKLINK_SRC (priv), S_OK); decklinksrc = GST_DECKLINK_SRC (priv); if (videoFrame == NULL) { GST_WARNING_OBJECT (decklinksrc, "video frame is NULL"); return S_OK; } if (audioFrame == NULL) { GST_WARNING_OBJECT (decklinksrc, "audio frame is NULL"); return S_OK; } if (videoFrame->GetFlags () & bmdFrameHasNoInputSource) { GST_DEBUG_OBJECT (decklinksrc, "Frame received - No input signal detected"); return S_OK; } /* FIXME: g_timecodeFormat is inited to 0 and never changed? dead code? */ if (g_timecodeFormat != 0) { IDeckLinkTimecode *timecode; if (videoFrame->GetTimecode (g_timecodeFormat, &timecode) == S_OK) { timecode->GetString (&timecodeString); CONVERT_COM_STRING (timecodeString); } } GST_DEBUG_OBJECT (decklinksrc, "Frame received [%s] - %s - Size: %li bytes", timecodeString != NULL ? timecodeString : "No timecode", "Valid Frame", videoFrame->GetRowBytes () * videoFrame->GetHeight ()); if (timecodeString) FREE_COM_STRING (timecodeString); g_mutex_lock (&decklinksrc->mutex); if (decklinksrc->video_frame != NULL) { decklinksrc->dropped_frames++; decklinksrc->video_frame->Release(); if (decklinksrc->audio_frame) { decklinksrc->audio_frame->Release(); } } videoFrame->AddRef (); decklinksrc->video_frame = videoFrame; if (audioFrame) { audioFrame->AddRef (); decklinksrc->audio_frame = audioFrame; } /* increment regardless whether frame was dropped or not */ decklinksrc->frame_num++; g_cond_signal (&decklinksrc->cond); g_mutex_unlock (&decklinksrc->mutex); return S_OK; } HRESULT DeckLinkCaptureDelegate::VideoInputFormatChanged ( BMDVideoInputFormatChangedEvents events, IDeckLinkDisplayMode * mode, BMDDetectedVideoInputFormatFlags) { GstDecklinkSrc *decklinksrc; g_return_val_if_fail (priv != NULL, S_OK); g_return_val_if_fail (GST_IS_DECKLINK_SRC (priv), S_OK); decklinksrc = GST_DECKLINK_SRC (priv); GST_ERROR_OBJECT (decklinksrc, "unimplemented: video input format changed"); return S_OK; }