Home | Contact Us | FAQ | Search & Site Map | Link to Us
Sign In | Join | Other 45 Sites in Network
HomeAnnouncementsFree MagazinesWhite PapersSubmit Content
Discussion GroupsASP.NETWindows FormsLanguages.NET FrameworkVisual Studio.NET
Articles.NET FrameworkASP.NETToolsWindows Forms
.NET DirectoryOpen Source ProjectsUser GroupsWeb Resources
Related Topics
Visual Basic 6SQL ServerMS AccessOther DB ProductsMS Server ProductsMore Topics ...

.NET Forum / .NET Framework / CLR / November 2007

Tip: Looking for answers? Try searching our database.

Unexpected reentrancy on Wait,etc...

Thread view: 
Enable EMail Alerts  Start New Thread
Thread rating: 
jaket - 03 Nov 2007 00:39 GMT
Hello,

I have run into a serious reentrancy problem.  It appears from a stack
trace that lock, WaitOne and Thread.Join do some servicing of the
message queue.  Here is the setup:

The GUI thread uses a combination of Application.Idle and a timer to
simulate the old MFC OnIdle.  The Idle handling will periodically pick
up some work that requires executing some non-reentrant code.  This
code uses several locks to synchronize with other threads.

Another background worker thread is calling thru the Application's
API.  The API must use Control.Invoke using the MainForm to marshal
the calls to the GUI thread.  It also calls the non-reentrant code.

While GUI thread is waiting on a lock (I presume that it is already
held by a 3rd thread), the Control.Invoke can occassionaly slip thru
and cause reentrancy.  Anyone familiar with the perils of DoEvents
will understand why this is a major problem.

Any suggestions?

Sincerely, Jake
Frank Hileman - 05 Nov 2007 23:31 GMT
Hi Jake,

Usually some code using locks can be reworked such that each thread works
with private copies of data only, returning data using a centralized queue,
with only the centralized queue being thread safe (using locks or other
techniques).

If you don't mind your reentrant function failing, set a boolean field to
true on entry of the function, and set it to false in a finally statement at
the end of the function (the body being in a try). Then add a check to the
top of the function to bail out if you are already executing the function.
That is the standard non-reentrancy guard.

Regards,
Frank Hileman

check out VG.net: http://www.vgdotnet.com
Animated vector graphics system
Integrated Visual Studio graphics editor

> Hello,
>
[quoted text clipped - 19 lines]
>
> Sincerely, Jake
jaket - 06 Nov 2007 22:14 GMT
I suppose the non-reentrant code could be moved into a new thread with
a queue.  This doesn't solve the general problem that the GUI can
reenter anytime it locks and it is very difficult to guarantee the
rest of the GUI code is reentrant.  Re your second point.  The code is
currently guarded and throws an exception indicating the caller
violated the transaction discipline.  Failure is not an option because
the receiver could be a customer calling thru the API.

Thanks for your suggestions,

Jake

On Nov 5, 3:31 pm, "Frank Hileman"
<frank...@no.spamming.prodigesoftware.com> wrote:
> Hi Jake,
>
[quoted text clipped - 41 lines]
>
> - Show quoted text -
Peter Ritchie [C# MVP] - 07 Nov 2007 15:09 GMT
The GUI thread is an STA thread.  A STA thread has an obligation to
responsively process its message pump.  Standard wait methods in .NET
recognize this and process the threads message pump while waiting (wait
methods that don't, like WaitAll, will cause an exception when executed on an
STA thread).

Given that this is a GUI thread, you shouldn't block the thread with
lock/Wait/Join as this will block the responsiveness of the UI.

Idle is raised when there's no more windows message in the current GUI
thread's message queue.  It's an indication that the GUI thread may not be
doing anything.  If you want to take the opportunity to do something, then I
would spawn a background thread to do the work.  If you need to communicate
something back to the GUI then I would recommend using a BackgroundWorker
object.

Signature

Browse http://connect.microsoft.com/VisualStudio/feedback/ and vote.
http://www.peterRitchie.com/blog/
Microsoft MVP, Visual Developer - Visual C#

> I suppose the non-reentrant code could be moved into a new thread with
> a queue.  This doesn't solve the general problem that the GUI can
[quoted text clipped - 55 lines]
> >
> > - Show quoted text -
jaket - 07 Nov 2007 19:53 GMT
Peter,

OnIdle is used to purposely slow updates to the GUI.  Computations are
happening on a separate thread.  Invoking onto the GUI thread from the
computation thread would not be feasable because the computations
sometimes happen too fast for the user's eye.  Instead, the results
are written to a mailbox and the GUI checks the mailbox during Idle
time and updates.  The locks are not held by either thread for long.
I don't understand how locks can be avoided in a multi-threaded
application, especially where data must be shared between the threads?

Thanks for your input, Jake

On Nov 7, 7:09 am, Peter Ritchie [C# MVP] <PRS...@newsgroups.nospam>
wrote:
> The GUI thread is an STA thread.  A STA thread has an obligation to
> responsively process its message pump.  Standard wait methods in .NET
[quoted text clipped - 81 lines]
>
> - Show quoted text -
jaket - 15 Nov 2007 19:06 GMT
I found an old blog entry from Chris Brumme discussing the exact
problem in grueling detail.  If anyone is interested:

http://blogs.msdn.com/cbrumme/archive/2004/02/02/66219.aspx

> Hello,
>
[quoted text clipped - 19 lines]
>
> Sincerely, Jake

Rate this thread:







Free Magazines

Get these publications absolutely FREE for up to 12 months. There are no hidden fees and no obligation. Simply choose a title, complete the application form and submit it. Read more ...

Oracle MagazineNetwork ComputingComputer WorldBio-IT WorldeWeekInformation WeekInfosecurity
 
Sign In
Join
My Latest Posts
My Monitored Threads
My Blog
My Photo Gallery
My Profile
My Homepage

Start New Thread
Enable EMail Alerts
Rate this Thread



©2008 Advenet LLC   Privacy Policy - Terms of Use
This website includes both content owned or controlled by Advenet as well as content owned or controlled by third parties.