gpt4 book ai didi

c++ - 调用 IConnectionPoint::Advise 导致崩溃

转载 作者:行者123 更新时间:2023-11-30 03:52:19 24 4
gpt4 key购买 nike

在我的项目中,我编写了一个将在服务(COM 服务器)中调用的组件。还有一个进程会获取这个组件的接口(interface),并通过连接点注册一个回调接口(interface)。因此服务可以使用回调接口(interface)做一些反馈。

但是我发现当我使用 IConnectionPoint::Advise 注册我的回调接口(interface)时会导致崩溃。

我手动实现连接点。这是我的部分代码:

class ATL_NO_VTABLE CUploadManager :
public CComObjectRootEx<CComMultiThreadModel>,
public CComCoClass<CUploadManager, &CLSID_UploadManager>,
public IUploadManager,
public IConnectionPointContainer,
public IConnectionPoint

...

// IConnectionPointContainer Functions
STDMETHODIMP EnumConnectionPoints(IEnumConnectionPoints **ppEnum);
STDMETHODIMP FindConnectionPoint(REFIID riid, IConnectionPoint **ppCP);

// IConnectionPoint Functions
STDMETHODIMP GetConnectionInterface(IID *pIID);
STDMETHODIMP GetConnectionPointContainer(IConnectionPointContainer **ppCPC);
STDMETHODIMP Advise(IUnknown *pUnkSink, DWORD *pdwCookie);
STDMETHODIMP Unadvise(DWORD dwCookie);
STDMETHODIMP EnumConnections(IEnumConnections **ppEnum);

在客户端中,我已经获取了组件接口(interface),然后我获取了连接点并尝试调用函数 Advise:

...
CComPtr<IConnectionPointContainer> pCPC;
hr = pUnknown->QueryInterface(IID_IConnectionPointContainer, (LPVOID *)&pCPC);
if (FAILED(hr))
{
return ;
}

CComPtr<IConnectionPoint> pConnectionPoint;
hr = pCPC->FindConnectionPoint(__uuidof(_IEvents), &pConnectionPoint);
if (SUCCEEDED(hr))
{
DWORD dwCookie = 0;
CDataCallback* pCallback = new CDataCallback();
IUnknown* pUnknown = NULL;
pCallback->QueryInterface(IID_IUnknown, (void**)&pUnknown);
hr = pConnectionPoint->Advise(pUnknown, &dwCookie);

...

所有接口(interface)指针都可以成功获取,但是当我调用Advise时,发生了崩溃。奇怪的是崩溃并没有发生在Advise实现中,似乎是发生在RPC调用过程中,调用栈是这样的:

003d0070() <----crash here
combase.dll!76eea3ab()
[Frames below may be incorrect and/or missing, no symbols loaded for combase.dll]
combase.dll!76ee9d00()
rpcrt4.dll!7612a53d()
mfc90ud.dll!CDialog::OnCmdMsg
...

但是如果Advise的第一个参数为NULL,则可以执行Advise函数。

这是客户端实现的Sink代码:

class CDataCallback : public _IEvents
{
public:
CDataCallback() {}
~CDataCallback() {}

HRESULT __stdcall QueryInterface(REFIID iid, LPVOID* ppInterface)
{
*ppInterface = this;
return S_OK;
}

ULONG STDMETHODCALLTYPE AddRef()
{
return 1;
}

ULONG STDMETHODCALLTYPE Release()
{
return 0;
}

HRESULT __stdcall TestFunc()
{
...
return S_OK;
}
};

接口(interface) _IEvents 在组件中生成,将用作连接点中的回调。我只是用这个类来测试连接点,所以实现起来很简单。

我不知道这次崩溃是如何发生的。感谢您的帮助!

最佳答案

问题很可能出在您的 CDataCallback 类中。它的 QueryInterface 实现是不安全的,并且会错误地返回接口(interface)指针。

class CDataCallback : public _IEvents

HRESULT __stdcall QueryInterface(REFIID iid, LPVOID* ppInterface)
{
*ppInterface = this;
return S_OK;
}

如果请求的接口(interface)不是_IEvents,您必须检查iid 并返回E_NOINTERFACE。基本上你可以在那里设置一个断点,看看在崩溃之前你是否在那里有一个调用并检查参数。

您的接收器可能会被查询其他接口(interface)(IMarhsal 等),您必须正确地表明您没有实现它们,而不是返回不匹配的指针,使用它会导致未定义的行为。

关于c++ - 调用 IConnectionPoint::Advise 导致崩溃,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30750806/

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