Signed-off-by: Michael Niedermayer <michaelni@gmx.at>tags/n0.8
| @@ -1463,6 +1463,8 @@ w64_demuxer_deps="wav_demuxer" | |||||
| alsa_indev_deps="alsa_asoundlib_h snd_pcm_htimestamp" | alsa_indev_deps="alsa_asoundlib_h snd_pcm_htimestamp" | ||||
| alsa_outdev_deps="alsa_asoundlib_h" | alsa_outdev_deps="alsa_asoundlib_h" | ||||
| bktr_indev_deps_any="dev_bktr_ioctl_bt848_h machine_ioctl_bt848_h dev_video_bktr_ioctl_bt848_h dev_ic_bt8xx_h" | bktr_indev_deps_any="dev_bktr_ioctl_bt848_h machine_ioctl_bt848_h dev_video_bktr_ioctl_bt848_h dev_ic_bt8xx_h" | ||||
| dshow_indev_deps="IBaseFilter" | |||||
| dshow_indev_extralibs="-lpsapi -lole32 -lstrmiids -luuid" | |||||
| dv1394_indev_deps="dv1394 dv_demuxer" | dv1394_indev_deps="dv1394 dv_demuxer" | ||||
| fbdev_indev_deps="linux_fb_h" | fbdev_indev_deps="linux_fb_h" | ||||
| jack_indev_deps="jack_jack_h sem_timedwait" | jack_indev_deps="jack_jack_h sem_timedwait" | ||||
| @@ -2979,6 +2981,8 @@ check_func_headers "windows.h vfw.h" capCreateCaptureWindow "$vfwcap_indev_extra | |||||
| # w32api 3.12 had it defined wrong | # w32api 3.12 had it defined wrong | ||||
| check_cpp_condition vfw.h "WM_CAP_DRIVER_CONNECT > WM_USER" && enable vfwcap_defines | check_cpp_condition vfw.h "WM_CAP_DRIVER_CONNECT > WM_USER" && enable vfwcap_defines | ||||
| check_type "dshow.h" IBaseFilter | |||||
| # check for ioctl_meteor.h, ioctl_bt848.h and alternatives | # check for ioctl_meteor.h, ioctl_bt848.h and alternatives | ||||
| { check_header dev/bktr/ioctl_meteor.h && | { check_header dev/bktr/ioctl_meteor.h && | ||||
| check_header dev/bktr/ioctl_bt848.h; } || | check_header dev/bktr/ioctl_bt848.h; } || | ||||
| @@ -13,6 +13,9 @@ OBJS-$(CONFIG_ALSA_INDEV) += alsa-audio-common.o \ | |||||
| OBJS-$(CONFIG_ALSA_OUTDEV) += alsa-audio-common.o \ | OBJS-$(CONFIG_ALSA_OUTDEV) += alsa-audio-common.o \ | ||||
| alsa-audio-enc.o | alsa-audio-enc.o | ||||
| OBJS-$(CONFIG_BKTR_INDEV) += bktr.o | OBJS-$(CONFIG_BKTR_INDEV) += bktr.o | ||||
| OBJS-$(CONFIG_DSHOW_INDEV) += dshow.o dshow_enummediatypes.o \ | |||||
| dshow_enumpins.o dshow_filter.o \ | |||||
| dshow_pin.o dshow_common.o | |||||
| OBJS-$(CONFIG_DV1394_INDEV) += dv1394.o | OBJS-$(CONFIG_DV1394_INDEV) += dv1394.o | ||||
| OBJS-$(CONFIG_FBDEV_INDEV) += fbdev.o | OBJS-$(CONFIG_FBDEV_INDEV) += fbdev.o | ||||
| OBJS-$(CONFIG_JACK_INDEV) += jack_audio.o | OBJS-$(CONFIG_JACK_INDEV) += jack_audio.o | ||||
| @@ -41,6 +41,7 @@ void avdevice_register_all(void) | |||||
| /* devices */ | /* devices */ | ||||
| REGISTER_INOUTDEV (ALSA, alsa); | REGISTER_INOUTDEV (ALSA, alsa); | ||||
| REGISTER_INDEV (BKTR, bktr); | REGISTER_INDEV (BKTR, bktr); | ||||
| REGISTER_INDEV (DSHOW, dshow); | |||||
| REGISTER_INDEV (DV1394, dv1394); | REGISTER_INDEV (DV1394, dv1394); | ||||
| REGISTER_INDEV (FBDEV, fbdev); | REGISTER_INDEV (FBDEV, fbdev); | ||||
| REGISTER_INDEV (JACK, jack); | REGISTER_INDEV (JACK, jack); | ||||
| @@ -0,0 +1,646 @@ | |||||
| /* | |||||
| * Directshow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "libavformat/avformat.h" | |||||
| #include "libavformat/timefilter.h" | |||||
| #include "dshow.h" | |||||
| struct dshow_ctx { | |||||
| IGraphBuilder *graph; | |||||
| char *device_name[2]; | |||||
| IBaseFilter *device_filter[2]; | |||||
| IPin *device_pin[2]; | |||||
| libAVFilter *capture_filter[2]; | |||||
| libAVPin *capture_pin[2]; | |||||
| HANDLE mutex; | |||||
| HANDLE event; | |||||
| AVPacketList *pktl; | |||||
| unsigned int curbufsize; | |||||
| unsigned int video_frame_num; | |||||
| IMediaControl *control; | |||||
| TimeFilter *timefilter; | |||||
| }; | |||||
| static enum PixelFormat dshow_pixfmt(DWORD biCompression, WORD biBitCount) | |||||
| { | |||||
| switch(biCompression) { | |||||
| case MKTAG('U', 'Y', 'V', 'Y'): | |||||
| return PIX_FMT_UYVY422; | |||||
| case MKTAG('Y', 'U', 'Y', '2'): | |||||
| return PIX_FMT_YUYV422; | |||||
| case MKTAG('I', '4', '2', '0'): | |||||
| return PIX_FMT_YUV420P; | |||||
| case BI_RGB: | |||||
| switch(biBitCount) { /* 1-8 are untested */ | |||||
| case 1: | |||||
| return PIX_FMT_MONOWHITE; | |||||
| case 4: | |||||
| return PIX_FMT_RGB4; | |||||
| case 8: | |||||
| return PIX_FMT_RGB8; | |||||
| case 16: | |||||
| return PIX_FMT_RGB555; | |||||
| case 24: | |||||
| return PIX_FMT_BGR24; | |||||
| case 32: | |||||
| return PIX_FMT_RGB32; | |||||
| } | |||||
| } | |||||
| return PIX_FMT_NONE; | |||||
| } | |||||
| static enum CodecID dshow_codecid(DWORD biCompression) | |||||
| { | |||||
| switch(biCompression) { | |||||
| case MKTAG('d', 'v', 's', 'd'): | |||||
| return CODEC_ID_DVVIDEO; | |||||
| case MKTAG('M', 'J', 'P', 'G'): | |||||
| case MKTAG('m', 'j', 'p', 'g'): | |||||
| return CODEC_ID_MJPEG; | |||||
| } | |||||
| return CODEC_ID_NONE; | |||||
| } | |||||
| static int | |||||
| dshow_read_close(AVFormatContext *s) | |||||
| { | |||||
| struct dshow_ctx *ctx = s->priv_data; | |||||
| AVPacketList *pktl; | |||||
| if (ctx->control) { | |||||
| IMediaControl_Stop(ctx->control); | |||||
| IMediaControl_Release(ctx->control); | |||||
| } | |||||
| if (ctx->graph) | |||||
| IGraphBuilder_Release(ctx->graph); | |||||
| /* FIXME remove filters from graph */ | |||||
| /* FIXME disconnect pins */ | |||||
| if (ctx->capture_pin[VideoDevice]) | |||||
| libAVPin_Release(ctx->capture_pin[VideoDevice]); | |||||
| if (ctx->capture_pin[AudioDevice]) | |||||
| libAVPin_Release(ctx->capture_pin[AudioDevice]); | |||||
| if (ctx->capture_filter[VideoDevice]) | |||||
| libAVFilter_Release(ctx->capture_filter[VideoDevice]); | |||||
| if (ctx->capture_filter[AudioDevice]) | |||||
| libAVFilter_Release(ctx->capture_filter[AudioDevice]); | |||||
| if (ctx->device_pin[VideoDevice]) | |||||
| IPin_Release(ctx->device_pin[VideoDevice]); | |||||
| if (ctx->device_pin[AudioDevice]) | |||||
| IPin_Release(ctx->device_pin[AudioDevice]); | |||||
| if (ctx->device_filter[VideoDevice]) | |||||
| IBaseFilter_Release(ctx->device_filter[VideoDevice]); | |||||
| if (ctx->device_filter[AudioDevice]) | |||||
| IBaseFilter_Release(ctx->device_filter[AudioDevice]); | |||||
| if (ctx->device_name[0]) | |||||
| av_free(ctx->device_name[0]); | |||||
| if (ctx->device_name[1]) | |||||
| av_free(ctx->device_name[1]); | |||||
| if(ctx->mutex) | |||||
| CloseHandle(ctx->mutex); | |||||
| if(ctx->event) | |||||
| CloseHandle(ctx->event); | |||||
| pktl = ctx->pktl; | |||||
| while (pktl) { | |||||
| AVPacketList *next = pktl->next; | |||||
| av_destruct_packet(&pktl->pkt); | |||||
| av_free(pktl); | |||||
| pktl = next; | |||||
| } | |||||
| return 0; | |||||
| } | |||||
| static char *dup_wchar_to_utf8(wchar_t *w) | |||||
| { | |||||
| char *s = NULL; | |||||
| int l = WideCharToMultiByte(CP_UTF8, 0, w, -1, 0, 0, 0, 0); | |||||
| s = av_malloc(l); | |||||
| if (s) | |||||
| WideCharToMultiByte(CP_UTF8, 0, w, -1, s, l, 0, 0); | |||||
| return s; | |||||
| } | |||||
| static int shall_we_drop(AVFormatContext *s) | |||||
| { | |||||
| struct dshow_ctx *ctx = s->priv_data; | |||||
| const uint8_t dropscore[] = {62, 75, 87, 100}; | |||||
| const int ndropscores = FF_ARRAY_ELEMS(dropscore); | |||||
| unsigned int buffer_fullness = (ctx->curbufsize*100)/s->max_picture_buffer; | |||||
| if(dropscore[++ctx->video_frame_num%ndropscores] <= buffer_fullness) { | |||||
| av_log(s, AV_LOG_ERROR, | |||||
| "real-time buffer %d%% full! frame dropped!\n", buffer_fullness); | |||||
| return 1; | |||||
| } | |||||
| return 0; | |||||
| } | |||||
| static void | |||||
| callback(void *priv_data, int index, uint8_t *buf, int buf_size, int64_t time) | |||||
| { | |||||
| AVFormatContext *s = priv_data; | |||||
| struct dshow_ctx *ctx = s->priv_data; | |||||
| AVPacketList **ppktl, *pktl_next; | |||||
| // dump_videohdr(s, vdhdr); | |||||
| if(shall_we_drop(s)) | |||||
| return; | |||||
| WaitForSingleObject(ctx->mutex, INFINITE); | |||||
| pktl_next = av_mallocz(sizeof(AVPacketList)); | |||||
| if(!pktl_next) | |||||
| goto fail; | |||||
| if(av_new_packet(&pktl_next->pkt, buf_size) < 0) { | |||||
| av_free(pktl_next); | |||||
| goto fail; | |||||
| } | |||||
| pktl_next->pkt.stream_index = index; | |||||
| pktl_next->pkt.pts = time; | |||||
| memcpy(pktl_next->pkt.data, buf, buf_size); | |||||
| for(ppktl = &ctx->pktl ; *ppktl ; ppktl = &(*ppktl)->next); | |||||
| *ppktl = pktl_next; | |||||
| ctx->curbufsize += buf_size; | |||||
| SetEvent(ctx->event); | |||||
| ReleaseMutex(ctx->mutex); | |||||
| return; | |||||
| fail: | |||||
| ReleaseMutex(ctx->mutex); | |||||
| return; | |||||
| } | |||||
| static int | |||||
| dshow_open_device(AVFormatContext *avctx, ICreateDevEnum *devenum, | |||||
| enum dshowDeviceType devtype) | |||||
| { | |||||
| struct dshow_ctx *ctx = avctx->priv_data; | |||||
| IBaseFilter *device_filter = NULL; | |||||
| IEnumMoniker *classenum = NULL; | |||||
| IGraphBuilder *graph = ctx->graph; | |||||
| IEnumPins *pins = 0; | |||||
| IMoniker *m = NULL; | |||||
| IPin *device_pin = NULL; | |||||
| libAVPin *capture_pin = NULL; | |||||
| libAVFilter *capture_filter = NULL; | |||||
| const char *device_name = ctx->device_name[devtype]; | |||||
| int ret = AVERROR(EIO); | |||||
| IPin *pin; | |||||
| int r, i; | |||||
| const GUID *device_guid[2] = { &CLSID_VideoInputDeviceCategory, | |||||
| &CLSID_AudioInputDeviceCategory }; | |||||
| const GUID *mediatype[2] = { &MEDIATYPE_Video, &MEDIATYPE_Audio }; | |||||
| const char *devtypename = (devtype == VideoDevice) ? "video" : "audio"; | |||||
| const wchar_t *filter_name[2] = { L"Audio capture filter", L"Video capture filter" }; | |||||
| r = ICreateDevEnum_CreateClassEnumerator(devenum, device_guid[devtype], | |||||
| (IEnumMoniker **) &classenum, 0); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not enumerate %s devices.\n", | |||||
| devtypename); | |||||
| goto error; | |||||
| } | |||||
| while (IEnumMoniker_Next(classenum, 1, &m, NULL) == S_OK && !device_filter) { | |||||
| IPropertyBag *bag = NULL; | |||||
| char *buf = NULL; | |||||
| VARIANT var; | |||||
| r = IMoniker_BindToStorage(m, 0, 0, &IID_IPropertyBag, (void *) &bag); | |||||
| if (r != S_OK) | |||||
| goto fail1; | |||||
| var.vt = VT_BSTR; | |||||
| r = IPropertyBag_Read(bag, L"FriendlyName", &var, NULL); | |||||
| if (r != S_OK) | |||||
| goto fail1; | |||||
| buf = dup_wchar_to_utf8(var.bstrVal); | |||||
| if (strcmp(device_name, buf)) | |||||
| goto fail1; | |||||
| IMoniker_BindToObject(m, 0, 0, &IID_IBaseFilter, (void *) &device_filter); | |||||
| fail1: | |||||
| if (buf) | |||||
| av_free(buf); | |||||
| if (bag) | |||||
| IPropertyBag_Release(bag); | |||||
| IMoniker_Release(m); | |||||
| } | |||||
| if (!device_filter) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not find %s device.\n", | |||||
| devtypename); | |||||
| goto error; | |||||
| } | |||||
| ctx->device_filter [devtype] = device_filter; | |||||
| r = IGraphBuilder_AddFilter(graph, device_filter, NULL); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not add device filter to graph.\n"); | |||||
| goto error; | |||||
| } | |||||
| r = IBaseFilter_EnumPins(device_filter, &pins); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not enumerate pins.\n"); | |||||
| goto error; | |||||
| } | |||||
| i = 0; | |||||
| while (IEnumPins_Next(pins, 1, &pin, NULL) == S_OK && !device_pin) { | |||||
| IKsPropertySet *p = NULL; | |||||
| IEnumMediaTypes *types; | |||||
| PIN_INFO info = {0}; | |||||
| AM_MEDIA_TYPE *type; | |||||
| GUID category; | |||||
| DWORD r2; | |||||
| IPin_QueryPinInfo(pin, &info); | |||||
| IBaseFilter_Release(info.pFilter); | |||||
| if (info.dir != PINDIR_OUTPUT) | |||||
| goto next; | |||||
| if (IPin_QueryInterface(pin, &IID_IKsPropertySet, (void **) &p) != S_OK) | |||||
| goto next; | |||||
| if (IKsPropertySet_Get(p, &ROPSETID_Pin, AMPROPERTY_PIN_CATEGORY, | |||||
| NULL, 0, &category, sizeof(GUID), &r2) != S_OK) | |||||
| goto next; | |||||
| if (!IsEqualGUID(&category, &PIN_CATEGORY_CAPTURE)) | |||||
| goto next; | |||||
| if (IPin_EnumMediaTypes(pin, &types) != S_OK) | |||||
| goto next; | |||||
| IEnumMediaTypes_Reset(types); | |||||
| while (IEnumMediaTypes_Next(types, 1, &type, NULL) == S_OK && !device_pin) { | |||||
| if (IsEqualGUID(&type->majortype, mediatype[devtype])) { | |||||
| device_pin = pin; | |||||
| goto next; | |||||
| } | |||||
| CoTaskMemFree(type); | |||||
| } | |||||
| next: | |||||
| if (types) | |||||
| IEnumMediaTypes_Release(types); | |||||
| if (p) | |||||
| IKsPropertySet_Release(p); | |||||
| if (device_pin != pin) | |||||
| IPin_Release(pin); | |||||
| } | |||||
| if (!device_pin) { | |||||
| av_log(avctx, AV_LOG_ERROR, | |||||
| "Could not find output pin from %s capture device.\n", devtypename); | |||||
| goto error; | |||||
| } | |||||
| ctx->device_pin[devtype] = device_pin; | |||||
| capture_filter = libAVFilter_Create(avctx, callback, devtype); | |||||
| if (!capture_filter) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not create grabber filter.\n"); | |||||
| goto error; | |||||
| } | |||||
| ctx->capture_filter[devtype] = capture_filter; | |||||
| r = IGraphBuilder_AddFilter(graph, (IBaseFilter *) capture_filter, | |||||
| filter_name[devtype]); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not add capture filter to graph\n"); | |||||
| goto error; | |||||
| } | |||||
| libAVPin_AddRef(capture_filter->pin); | |||||
| capture_pin = capture_filter->pin; | |||||
| ctx->capture_pin[devtype] = capture_pin; | |||||
| r = IGraphBuilder_ConnectDirect(graph, device_pin, (IPin *) capture_pin, NULL); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not connect pins\n"); | |||||
| goto error; | |||||
| } | |||||
| ret = 0; | |||||
| error: | |||||
| if (pins) | |||||
| IEnumPins_Release(pins); | |||||
| if (classenum) | |||||
| IEnumMoniker_Release(classenum); | |||||
| return ret; | |||||
| } | |||||
| static enum CodecID waveform_codec_id(enum AVSampleFormat sample_fmt) | |||||
| { | |||||
| switch (sample_fmt) { | |||||
| case AV_SAMPLE_FMT_U8: return CODEC_ID_PCM_U8; | |||||
| case AV_SAMPLE_FMT_S16: return CODEC_ID_PCM_S16LE; | |||||
| case AV_SAMPLE_FMT_S32: return CODEC_ID_PCM_S32LE; | |||||
| default: return CODEC_ID_NONE; /* Should never happen. */ | |||||
| } | |||||
| } | |||||
| static enum SampleFormat sample_fmt_bits_per_sample(int bits) | |||||
| { | |||||
| switch (bits) { | |||||
| case 8: return AV_SAMPLE_FMT_U8; | |||||
| case 16: return AV_SAMPLE_FMT_S16; | |||||
| case 32: return AV_SAMPLE_FMT_S32; | |||||
| default: return AV_SAMPLE_FMT_NONE; /* Should never happen. */ | |||||
| } | |||||
| } | |||||
| static int | |||||
| dshow_add_device(AVFormatContext *avctx, AVFormatParameters *ap, | |||||
| enum dshowDeviceType devtype) | |||||
| { | |||||
| struct dshow_ctx *ctx = avctx->priv_data; | |||||
| AM_MEDIA_TYPE type; | |||||
| AVCodecContext *codec; | |||||
| AVStream *st; | |||||
| int ret = AVERROR(EIO); | |||||
| st = av_new_stream(avctx, devtype); | |||||
| if (!st) { | |||||
| ret = AVERROR(ENOMEM); | |||||
| goto error; | |||||
| } | |||||
| ctx->capture_filter[devtype]->stream_index = st->index; | |||||
| libAVPin_ConnectionMediaType(ctx->capture_pin[devtype], &type); | |||||
| codec = st->codec; | |||||
| if (devtype == VideoDevice) { | |||||
| BITMAPINFOHEADER *bih = NULL; | |||||
| if (IsEqualGUID(&type.formattype, &FORMAT_VideoInfo)) { | |||||
| VIDEOINFOHEADER *v = (void *) type.pbFormat; | |||||
| bih = &v->bmiHeader; | |||||
| } else if (IsEqualGUID(&type.formattype, &FORMAT_VideoInfo2)) { | |||||
| VIDEOINFOHEADER2 *v = (void *) type.pbFormat; | |||||
| bih = &v->bmiHeader; | |||||
| } | |||||
| if (!bih) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not get media type.\n"); | |||||
| goto error; | |||||
| } | |||||
| codec->time_base = ap->time_base; | |||||
| codec->codec_type = AVMEDIA_TYPE_VIDEO; | |||||
| codec->width = bih->biWidth; | |||||
| codec->height = bih->biHeight; | |||||
| codec->pix_fmt = dshow_pixfmt(bih->biCompression, bih->biBitCount); | |||||
| if (codec->pix_fmt == PIX_FMT_NONE) { | |||||
| codec->codec_id = dshow_codecid(bih->biCompression); | |||||
| if (codec->codec_id == CODEC_ID_NONE) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Unknown compression type. " | |||||
| "Please report verbose (-v 9) debug information.\n"); | |||||
| dshow_read_close(avctx); | |||||
| return AVERROR_PATCHWELCOME; | |||||
| } | |||||
| codec->bits_per_coded_sample = bih->biBitCount; | |||||
| } else { | |||||
| codec->codec_id = CODEC_ID_RAWVIDEO; | |||||
| if (bih->biCompression == BI_RGB) { | |||||
| codec->bits_per_coded_sample = bih->biBitCount; | |||||
| codec->extradata = av_malloc(9 + FF_INPUT_BUFFER_PADDING_SIZE); | |||||
| if (codec->extradata) { | |||||
| codec->extradata_size = 9; | |||||
| memcpy(codec->extradata, "BottomUp", 9); | |||||
| } | |||||
| } | |||||
| } | |||||
| } else { | |||||
| WAVEFORMATEX *fx = NULL; | |||||
| if (IsEqualGUID(&type.formattype, &FORMAT_WaveFormatEx)) { | |||||
| fx = (void *) type.pbFormat; | |||||
| } | |||||
| if (!fx) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not get media type.\n"); | |||||
| goto error; | |||||
| } | |||||
| codec->codec_type = CODEC_TYPE_AUDIO; | |||||
| codec->sample_fmt = sample_fmt_bits_per_sample(fx->wBitsPerSample); | |||||
| codec->codec_id = waveform_codec_id(codec->sample_fmt); | |||||
| codec->sample_rate = fx->nSamplesPerSec; | |||||
| codec->channels = fx->nChannels; | |||||
| } | |||||
| av_set_pts_info(st, 64, 1, 10000000); | |||||
| ret = 0; | |||||
| error: | |||||
| return ret; | |||||
| } | |||||
| static int parse_device_name(AVFormatContext *avctx) | |||||
| { | |||||
| struct dshow_ctx *ctx = avctx->priv_data; | |||||
| char **device_name = ctx->device_name; | |||||
| char *name = av_strdup(avctx->filename); | |||||
| char *tmp = name; | |||||
| int ret = 1; | |||||
| char *type; | |||||
| while ((type = strtok(tmp, "="))) { | |||||
| char *token = strtok(NULL, ":"); | |||||
| tmp = NULL; | |||||
| if (!strcmp(type, "video")) { | |||||
| device_name[0] = token; | |||||
| } else if (!strcmp(type, "audio")) { | |||||
| device_name[1] = token; | |||||
| } else { | |||||
| device_name[0] = NULL; | |||||
| device_name[1] = NULL; | |||||
| break; | |||||
| } | |||||
| } | |||||
| if (!device_name[0] && !device_name[1]) { | |||||
| ret = 0; | |||||
| } else { | |||||
| if (device_name[0]) | |||||
| device_name[0] = av_strdup(device_name[0]); | |||||
| if (device_name[1]) | |||||
| device_name[1] = av_strdup(device_name[1]); | |||||
| } | |||||
| av_free(name); | |||||
| return ret; | |||||
| } | |||||
| static int dshow_read_header(AVFormatContext *avctx, AVFormatParameters *ap) | |||||
| { | |||||
| struct dshow_ctx *ctx = avctx->priv_data; | |||||
| IGraphBuilder *graph = NULL; | |||||
| ICreateDevEnum *devenum = NULL; | |||||
| IMediaControl *control = NULL; | |||||
| int ret = AVERROR(EIO); | |||||
| int r; | |||||
| if (!parse_device_name(avctx)) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Malformed dshow input string.\n"); | |||||
| goto error; | |||||
| } | |||||
| CoInitialize(0); | |||||
| r = CoCreateInstance(&CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, | |||||
| &IID_IGraphBuilder, (void **) &graph); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not create capture graph.\n"); | |||||
| goto error; | |||||
| } | |||||
| ctx->graph = graph; | |||||
| r = CoCreateInstance(&CLSID_SystemDeviceEnum, NULL, CLSCTX_INPROC_SERVER, | |||||
| &IID_ICreateDevEnum, (void **) &devenum); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not enumerate system devices.\n"); | |||||
| goto error; | |||||
| } | |||||
| if (ctx->device_name[VideoDevice]) { | |||||
| ret = dshow_open_device(avctx, devenum, VideoDevice); | |||||
| if (ret < 0) | |||||
| goto error; | |||||
| ret = dshow_add_device(avctx, ap, VideoDevice); | |||||
| if (ret < 0) | |||||
| goto error; | |||||
| } | |||||
| if (ctx->device_name[AudioDevice]) { | |||||
| ret = dshow_open_device(avctx, devenum, AudioDevice); | |||||
| if (ret < 0) | |||||
| goto error; | |||||
| ret = dshow_add_device(avctx, ap, AudioDevice); | |||||
| if (ret < 0) | |||||
| goto error; | |||||
| } | |||||
| ctx->mutex = CreateMutex(NULL, 0, NULL); | |||||
| if (!ctx->mutex) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not create Mutex\n"); | |||||
| goto error; | |||||
| } | |||||
| ctx->event = CreateEvent(NULL, 1, 0, NULL); | |||||
| if (!ctx->event) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not create Event\n"); | |||||
| goto error; | |||||
| } | |||||
| r = IGraphBuilder_QueryInterface(graph, &IID_IMediaControl, (void **) &control); | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not get media control.\n"); | |||||
| goto error; | |||||
| } | |||||
| ctx->control = control; | |||||
| r = IMediaControl_Run(control); | |||||
| if (r == S_FALSE) { | |||||
| OAFilterState pfs; | |||||
| r = IMediaControl_GetState(control, 0, &pfs); | |||||
| } | |||||
| if (r != S_OK) { | |||||
| av_log(avctx, AV_LOG_ERROR, "Could not run filter\n"); | |||||
| goto error; | |||||
| } | |||||
| ret = 0; | |||||
| error: | |||||
| if (ret < 0) | |||||
| dshow_read_close(avctx); | |||||
| if (devenum) | |||||
| ICreateDevEnum_Release(devenum); | |||||
| return ret; | |||||
| } | |||||
| static int dshow_read_packet(AVFormatContext *s, AVPacket *pkt) | |||||
| { | |||||
| struct dshow_ctx *ctx = s->priv_data; | |||||
| AVPacketList *pktl = NULL; | |||||
| while (!pktl) { | |||||
| WaitForSingleObject(ctx->mutex, INFINITE); | |||||
| pktl = ctx->pktl; | |||||
| if (ctx->pktl) { | |||||
| *pkt = ctx->pktl->pkt; | |||||
| ctx->pktl = ctx->pktl->next; | |||||
| av_free(pktl); | |||||
| } | |||||
| ResetEvent(ctx->event); | |||||
| ReleaseMutex(ctx->mutex); | |||||
| if (!pktl) { | |||||
| if (s->flags & AVFMT_FLAG_NONBLOCK) { | |||||
| return AVERROR(EAGAIN); | |||||
| } else { | |||||
| WaitForSingleObject(ctx->event, INFINITE); | |||||
| } | |||||
| } | |||||
| } | |||||
| ctx->curbufsize -= pkt->size; | |||||
| return pkt->size; | |||||
| } | |||||
| AVInputFormat dshow_demuxer = { | |||||
| "dshow", | |||||
| NULL_IF_CONFIG_SMALL("DirectShow capture"), | |||||
| sizeof(struct dshow_ctx), | |||||
| NULL, | |||||
| dshow_read_header, | |||||
| dshow_read_packet, | |||||
| dshow_read_close, | |||||
| .flags = AVFMT_NOFILE, | |||||
| }; | |||||
| @@ -0,0 +1,266 @@ | |||||
| /* | |||||
| * DirectShow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #define DSHOWDEBUG 0 | |||||
| #include "libavformat/avformat.h" | |||||
| #define COBJMACROS | |||||
| #include <windows.h> | |||||
| #include <dshow.h> | |||||
| #include <dvdmedia.h> | |||||
| long ff_copy_dshow_media_type(AM_MEDIA_TYPE *dst, const AM_MEDIA_TYPE *src); | |||||
| void ff_print_AM_MEDIA_TYPE(const AM_MEDIA_TYPE *type); | |||||
| void ff_printGUID(const GUID *g); | |||||
| #if DSHOWDEBUG | |||||
| extern const AVClass *ff_dshow_context_class_ptr; | |||||
| #define dshowdebug(...) av_log(&ff_dshow_context_class_ptr, AV_LOG_DEBUG, __VA_ARGS__) | |||||
| #else | |||||
| #define dshowdebug(...) | |||||
| #endif | |||||
| static inline void nothing(void *foo) | |||||
| { | |||||
| } | |||||
| struct GUIDoffset { | |||||
| const GUID *iid; | |||||
| int offset; | |||||
| }; | |||||
| enum dshowDeviceType { | |||||
| VideoDevice = 0, | |||||
| AudioDevice = 1, | |||||
| }; | |||||
| #define DECLARE_QUERYINTERFACE(class, ...) \ | |||||
| long WINAPI \ | |||||
| class##_QueryInterface(class *this, const GUID *riid, void **ppvObject) \ | |||||
| { \ | |||||
| struct GUIDoffset ifaces[] = __VA_ARGS__; \ | |||||
| int i; \ | |||||
| dshowdebug(AV_STRINGIFY(class)"_QueryInterface(%p, %p, %p)\n", this, riid, ppvObject); \ | |||||
| ff_printGUID(riid); \ | |||||
| if (!ppvObject) \ | |||||
| return E_POINTER; \ | |||||
| for (i = 0; i < sizeof(ifaces)/sizeof(ifaces[0]); i++) { \ | |||||
| if (IsEqualGUID(riid, ifaces[i].iid)) { \ | |||||
| void *obj = (void *) ((uint8_t *) this + ifaces[i].offset); \ | |||||
| class##_AddRef(this); \ | |||||
| dshowdebug("\tfound %d with offset %d\n", i, ifaces[i].offset); \ | |||||
| *ppvObject = (void *) obj; \ | |||||
| return S_OK; \ | |||||
| } \ | |||||
| } \ | |||||
| dshowdebug("\tE_NOINTERFACE\n"); \ | |||||
| *ppvObject = NULL; \ | |||||
| return E_NOINTERFACE; \ | |||||
| } | |||||
| #define DECLARE_ADDREF(class) \ | |||||
| unsigned long WINAPI \ | |||||
| class##_AddRef(class *this) \ | |||||
| { \ | |||||
| dshowdebug(AV_STRINGIFY(class)"_AddRef(%p)\t%ld\n", this, this->ref+1); \ | |||||
| return InterlockedIncrement(&this->ref); \ | |||||
| } | |||||
| #define DECLARE_RELEASE(class) \ | |||||
| unsigned long WINAPI \ | |||||
| class##_Release(class *this) \ | |||||
| { \ | |||||
| long ref = InterlockedDecrement(&this->ref); \ | |||||
| dshowdebug(AV_STRINGIFY(class)"_Release(%p)\t%ld\n", this, ref); \ | |||||
| if (!ref) \ | |||||
| class##_Destroy(this); \ | |||||
| return ref; \ | |||||
| } | |||||
| #define DECLARE_DESTROY(class, func) \ | |||||
| void class##_Destroy(class *this) \ | |||||
| { \ | |||||
| dshowdebug(AV_STRINGIFY(class)"_Destroy(%p)\n", this); \ | |||||
| func(this); \ | |||||
| if (this) { \ | |||||
| if (this->vtbl) \ | |||||
| CoTaskMemFree(this->vtbl); \ | |||||
| CoTaskMemFree(this); \ | |||||
| } \ | |||||
| } | |||||
| #define DECLARE_CREATE(class, setup, ...) \ | |||||
| class *class##_Create(__VA_ARGS__) \ | |||||
| { \ | |||||
| class *this = CoTaskMemAlloc(sizeof(class)); \ | |||||
| void *vtbl = CoTaskMemAlloc(sizeof(*this->vtbl)); \ | |||||
| dshowdebug(AV_STRINGIFY(class)"_Create(%p)\n", this); \ | |||||
| if (!this || !vtbl) \ | |||||
| goto fail; \ | |||||
| ZeroMemory(this, sizeof(class)); \ | |||||
| ZeroMemory(vtbl, sizeof(*this->vtbl)); \ | |||||
| this->ref = 1; \ | |||||
| this->vtbl = vtbl; \ | |||||
| if (!setup) \ | |||||
| goto fail; \ | |||||
| dshowdebug("created "AV_STRINGIFY(class)" %p\n", this); \ | |||||
| return this; \ | |||||
| fail: \ | |||||
| class##_Destroy(this); \ | |||||
| dshowdebug("could not create "AV_STRINGIFY(class)"\n"); \ | |||||
| return NULL; \ | |||||
| } | |||||
| #define SETVTBL(vtbl, class, fn) \ | |||||
| do { (vtbl)->fn = (void *) class##_##fn; } while(0) | |||||
| /***************************************************************************** | |||||
| * Forward Declarations | |||||
| ****************************************************************************/ | |||||
| typedef struct libAVPin libAVPin; | |||||
| typedef struct libAVMemInputPin libAVMemInputPin; | |||||
| typedef struct libAVEnumPins libAVEnumPins; | |||||
| typedef struct libAVEnumMediaTypes libAVEnumMediaTypes; | |||||
| typedef struct libAVFilter libAVFilter; | |||||
| /***************************************************************************** | |||||
| * libAVPin | |||||
| ****************************************************************************/ | |||||
| struct libAVPin { | |||||
| IPinVtbl *vtbl; | |||||
| long ref; | |||||
| libAVFilter *filter; | |||||
| IPin *connectedto; | |||||
| AM_MEDIA_TYPE type; | |||||
| IMemInputPinVtbl *imemvtbl; | |||||
| }; | |||||
| long WINAPI libAVPin_QueryInterface (libAVPin *, const GUID *, void **); | |||||
| unsigned long WINAPI libAVPin_AddRef (libAVPin *); | |||||
| unsigned long WINAPI libAVPin_Release (libAVPin *); | |||||
| long WINAPI libAVPin_Connect (libAVPin *, IPin *, const AM_MEDIA_TYPE *); | |||||
| long WINAPI libAVPin_ReceiveConnection (libAVPin *, IPin *, const AM_MEDIA_TYPE *); | |||||
| long WINAPI libAVPin_Disconnect (libAVPin *); | |||||
| long WINAPI libAVPin_ConnectedTo (libAVPin *, IPin **); | |||||
| long WINAPI libAVPin_ConnectionMediaType (libAVPin *, AM_MEDIA_TYPE *); | |||||
| long WINAPI libAVPin_QueryPinInfo (libAVPin *, PIN_INFO *); | |||||
| long WINAPI libAVPin_QueryDirection (libAVPin *, PIN_DIRECTION *); | |||||
| long WINAPI libAVPin_QueryId (libAVPin *, wchar_t **); | |||||
| long WINAPI libAVPin_QueryAccept (libAVPin *, const AM_MEDIA_TYPE *); | |||||
| long WINAPI libAVPin_EnumMediaTypes (libAVPin *, IEnumMediaTypes **); | |||||
| long WINAPI libAVPin_QueryInternalConnections(libAVPin *, IPin **, unsigned long *); | |||||
| long WINAPI libAVPin_EndOfStream (libAVPin *); | |||||
| long WINAPI libAVPin_BeginFlush (libAVPin *); | |||||
| long WINAPI libAVPin_EndFlush (libAVPin *); | |||||
| long WINAPI libAVPin_NewSegment (libAVPin *, REFERENCE_TIME, REFERENCE_TIME, double); | |||||
| long WINAPI libAVMemInputPin_QueryInterface (libAVMemInputPin *, const GUID *, void **); | |||||
| unsigned long WINAPI libAVMemInputPin_AddRef (libAVMemInputPin *); | |||||
| unsigned long WINAPI libAVMemInputPin_Release (libAVMemInputPin *); | |||||
| long WINAPI libAVMemInputPin_GetAllocator (libAVMemInputPin *, IMemAllocator **); | |||||
| long WINAPI libAVMemInputPin_NotifyAllocator (libAVMemInputPin *, IMemAllocator *, WINBOOL); | |||||
| long WINAPI libAVMemInputPin_GetAllocatorRequirements(libAVMemInputPin *, ALLOCATOR_PROPERTIES *); | |||||
| long WINAPI libAVMemInputPin_Receive (libAVMemInputPin *, IMediaSample *); | |||||
| long WINAPI libAVMemInputPin_ReceiveMultiple (libAVMemInputPin *, IMediaSample **, long, long *); | |||||
| long WINAPI libAVMemInputPin_ReceiveCanBlock (libAVMemInputPin *); | |||||
| void libAVPin_Destroy(libAVPin *); | |||||
| libAVPin *libAVPin_Create (libAVFilter *filter); | |||||
| void libAVMemInputPin_Destroy(libAVMemInputPin *); | |||||
| /***************************************************************************** | |||||
| * libAVEnumPins | |||||
| ****************************************************************************/ | |||||
| struct libAVEnumPins { | |||||
| IEnumPinsVtbl *vtbl; | |||||
| long ref; | |||||
| int pos; | |||||
| libAVPin *pin; | |||||
| libAVFilter *filter; | |||||
| }; | |||||
| long WINAPI libAVEnumPins_QueryInterface(libAVEnumPins *, const GUID *, void **); | |||||
| unsigned long WINAPI libAVEnumPins_AddRef (libAVEnumPins *); | |||||
| unsigned long WINAPI libAVEnumPins_Release (libAVEnumPins *); | |||||
| long WINAPI libAVEnumPins_Next (libAVEnumPins *, unsigned long, IPin **, unsigned long *); | |||||
| long WINAPI libAVEnumPins_Skip (libAVEnumPins *, unsigned long); | |||||
| long WINAPI libAVEnumPins_Reset (libAVEnumPins *); | |||||
| long WINAPI libAVEnumPins_Clone (libAVEnumPins *, libAVEnumPins **); | |||||
| void libAVEnumPins_Destroy(libAVEnumPins *); | |||||
| libAVEnumPins *libAVEnumPins_Create (libAVPin *pin, libAVFilter *filter); | |||||
| /***************************************************************************** | |||||
| * libAVEnumMediaTypes | |||||
| ****************************************************************************/ | |||||
| struct libAVEnumMediaTypes { | |||||
| IEnumPinsVtbl *vtbl; | |||||
| long ref; | |||||
| int pos; | |||||
| AM_MEDIA_TYPE type; | |||||
| }; | |||||
| long WINAPI libAVEnumMediaTypes_QueryInterface(libAVEnumMediaTypes *, const GUID *, void **); | |||||
| unsigned long WINAPI libAVEnumMediaTypes_AddRef (libAVEnumMediaTypes *); | |||||
| unsigned long WINAPI libAVEnumMediaTypes_Release (libAVEnumMediaTypes *); | |||||
| long WINAPI libAVEnumMediaTypes_Next (libAVEnumMediaTypes *, unsigned long, AM_MEDIA_TYPE **, unsigned long *); | |||||
| long WINAPI libAVEnumMediaTypes_Skip (libAVEnumMediaTypes *, unsigned long); | |||||
| long WINAPI libAVEnumMediaTypes_Reset (libAVEnumMediaTypes *); | |||||
| long WINAPI libAVEnumMediaTypes_Clone (libAVEnumMediaTypes *, libAVEnumMediaTypes **); | |||||
| void libAVEnumMediaTypes_Destroy(libAVEnumMediaTypes *); | |||||
| libAVEnumMediaTypes *libAVEnumMediaTypes_Create(const AM_MEDIA_TYPE *type); | |||||
| /***************************************************************************** | |||||
| * libAVFilter | |||||
| ****************************************************************************/ | |||||
| struct libAVFilter { | |||||
| IBaseFilterVtbl *vtbl; | |||||
| long ref; | |||||
| const wchar_t *name; | |||||
| libAVPin *pin; | |||||
| FILTER_INFO info; | |||||
| FILTER_STATE state; | |||||
| IReferenceClock *clock; | |||||
| enum dshowDeviceType type; | |||||
| void *priv_data; | |||||
| int stream_index; | |||||
| int64_t start_time; | |||||
| void (*callback)(void *priv_data, int index, uint8_t *buf, int buf_size, int64_t time); | |||||
| }; | |||||
| long WINAPI libAVFilter_QueryInterface (libAVFilter *, const GUID *, void **); | |||||
| unsigned long WINAPI libAVFilter_AddRef (libAVFilter *); | |||||
| unsigned long WINAPI libAVFilter_Release (libAVFilter *); | |||||
| long WINAPI libAVFilter_GetClassID (libAVFilter *, CLSID *); | |||||
| long WINAPI libAVFilter_Stop (libAVFilter *); | |||||
| long WINAPI libAVFilter_Pause (libAVFilter *); | |||||
| long WINAPI libAVFilter_Run (libAVFilter *, REFERENCE_TIME); | |||||
| long WINAPI libAVFilter_GetState (libAVFilter *, DWORD, FILTER_STATE *); | |||||
| long WINAPI libAVFilter_SetSyncSource (libAVFilter *, IReferenceClock *); | |||||
| long WINAPI libAVFilter_GetSyncSource (libAVFilter *, IReferenceClock **); | |||||
| long WINAPI libAVFilter_EnumPins (libAVFilter *, IEnumPins **); | |||||
| long WINAPI libAVFilter_FindPin (libAVFilter *, const wchar_t *, IPin **); | |||||
| long WINAPI libAVFilter_QueryFilterInfo(libAVFilter *, FILTER_INFO *); | |||||
| long WINAPI libAVFilter_JoinFilterGraph(libAVFilter *, IFilterGraph *, const wchar_t *); | |||||
| long WINAPI libAVFilter_QueryVendorInfo(libAVFilter *, wchar_t **); | |||||
| void libAVFilter_Destroy(libAVFilter *); | |||||
| libAVFilter *libAVFilter_Create (void *, void *, enum dshowDeviceType); | |||||
| @@ -0,0 +1,141 @@ | |||||
| /* | |||||
| * Directshow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "dshow.h" | |||||
| long ff_copy_dshow_media_type(AM_MEDIA_TYPE *dst, const AM_MEDIA_TYPE *src) | |||||
| { | |||||
| uint8_t *pbFormat = NULL; | |||||
| if (src->cbFormat) { | |||||
| pbFormat = CoTaskMemAlloc(src->cbFormat); | |||||
| if (!pbFormat) | |||||
| return E_OUTOFMEMORY; | |||||
| memcpy(pbFormat, src->pbFormat, src->cbFormat); | |||||
| } | |||||
| *dst = *src; | |||||
| dst->pUnk = NULL; | |||||
| dst->pbFormat = pbFormat; | |||||
| return S_OK; | |||||
| } | |||||
| void ff_printGUID(const GUID *g) | |||||
| { | |||||
| #if DSHOWDEBUG | |||||
| const uint32_t *d = (const uint32_t *) &g->Data1; | |||||
| const uint16_t *w = (const uint16_t *) &g->Data2; | |||||
| const uint8_t *c = (const uint8_t *) &g->Data4; | |||||
| dshowdebug("0x%08x 0x%04x 0x%04x %02x%02x%02x%02x%02x%02x%02x%02x", | |||||
| d[0], w[0], w[1], | |||||
| c[0], c[1], c[2], c[3], c[4], c[5], c[6], c[7]); | |||||
| #endif | |||||
| } | |||||
| static const char *dshow_context_to_name(void *ptr) | |||||
| { | |||||
| return "dshow"; | |||||
| } | |||||
| static const AVClass ff_dshow_context_class = { "DirectShow", dshow_context_to_name }; | |||||
| const AVClass *ff_dshow_context_class_ptr = &ff_dshow_context_class; | |||||
| #define dstruct(pctx, sname, var, type) \ | |||||
| dshowdebug(" "#var":\t%"type"\n", sname->var) | |||||
| #if DSHOWDEBUG | |||||
| static void dump_bih(void *s, BITMAPINFOHEADER *bih) | |||||
| { | |||||
| dshowdebug(" BITMAPINFOHEADER\n"); | |||||
| dstruct(s, bih, biSize, "lu"); | |||||
| dstruct(s, bih, biWidth, "ld"); | |||||
| dstruct(s, bih, biHeight, "ld"); | |||||
| dstruct(s, bih, biPlanes, "d"); | |||||
| dstruct(s, bih, biBitCount, "d"); | |||||
| dstruct(s, bih, biCompression, "lu"); | |||||
| dshowdebug(" biCompression:\t\"%.4s\"\n", | |||||
| (char*) &bih->biCompression); | |||||
| dstruct(s, bih, biSizeImage, "lu"); | |||||
| dstruct(s, bih, biXPelsPerMeter, "lu"); | |||||
| dstruct(s, bih, biYPelsPerMeter, "lu"); | |||||
| dstruct(s, bih, biClrUsed, "lu"); | |||||
| dstruct(s, bih, biClrImportant, "lu"); | |||||
| } | |||||
| #endif | |||||
| void ff_print_AM_MEDIA_TYPE(const AM_MEDIA_TYPE *type) | |||||
| { | |||||
| #if DSHOWDEBUG | |||||
| dshowdebug(" majortype\t"); | |||||
| ff_printGUID(&type->majortype); | |||||
| dshowdebug("\n"); | |||||
| dshowdebug(" subtype\t"); | |||||
| ff_printGUID(&type->subtype); | |||||
| dshowdebug("\n"); | |||||
| dshowdebug(" bFixedSizeSamples\t%d\n", type->bFixedSizeSamples); | |||||
| dshowdebug(" bTemporalCompression\t%d\n", type->bTemporalCompression); | |||||
| dshowdebug(" lSampleSize\t%lu\n", type->lSampleSize); | |||||
| dshowdebug(" formattype\t"); | |||||
| ff_printGUID(&type->formattype); | |||||
| dshowdebug("\n"); | |||||
| dshowdebug(" pUnk\t%p\n", type->pUnk); | |||||
| dshowdebug(" cbFormat\t%lu\n", type->cbFormat); | |||||
| dshowdebug(" pbFormat\t%p\n", type->pbFormat); | |||||
| if (IsEqualGUID(&type->formattype, &FORMAT_VideoInfo)) { | |||||
| VIDEOINFOHEADER *v = (void *) type->pbFormat; | |||||
| dshowdebug(" rcSource: left %ld top %ld right %ld bottom %ld\n", | |||||
| v->rcSource.left, v->rcSource.top, v->rcSource.right, v->rcSource.bottom); | |||||
| dshowdebug(" rcTarget: left %ld top %ld right %ld bottom %ld\n", | |||||
| v->rcTarget.left, v->rcTarget.top, v->rcTarget.right, v->rcTarget.bottom); | |||||
| dshowdebug(" dwBitRate: %lu\n", v->dwBitRate); | |||||
| dshowdebug(" dwBitErrorRate: %lu\n", v->dwBitErrorRate); | |||||
| dshowdebug(" AvgTimePerFrame: %"PRId64"\n", v->AvgTimePerFrame); | |||||
| dump_bih(NULL, &v->bmiHeader); | |||||
| } else if (IsEqualGUID(&type->formattype, &FORMAT_VideoInfo2)) { | |||||
| VIDEOINFOHEADER2 *v = (void *) type->pbFormat; | |||||
| dshowdebug(" rcSource: left %ld top %ld right %ld bottom %ld\n", | |||||
| v->rcSource.left, v->rcSource.top, v->rcSource.right, v->rcSource.bottom); | |||||
| dshowdebug(" rcTarget: left %ld top %ld right %ld bottom %ld\n", | |||||
| v->rcTarget.left, v->rcTarget.top, v->rcTarget.right, v->rcTarget.bottom); | |||||
| dshowdebug(" dwBitRate: %lu\n", v->dwBitRate); | |||||
| dshowdebug(" dwBitErrorRate: %lu\n", v->dwBitErrorRate); | |||||
| dshowdebug(" AvgTimePerFrame: %"PRId64"\n", v->AvgTimePerFrame); | |||||
| dshowdebug(" dwInterlaceFlags: %lu\n", v->dwInterlaceFlags); | |||||
| dshowdebug(" dwCopyProtectFlags: %lu\n", v->dwCopyProtectFlags); | |||||
| dshowdebug(" dwPictAspectRatioX: %lu\n", v->dwPictAspectRatioX); | |||||
| dshowdebug(" dwPictAspectRatioY: %lu\n", v->dwPictAspectRatioY); | |||||
| // dshowdebug(" dwReserved1: %lu\n", v->u.dwReserved1); /* mingw-w64 is buggy and doesn't name unnamed unions */ | |||||
| dshowdebug(" dwReserved2: %lu\n", v->dwReserved2); | |||||
| dump_bih(NULL, &v->bmiHeader); | |||||
| } else if (IsEqualGUID(&type->formattype, &FORMAT_WaveFormatEx)) { | |||||
| WAVEFORMATEX *fx = (void *) type->pbFormat; | |||||
| dshowdebug(" wFormatTag: %u\n", fx->wFormatTag); | |||||
| dshowdebug(" nChannels: %u\n", fx->nChannels); | |||||
| dshowdebug(" nSamplesPerSec: %lu\n", fx->nSamplesPerSec); | |||||
| dshowdebug(" nAvgBytesPerSec: %lu\n", fx->nAvgBytesPerSec); | |||||
| dshowdebug(" nBlockAlign: %u\n", fx->nBlockAlign); | |||||
| dshowdebug(" wBitsPerSample: %u\n", fx->wBitsPerSample); | |||||
| dshowdebug(" cbSize: %u\n", fx->cbSize); | |||||
| } | |||||
| #endif | |||||
| } | |||||
| @@ -0,0 +1,103 @@ | |||||
| /* | |||||
| * DirectShow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "dshow.h" | |||||
| DECLARE_QUERYINTERFACE(libAVEnumMediaTypes, | |||||
| { {&IID_IUnknown,0}, {&IID_IEnumPins,0} }) | |||||
| DECLARE_ADDREF(libAVEnumMediaTypes) | |||||
| DECLARE_RELEASE(libAVEnumMediaTypes) | |||||
| long WINAPI | |||||
| libAVEnumMediaTypes_Next(libAVEnumMediaTypes *this, unsigned long n, | |||||
| AM_MEDIA_TYPE **types, unsigned long *fetched) | |||||
| { | |||||
| int count = 0; | |||||
| dshowdebug("libAVEnumMediaTypes_Next(%p)\n", this); | |||||
| if (!types) | |||||
| return E_POINTER; | |||||
| if (!this->pos && n == 1) { | |||||
| if (!IsEqualGUID(&this->type.majortype, &GUID_NULL)) { | |||||
| AM_MEDIA_TYPE *type = av_malloc(sizeof(AM_MEDIA_TYPE)); | |||||
| ff_copy_dshow_media_type(type, &this->type); | |||||
| *types = type; | |||||
| count = 1; | |||||
| } | |||||
| this->pos = 1; | |||||
| } | |||||
| if (fetched) | |||||
| *fetched = count; | |||||
| if (!count) | |||||
| return S_FALSE; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumMediaTypes_Skip(libAVEnumMediaTypes *this, unsigned long n) | |||||
| { | |||||
| dshowdebug("libAVEnumMediaTypes_Skip(%p)\n", this); | |||||
| if (n) /* Any skip will always fall outside of the only valid type. */ | |||||
| return S_FALSE; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumMediaTypes_Reset(libAVEnumMediaTypes *this) | |||||
| { | |||||
| dshowdebug("libAVEnumMediaTypes_Reset(%p)\n", this); | |||||
| this->pos = 0; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumMediaTypes_Clone(libAVEnumMediaTypes *this, libAVEnumMediaTypes **enums) | |||||
| { | |||||
| libAVEnumMediaTypes *new; | |||||
| dshowdebug("libAVEnumMediaTypes_Clone(%p)\n", this); | |||||
| if (!enums) | |||||
| return E_POINTER; | |||||
| new = libAVEnumMediaTypes_Create(&this->type); | |||||
| if (!new) | |||||
| return E_OUTOFMEMORY; | |||||
| new->pos = this->pos; | |||||
| *enums = new; | |||||
| return S_OK; | |||||
| } | |||||
| static int | |||||
| libAVEnumMediaTypes_Setup(libAVEnumMediaTypes *this, const AM_MEDIA_TYPE *type) | |||||
| { | |||||
| IEnumPinsVtbl *vtbl = this->vtbl; | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, QueryInterface); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, AddRef); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, Release); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, Next); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, Skip); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, Reset); | |||||
| SETVTBL(vtbl, libAVEnumMediaTypes, Clone); | |||||
| if (!type) { | |||||
| this->type.majortype = GUID_NULL; | |||||
| } else { | |||||
| ff_copy_dshow_media_type(&this->type, type); | |||||
| } | |||||
| return 1; | |||||
| } | |||||
| DECLARE_CREATE(libAVEnumMediaTypes, libAVEnumMediaTypes_Setup(this, type), const AM_MEDIA_TYPE *type) | |||||
| DECLARE_DESTROY(libAVEnumMediaTypes, nothing) | |||||
| @@ -0,0 +1,99 @@ | |||||
| /* | |||||
| * DirectShow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "dshow.h" | |||||
| DECLARE_QUERYINTERFACE(libAVEnumPins, | |||||
| { {&IID_IUnknown,0}, {&IID_IEnumPins,0} }) | |||||
| DECLARE_ADDREF(libAVEnumPins) | |||||
| DECLARE_RELEASE(libAVEnumPins) | |||||
| long WINAPI | |||||
| libAVEnumPins_Next(libAVEnumPins *this, unsigned long n, IPin **pins, | |||||
| unsigned long *fetched) | |||||
| { | |||||
| int count = 0; | |||||
| dshowdebug("libAVEnumPins_Next(%p)\n", this); | |||||
| if (!pins) | |||||
| return E_POINTER; | |||||
| if (!this->pos && n == 1) { | |||||
| libAVPin_AddRef(this->pin); | |||||
| *pins = (IPin *) this->pin; | |||||
| count = 1; | |||||
| this->pos = 1; | |||||
| } | |||||
| if (fetched) | |||||
| *fetched = count; | |||||
| if (!count) | |||||
| return S_FALSE; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumPins_Skip(libAVEnumPins *this, unsigned long n) | |||||
| { | |||||
| dshowdebug("libAVEnumPins_Skip(%p)\n", this); | |||||
| if (n) /* Any skip will always fall outside of the only valid pin. */ | |||||
| return S_FALSE; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumPins_Reset(libAVEnumPins *this) | |||||
| { | |||||
| dshowdebug("libAVEnumPins_Reset(%p)\n", this); | |||||
| this->pos = 0; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVEnumPins_Clone(libAVEnumPins *this, libAVEnumPins **pins) | |||||
| { | |||||
| libAVEnumPins *new; | |||||
| dshowdebug("libAVEnumPins_Clone(%p)\n", this); | |||||
| if (!pins) | |||||
| return E_POINTER; | |||||
| new = libAVEnumPins_Create(this->pin, this->filter); | |||||
| if (!new) | |||||
| return E_OUTOFMEMORY; | |||||
| new->pos = this->pos; | |||||
| *pins = new; | |||||
| return S_OK; | |||||
| } | |||||
| static int | |||||
| libAVEnumPins_Setup(libAVEnumPins *this, libAVPin *pin, libAVFilter *filter) | |||||
| { | |||||
| IEnumPinsVtbl *vtbl = this->vtbl; | |||||
| SETVTBL(vtbl, libAVEnumPins, QueryInterface); | |||||
| SETVTBL(vtbl, libAVEnumPins, AddRef); | |||||
| SETVTBL(vtbl, libAVEnumPins, Release); | |||||
| SETVTBL(vtbl, libAVEnumPins, Next); | |||||
| SETVTBL(vtbl, libAVEnumPins, Skip); | |||||
| SETVTBL(vtbl, libAVEnumPins, Reset); | |||||
| SETVTBL(vtbl, libAVEnumPins, Clone); | |||||
| this->pin = pin; | |||||
| this->filter = filter; | |||||
| libAVFilter_AddRef(this->filter); | |||||
| return 1; | |||||
| } | |||||
| DECLARE_CREATE(libAVEnumPins, libAVEnumPins_Setup(this, pin, filter), | |||||
| libAVPin *pin, libAVFilter *filter) | |||||
| DECLARE_DESTROY(libAVEnumPins, nothing) | |||||
| @@ -0,0 +1,196 @@ | |||||
| /* | |||||
| * DirectShow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "dshow.h" | |||||
| DECLARE_QUERYINTERFACE(libAVFilter, | |||||
| { {&IID_IUnknown,0}, {&IID_IBaseFilter,0} }) | |||||
| DECLARE_ADDREF(libAVFilter) | |||||
| DECLARE_RELEASE(libAVFilter) | |||||
| long WINAPI | |||||
| libAVFilter_GetClassID(libAVFilter *this, CLSID *id) | |||||
| { | |||||
| dshowdebug("libAVFilter_GetClassID(%p)\n", this); | |||||
| /* I'm not creating a ClassID just for this. */ | |||||
| return E_FAIL; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_Stop(libAVFilter *this) | |||||
| { | |||||
| dshowdebug("libAVFilter_Stop(%p)\n", this); | |||||
| this->state = State_Stopped; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_Pause(libAVFilter *this) | |||||
| { | |||||
| dshowdebug("libAVFilter_Pause(%p)\n", this); | |||||
| this->state = State_Paused; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_Run(libAVFilter *this, REFERENCE_TIME start) | |||||
| { | |||||
| dshowdebug("libAVFilter_Run(%p) %"PRId64"\n", this, start); | |||||
| this->state = State_Running; | |||||
| this->start_time = start; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_GetState(libAVFilter *this, DWORD ms, FILTER_STATE *state) | |||||
| { | |||||
| dshowdebug("libAVFilter_GetState(%p)\n", this); | |||||
| if (!state) | |||||
| return E_POINTER; | |||||
| *state = this->state; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_SetSyncSource(libAVFilter *this, IReferenceClock *clock) | |||||
| { | |||||
| dshowdebug("libAVFilter_SetSyncSource(%p)\n", this); | |||||
| if (this->clock != clock) { | |||||
| if (this->clock) | |||||
| IReferenceClock_Release(this->clock); | |||||
| this->clock = clock; | |||||
| if (clock) | |||||
| IReferenceClock_AddRef(clock); | |||||
| } | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_GetSyncSource(libAVFilter *this, IReferenceClock **clock) | |||||
| { | |||||
| dshowdebug("libAVFilter_GetSyncSource(%p)\n", this); | |||||
| if (!clock) | |||||
| return E_POINTER; | |||||
| if (this->clock) | |||||
| IReferenceClock_AddRef(this->clock); | |||||
| *clock = this->clock; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_EnumPins(libAVFilter *this, IEnumPins **enumpin) | |||||
| { | |||||
| libAVEnumPins *new; | |||||
| dshowdebug("libAVFilter_EnumPins(%p)\n", this); | |||||
| if (!enumpin) | |||||
| return E_POINTER; | |||||
| new = libAVEnumPins_Create(this->pin, this); | |||||
| if (!new) | |||||
| return E_OUTOFMEMORY; | |||||
| *enumpin = (IEnumPins *) new; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_FindPin(libAVFilter *this, const wchar_t *id, IPin **pin) | |||||
| { | |||||
| libAVPin *found = NULL; | |||||
| dshowdebug("libAVFilter_FindPin(%p)\n", this); | |||||
| if (!id || !pin) | |||||
| return E_POINTER; | |||||
| if (!wcscmp(id, L"In")) { | |||||
| found = this->pin; | |||||
| libAVPin_AddRef(found); | |||||
| } | |||||
| *pin = (IPin *) found; | |||||
| if (!found) | |||||
| return VFW_E_NOT_FOUND; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_QueryFilterInfo(libAVFilter *this, FILTER_INFO *info) | |||||
| { | |||||
| dshowdebug("libAVFilter_QueryFilterInfo(%p)\n", this); | |||||
| if (!info) | |||||
| return E_POINTER; | |||||
| if (this->info.pGraph) | |||||
| IFilterGraph_AddRef(this->info.pGraph); | |||||
| *info = this->info; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_JoinFilterGraph(libAVFilter *this, IFilterGraph *graph, | |||||
| const wchar_t *name) | |||||
| { | |||||
| dshowdebug("libAVFilter_JoinFilterGraph(%p)\n", this); | |||||
| this->info.pGraph = graph; | |||||
| if (name) | |||||
| wcscpy(this->info.achName, name); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVFilter_QueryVendorInfo(libAVFilter *this, wchar_t **info) | |||||
| { | |||||
| dshowdebug("libAVFilter_QueryVendorInfo(%p)\n", this); | |||||
| if (!info) | |||||
| return E_POINTER; | |||||
| *info = wcsdup(L"libAV"); | |||||
| return S_OK; | |||||
| } | |||||
| static int | |||||
| libAVFilter_Setup(libAVFilter *this, void *priv_data, void *callback, | |||||
| enum dshowDeviceType type) | |||||
| { | |||||
| IBaseFilterVtbl *vtbl = this->vtbl; | |||||
| SETVTBL(vtbl, libAVFilter, QueryInterface); | |||||
| SETVTBL(vtbl, libAVFilter, AddRef); | |||||
| SETVTBL(vtbl, libAVFilter, Release); | |||||
| SETVTBL(vtbl, libAVFilter, GetClassID); | |||||
| SETVTBL(vtbl, libAVFilter, Stop); | |||||
| SETVTBL(vtbl, libAVFilter, Pause); | |||||
| SETVTBL(vtbl, libAVFilter, Run); | |||||
| SETVTBL(vtbl, libAVFilter, GetState); | |||||
| SETVTBL(vtbl, libAVFilter, SetSyncSource); | |||||
| SETVTBL(vtbl, libAVFilter, GetSyncSource); | |||||
| SETVTBL(vtbl, libAVFilter, EnumPins); | |||||
| SETVTBL(vtbl, libAVFilter, FindPin); | |||||
| SETVTBL(vtbl, libAVFilter, QueryFilterInfo); | |||||
| SETVTBL(vtbl, libAVFilter, JoinFilterGraph); | |||||
| SETVTBL(vtbl, libAVFilter, QueryVendorInfo); | |||||
| this->pin = libAVPin_Create(this); | |||||
| this->priv_data = priv_data; | |||||
| this->callback = callback; | |||||
| this->type = type; | |||||
| return 1; | |||||
| } | |||||
| DECLARE_CREATE(libAVFilter, libAVFilter_Setup(this, priv_data, callback, type), | |||||
| void *priv_data, void *callback, enum dshowDeviceType type) | |||||
| DECLARE_DESTROY(libAVFilter, nothing) | |||||
| @@ -0,0 +1,361 @@ | |||||
| /* | |||||
| * DirectShow capture interface | |||||
| * Copyright (c) 2010 Ramiro Polla | |||||
| * | |||||
| * 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 | |||||
| */ | |||||
| #include "dshow.h" | |||||
| #include <stddef.h> | |||||
| #define imemoffset offsetof(libAVPin, imemvtbl) | |||||
| DECLARE_QUERYINTERFACE(libAVPin, | |||||
| { {&IID_IUnknown,0}, {&IID_IPin,0}, {&IID_IMemInputPin,imemoffset} }) | |||||
| DECLARE_ADDREF(libAVPin) | |||||
| DECLARE_RELEASE(libAVPin) | |||||
| long WINAPI | |||||
| libAVPin_Connect(libAVPin *this, IPin *pin, const AM_MEDIA_TYPE *type) | |||||
| { | |||||
| dshowdebug("libAVPin_Connect(%p, %p, %p)\n", this, pin, type); | |||||
| /* Input pins receive connections. */ | |||||
| return S_FALSE; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_ReceiveConnection(libAVPin *this, IPin *pin, | |||||
| const AM_MEDIA_TYPE *type) | |||||
| { | |||||
| enum dshowDeviceType devtype = this->filter->type; | |||||
| dshowdebug("libAVPin_ReceiveConnection(%p)\n", this); | |||||
| if (!pin) | |||||
| return E_POINTER; | |||||
| if (this->connectedto) | |||||
| return VFW_E_ALREADY_CONNECTED; | |||||
| ff_print_AM_MEDIA_TYPE(type); | |||||
| if (devtype == VideoDevice) { | |||||
| if (!IsEqualGUID(&type->majortype, &MEDIATYPE_Video)) | |||||
| return VFW_E_TYPE_NOT_ACCEPTED; | |||||
| } else { | |||||
| if (!IsEqualGUID(&type->majortype, &MEDIATYPE_Audio)) | |||||
| return VFW_E_TYPE_NOT_ACCEPTED; | |||||
| } | |||||
| IPin_AddRef(pin); | |||||
| this->connectedto = pin; | |||||
| ff_copy_dshow_media_type(&this->type, type); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_Disconnect(libAVPin *this) | |||||
| { | |||||
| dshowdebug("libAVPin_Disconnect(%p)\n", this); | |||||
| if (this->filter->state != State_Stopped) | |||||
| return VFW_E_NOT_STOPPED; | |||||
| if (!this->connectedto) | |||||
| return S_FALSE; | |||||
| this->connectedto = NULL; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_ConnectedTo(libAVPin *this, IPin **pin) | |||||
| { | |||||
| dshowdebug("libAVPin_ConnectedTo(%p)\n", this); | |||||
| if (!pin) | |||||
| return E_POINTER; | |||||
| if (!this->connectedto) | |||||
| return VFW_E_NOT_CONNECTED; | |||||
| IPin_AddRef(this->connectedto); | |||||
| *pin = this->connectedto; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_ConnectionMediaType(libAVPin *this, AM_MEDIA_TYPE *type) | |||||
| { | |||||
| dshowdebug("libAVPin_ConnectionMediaType(%p)\n", this); | |||||
| if (!type) | |||||
| return E_POINTER; | |||||
| if (!this->connectedto) | |||||
| return VFW_E_NOT_CONNECTED; | |||||
| return ff_copy_dshow_media_type(type, &this->type); | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_QueryPinInfo(libAVPin *this, PIN_INFO *info) | |||||
| { | |||||
| dshowdebug("libAVPin_QueryPinInfo(%p)\n", this); | |||||
| if (!info) | |||||
| return E_POINTER; | |||||
| if (this->filter) | |||||
| libAVFilter_AddRef(this->filter); | |||||
| info->pFilter = (IBaseFilter *) this->filter; | |||||
| info->dir = PINDIR_INPUT; | |||||
| wcscpy(info->achName, L"Capture"); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_QueryDirection(libAVPin *this, PIN_DIRECTION *dir) | |||||
| { | |||||
| dshowdebug("libAVPin_QueryDirection(%p)\n", this); | |||||
| if (!dir) | |||||
| return E_POINTER; | |||||
| *dir = PINDIR_INPUT; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_QueryId(libAVPin *this, wchar_t **id) | |||||
| { | |||||
| dshowdebug("libAVPin_QueryId(%p)\n", this); | |||||
| if (!id) | |||||
| return E_POINTER; | |||||
| *id = wcsdup(L"libAV Pin"); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_QueryAccept(libAVPin *this, const AM_MEDIA_TYPE *type) | |||||
| { | |||||
| dshowdebug("libAVPin_QueryAccept(%p)\n", this); | |||||
| return S_FALSE; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_EnumMediaTypes(libAVPin *this, IEnumMediaTypes **enumtypes) | |||||
| { | |||||
| const AM_MEDIA_TYPE *type = NULL; | |||||
| libAVEnumMediaTypes *new; | |||||
| dshowdebug("libAVPin_EnumMediaTypes(%p)\n", this); | |||||
| if (!enumtypes) | |||||
| return E_POINTER; | |||||
| new = libAVEnumMediaTypes_Create(type); | |||||
| if (!new) | |||||
| return E_OUTOFMEMORY; | |||||
| *enumtypes = (IEnumMediaTypes *) new; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_QueryInternalConnections(libAVPin *this, IPin **pin, | |||||
| unsigned long *npin) | |||||
| { | |||||
| dshowdebug("libAVPin_QueryInternalConnections(%p)\n", this); | |||||
| return E_NOTIMPL; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_EndOfStream(libAVPin *this) | |||||
| { | |||||
| dshowdebug("libAVPin_EndOfStream(%p)\n", this); | |||||
| /* I don't care. */ | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_BeginFlush(libAVPin *this) | |||||
| { | |||||
| dshowdebug("libAVPin_BeginFlush(%p)\n", this); | |||||
| /* I don't care. */ | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_EndFlush(libAVPin *this) | |||||
| { | |||||
| dshowdebug("libAVPin_EndFlush(%p)\n", this); | |||||
| /* I don't care. */ | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVPin_NewSegment(libAVPin *this, REFERENCE_TIME start, REFERENCE_TIME stop, | |||||
| double rate) | |||||
| { | |||||
| dshowdebug("libAVPin_NewSegment(%p)\n", this); | |||||
| /* I don't care. */ | |||||
| return S_OK; | |||||
| } | |||||
| static int | |||||
| libAVPin_Setup(libAVPin *this, libAVFilter *filter) | |||||
| { | |||||
| IPinVtbl *vtbl = this->vtbl; | |||||
| IMemInputPinVtbl *imemvtbl; | |||||
| if (!filter) | |||||
| return 0; | |||||
| imemvtbl = av_malloc(sizeof(IMemInputPinVtbl)); | |||||
| if (!imemvtbl) | |||||
| return 0; | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, QueryInterface); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, AddRef); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, Release); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, GetAllocator); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, NotifyAllocator); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, GetAllocatorRequirements); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, Receive); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, ReceiveMultiple); | |||||
| SETVTBL(imemvtbl, libAVMemInputPin, ReceiveCanBlock); | |||||
| this->imemvtbl = imemvtbl; | |||||
| SETVTBL(vtbl, libAVPin, QueryInterface); | |||||
| SETVTBL(vtbl, libAVPin, AddRef); | |||||
| SETVTBL(vtbl, libAVPin, Release); | |||||
| SETVTBL(vtbl, libAVPin, Connect); | |||||
| SETVTBL(vtbl, libAVPin, ReceiveConnection); | |||||
| SETVTBL(vtbl, libAVPin, Disconnect); | |||||
| SETVTBL(vtbl, libAVPin, ConnectedTo); | |||||
| SETVTBL(vtbl, libAVPin, ConnectionMediaType); | |||||
| SETVTBL(vtbl, libAVPin, QueryPinInfo); | |||||
| SETVTBL(vtbl, libAVPin, QueryDirection); | |||||
| SETVTBL(vtbl, libAVPin, QueryId); | |||||
| SETVTBL(vtbl, libAVPin, QueryAccept); | |||||
| SETVTBL(vtbl, libAVPin, EnumMediaTypes); | |||||
| SETVTBL(vtbl, libAVPin, QueryInternalConnections); | |||||
| SETVTBL(vtbl, libAVPin, EndOfStream); | |||||
| SETVTBL(vtbl, libAVPin, BeginFlush); | |||||
| SETVTBL(vtbl, libAVPin, EndFlush); | |||||
| SETVTBL(vtbl, libAVPin, NewSegment); | |||||
| this->filter = filter; | |||||
| return 1; | |||||
| } | |||||
| DECLARE_CREATE(libAVPin, libAVPin_Setup(this, filter), libAVFilter *filter) | |||||
| DECLARE_DESTROY(libAVPin, nothing) | |||||
| /***************************************************************************** | |||||
| * libAVMemInputPin | |||||
| ****************************************************************************/ | |||||
| long WINAPI | |||||
| libAVMemInputPin_QueryInterface(libAVMemInputPin *this, const GUID *riid, | |||||
| void **ppvObject) | |||||
| { | |||||
| libAVPin *pin = (libAVPin *) ((uint8_t *) this - imemoffset); | |||||
| dshowdebug("libAVMemInputPin_QueryInterface(%p)\n", this); | |||||
| return libAVPin_QueryInterface(pin, riid, ppvObject); | |||||
| } | |||||
| unsigned long WINAPI | |||||
| libAVMemInputPin_AddRef(libAVMemInputPin *this) | |||||
| { | |||||
| libAVPin *pin = (libAVPin *) ((uint8_t *) this - imemoffset); | |||||
| dshowdebug("libAVMemInputPin_AddRef(%p)\n", this); | |||||
| return libAVPin_AddRef(pin); | |||||
| } | |||||
| unsigned long WINAPI | |||||
| libAVMemInputPin_Release(libAVMemInputPin *this) | |||||
| { | |||||
| libAVPin *pin = (libAVPin *) ((uint8_t *) this - imemoffset); | |||||
| dshowdebug("libAVMemInputPin_Release(%p)\n", this); | |||||
| return libAVPin_Release(pin); | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_GetAllocator(libAVMemInputPin *this, IMemAllocator **alloc) | |||||
| { | |||||
| dshowdebug("libAVMemInputPin_GetAllocator(%p)\n", this); | |||||
| return VFW_E_NO_ALLOCATOR; | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_NotifyAllocator(libAVMemInputPin *this, IMemAllocator *alloc, | |||||
| WINBOOL rdwr) | |||||
| { | |||||
| dshowdebug("libAVMemInputPin_NotifyAllocator(%p)\n", this); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_GetAllocatorRequirements(libAVMemInputPin *this, | |||||
| ALLOCATOR_PROPERTIES *props) | |||||
| { | |||||
| dshowdebug("libAVMemInputPin_GetAllocatorRequirements(%p)\n", this); | |||||
| return E_NOTIMPL; | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_Receive(libAVMemInputPin *this, IMediaSample *sample) | |||||
| { | |||||
| libAVPin *pin = (libAVPin *) ((uint8_t *) this - imemoffset); | |||||
| enum dshowDeviceType devtype = pin->filter->type; | |||||
| void *priv_data; | |||||
| uint8_t *buf; | |||||
| int buf_size; | |||||
| int index; | |||||
| int64_t curtime; | |||||
| dshowdebug("libAVMemInputPin_Receive(%p)\n", this); | |||||
| if (!sample) | |||||
| return E_POINTER; | |||||
| if (devtype == VideoDevice) { | |||||
| /* PTS from video devices is unreliable. */ | |||||
| IReferenceClock *clock = pin->filter->clock; | |||||
| IReferenceClock_GetTime(clock, &curtime); | |||||
| } else { | |||||
| int64_t dummy; | |||||
| IMediaSample_GetTime(sample, &curtime, &dummy); | |||||
| curtime += pin->filter->start_time; | |||||
| } | |||||
| buf_size = IMediaSample_GetActualDataLength(sample); | |||||
| IMediaSample_GetPointer(sample, &buf); | |||||
| priv_data = pin->filter->priv_data; | |||||
| index = pin->filter->stream_index; | |||||
| pin->filter->callback(priv_data, index, buf, buf_size, curtime); | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_ReceiveMultiple(libAVMemInputPin *this, | |||||
| IMediaSample **samples, long n, long *nproc) | |||||
| { | |||||
| int i; | |||||
| dshowdebug("libAVMemInputPin_ReceiveMultiple(%p)\n", this); | |||||
| for (i = 0; i < n; i++) | |||||
| libAVMemInputPin_Receive(this, samples[i]); | |||||
| *nproc = n; | |||||
| return S_OK; | |||||
| } | |||||
| long WINAPI | |||||
| libAVMemInputPin_ReceiveCanBlock(libAVMemInputPin *this) | |||||
| { | |||||
| dshowdebug("libAVMemInputPin_ReceiveCanBlock(%p)\n", this); | |||||
| /* I swear I will not block. */ | |||||
| return S_FALSE; | |||||
| } | |||||
| void | |||||
| libAVMemInputPin_Destroy(libAVMemInputPin *this) | |||||
| { | |||||
| libAVPin *pin = (libAVPin *) ((uint8_t *) this - imemoffset); | |||||
| dshowdebug("libAVMemInputPin_Destroy(%p)\n", this); | |||||
| return libAVPin_Destroy(pin); | |||||
| } | |||||
| @@ -29,8 +29,6 @@ | |||||
| * Remove this when MinGW incorporates them. */ | * Remove this when MinGW incorporates them. */ | ||||
| #define HWND_MESSAGE ((HWND)-3) | #define HWND_MESSAGE ((HWND)-3) | ||||
| #define BI_RGB 0 | |||||
| /* End of missing MinGW defines */ | /* End of missing MinGW defines */ | ||||
| struct vfw_ctx { | struct vfw_ctx { | ||||