akvirtualcamera/dshow/VirtualCamera/src/enumpins.cpp

187 lines
4 KiB
C++
Raw Normal View History

/* akvirtualcamera, virtual camera for Mac and Windows.
* Copyright (C) 2020 Gonzalo Exequiel Pedone
*
* akvirtualcamera is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* akvirtualcamera 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 General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with akvirtualcamera. If not, see <http://www.gnu.org/licenses/>.
*
* Web-Site: http://webcamoid.github.io/
*/
#include <vector>
#include <dshow.h>
#include "enumpins.h"
#include "pin.h"
#include "PlatformUtils/src/utils.h"
#include "VCamUtils/src/utils.h"
#define AK_CUR_INTERFACE "EnumPins"
namespace AkVCam
{
class EnumPinsPrivate
{
public:
std::vector<IPin *> m_pins;
size_t m_position {0};
bool m_changed {false};
};
}
AkVCam::EnumPins::EnumPins():
CUnknown(this, IID_IEnumPins)
{
this->d = new EnumPinsPrivate;
}
AkVCam::EnumPins::EnumPins(const EnumPins &other):
CUnknown(this, IID_IEnumPins)
{
this->d = new EnumPinsPrivate;
this->d->m_position = other.d->m_position;
this->d->m_changed = other.d->m_changed;
for (auto &pin: other.d->m_pins) {
this->d->m_pins.push_back(pin);
this->d->m_pins.back()->AddRef();
}
}
AkVCam::EnumPins::~EnumPins()
{
for (auto &pin: this->d->m_pins)
pin->Release();
delete this->d;
}
size_t AkVCam::EnumPins::count() const
{
return this->d->m_pins.size();
}
void AkVCam::EnumPins::addPin(IPin *pin, bool changed)
{
this->d->m_pins.push_back(pin);
this->d->m_pins.back()->AddRef();
this->d->m_changed = changed;
}
void AkVCam::EnumPins::removePin(IPin *pin, bool changed)
{
for (auto it = this->d->m_pins.begin(); it != this->d->m_pins.end(); it++)
if (*it == pin) {
this->d->m_pins.erase(it);
pin->Release();
this->d->m_changed = changed;
break;
}
}
void AkVCam::EnumPins::setBaseFilter(BaseFilter *baseFilter)
{
for (auto &pin: this->d->m_pins) {
auto akPin = static_cast<Pin *>(pin);
akPin->setBaseFilter(baseFilter);
}
}
HRESULT AkVCam::EnumPins::Next(ULONG cPins, IPin **ppPins, ULONG *pcFetched)
{
AkLogMethod();
if (pcFetched)
*pcFetched = 0;
if (!cPins)
return E_INVALIDARG;
if (!ppPins)
return E_POINTER;
memset(ppPins, 0, cPins * sizeof(IPin *));
if (this->d->m_changed) {
this->d->m_changed = false;
return VFW_E_ENUM_OUT_OF_SYNC;
}
ULONG fetched = 0;
for (;
fetched < cPins
&& this->d->m_position < this->d->m_pins.size();
fetched++, this->d->m_position++) {
*ppPins = this->d->m_pins[this->d->m_position];
(*ppPins)->AddRef();
if (*ppPins)
ppPins++;
}
if (pcFetched)
*pcFetched = fetched;
return fetched == cPins? S_OK: S_FALSE;
}
HRESULT AkVCam::EnumPins::Skip(ULONG cPins)
{
AkLogMethod();
AkLoggerLog("Skip ", cPins, " pins");
if (this->d->m_changed) {
this->d->m_changed = false;
return VFW_E_ENUM_OUT_OF_SYNC;
}
auto position = this->d->m_position + cPins;
if (position > this->d->m_pins.size())
return S_FALSE;
this->d->m_position = position;
return S_OK;
}
HRESULT AkVCam::EnumPins::Reset()
{
AkLogMethod();
this->d->m_position = 0;
return S_OK;
}
HRESULT AkVCam::EnumPins::Clone(IEnumPins **ppEnum)
{
AkLogMethod();
if (!ppEnum)
return E_POINTER;
if (this->d->m_changed) {
this->d->m_changed = false;
return VFW_E_ENUM_OUT_OF_SYNC;
}
*ppEnum = new EnumPins(*this);
(*ppEnum)->AddRef();
return S_OK;
}