<html xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office" xmlns:w="urn:schemas-microsoft-com:office:word" xmlns:x="urn:schemas-microsoft-com:office:excel" xmlns:p="urn:schemas-microsoft-com:office:powerpoint" xmlns:a="urn:schemas-microsoft-com:office:access" xmlns:dt="uuid:C2F41010-65B3-11d1-A29F-00AA00C14882" xmlns:s="uuid:BDC6E3F0-6DA3-11d1-A2A3-00AA00C14882" xmlns:rs="urn:schemas-microsoft-com:rowset" xmlns:z="#RowsetSchema" xmlns:b="urn:schemas-microsoft-com:office:publisher" xmlns:ss="urn:schemas-microsoft-com:office:spreadsheet" xmlns:c="urn:schemas-microsoft-com:office:component:spreadsheet" xmlns:oa="urn:schemas-microsoft-com:office:activation" xmlns:html="http://www.w3.org/TR/REC-html40" xmlns:q="http://schemas.xmlsoap.org/soap/envelope/" xmlns:D="DAV:" xmlns:x2="http://schemas.microsoft.com/office/excel/2003/xml" xmlns:ois="http://schemas.microsoft.com/sharepoint/soap/ois/" xmlns:dir="http://schemas.microsoft.com/sharepoint/soap/directory/" xmlns:ds="http://www.w3.org/2000/09/xmldsig#" xmlns:dsp="http://schemas.microsoft.com/sharepoint/dsp" xmlns:udc="http://schemas.microsoft.com/data/udc" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:sps="http://schemas.microsoft.com/sharepoint/soap/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:udcxf="http://schemas.microsoft.com/data/udc/xmlfile" xmlns:m="http://schemas.microsoft.com/office/2004/12/omml" xmlns="http://www.w3.org/TR/REC-html40">
<head>
<META HTTP-EQUIV="Content-Type" CONTENT="text/html; charset=us-ascii">
<meta name=Generator content="Microsoft Word 12 (filtered medium)">
<!--[if !mso]>
<style>
v\:* {behavior:url(#default#VML);}
o\:* {behavior:url(#default#VML);}
w\:* {behavior:url(#default#VML);}
.shape {behavior:url(#default#VML);}
</style>
<![endif]-->
<style>
<!--
/* Font Definitions */
@font-face
        {font-family:Courier;
        panose-1:2 7 4 9 2 2 5 2 4 4;}
@font-face
        {font-family:"Cambria Math";
        panose-1:2 4 5 3 5 4 6 3 2 4;}
@font-face
        {font-family:Calibri;
        panose-1:2 15 5 2 2 2 4 3 2 4;}
@font-face
        {font-family:Tahoma;
        panose-1:2 11 6 4 3 5 4 4 2 4;}
@font-face
        {font-family:Consolas;
        panose-1:2 11 6 9 2 2 4 3 2 4;}
/* Style Definitions */
p.MsoNormal, li.MsoNormal, div.MsoNormal
        {margin:0in;
        margin-bottom:.0001pt;
        font-size:12.0pt;
        font-family:"Times New Roman","serif";}
a:link, span.MsoHyperlink
        {mso-style-priority:99;
        color:blue;
        text-decoration:underline;}
a:visited, span.MsoHyperlinkFollowed
        {mso-style-priority:99;
        color:purple;
        text-decoration:underline;}
p.MsoPlainText, li.MsoPlainText, div.MsoPlainText
        {mso-style-priority:99;
        mso-style-link:"Plain Text Char";
        margin:0in;
        margin-bottom:.0001pt;
        font-size:10.5pt;
        font-family:Consolas;}
pre
        {mso-style-priority:99;
        mso-style-link:"HTML Preformatted Char";
        margin:0in;
        margin-bottom:.0001pt;
        font-size:10.0pt;
        font-family:"Courier New";}
span.HTMLPreformattedChar
        {mso-style-name:"HTML Preformatted Char";
        mso-style-priority:99;
        mso-style-link:"HTML Preformatted";
        font-family:"Courier New";}
span.PlainTextChar
        {mso-style-name:"Plain Text Char";
        mso-style-priority:99;
        mso-style-link:"Plain Text";
        font-family:Consolas;}
span.EmailStyle21
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle22
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle23
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle24
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle25
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle26
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle27
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle28
        {mso-style-type:personal;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
span.EmailStyle30
        {mso-style-type:personal-reply;
        font-family:"Calibri","sans-serif";
        color:#1F497D;}
.MsoChpDefault
        {mso-style-type:export-only;
        font-size:10.0pt;}
@page Section1
        {size:8.5in 11.0in;
        margin:1.0in 1.0in 1.0in 1.0in;}
div.Section1
        {page:Section1;}
-->
</style>
<!--[if gte mso 9]><xml>
<o:shapedefaults v:ext="edit" spidmax="1026" />
</xml><![endif]--><!--[if gte mso 9]><xml>
<o:shapelayout v:ext="edit">
<o:idmap v:ext="edit" data="1" />
</o:shapelayout></xml><![endif]-->
</head>
<body lang=EN-US link=blue vlink=purple>
<div class=Section1>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Hi,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Great work! Thanks for making the test. Can you put this in
bugzilla (<a href="http://deuce.doc.wustl.edu/bugzilla/index.cgi">http://deuce.doc.wustl.edu/bugzilla/index.cgi</a>).
Be aware of our bug fixing policies as described in the docs/ACE-bug-process.html
document<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Regards,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;color:#1F497D'>Johnny
Willemsen<br>
Remedy IT<br>
Postbus 101<br>
2650 AC Berkel en Rodenrijs<br>
The Netherlands<br>
www.theaceorb.nl / www.remedy.nl <br>
<br>
*** Integrated compile and test statistics see <a
href="http://scoreboard.theaceorb.nl">http://scoreboard.theaceorb.nl</a> ***<br>
*** Commercial service and support for
ACE/TAO/CIAO
***<br>
*** See <a href="http://www.theaceorb.nl/en/support.html">http://www.theaceorb.nl/en/support.html</a>
***</span><span style='color:#1F497D'> </span><span style='font-size:11.0pt;
font-family:"Calibri","sans-serif";color:#1F497D'><o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div style='border:none;border-left:solid blue 1.5pt;padding:0in 0in 0in 4.0pt'>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Rudy Pot
[mailto:rpot@aweta.nl] <br>
<b>Sent:</b> Tuesday, March 11, 2008 3:18 PM<br>
<b>To:</b> Johnny Willemsen; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Hi,</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Below you can find an extended version of the
ACE_wrappers/tests/Message_Queue_Notifications_Test.cpp</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>The original source was taken from the head of <a
href="https://svn.dre.vanderbilt.edu/viewvc/Middleware/"><strong><span
style='font-size:12.0pt'>[Middleware]</span></strong></a></span><strong><span
style='color:black'> / </span></strong><span style='font-size:10.0pt;
font-family:"Arial","sans-serif";color:blue'><a
href="https://svn.dre.vanderbilt.edu/viewvc/Middleware/trunk/"><strong><span
style='font-size:12.0pt'>trunk</span></strong></a></span><strong><span
style='color:black'> / </span></strong><span style='font-size:10.0pt;
font-family:"Arial","sans-serif";color:blue'><a
href="https://svn.dre.vanderbilt.edu/viewvc/Middleware/trunk/ACE/"><strong><span
style='font-size:12.0pt'>ACE</span></strong></a></span><strong><span
style='color:black'> / </span></strong><span style='font-size:10.0pt;
font-family:"Arial","sans-serif";color:blue'><a
href="https://svn.dre.vanderbilt.edu/viewvc/Middleware/trunk/ACE/tests/"><strong><span
style='font-size:12.0pt'>tests</span></strong></a></span><strong><span
style='color:black'> / Message_Queue_Notifications_Test.cpp</span></strong><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>(Subversion web)</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>NEW TESTS</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Two new tests are added which both can be (de)selected with
defines:</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>1) TEST_CLOSEHOOK: This should test the usecase where notify() race
conditions can occur when an Task EH uses "delete this" in its close
hook. </span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>However, the race condition is not reproducable for
sure. </span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>I have had a segmentation fault only once and unfortunately at
that time coredump was not active on my test system.</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>I tried to increase the chance of race condition by adding an
iteration loop but that didn't help either...</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Any ideas for getting the race condition addressed for sure might
be useful.</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>2) TEST_FULLNOTIFY: This tests the deadlock-usecase (notified
message enqueue with full reactor notification pipe).</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>On our test scoreboard this test will timeout when the
deadlock occurs (maximum run time per test allowed)</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>The deadlock is always adressed whith the latest version of ACE.</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>NOTES:</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>If you want to use this, two things have to be looked at before
checking in:</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>1) #includes section: I had to change some includes in order to get
the test through our compilers and the ACE versions used for the different
platforms (W32 BCB ACE-5.3.1, RH9 gcc ACE-5.3.1, FC6 gcc
ACE-5.5.6)</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>2) "ACE_TMAIN (int, ACE_TCHAR *[])" probably to be
replaced with "run_main (int, ACE_TCHAR *[])" ?<br>
(In subversion it was "run_main" but that doesn't work with the ACE
versions we use, which are probably already very old...).</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>By the way, some minor change has been
made in the code so the default_message = "ACE RULES" is
used for all the test cases which enqueue messages...</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Message_Queue_T.cpp:</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>For now, I've patched this code to avoid deadlocks. In our
applications we don't use "delete this" on any EH so we won't get
struck by the race conditions.</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>A suggested fix for both of the Message_Queue_T.cpp notify problems
(race / deadlock) is in my second post in this bug-thread (Feb 27 2008 3:37pm).
See also FIX CODE EXAMPLE below in this post. However that is only
pseudo code, not fully worked out. It probably also needs modifications in the
Notification_Strategy and Reactor classes.</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Regards,</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><strong><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Rudy Pot</span></strong><o:p></o:p></p>
<div>
<div>
<div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Embedded Computer Systems</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><strong><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>AWETA G&P b.v</span></strong><o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Postbox 17</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>NL-2630 AA Nootdorp</span><o:p></o:p></p>
</div>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>tel +31 (0)15 3109961 </span><o:p></o:p></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>fax +31 (0)15 3107321 </span><o:p></o:p></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>mail </span><a href="mailto:rpot@aweta.nl"><span
style='font-size:10.0pt;font-family:"Arial","sans-serif"'>rpot@aweta.nl</span></a><o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>web <a href="blocked::http://www.aweta.com">www.aweta.com</a></span><o:p></o:p></p>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>// ============================================================================</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>// $Id: Message_Queue_Notifications_Test.cpp 10757 2008-03-10
11:41:40Z rpot $</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//
============================================================================<br>
//<br>
// = LIBRARY<br>
// tests<br>
//<br>
// = FILENAME<br>
// Message_Queue_Notification_Test.cpp<br>
//<br>
// = DESCRIPTION<br>
// There are two tests that test 2 different
notification<br>
// mechanisms in Message Queue.<br>
//<br>
// The first test illustrates the notification
mechanisms in<br>
// Message_Queue and its integration with
Reactor.<br>
//<br>
// Note the following things about this part of
the test:<br>
//<br>
// 1. Multiple threads are not required.<br>
// 2. You do not have to explicitly notify the
Reactor<br>
// 3. This code will work the same with any
Reactor Implementation<br>
// 4. handle_input, handle_exception,
handle_output are the only<br>
// callbacks supported by this
mechanism<br>
// 5. The notification mechanism need not notify
the Reactor. You can<br>
// write your own strategy
classes that can do whatever application<br>
// specific behavior you want.<br>
//<br>
// The second test also makes sure the high/low
water mark<br>
// signaling mechanism works flawlessly.<br>
//<br>
// The third test handles the use case of
notification strategy in <br>
// combination with EH "delete this"
termination.<br>
//<br>
// There is also a fourth stress test added for
addressing possible <br>
// notification deadlocks. It look similar as the
first test but needs <br>
// threads. If deadlock occurs, this test has to
be killed to terminate.<br>
//<br>
// = AUTHOR<br>
// Irfan Pyarali <<a href="mailto:irfan@cs.wustl.edu">irfan@cs.wustl.edu</a>>
and Nanbor Wang <<a href="mailto:nanbor@cs.wustl.edu">nanbor@cs.wustl.edu</a>><br>
//<br>
// ============================================================================<br>
#include "test_config.h"<br>
#include "ace/Reactor.h"<br>
#include "ace/Task.h"<br>
#include "ace/Reactor_Notification_Strategy.h"<br>
#include "ace/Atomic_Op.h"</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#ifndef ACE_SYNCH_H<br>
#include "ace/Synch.h"<br>
#endif<br>
//#include "ace/Barrier.h"<br>
//#include "ace/Synch_Traits.h"<br>
//#include "ace/Null_Condition.h"<br>
//#include "ace/Null_Mutex.h"</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>// Are below includes needed?<br>
//#include "ace/OS_NS_string.h"<br>
//#include "ace/OS_NS_unistd.h"</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>ACE_RCSID(tests, Message_Queue_Notifications_Test, "$Id:
Message_Queue_Notifications_Test.cpp 10757 2008-03-10 11:41:40Z rpot $")</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>static int iterations = 10;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>static const size_t worker_threads = 2;<br>
static const char * default_message = "ACE RULES";<br>
static const size_t default_high_water_mark = 20;<br>
static const size_t default_low_water_mark = 10;<br>
static const int watermark_iterations = 2 * default_high_water_mark;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#define TEST_CLOSEHOOK 1 // Test for Message_Queue_T
notify race conditions.<br>
#define TEST_FULLNOTIFY 1 // Test for Message_Queue_T notify deadlock.</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
class Message_Handler : public ACE_Task<ACE_NULL_SYNCH><br>
{<br>
// = TITLE<br>
// This class implements a notification strategy for the
Reactor.<br>
public:<br>
// = Initialization and termination.<br>
Message_Handler (ACE_Reactor &reactor);<br>
// Constructor.</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Demuxing hooks.<br>
virtual int handle_input (ACE_HANDLE);<br>
virtual int handle_output (ACE_HANDLE fd = ACE_INVALID_HANDLE);<br>
virtual int handle_exception (ACE_HANDLE fd = ACE_INVALID_HANDLE);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>private:<br>
int process_message (void);<br>
void make_message (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_Reactor_Notification_Strategy notification_strategy_;<br>
};</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
class Watermark_Test : public ACE_Task<ACE_SYNCH><br>
{<br>
// = TITLE<br>
// This class test the correct functioning of
build-in flow<br>
// control machanism in ACE_Task.<br>
public:<br>
Watermark_Test (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> virtual int svc (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> int consumer (void);<br>
int producer (void);<br>
int put_message (ACE_Time_Value* timeout = 0);<br>
int get_message (void);<br>
void print_producer_debug_message (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>private:<br>
const size_t len_;<br>
const size_t hwm_;<br>
const size_t lwm_;<br>
ACE_Atomic_Op <ACE_SYNCH_MUTEX, int> role_;<br>
#if defined (ACE_HAS_THREADS)<br>
ACE_Barrier mq_full_;<br>
ACE_Barrier mq_low_water_mark_hit_;<br>
#endif /* ACE_HAS_THREADS */<br>
};</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
#if defined (ACE_HAS_THREADS)</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#if defined (TEST_CLOSEHOOK)<br>
class CloseHook_Handler : public ACE_Task<ACE_MT_SYNCH><br>
{<br>
// = TITLE<br>
// This class tests the behaviour of the Task <close>
hook called in <br>
// combination with notified MB_STOP message enqueing.<br>
// It should normally cause this dynamically created task
(EH) to <br>
// gracefully delete itself.<br>
public:<br>
// = Initialization and termination.<br>
CloseHook_Handler (ACE_Reactor &reactor);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Demuxing hooks.<br>
virtual int handle_input (ACE_HANDLE);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Starts the threads needed for testing.<br>
virtual int open (void *args = 0); </span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>private:<br>
// = Ensure dynamic creation.<br>
virtual ~CloseHook_Handler ();</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Svc hook.<br>
virtual int svc (void);<br>
// = Hook called from <ACE_Thread_Exit> during thread exit.<br>
virtual int close (u_long flags = 0);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_Reactor_Notification_Strategy
notification_strategy_;<br>
ACE_Barrier
startupBarrier_; <br>
ACE_Message_Block::ACE_Message_Type messageType_;<br>
};<br>
#endif /* TEST_CLOSEHOOK */</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
#if defined (TEST_FULLNOTIFY)<br>
class FullNotify_Handler : public ACE_Task<ACE_MT_SYNCH><br>
{<br>
// = TITLE<br>
// This class tests the behaviour of (notified) message
enqueing at a point <br>
// when the Reactor notification pipe is full. It should
then just block.<br>
// After the (consumer) Reactor eventloop has handled some
pending <br>
// notifications, the notifiers (producer threads) should
just get unblocked.<br>
public:<br>
// = Initialization and termination.<br>
FullNotify_Handler (ACE_Reactor &reactor);<br>
virtual ~FullNotify_Handler ();</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Demuxing hooks.<br>
virtual int handle_input (ACE_HANDLE);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // = Starts the threads needed for testing.<br>
virtual int open (void *args = 0); </span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>private:<br>
// = Svc hook.<br>
virtual int svc (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> void make_message (void);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_Reactor_Notification_Strategy notification_strategy_;<br>
static const int
iMAX_NR_THREADS = 2;
<br>
// Identifiers needed when we have more than one thread active.<br>
ACE_thread_t
thread_ids_[iMAX_NR_THREADS]; <br>
};<br>
#endif /* TEST_FULLNOTIFY */</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#endif /* ACE_HAS_THREADS */</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
Message_Handler::Message_Handler (ACE_Reactor &reactor)<br>
// First time handle_input will be called<br>
: notification_strategy_ (&reactor,<br>
this,<br>
ACE_Event_Handler::READ_MASK)<br>
{<br>
this->msg_queue ()->notification_strategy
(&this->notification_strategy_);<br>
this->make_message ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Message_Handler::handle_input (ACE_HANDLE)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("Message_Handler::handle_input\n")));</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Next time handle_output will be called.<br>
this->notification_strategy_.mask (ACE_Event_Handler::WRITE_MASK);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return process_message ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Message_Handler::handle_output (ACE_HANDLE fd)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("Message_Handler::handle_output\n")));<br>
ACE_UNUSED_ARG (fd);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Next time handle_exception will be called.<br>
this->notification_strategy_.mask (ACE_Event_Handler::EXCEPT_MASK);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return process_message ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Message_Handler::handle_exception (ACE_HANDLE fd)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("Message_Handler::handle_exception\n")));<br>
ACE_UNUSED_ARG (fd);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Next time handle_input will be called.<br>
this->notification_strategy_.mask (ACE_Event_Handler::READ_MASK);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return this->process_message ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Message_Handler::process_message (void)<br>
{<br>
ACE_Message_Block *mb;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (this->getq (mb,<br>
(ACE_Time_Value *) &ACE_Time_Value::zero) == -1)<br>
ACE_ERROR_RETURN ((LM_ERROR,<br>
ACE_TEXT ("%p\n"),<br>
ACE_TEXT ("dequeue_head")),<br>
-1);<br>
else<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("message received = %s\n"),<br>
mb->rd_ptr ()));<br>
mb->release ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> this->make_message ();<br>
return 0;<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>void<br>
Message_Handler::make_message (void)<br>
{<br>
if (--iterations > 0)<br>
{<br>
ACE_Message_Block *mb;<br>
ACE_NEW (mb,<br>
ACE_Message_Block (default_message));</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("sending message\n")));<br>
this->putq (mb);<br>
}<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
Watermark_Test::Watermark_Test (void)<br>
: len_ (ACE_OS::strlen (default_message) + 1),<br>
hwm_ (this->len_ * default_high_water_mark),<br>
lwm_ (this->len_ * default_low_water_mark),<br>
role_ (0)<br>
#if defined (ACE_HAS_THREADS)<br>
, mq_full_ (worker_threads),<br>
mq_low_water_mark_hit_ (worker_threads)<br>
#endif /* ACE_HAS_THREADS */<br>
{<br>
this->water_marks (ACE_IO_Cntl_Msg::SET_LWM,<br>
this->lwm_);<br>
this->water_marks (ACE_IO_Cntl_Msg::SET_HWM,<br>
this->hwm_);<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Watermark_Test::producer (void)<br>
{<br>
int i = watermark_iterations;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> for (ssize_t hwm = this->hwm_;<br>
hwm >= 0 ;<br>
hwm -= this->len_)<br>
{<br>
this->put_message ();<br>
this->print_producer_debug_message ();<br>
i--;<br>
if (this->msg_queue ()->is_full ())<br>
break;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> }<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("(%P|%t) Producer: High water mark hit ---- \n")));</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_MT (this->mq_full_.wait ());</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // The following put_message should block until the message
queue<br>
// has dropped under the lwm.<br>
this->put_message ();</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_ASSERT (this->msg_queue ()-> message_bytes ()
<= this->lwm_ + this->len_);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> this->print_producer_debug_message ();</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> for (i--; i >= 0 ; i--)<br>
{<br>
this->put_message ();<br>
this->print_producer_debug_message ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return 0;<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Watermark_Test::consumer (void)<br>
{<br>
ACE_MT (this->mq_full_.wait ());</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_OS::sleep (1);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Let producer proceed and block in putq.</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> for (int i = watermark_iterations; i >= 0; i--)<br>
{<br>
this->get_message ();<br>
ACE_OS::sleep (0);<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return 0;<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Watermark_Test::get_message (void)<br>
{<br>
ACE_Message_Block *mb;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (this->getq (mb) == -1)<br>
ACE_ERROR_RETURN ((LM_ERROR,<br>
ACE_TEXT ("%p\n"),<br>
ACE_TEXT ("dequeue_head")),<br>
-1);<br>
else<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("(%P|%t) Consumer: message size = %3d, ")<br>
ACE_TEXT ("message count = %3d\n"),<br>
this->msg_queue ()-> message_bytes (),<br>
this->msg_queue ()-> message_count ()));<br>
mb->release ();<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return 0;<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Watermark_Test::put_message (ACE_Time_Value *timeout)<br>
{<br>
ACE_Message_Block *mb;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_NEW_RETURN (mb,<br>
ACE_Message_Block (default_message,<br>
this->len_),<br>
-1);</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return this->putq (mb, timeout);<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>void<br>
Watermark_Test::print_producer_debug_message (void)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("(%P|%t) Producer: message size = %3d, ")<br>
ACE_TEXT ("message count = %3d\n"),<br>
this->msg_queue ()-> message_bytes (),<br>
this->msg_queue ()-> message_count ()));<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
Watermark_Test::svc (void)<br>
{<br>
// this->role_ is an Atomic_Op object.<br>
int role = this->role_++;</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> switch (role)<br>
{<br>
case 0:<br>
this->producer ();<br>
break;<br>
case 1:<br>
this->consumer ();<br>
break;<br>
default:<br>
break;<br>
}<br>
return 0;<br>
}</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'><br>
//-----------------------------------------------------------------------------<br>
#if defined (ACE_HAS_THREADS)<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#if defined (TEST_CLOSEHOOK)<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>CloseHook_Handler::CloseHook_Handler (ACE_Reactor &reactor)<br>
: notification_strategy_ (&reactor,<br>
this,<br>
ACE_Event_Handler::READ_MASK)<br>
, startupBarrier_(2) //
Barrier for ourself and the new thread.<br>
, messageType_(ACE_Message_Block::MB_NORMAL)<br>
{<br>
this->msg_queue()->notification_strategy
(&this->notification_strategy_);<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>CloseHook_Handler::~CloseHook_Handler()<br>
{<br>
if (this->reactor() != 0)<br>
{<br>
// Remove ourself from reactor Eh queue (don't call
handle_close).<br>
this->reactor()->remove_handler(this,<br>
ACE_Event_Handler::READ_MASK |<br>
ACE_Event_Handler::DONT_CALL );<br>
this->reactor(0); <br>
};<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
CloseHook_Handler::handle_input (ACE_HANDLE)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("CloseHook_Handler::handle_input\n")));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // All messages handled in svc().<br>
return 0; // Stay registered.<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
CloseHook_Handler::open (void *args)<br>
{<br>
ACE_UNUSED_ARG (args);<br>
if (this->activate(THR_NEW_LWP | THR_DETACHED) < 0)<br>
{<br>
ACE_ERROR_RETURN ((LM_ERROR<br>
,ACE_TEXT ("CloseHook_Handler::open, activate ERROR(%s)!\n")<br>
,strerror(ACE_OS::last_error())<br>
),-1);<br>
};<br>
startupBarrier_.wait();<br>
return 0;<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
CloseHook_Handler::svc (void)<br>
{<br>
ACE_Message_Block *pMb=0;<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> startupBarrier_.wait();<br>
while (messageType_ != ACE_Message_Block::MB_STOP)<br>
{<br>
if (this->msg_queue()->dequeue_head(pMb) >= 0)
// Block on input queue.<br>
{<br>
messageType_ = pMb->msg_type();<br>
pMb->release();<br>
ACE_DEBUG ((LM_DEBUG, ACE_TEXT
("CloseHook_Handler::svc[t=%t]: pMb->release()\n")));<br>
}<br>
else<br>
{<br>
ACE_DEBUG ((LM_WARNING,<br>
ACE_TEXT ("CloseHook_Handler::svc: Invalid message dequeued!\n")<br>
));<br>
};<br>
};<br>
return 0;<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int <br>
CloseHook_Handler::close (u_long flags)<br>
{<br>
ACE_UNUSED_ARG (flags);<br>
ACE_DEBUG ((LM_DEBUG,ACE_TEXT ("CloseHook_Handler::close hook
called.\n")));<br>
// Service thread has exited, and this object is now invalid.<br>
delete this;<br>
return 0;<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#endif /* TEST_CLOSEHOOK */<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
#if defined (TEST_FULLNOTIFY)<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>FullNotify_Handler::FullNotify_Handler (ACE_Reactor &reactor)<br>
: notification_strategy_ (&reactor,<br>
this,<br>
ACE_Event_Handler::READ_MASK)<br>
{<br>
this->reactor(&reactor);<br>
this->msg_queue()->notification_strategy (&this->notification_strategy_);<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>FullNotify_Handler::~FullNotify_Handler()<br>
{<br>
this->wait();<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
FullNotify_Handler::handle_input (ACE_HANDLE)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("FullNotify_Handler::handle_input\n")));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_Message_Block *mb;<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (this->msg_queue()->is_empty())<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("Message queue is empty!\n")<br>
));<br>
else if (this->getq (mb,<br>
(ACE_Time_Value *) &ACE_Time_Value::zero) == -1)<br>
ACE_ERROR_RETURN ((LM_ERROR,<br>
ACE_TEXT ("%p\n"),<br>
ACE_TEXT ("dequeue_head")),<br>
-1);<br>
else<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("message received = %s\n"),<br>
mb->rd_ptr ()));<br>
mb->release ();<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> return 0; // Stay registered.<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
FullNotify_Handler::open (void *args)<br>
{<br>
ACE_UNUSED_ARG (args);<br>
// Activate the threads: Most are default parms, except n_threads <br>
// and thread_ids_.<br>
// thread_ids_ is used to distinguise between the threads.<br>
if
(this->activate(
// Activate the svc() <br>
THR_NEW_LWP | THR_JOINABLE // (default) flags.<br>
,iMAX_NR_THREADS
// - n_threads <br>
,0
// (default) force_active<br>
,ACE_DEFAULT_THREAD_PRIORITY // (default) priority<br>
,-1
// (default) grp_id<br>
,0
// (default) *task<br>
,0
// (default) thread_handles[]<br>
,0
// (default) *stack[]<br>
,0
// (default) stack_size[]<br>
,thread_ids_
// - thread_ids[] <br>
) < 0)<br>
{<br>
ACE_ERROR_RETURN ((LM_ERROR<br>
,ACE_TEXT ("FullNotify_Handler::open, activate ERROR(%s)!\n")<br>
,strerror(ACE_OS::last_error())<br>
),-1);<br>
};<br>
return 0;<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>int<br>
FullNotify_Handler::svc (void)<br>
{<br>
int iNotifications=0;<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> //ACE_DEBUG((LM_ERROR, ">>
FullNotify_Handler::svc(%t)\n"));<br>
//--------------------------------------------<br>
if (ACE_Thread::self() == thread_ids_[0]) <br>
{<br>
// [Thread 0]<br>
// Hit the reactor's notification queue until it is full.<br>
// We can only continue if main thread handles the reactor
notifications.<br>
// Global 'iterations' will countdown if next thread
performs 'make_message'.<br>
while (iterations > 0)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("(%t) Reactor notification (%d)\n")<br>
,iNotifications++<br>
));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Notify the Reactor, which will
call our <handle_input>.<br>
// Timeout when notify pipe is full? Somehow it
isn't full when timeout<br>
// occurs.<br>
// FC6: <br>
// - With timeout parm:
108 x notify until timeout occurs.<br>
// - Without timeout parm: 432 x notify
until it blocks.<br>
//ACE_Time_Value timeout (10);<br>
//this-reactor()->notify (this,
ACE_Event_Handler::READ_MASK, &timeout);<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> this->reactor()->notify (this,
ACE_Event_Handler::READ_MASK);<br>
};<br>
}<br>
//--------------------------------------------<br>
else if (ACE_Thread::self() == thread_ids_[1])<br>
{<br>
// [Thread 1]<br>
// Use Message_Queue and hit the notification queue again
(blocks).<br>
// We can only continue if main thread handles the reactor
notifications.<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Sleep some time until notification pipe is
full?<br>
ACE_OS::sleep(2); <br>
while (iterations > 0)<br>
{<br>
this->make_message();<br>
};<br>
};<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> //ACE_DEBUG((LM_ERROR, "<<
FullNotify_Handler::svc(%t)\n"));<br>
return 0;<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>void<br>
FullNotify_Handler::make_message (void)<br>
{<br>
if (--iterations > 0)<br>
{<br>
ACE_Message_Block *mb;<br>
ACE_NEW (mb,<br>
ACE_Message_Block (default_message));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("sending message\n")));<br>
this->putq (mb);<br>
}<br>
}<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#endif /* TEST_FULLNOTIFY */<br>
#endif /* ACE_HAS_THREADS */<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//-----------------------------------------------------------------------------<br>
int<br>
//run_main (int, ACE_TCHAR *[])<br>
ACE_TMAIN (int, ACE_TCHAR *[])<br>
{<br>
ACE_START_TEST (ACE_TEXT ("Message_Queue_Notifications_Test"));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> {<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("\nStarting message queue reactive notification
test...\n")));<br>
<br>
ACE_Reactor reactor;<br>
Message_Handler mh (reactor);<br>
<br>
while (iterations > 0)<br>
reactor.handle_events ();<br>
}<br>
#if defined (ACE_HAS_THREADS)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("\nStarting message queue watermark test...\n")));<br>
Watermark_Test watermark_test;<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("High water mark is %d\n")<br>
ACE_TEXT ("Low water mark is %d\n"),<br>
default_high_water_mark,<br>
default_low_water_mark));<br>
<br>
watermark_test.activate (THR_NEW_LWP | THR_DETACHED,<br>
worker_threads);<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // This frees the memory used by the DETACHED
thread.<br>
ACE_Thread_Manager::instance ()->wait ();<br>
}<br>
#else<br>
ACE_DEBUG ((LM_INFO,<br>
ACE_TEXT ("Message queue watermark test not performed because threads are
not supported\n")));<br>
#endif /* ACE_HAS_THREADS */<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#if !defined (TEST_CLOSEHOOK)<br>
ACE_DEBUG ((LM_INFO,<br>
ACE_TEXT ("Task <close> hook trigger test not used.\n")));<br>
#elif defined (ACE_HAS_THREADS)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("\nStarting test: Task <close> hook triggered by notified
message enqueue...\n")));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>
ACE_Reactor reactor;<br>
CloseHook_Handler* pChh = 0;<br>
ACE_Message_Block* pMBlk = 0;<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> for (int iIterations=0; iIterations<10;
iIterations++)<br>
{<br>
// Dynamically allocate Task EH which deletes
itself after MB_STOP <br>
// is received.<br>
ACE_NEW_RETURN (pChh,
CloseHook_Handler(reactor), -1);<br>
<br>
// Start svc thread.<br>
pChh->open();<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Just enqueue some normal
messages.<br>
ACE_NEW_RETURN( pMBlk, ACE_Message_Block(0,
ACE_Message_Block::MB_NORMAL), -1);<br>
pChh->putq( pMBlk );<br>
ACE_NEW_RETURN( pMBlk, ACE_Message_Block(0,
ACE_Message_Block::MB_NORMAL), -1);<br>
pChh->putq( pMBlk );<br>
ACE_NEW_RETURN( pMBlk, ACE_Message_Block(0,
ACE_Message_Block::MB_NORMAL), -1);<br>
pChh->putq( pMBlk );<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Quit svc thread and let task
delete itself (and see what happens).<br>
ACE_NEW_RETURN( pMBlk, ACE_Message_Block(0,
ACE_Message_Block::MB_STOP), -1);<br>
pChh->putq( pMBlk );<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // This frees the memory used by the
DETACHED thread.<br>
ACE_Thread_Manager::instance ()->wait ();<br>
};<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> }<br>
#else<br>
ACE_DEBUG ((LM_INFO,<br>
ACE_TEXT ("Task <close> hook trigger test not performed because
threads are not supported\n")));<br>
#endif /* ACE_HAS_THREADS */<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>#if !defined (TEST_FULLNOTIFY)<br>
ACE_DEBUG ((LM_INFO,<br>
ACE_TEXT ("Blocking (full) notification test not used.\n")));<br>
#elif defined (ACE_HAS_THREADS)<br>
{<br>
ACE_DEBUG ((LM_DEBUG,<br>
ACE_TEXT ("\nStarting test: message enqueue with blocking (full)
notification...\n")));<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Global message enqueue iterations.<br>
iterations = 10; <o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_Reactor reactor;<br>
FullNotify_Handler fnh (reactor); <o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Start producer threads.<br>
fnh.open();<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Sleep until notification pipe is full and all
producer threads block.<br>
ACE_OS::sleep(4); <o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> // Empty the notification pipe so producers
should normally unblock.<br>
while (iterations > 0)<br>
reactor.handle_events ();<br>
}<br>
#else<br>
ACE_DEBUG ((LM_INFO,<br>
ACE_TEXT ("Message enqueue with full notification test not performed
because threads are not supported\n")));<br>
#endif /* ACE_HAS_THREADS */<o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> ACE_END_TEST;<br>
return 0;<br>
}</span><o:p></o:p></p>
<p class=MsoNormal><o:p> </o:p></p>
<div class=MsoNormal align=center style='text-align:center'>
<hr size=2 width="100%" align=center>
</div>
<p class=MsoNormal style='margin-bottom:12.0pt'><b><span style='font-size:10.0pt;
font-family:"Tahoma","sans-serif"'>From:</span></b><span style='font-size:10.0pt;
font-family:"Tahoma","sans-serif"'> Greg Popovitch [mailto:gpy@altair.com] <br>
<b>Sent:</b> woensdag 27 februari 2008 21:35<br>
<b>To:</b> Johnny Willemsen; Rudy Pot; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny, I understand completely. I hope someone will provide the
regression test (I can’t). Thanks for the help.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>greg<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Johnny Willemsen
[mailto:jwillemsen@remedy.nl] <br>
<b>Sent:</b> Wednesday, February 27, 2008 3:32 PM<br>
<b>To:</b> Greg Popovitch; 'Rudy Pot'; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Hi,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Even when a deadlock can be shown on paper, we do need to have a
regression test that triggers it. That way we can see it ourselves as core
developers and we can also make sure we don't introduce the problem back in the
future by any other change.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div style='border:none;border-left:solid blue 1.5pt;padding:0in 0in 0in 4.0pt'>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Greg Popovitch
[mailto:gpy@altair.com] <br>
<b>Sent:</b> Wednesday, February 27, 2008 8:14 PM<br>
<b>To:</b> Johnny Willemsen; Rudy Pot; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny, I agree with the usefulness of having regression tests. <o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Still, this issue is a clear deadlock situation, which can be
easily explained and understood (see below). I think it is a no-brainer
to remove a known deadlock, occuring when the program is running and busy, even
if there is a chance of a race condition at exit. <o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Deadlock explaination:<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoPlainText>In my enqueuing thread:<o:p></o:p></p>
<p class=MsoPlainText>----------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>1/ enqueue a message for the other thread on my own
message queue<o:p></o:p></p>
<p class=MsoPlainText>------------------------------------------------------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if
(_msg_queue) _msg_queue->enqueue_head(mb);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>2/ in ACE DLL, enqueue and notify reactor<o:p></o:p></p>
<p class=MsoPlainText>-----------------------------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>template <ACE_SYNCH_DECL> int<o:p></o:p></p>
<p class=MsoPlainText>ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_head
(ACE_Message_Block *new_item,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Time_Value *timeout) {<o:p></o:p></p>
<p class=MsoPlainText> ACE_TRACE ("ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_head");<o:p></o:p></p>
<p class=MsoPlainText> int queue_count = 0;<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T,
ace_mon, this->lock_, -1);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (this->state_ ==
ACE_Message_Queue_Base::DEACTIVATED)<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> errno =
ESHUTDOWN;<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (this->wait_not_full_cond
(ace_mon, timeout) == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> queue_count = this->enqueue_head_i
(new_item);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (queue_count == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>=> this->notify ();<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText> return queue_count;<o:p></o:p></p>
<p class=MsoPlainText>}<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>3/ notification proceeds through, nothing interesting
there<o:p></o:p></p>
<p class=MsoPlainText>-----------------------------------------------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>
ACEd.dll!ACE_Reactor_Notification_Strategy::notify() Line 29<o:p></o:p></p>
<p class=MsoPlainText>C++<o:p></o:p></p>
<p class=MsoPlainText>
ACEd.dll!ACE_Reactor::notify(ACE_Event_Handler * event_handler=0x000000002971bc70,
unsigned long mask=1, ACE_Time_Value * tv=0x00<o:p></o:p></p>
<p class=MsoPlainText>
ACEd.dll!ACE_WFMO_Reactor::notify(ACE_Event_Handler *
event_handler=0x000000002971bc70, unsigned long mask=1, ACE_Time_Value * timeo<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>4/ getting into WFMO reactor which enqueues on its own
message queue<o:p></o:p></p>
<p class=MsoPlainText>--------------------------------------------------------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>int<o:p></o:p></p>
<p class=MsoPlainText>ACE_WFMO_Reactor_Notify::notify (ACE_Event_Handler
*event_handler,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Reactor_Mask mask,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Time_Value *timeout) {<o:p></o:p></p>
<p class=MsoPlainText> if (event_handler != 0)<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> ACE_Message_Block *mb = 0;<o:p></o:p></p>
<p class=MsoPlainText> ACE_NEW_RETURN (mb,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Message_Block (sizeof (ACE_Notification_Buffer)),<o:p></o:p></p>
<p class=MsoPlainText>
-1);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> ACE_Notification_Buffer
*buffer =<o:p></o:p></p>
<p class=MsoPlainText>
(ACE_Notification_Buffer *) mb->base ();<o:p></o:p></p>
<p class=MsoPlainText> buffer->eh_ =
event_handler;<o:p></o:p></p>
<p class=MsoPlainText> buffer->mask_ = mask;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> // Convert from relative
time to absolute time by adding the<o:p></o:p></p>
<p class=MsoPlainText> // current time of
day. This is what <ACE_Message_Queue><o:p></o:p></p>
<p class=MsoPlainText> // expects.<o:p></o:p></p>
<p class=MsoPlainText> if (timeout != 0)<o:p></o:p></p>
<p class=MsoPlainText> *timeout +=
timer_queue_->gettimeofday ();<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>=> if
(this->message_queue_.enqueue_tail<o:p></o:p></p>
<p class=MsoPlainText>
(mb, timeout) == -1)<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText>
mb->release ();<o:p></o:p></p>
<p class=MsoPlainText>
return -1;<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> event_handler->add_reference
();<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> return this->wakeup_one_thread_.signal (); }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>5/ enqueue blocks because queue is full (deadlock because
calling thread is hung in step 1 and will never dequeue)<o:p></o:p></p>
<p class=MsoPlainText>---------------------------------------<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>
message_queue_ {head_=0x000000006deaf1d0<o:p></o:p></p>
<p class=MsoPlainText>tail_=0x000000007a5826b0 low_water_mark_=16384 ...}<o:p></o:p></p>
<p class=MsoPlainText>ACE_Message_Queue<ACE_MT_SYNCH>
ACE_Message_Queue_Base<o:p></o:p></p>
<p class=MsoPlainText>{state_=1 } ACE_Message_Queue_Base<o:p></o:p></p>
<p class=MsoPlainText>
head_ 0x000000006deaf1d0 {rd_ptr_=0 wr_ptr_=0<o:p></o:p></p>
<p class=MsoPlainText>priority_=0 ...} ACE_Message_Block *<o:p></o:p></p>
<p class=MsoPlainText>
tail_ 0x000000007a5826b0 {rd_ptr_=0 wr_ptr_=0<o:p></o:p></p>
<p class=MsoPlainText>priority_=0 ...} ACE_Message_Block *<o:p></o:p></p>
<p class=MsoPlainText>
low_water_mark_ 16384 unsigned __int64<o:p></o:p></p>
<p class=MsoPlainText>
high_water_mark_ 16384 unsigned __int64<o:p></o:p></p>
<p class=MsoPlainText>
cur_bytes_ 16384 unsigned __int64<o:p></o:p></p>
<p class=MsoPlainText>
cur_length_ 0 unsigned __int64<o:p></o:p></p>
<p class=MsoPlainText>
cur_count_ 1024 unsigned __int64<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>template <ACE_SYNCH_DECL> int<o:p></o:p></p>
<p class=MsoPlainText>ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_tail
(ACE_Message_Block *new_item,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Time_Value *timeout) {<o:p></o:p></p>
<p class=MsoPlainText> ACE_TRACE
("ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_tail");<o:p></o:p></p>
<p class=MsoPlainText> int queue_count = 0;<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T,
ace_mon, this->lock_, -1);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (this->state_ ==
ACE_Message_Queue_Base::DEACTIVATED)<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> errno =
ESHUTDOWN;<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>=> if (this->wait_not_full_cond
(ace_mon, timeout) == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> queue_count = this->enqueue_tail_i
(new_item);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (queue_count == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> this->notify ();<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText> return queue_count;<o:p></o:p></p>
<p class=MsoPlainText>}<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>greg<o:p></o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Johnny Willemsen
[mailto:jwillemsen@remedy.nl] <br>
<b>Sent:</b> Wednesday, February 27, 2008 1:22 PM<br>
<b>To:</b> Greg Popovitch; 'Rudy Pot'; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block causesSelect_Reactor
deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Hi,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Also, in order to make such an important change to the core of
ACE, we do need an automated regression test that reproduces the problem. This
must be written like the ACE regression tests under ACE_wrappers/tests. If
someone can make this test and add it to bugzilla, the issue gets higher in the
todo list, without such a test we have to write one, which consumes more time.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div style='border:none;border-left:solid blue 1.5pt;padding:0in 0in 0in 4.0pt'>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Greg Popovitch
[mailto:gpy@altair.com] <br>
<b>Sent:</b> Wednesday, February 27, 2008 5:37 PM<br>
<b>To:</b> Johnny Willemsen; Rudy Pot; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny, I didn’t mean any offense. You do confirm my observation
with the statement: “people don't have time anymore to just fix these kind of
issues without funding”. I am sorry that this is the case, as I really like
ACE. But I strongly feel that if there is not anyone who feels
enough sense of ownership to fix clear bugs like this one, then it is hard to
justify using ACE in any project which doesn’t allocate specific funding for
ACE maintenance.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Now again that’s fine with me, and I certainly don’t ask anyone
to provide a service for free. However in the light of my own experience
(serious bug fixed in 2002, reintroduced in 2003 and still there in 2008), I
tend to disagree with the statement in ACE’s overview: “</span>ACE continues to
improve and its <a href="http://www.cs.wustl.edu/%7Eschmidt/ACE-future.html">future</a>
is bright”.<span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>greg<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Johnny Willemsen
[mailto:jwillemsen@remedy.nl] <br>
<b>Sent:</b> Wednesday, February 27, 2008 11:20 AM<br>
<b>To:</b> Greg Popovitch; 'Rudy Pot'; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Hi,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Please read docs/ACE-bug-process.html, the issue is more that
people don't have time anymore to just fix these kind of issues without
funding. This is a change in the core of ACE which has to be investigated,
reviewed, tested, all taking time. At the moment someone is willing to fund
such a change there are multiple people that would be willing to work on this.
We as Remedy IT do fix issues on the doc_group release without funding, but our
budgets for doing that are limited, at the end we also have to pay our bills.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div style='border:none;border-left:solid blue 1.5pt;padding:0in 0in 0in 4.0pt'>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'> Greg Popovitch
[mailto:gpy@altair.com] <br>
<b>Sent:</b> Wednesday, February 27, 2008 5:13 PM<br>
<b>To:</b> Rudy Pot; Johnny Willemsen; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Rudy, I reported the exact same issue and diagnostic a couple of
months ago (see mailing list on 11/29/2007 and 11/30/2007). Even though this is
a clear bug, it seemed to me that there was no one in particular planning to
fix the code. I ended up patching my own version, moving the notify() in the
code below one line down after the closing brace, so it would be out of the
scope of the ACE_GUARD. This fixes the issue and I have not seen any
termination race conditions.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoPlainText>template <ACE_SYNCH_DECL> int<o:p></o:p></p>
<p class=MsoPlainText>ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_head
(ACE_Message_Block *new_item,<o:p></o:p></p>
<p class=MsoPlainText>
ACE_Time_Value *timeout) {<o:p></o:p></p>
<p class=MsoPlainText> ACE_TRACE
("ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_head");<o:p></o:p></p>
<p class=MsoPlainText> int queue_count = 0;<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T,
ace_mon, this->lock_, -1);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (this->state_ ==
ACE_Message_Queue_Base::DEACTIVATED)<o:p></o:p></p>
<p class=MsoPlainText> {<o:p></o:p></p>
<p class=MsoPlainText> errno =
ESHUTDOWN;<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (this->wait_not_full_cond
(ace_mon, timeout) == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> queue_count = this->enqueue_head_i
(new_item);<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText> if (queue_count == -1)<o:p></o:p></p>
<p class=MsoPlainText> return -1;<o:p></o:p></p>
<p class=MsoPlainText><o:p> </o:p></p>
<p class=MsoPlainText>=> this->notify ();<o:p></o:p></p>
<p class=MsoPlainText> }<o:p></o:p></p>
<p class=MsoPlainText> return queue_count;<o:p></o:p></p>
<p class=MsoPlainText>}<o:p></o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>
ace-bugs-bounces@cse.wustl.edu [mailto:ace-bugs-bounces@cse.wustl.edu] <b>On
Behalf Of </b>Rudy Pot<br>
<b>Sent:</b> Wednesday, February 27, 2008 10:30 AM<br>
<b>To:</b> Johnny Willemsen; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> Re: [ace-bugs] [ACE_Message_Queue] notify PIPE block
causesSelect_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Hi Johnny,</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Thanks for your quick response!</span><o:p></o:p></p>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>I have had a closer look at all related bug messages and now I see
that my <br>
problem is already quite old (sorry for that).</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Correct me if I'm wrong but as far as I can see now there are two
problems <br>
with ace/Message_Queue_T.cpp:</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>1) Sun May 5 19:14:34 2002 Douglas C. Schmidt
<<a href="mailto:schm...@macarena.cs.wustl.edu">schm...@macarena.cs.wustl.edu</a>></span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> *
ace/Message_Queue_T.cpp: Modified all the enqueue*() methods so that<br>
their calls to notify()
occur *outside* of the monitor lock.<br>
This change prevents
deadlock from occurring when a reactor's<br>
notification pipe is
full. Thanks to Sasha Agranov<br>
<<a
href="mailto:sagra...@COMGATES.co.il">sagra...@COMGATES.co.il</a>> for
reporting this.</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>2) Sat Mar 22 11:58:12 2003 Douglas C. Schmidt <<a
href="mailto:schm...@tango.doc.wustl.edu">schm...@tango.doc.wustl.edu</a>></span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> *
ace/Message_Queue_T.cpp: Moved the notify() hook calls within<br>
the protection of the
guard lock critical section to prevent<br>
race conditions on
cleanup. Thanks to Ron Muck <<a href="mailto:r...@sdiusa.com">r...@sdiusa.com</a>><br>
for this suggestion. </span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Currently, the latter (2) is solved but because of that (1), which
is my <br>
problem, has returned.</span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Is it not possible to solve both problems with adding an extra
notify lock? </span><o:p></o:p></p>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'><br>
</span><span style='font-size:10.0pt;font-family:Courier;color:blue'>
<MessageQueue1, lock1>
handle_*<br>
[Producer-1]-enqueue-->[|||||||||||||||||||||||||]--->[EH-1] <-----+<br>
|
\<br>
\ <notifyLock>
<PIPE?>
|<br>
+-notify----->[|||||||||||||||||||||]--->[Reactor]---+<br>
/
(consumer) |<br>
|
/<br>
| <MessageQueue-n,
lock-n>
/<br>
[Producer-n]-enqueue-->[|||||||||||||||||||||||||]--->[EH-n] <----+ <br>
handle_*</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'><br>
</span><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>I will point out my idea about this in the following fix code
example.</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>This fix might be a problem because I think it needs some interface
change in <br>
ACE_Notification_Strategy too (to get a notification lock somehow) but you have<br>
more insight in the concequences of that. </span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Maybe you first could have a look at this proposal?</span><span
style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>Thanks in advance,<br>
Rudy Pot</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>//==============================================================================<br>
FIX CODE EXAMPLE (changes applicable for: enqueue_prio, enqueue_head, <br>
enqueue_deadline, enqueue_tail )</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'><br>
</span><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>template <ACE_SYNCH_DECL> int<br>
ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_prio (ACE_Message_Block
*new_item,<br>
ACE_Time_Value *timeout)<br>
{<br>
ACE_TRACE
("ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_prio");<br>
int queue_count = 0;<br>
{<br>
ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T, ace_mon,
this->lock_, -1);</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (this->state_ == ACE_Message_Queue_Base::DEACTIVATED)<br>
{<br>
errno = ESHUTDOWN;<br>
return -1;<br>
}</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (this->wait_not_full_cond (ace_mon,
timeout) == -1)<br>
return -1;</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>+ {<br>
+ ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T<br>
+
,ace_notify_mon<br>
+
// Would be handy if ACE_Notification_Strategy<br>
+
// has a lock. To be gathered from notify target<br>
+
// (reactor?)<br>
+
// Problem: ACE_Reactor_Notification_Strategy has stored <br>
+
// Reactor reference but ACE_Notification_Strategy not.<br>
+
,this->notification_strategy->lock()<br>
+
, -1);</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>+ // Somehow check if the (reactor?)
notification channel is full.<br>
+ // Probably needs this->notification_strategy too.<br>
+ if (this->wait_no_notify_full_cond
(ace_notify_mon, timeout) == -1)<br>
+ return -1; // We cannot enqueue without
causing a deadlock, so exit.</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>+ // Because we hold the notify lock, no new
notifications will occur on our<br>
+ // target and it is safe to enqueue & notify
hereafter...</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> queue_count = this->enqueue_i
(new_item);</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> if (queue_count == -1)<br>
return -1;</span><span
style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> this->notify (); </span><span
style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>+ }; // e.o. notify guarded area.</span><span
style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'> }<br>
return queue_count;<br>
}</span><span style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif";
color:blue'>END FIX CODE EXAMPLE <br>
//==============================================================================</span><span
style='font-family:"Arial","sans-serif"'><o:p></o:p></span></p>
<div>
<p class=MsoNormal><span style='font-family:"Arial","sans-serif"'> <o:p></o:p></span></p>
</div>
<p class=MsoNormal> <o:p></o:p></p>
<p class=MsoNormal><span style='font-family:"Courier New";color:blue'> <o:p></o:p></span></p>
<p class=MsoNormal> <o:p></o:p></p>
<div class=MsoNormal align=center style='text-align:center'>
<hr size=2 width="100%" align=center>
</div>
<p class=MsoNormal style='margin-bottom:12.0pt'><b><span style='font-size:10.0pt;
font-family:"Tahoma","sans-serif"'>From:</span></b><span style='font-size:10.0pt;
font-family:"Tahoma","sans-serif"'> Johnny Willemsen
[mailto:jwillemsen@remedy.nl] <br>
<b>Sent:</b> woensdag 27 februari 2008 13:08<br>
<b>To:</b> Rudy Pot; ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> RE: [ace-bugs] [ACE_Message_Queue] notify PIPE block causes
Select_Reactor deadlock.</span><o:p></o:p></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Hi,<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Thanks for using the PRF form. I found the change below in svn <o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'>Johnny<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Courier New";
color:rosybrown'>Sat Mar 22 11:58:12 2003 </span></b><b><span
style='font-size:10.0pt;font-family:"Courier New";color:cadetblue'>Douglas C.
Schmidt </span></b><span style='font-size:10.0pt;font-family:"Courier New"'><<span
style='color:darkgoldenrod'>schmidt@tango.doc.wustl.edu</span>><o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Courier New"'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Courier New"'>
* <b><span style='color:blue'>ace/Message_Queue_T.cpp</span></b>: Moved the
notify() hook calls within<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Courier New"'>
the protection of the guard lock critical section to prevent<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Courier New"'>
race conditions on cleanup. Thanks to Ron Muck <rlm@sdiusa.com>
<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Courier New"'>
for this suggestion.<o:p></o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<p class=MsoNormal><span style='font-size:11.0pt;font-family:"Calibri","sans-serif";
color:#1F497D'><o:p> </o:p></span></p>
<div style='border:none;border-left:solid blue 1.5pt;padding:0in 0in 0in 4.0pt'>
<div>
<div style='border:none;border-top:solid #B5C4DF 1.0pt;padding:3.0pt 0in 0in 0in'>
<p class=MsoNormal><b><span style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>From:</span></b><span
style='font-size:10.0pt;font-family:"Tahoma","sans-serif"'>
ace-bugs-bounces@cse.wustl.edu [mailto:ace-bugs-bounces@cse.wustl.edu] <b>On
Behalf Of </b>Rudy Pot<br>
<b>Sent:</b> Wednesday, February 27, 2008 11:29 AM<br>
<b>To:</b> ace-bugs@cs.wustl.edu<br>
<b>Subject:</b> [ace-bugs] [ACE_Message_Queue] notify PIPE block causes
Select_Reactor deadlock.<o:p></o:p></span></p>
</div>
</div>
<p class=MsoNormal><o:p> </o:p></p>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
ACE VERSION: 5.5.6</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
HOST MACHINE and OPERATING SYSTEM: i386, Linux 2.6.20-1.2933.fc6 </span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
TARGET MACHINE and OPERATING SYSTEM, if different from HOST: same.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'><br>
COMPILER NAME AND VERSION (AND PATCHLEVEL): gcc 4.1.1</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
CONTENTS OF $ACE_ROOT/ace/config.h: config-linux.h</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
CONTENTS OF $ACE_ROOT/include/makeinclude/platform_macros.GNU (unless<br>
this isn't used in this case, e.g., with Microsoft Visual
C++):<br>
platform-linux.GNU<br>
<br>
AREA/CLASS/EXAMPLE AFFECTED: <br>
Message_Queue_T.cpp / ACE_Message_Queue<ACE_SYNCH_USE>
/ enqueue_prio<br>
( But I think also:<br>
::enqueue_head, <br>
::enqueue_deadline,<br>
::enqueue_tail )</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
See DESCRIPTION. </span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
DOES THE PROBLEM AFFECT:<br>
COMPILATION? NO<br>
LINKING? NO<br>
EXECUTION? YES<br>
OTHER (please specify)? NO</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
SYNOPSIS:<br>
Application sometimes hangs when main Reactor gets more
busy. <br>
(e.g. due to significant increasing external events)<br>
Coredump shows:<br>
1) Main Reactor thread is blocking to get the
ACE_Message_Queue lock.<br>
2) Multiple threads block on ACE_Select_Reactor_Notify->
which ends up in<br>
->ACE::send() -> write().<br>
Result is deadlock. We have to kill the application for
shutdown.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
I've read Sidebar17 of C++ Network Programming Volume 2 but our situation <br>
is different from what is stated there (no notify called
from handle_* of<br>
event_handler but producer-consumer deadlock).</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
DESCRIPTION:</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
The application in short.<br>
-------------------------<br>
The part of our program where this deadlock appears is
dealing with <br>
processing messages from an embedded (CAN) network. <br>
There is a thread per CAN message center (hardware
communication channel)<br>
which puts the received message into ACE_Message_Queue's,
depending on who<br>
wants to observe the messages. These are the producer
threads.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
The message queues belongs to observers who all have registered to get<br>
notified by one Reactor which runs in one main Reactor
thread (consumer).<br>
(they also have registered themselves by the message center
threads as being<br>
interested in the messages).</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
Problem cause<br>
--------------<br>
What can happen now is that the main Reactor, which is used
for many other <br>
things in our application, temporarily got other work todo,
and therefore<br>
the message center threads may fill up the
ACE_Select_Reactor notification<br>
PIPE. This causes the message center threads (producers) to
block on the <br>
(FULL) PIPE write().</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
The main Reactor thread (consumer), when ready with the other work, wants to <br>
proceed with handling the pending notifications, and so
emptying the PIPE, <br>
but cannot do this because the current notification code
also holds the <br>
message QUEUE lock! <br>
<br>
See code description below:</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
In Message_Queue_T.cpp<br>
======================<br>
ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_prio,<br>
( But also the code reveals:<br>
::enqueue_head, <br>
::enqueue_deadline,<br>
::enqueue_tail )</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>-------------------------------<br>
DEADLOCK CODE: (Above rev. 46096 until HEAD code)</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>template
<ACE_SYNCH_DECL> int<br>
ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_prio (ACE_Message_Block
*new_item,<br>
ACE_Time_Value *timeout)<br>
{<br>
ACE_TRACE
("ACE_Message_Queue<ACE_SYNCH_USE>::enqueue_prio");<br>
int queue_count = 0;<br>
{<br>
ACE_GUARD_RETURN (ACE_SYNCH_MUTEX_T, ace_mon, this->lock_,
-1);</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
if (this->state_ == ACE_Message_Queue_Base::DEACTIVATED)<br>
{<br>
errno = ESHUTDOWN;<br>
return -1;<br>
}</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
if (this->wait_not_full_cond (ace_mon, timeout) == -1)<br>
return -1;</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
queue_count = this->enqueue_i (new_item);</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
if (queue_count == -1)<br>
return -1;</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
this->notify (); <<<< ERROR DEADLOCK (When blocking on
notify in scope of <br>
buffer lock...)<br>
}<br>
return queue_count;<br>
}</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
In above code snippet: <br>
this->notify (); Causes DEADLOCK when blocking on full
notification pipe.<br>
This happens because <notify()> is now called within
the scope of the <br>
ACE_GUARD_RETURN.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
In older versions of ACE, e.g. 5.3.1, the <notify()> was outside the
scope<br>
of the GUARD and we never had this deadlock.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
In SUBVERSION ACE, I can see this has been changed after revision<br>
r.46096 of Message_Queue_T.cpp (ChangeLogTag:Sat Mar
22 11:58:12 2003)<br>
But I don't know why.</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
See Message_Queue_T.cpp <br>
: enqueue_prio, enqueue_head, enqueue_deadline,
enqueue_tail</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>-------------------------------<br>
NON DEADLOCK CODE (rev. 46096)<br>
... <br>
... <br>
if (queue_count == -1)<br>
return -1;</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
} // e.o. scope ACE_GUARD_RETURN for queue lock.<br>
this->notify (); // NO deadlock here, notify will unblock as
soon PIPE is <br>
// emptied.<br>
return queue_count;<br>
}</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>-------------------------------<br>
Concerning Coredump parts:</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>[
consumer ]</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Thread
1 (process 16785):<br>
#0 0x009bc5d9 in __lll_mutex_lock_wait () from /lib/libpthread.so.0<br>
#1 0x009b8636 in _L_mutex_lock_85 () from /lib/libpthread.so.0<br>
#2 0x009b817d in pthread_mutex_lock () from /lib/libpthread.so.0<br>
#3 0x00a9f922 in ACE_OS::mutex_lock () from /usr/lib/libACE.so.5.5.6<br>
#4 0x00fd83c8 in ACE_Message_Queue<ACE_MT_SYNCH>::is_empty ()<br>
from /opt/lib/libgcp_datadump.so<br>
#5 0x0142d881 in can::CCanSvcDriverObserver::handle_output ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#6 0x00ac03f6 in ACE_Select_Reactor_Notify::dispatch_notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#7 0x00ac057a in ACE_Select_Reactor_Notify::handle_input ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#8 0x00ac14da in ACE_Select_Reactor_Notify::dispatch_notifications ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#9 0x00a5ef8e in ACE_Asynch_Pseudo_Task::ACE_Asynch_Pseudo_Task$base ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#10 0x00a5f5fd in ACE_Asynch_Pseudo_Task::ACE_Asynch_Pseudo_Task$base ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#11 0x00a65b03 in ACE_OS::gettimeofday () from /usr/lib/libACE.so.5.5.6<br>
#12 0x00abd0d3 in ACE_Reactor::run_reactor_event_loop ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#13 0x0804ae36 in main ()</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'><o:p> </o:p></span></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>[ producer
]<o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Thread
28 (process 17023):<br>
#0 0x009bc8f1 in write () from /lib/libpthread.so.0<br>
#1 0x00a55a3a in ACE::send () from /usr/lib/libACE.so.5.5.6<br>
#2 0x00ac082e in ACE_Select_Reactor_Notify::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#3 0x00a5edcc in ACE_Asynch_Pseudo_Task::ACE_Asynch_Pseudo_Task$base ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#4 0x00abdce6 in ACE_Reactor::notify () from /usr/lib/libACE.so.5.5.6<br>
#5 0x00abeace in ACE_Reactor_Notification_Strategy::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#6 0x0804b42a in ACE_Message_Queue<ACE_MT_SYNCH>::notify ()<br>
#7 0x00fd7fa2 in ACE_Message_Queue<ACE_MT_SYNCH>::enqueue_prio ()<br>
from /opt/lib/libgcp_datadump.so<br>
#8 0x0142d6f3 in can::CCanSvcDriverObserver::update ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#9 0x0143078d in can::CCanSvcDriver_Base::svc_Read ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#10 0x0143087f in can::CCanSvcDriverRemoteRequestImpl::svc ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#11 0x00ad3026 in ACE_Task_Base::svc_run () from /usr/lib/libACE.so.5.5.6<br>
#12 0x00ad39e8 in ACE_Thread_Adapter::invoke_i ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#13 0x00ad3bb6 in ACE_Thread_Adapter::invoke () from /usr/lib/libACE.so.5.5.6<br>
#14 0x00a67511 in ace_thread_adapter () from /usr/lib/libACE.so.5.5.6<br>
#15 0x009b626a in start_thread () from /lib/libpthread.so.0<br>
#16 0x92fff470 in ?? ()<br>
#17 0x92fff470 in ?? ()<br>
#18 0x92fff470 in ?? ()<br>
#19 0x92fff470 in ?? ()<br>
#20 0x00000000 in ?? ()</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>[ producer
]<o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Thread
34 (process 17012):<br>
#0 0x009bc8f1 in write () from /lib/libpthread.so.0<br>
#1 0x00a55a3a in ACE::send () from /usr/lib/libACE.so.5.5.6<br>
#2 0x00ac082e in ACE_Select_Reactor_Notify::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#3 0x00a5edcc in ACE_Asynch_Pseudo_Task::ACE_Asynch_Pseudo_Task$base ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#4 0x00abdce6 in ACE_Reactor::notify () from /usr/lib/libACE.so.5.5.6<br>
#5 0x00abeace in ACE_Reactor_Notification_Strategy::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#6 0x0804b42a in ACE_Message_Queue<ACE_MT_SYNCH>::notify ()<br>
#7 0x00fd7fa2 in ACE_Message_Queue<ACE_MT_SYNCH>::enqueue_prio ()<br>
from /opt/lib/libgcp_datadump.so<br>
#8 0x0142d6f3 in can::CCanSvcDriverObserver::update ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#9 0x0143062b in can::CCanSvcDriver_Base::svc_Read ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#10 0x014308cd in can::CCanSvcDriverReadImpl::svc ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#11 0x00ad3026 in ACE_Task_Base::svc_run () from /usr/lib/libACE.so.5.5.6<br>
#12 0x00ad39e8 in ACE_Thread_Adapter::invoke_i ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#13 0x00ad3bb6 in ACE_Thread_Adapter::invoke () from /usr/lib/libACE.so.5.5.6<br>
#14 0x00a67511 in ace_thread_adapter () from /usr/lib/libACE.so.5.5.6<br>
#15 0x009b626a in start_thread () from /lib/libpthread.so.0<br>
#16 0x96bff470 in ?? ()<br>
#17 0x96bff470 in ?? ()<br>
#18 0x96bff470 in ?? ()<br>
#19 0x96bff470 in ?? ()<br>
#20 0x00000000 in ?? ()</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>[ producer
]<o:p></o:p></span></p>
</div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Thread
26 (process 17025):<br>
#0 0x009bc8f1 in write () from /lib/libpthread.so.0<br>
#1 0x00a55a3a in ACE::send () from /usr/lib/libACE.so.5.5.6<br>
#2 0x00ac082e in ACE_Select_Reactor_Notify::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#3 0x00a5edcc in ACE_Asynch_Pseudo_Task::ACE_Asynch_Pseudo_Task$base ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#4 0x00abdce6 in ACE_Reactor::notify () from /usr/lib/libACE.so.5.5.6<br>
#5 0x00abeace in ACE_Reactor_Notification_Strategy::notify ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#6 0x0804b42a in ACE_Message_Queue<ACE_MT_SYNCH>::notify ()<br>
#7 0x00fd7fa2 in ACE_Message_Queue<ACE_MT_SYNCH>::enqueue_prio ()<br>
from /opt/lib/libgcp_datadump.so<br>
#8 0x0142d6f3 in can::CCanSvcDriverObserver::update ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#9 0x0143078d in can::CCanSvcDriver_Base::svc_Read ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#10 0x0143087f in can::CCanSvcDriverRemoteRequestImpl::svc ()<br>
from /opt/lib/libcanResourceManager.so.2<br>
#11 0x00ad3026 in ACE_Task_Base::svc_run () from /usr/lib/libACE.so.5.5.6<br>
#12 0x00ad39e8 in ACE_Thread_Adapter::invoke_i ()<br>
from /usr/lib/libACE.so.5.5.6<br>
#13 0x00ad3bb6 in ACE_Thread_Adapter::invoke () from /usr/lib/libACE.so.5.5.6<br>
#14 0x00a67511 in ace_thread_adapter () from /usr/lib/libACE.so.5.5.6<br>
#15 0x009b626a in start_thread () from /lib/libpthread.so.0<br>
#16 0x91bff470 in ?? ()<br>
#17 0x91bff470 in ?? ()<br>
#18 0x91bff470 in ?? ()<br>
#19 0x91bff470 in ?? ()<br>
#20 0x00000000 in ?? ()</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><br>
<span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
REPEAT BY:<br>
See description</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>
SAMPLE FIX/WORKAROUND:<br>
Change ACE code so that <this->notify();> is
outside of GUARD scope?<br>
(like it was before in rev. 46096).</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>[*
end of PRF *]</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Aside
from this deadlock problem:</span><o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>One
thing what also comes up now is that I have to look at the notification pipe</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>buffer
length too when I want to increase the ACE_Message_Queue size?</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>(enqueue
will block if notification pipe is full). </span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Or, I
have to use #define ACE_HAS_REACTOR_NOTIFICATION_QUEUE and</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>recompile
ACE?</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Best
regards,</span><o:p></o:p></p>
</div>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
<div>
<p class=MsoNormal><strong><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Rudy Pot</span></strong><o:p></o:p></p>
</div>
<div>
<div>
<div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Embedded
Computer Systems</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal> <o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><strong><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>AWETA
G&P b.v</span></strong><o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>Postbox
17</span><o:p></o:p></p>
</div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>NL-2630
AA Nootdorp</span><o:p></o:p></p>
</div>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>tel
+31 (0)15 3109961 </span><o:p></o:p></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>fax +31
(0)15 3107321 </span><o:p></o:p></p>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>mail </span><a
href="mailto:rpot@aweta.nl"><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>rpot@aweta.nl</span></a><o:p></o:p></p>
</div>
<div>
<div>
<p class=MsoNormal><span style='font-size:10.0pt;font-family:"Arial","sans-serif"'>web
<a href="blocked::http://www.aweta.com">www.aweta.com</a></span><o:p></o:p></p>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</body>
</html>