Пин-код пользовательского источника DirectShow

Я новичок в DirectShow. Я, как и многие другие, пытаюсь создать решение для потоковой передачи P2P на основе сокетов для карточной игры на основе WPF. Я хочу, чтобы каждый игрок мог видеть друг друга через маленькие видеоокна.

Мои вопросы двоякие. Во-первых, как снизить частоту дискретизации кадров и разрешение? Я считаю, что 320x200 x 15-20 кадров в секунду должно быть достаточно. Я использую обратный вызов SampleGrabber для захвата данных кадра и отправки их через сокет; который на самом деле работает без сжатия при разрешении 640x480.

Мой второй вопрос заключается в том, что, поскольку каждый кадр содержит 921 600 байт, это действительно увязает, и я получаю очень медленный рендеринг только через мою локальную сеть, подключенную к Wi-Fi. Я добавил простое сжатие MJPEG (желая позже переключиться на h.264) и заметил, что байты упали примерно до 330-360 КБ. Не плохое улучшение.

На принимающей стороне Нужно ли мне создавать пользовательский пин-код источника DirectShow, чтобы обслуживать байты, полученные из сокета, чтобы я мог подключить декодер и отображать байты в окне?

Я просто хотел сначала спросить об этом, так как кажется, что нужно много работать, чтобы создать новый COM-объект (не делал этого примерно 15 лет!), зарегистрировать его и использовать/отладить.

Возможно, есть другой способ?

Кроме того, если это путь, я должен использовать SampleGrabber на принимающей стороне и создать BitmapSource из распакованных байтов, или я должен позволить DirectShow создать дочернее окно? Дело в том, что я хочу иметь более одного другого игрока, и я устанавливаю дополнительный байт в сокете, чтобы указать, в какой позиции таблицы они находятся. Как я могу отображать каждую позицию по очереди?


person NickV    schedule 29.10.2013    source источник


Ответы (1)


Для тех, кому интересно, вот как вы устанавливаете разрешение и добавляете кодировщик/компрессор:

        // Create a graph builder
        int hr = captureGraphBuilder.SetFiltergraph(graphBuilder);

        // Find a capture device (WebCam) and attach it to the graph
        sourceFilter = FindCaptureDevice();
        hr = graphBuilder.AddFilter(sourceFilter, "Video Capture");

        // Get the source output Pin
        IPin sourcePin = DsFindPin.ByDirection((IBaseFilter)sourceFilter, PinDirection.Output, 0);
        IAMStreamConfig sc = (IAMStreamConfig)sourcePin;
        int count;
        int size;
        sc.GetNumberOfCapabilities(out count, out size);
        VideoInfoHeader v;
        AMMediaType media2 = null;
        IntPtr memPtr = Marshal.AllocCoTaskMem(size);
        for (int i = 0; i < count; ++i)
        {
            sc.GetStreamCaps(i, out media2, memPtr);

            v = (VideoInfoHeader)Marshal.PtrToStructure(media2.formatPtr, typeof(VideoInfoHeader));
            // Break when width is 160
            if (v.BmiHeader.Width == 160)
                break;
        }

        // Set the new media format t0 160 x 120
        hr = sc.SetFormat(media2);
        Marshal.FreeCoTaskMem(memPtr);
        DsUtils.FreeAMMediaType(media2);

        // Create a FramGrabber
        IBaseFilter grabberF = (IBaseFilter)new SampleGrabber();
        ISampleGrabber grabber = (ISampleGrabber)grabberF;

        // Set the media type
        var media = new AMMediaType
        {
            majorType = MediaType.Video,
            subType = MediaSubType.MJPG
            //subType = MediaSubType.RGB24
        };

        // The media sub type will be MJPG
        hr = grabber.SetMediaType(media);
        DsUtils.FreeAMMediaType(media);
        hr = grabber.SetCallback(this, 1);
        hr = graphBuilder.AddFilter(grabberF, "Sample Grabber");
        IPin grabberPin = DsFindPin.ByDirection(grabberF, PinDirection.Input, 0);

        // Get the MPEG compressor
        Guid iid = typeof(IBaseFilter).GUID;
        object compressor = null;
        foreach (DsDevice device in DsDevice.GetDevicesOfCat(FilterCategory.VideoCompressorCategory))//.MediaEncoderCategory))
        {
            if (device.Name == "MJPEG Compressor")
            {
                device.Mon.BindToObject(null, null, ref iid, out compressor);
                hr = graphBuilder.AddFilter((IBaseFilter)compressor, "Compressor");
                break;
            }
            string name = device.Name;
        }

        // This also works!
        //IBaseFilter enc = (IBaseFilter)new MJPGEnc();
        //graphBuilder.AddFilter(enc, "MJPEG Encoder");

        // Get the input and out pins of the compressor
        IBaseFilter enc = (IBaseFilter)compressor;
        IPin encPinIn = DsFindPin.ByDirection(enc, PinDirection.Input, 0);
        IPin encPinOut = DsFindPin.ByDirection(enc, PinDirection.Output, 0);

        // Attach the pins: source to input, output to grabber
        hr = graphBuilder.Connect(sourcePin, encPinIn);
        hr = graphBuilder.Connect(encPinOut, grabberPin);

        // Free the pin resources
        Marshal.ReleaseComObject(sourcePin);
        Marshal.ReleaseComObject(enc);
        Marshal.ReleaseComObject(encPinIn);
        Marshal.ReleaseComObject(encPinOut);
        Marshal.ReleaseComObject(grabberPin);

        // Create a render stream
        hr = captureGraphBuilder.RenderStream(PinCategory.Preview, MediaType.Video, sourceFilter, null, grabberF);
        Marshal.ReleaseComObject(sourceFilter);
        Configure(grabber);
person NickV    schedule 06.11.2013