gpt4 book ai didi

delphi - 在 DirectShow Filter 的 EnumPins 方法(Delphi/DSPACK)中处理 NULL ppPins 参数的正确方法?

转载 作者:行者123 更新时间:2023-12-01 18:30:36 26 4
gpt4 key购买 nike

我有一个在 Windows XP/32 计算机上使用 DSPACK DirectShow 组件库用 Delphi 6 编写的自定义推送源过滤器。在编码过程中,在我打开范围检查后,我遇到了属于 DSPACK (BaseClass.pas) 的 BaseClass 单元中发生范围检查错误的问题。 em> 在我的应用程序的编译器选项中。

在我的 EnumPins() 方法期间发生错误。请注意,此方法驻留在 DSPACK 的 BaseClass 单元中,而不是我的应用程序中。我跟踪了这​​个问题,发现当我构建的使用我的过滤器的过滤器图表播放过滤器时,就会出现问题。请注意,我的过滤器作为私有(private)未注册过滤器而不是外部 AX 直接合并到我的应用程序中。当DirectShow调用基类方法TBCEnumPins.Next()时,如果ppPins参数为NIL,则会发生范围检查错误。由于我不是 DirectShow 专家,因此我不确定在不干扰 DirectShow 引脚枚举过程的正确流程的情况下修复此错误的正确方法是什么。相反,如果它是一个不可忽略的真实错误条件,那么我需要知道在此事件中抛出的正确异常或返回的 HRESULT 代码是什么。谁能告诉我为 NIL ppPins 参数调整此代码的正确方法?完整的方法代码如下,其中突出显示了发生范围检查错误的行:

function TBCEnumPins.Next(cPins: ULONG; out ppPins: IPin; pcFetched: PULONG): HRESULT;
type
TPointerDynArray = array of Pointer;
TIPinDynArray = array of IPin;
var
Fetched: cardinal;
RealPins: integer;
Pin: TBCBasePin;
begin
// ATI: Debugging range check error.
try
if pcFetched <> nil then
pcFetched^ := 0
else
if (cPins>1) then
begin
result := E_INVALIDARG;
exit;
end;
Fetched := 0; // increment as we get each one.

// Check we are still in sync with the filter
// If we are out of sync, we should refresh the enumerator.
// This will reset the position and update the other members, but
// will not clear cache of pins we have already returned.
if AreWeOutOfSync then
Refresh;

// Calculate the number of available pins
RealPins := min(FPinCount - FPosition, cPins);
if RealPins = 0 then
begin
result := S_FALSE;
exit;
end;

{ Return each pin interface NOTE GetPin returns CBasePin * not addrefed
so we must QI for the IPin (which increments its reference count)
If while we are retrieving a pin from the filter an error occurs we
assume that our internal state is stale with respect to the filter
(for example someone has deleted a pin) so we
return VFW_E_ENUM_OUT_OF_SYNC }

while RealPins > 0 do
begin
// Get the next pin object from the filter */
inc(FPosition);
Pin := FFilter.GetPin(FPosition-1);
if Pin = nil then
begin
// If this happend, and it's not the first time through, then we've got a problem,
// since we should really go back and release the iPins, which we have previously
// AddRef'ed.
ASSERT(Fetched = 0);
result := VFW_E_ENUM_OUT_OF_SYNC;
exit;
end;

// We only want to return this pin, if it is not in our cache
if FPinCache.IndexOf(Pin) = -1 then
begin
// From the object get an IPin interface
TPointerDynArray(@ppPins)[Fetched] := nil; // <<<<<< THIS IS WHERE THE RANGE CHECK ERROR OCCURS.
TIPinDynArray(@ppPins)[Fetched] := Pin;
inc(Fetched);
FPinCache.Add(Pin);
dec(RealPins);
end;
end; // while RealPins > 0 do

if (pcFetched <> nil) then pcFetched^ := Fetched;

if (cPins = Fetched) then result := NOERROR else result := S_FALSE;
except
On E: Exception do
begin
OutputDebugString(PChar(
'(TBCEnumPins.Next) Exception class name(' + E.ClassName + ') message: ' + E.Message
));

raise;
end;
end;
end;

更新:从技术角度来看,DSPACK 代码似乎是合理的,但从编码习惯的角度来看有点奇怪,并且其结构方式与范围检查不兼容。通过 ppPins“out”参数传入的 NIL 映射到调用者传递给 TBCEnumPins.Next() 作为 ppPins 参数的目标缓冲区。例如,下面的代码来自此页面:

http://tanvon.wordpress.com/2008/09/07/enumerating-the-directshow-filter-pin/

该页面上的以下代码与 DirectShow Filter 交互以枚举 Filter 的引脚:

IEnumPins * pEP;
pSF->EnumPins(&pEP);
IPin * pOutPin;
while(pEP->Next(1,&pOutPin,0) == S_OK)
{
PIN_DIRECTION pDir;
pOutPin->QueryDirection(&pDir);
if(pDir == PINDIR_OUTPUT)
break;// success
pOutPin->Release();
}
pEP->Release();

通过告诉 Next() 方法要检索多少个引脚,TBCEnumPins.Next() 方法代码及其不寻常的动态数组转换是安全的,因为它只会复制到 ppPins“out”参数中Next() 函数“cPins”参数中请求的引脚。只要调用者传递一个可以保存“cPins”中请求的引脚数量的目标缓冲区,一切就可以正常工作(只要范围检查关闭)。请注意,在这种情况下,名为“outPin”的 IPin 变量是目标缓冲区。如果打开范围检查,则会发生范围检查错误,因为 Delphi 将 NIL 视为零长度数组。

最佳答案

我对您对 ppPins 参数的声明以及对 TPointerDynArrayTIPinDynArray 的强制转换表示怀疑。

如果您假装传入的数组是 Delphi 动态数组,而实际上它不是,则会出现范围错误。 Delphi 动态数组有一个位于数组第一项之前的内存块,用于标识数组的引用计数及其长度。由于您的数组中缺少此 block ,因此欺骗编译器相信它在那里将导致您看到的错误。

我会这样做,如下所示。这与 MSDN 中使用的声明相匹配。您必须使用一些指针算术,但我相信这样做更容易,因为您的代码现在很容易与您可以在网上找到的任何 C++ 示例相关联。

type
PIPin = ^IPin;

function TBCEnumPins.Next(cPins: ULONG; ppPins: PIPin; pcFetched: PULONG): HRESULT;
begin
...
// this is the meat of the loop, assign to the output array, and increment
Pointer(ppPins)^ := nil; // initialise the intf ref to avoid bogus _Release
ppPins^ := Pin;
inc(ppPins);
...
end;

关于delphi - 在 DirectShow Filter 的 EnumPins 方法(Delphi/DSPACK)中处理 NULL ppPins 参数的正确方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7935896/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com