Re: A _very_ basic question about WM_TIMER and threading
On Sep 3, 3:30 pm, "andrew" <t...@test.com> wrote:
Hi,
If I create a timer by calling CWnd::SetTimer() (e.g. in my
CView::OnInitialUpdate()) why do I need to deal with re-entrancy issues i=
n
the corresponding OnTimer() handler? I thought that all calls to OnTim=
er()
handlers execute on the same thread (main GUI thread?). How is it poss=
ible
then for the handler to be called while the thread is already executing t=
he
handler?
I've run into this problem recently while debugging my app.
I'd welcome any pointers and/or explanations.
Thanks.
As already noted here, your handler code probably spun a message loop
(through e.g. a modal dialog box, call to a COM component in another
apartment etc).
What you could try is to kill/re-enable a timer inside OnTimer.
Imagine (warning: compiled with head-compiler and tried with head-
debugger ;-) ):
class CTempTimerDisable
{
public:
CTempTimerDisable(CWnd* pWnd, UINT nTimer, UINT nPeriod) :
m_pWnd(pWnd), m_nTimer(nTimer), m_nPeriod(nPeriod)
{
VERIFY(m_pWnd->KillTimer(m_nTimer));
}
~CTempTimerDisable()
{
VERIFY(m_pWnd->SetTimer(m_nTimer, m_nPeriod) == m_nTimer);
}
// Maybe useful: void DisablePermanently() { /*set a flag to suppress
setting a timer in dtor*/ }
private:
// needed data members here.
// good idea to make this non-copyable
};
Then, in OnTimer
void CMyWnd::OnTimer(UINT nTimer)
{
CTempTimerDisable DisableCurrentTimer(this, nTimer, nTimerPeriod);
// Handler, without fear of nTimer firing again.
}
You may want additional error handling in CTempTimerDisable ctor. You
may need to account for SetTimer failing (I've never seen this
actually happen, but...).
Note that other your other timers still may fire, which may be
undesirable.
HTH,
Goran.