Re: threads/marshalling/callbacks
"Jason S" <jmsachs@gmail.com> wrote in message
news:1178231765.593529.309230@y5g2000hsa.googlegroups.com
On May 3, 1:44 pm, "Igor Tandetnik" <itandet...@mvps.org> wrote:
I'm confused. You have the interface pointer (passed to you in
Advise). What's stopping you from making a call on this poiner? What
exactly seems to be the problem?
The thing that was confusing me was who is able to do the Notify()
call (e.g. which thread is the "you" in your above sentences)
If you don't do explicit marshaling, then it would be the thread (to be
precise, the apartment) that created the object on which Advise was
called. In other words, the apartment in the server process that
originally received the sink pointer.
how to
minimize unnecessary marshalling, or even if I should try to minimize
unnecessary marshalling.
Marshalling is fairly expensive, but you only need to marshal once (if
you choose to do it at all), when Advise is called.
When I think about which thread in server S should be making an
outgoing COM call, I am still a little confused. I have at least two
threads:
1) Worker thread SW is probably not the one to do it. If the outgoing
Notify() calls are inter-apartment, or there are multiple Notify()
recipients, then that will likely take too long.
Yes, I believe I made this point, too.
2) Main server message pump thread (what is the standard terminology
for a COM server's main thread?) SA, which was in the apartment which
got the Advise() call. Ideally it would be the one to make the
outgoing call since it already has an interface pointer. But then I
have to figure out how to work out a method for SW to communicate to
SA (PostMessage? SetEvent()? whatever it is has to be really quick so
SW doesn't get blocked for long)
Personally, I suggest PostMessage from SW to SA. Note that PostMessage
doesn't block the caller at all (except for the brief time it takes to
place the message into the queue).
I'm using ATL and I'm not quite sure how to work that into the
EXE server's main message loop without screwing something up.
Well, if you go PostMessage route, you don't need to change the message
loop at all. It already calls GetMessage and DispatchMessage, right? All
you need is to create a window before starting the pump, and to
communicate its HWND handle to SW so it can call PostMessage on it. And,
of course, have this window actually handle the message.
3) Another worker thread in apartment SB whose sole purpose is to make
the outgoing Notify() calls & can sit in a WaitXXX() loop.
If you go this route, make sure SB joins MTA, not STA. STA thread
_must_, I repeat _must_, run a message pump.
Alternatively, you can have a message pump based on
MsgWaitForMultipleObjects instead of GetMessage. This function can wait
on handles and monitor the message queue at the same time. It's somewhat
tricky to use right, be careful.
Personally, I don't see the point of creating yet another thread. If you
insist on using event handles, SA can run a
MsgWaitForMultipleObjects-based message pump just as well.
I suggest you keep it simple and just have the worker thread
post a user-defined message to a hidden window created by SA.
Okay, if I understand this correctly, you're referring to my approach
#2 above, using PostMessage rather than mucking about with SetEvent.
Correct.
So my worker thread should call PostMessage(hWndEventHandler, WM_XXXX,
wParamOfMyChoice, lParamOfMyChoice) where WM_XXXX is a user-defined
message (do I have to call RegisterWindowMessage() or can I just do
WM_USER+N?)
WM_USER is fine.
and hWndEventHandler is some kind of hidden window that I
create somehow, that calls pCallbackSink->Notify() when it receives
the WM_XXXX message....
Correct.
what's the easiest way to set up a hidden
window using WTL in conjunction with the basic ATL COM EXE server?
Write a class derived from CWindowImpl.
I've used ATL for COM servers & WTL for an application GUI, but have
never used them together. Is this faq what you mean:
http://vcfaq.mvps.org/com/11.htm
Yes, this article uses the same idea, but its approach is a lot more
flexible than the minimum you need. Studying the code should prove to be
instructive.
--
With best wishes,
Igor Tandetnik
With sufficient thrust, pigs fly just fine. However, this is not
necessarily a good idea. It is hard to be sure where they are going to
land, and it could be dangerous sitting under them as they fly
overhead. -- RFC 1925