summaryrefslogtreecommitdiff
path: root/ACE/tests/Reactor_Notification_Queue_Test.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'ACE/tests/Reactor_Notification_Queue_Test.cpp')
-rw-r--r--ACE/tests/Reactor_Notification_Queue_Test.cpp223
1 files changed, 223 insertions, 0 deletions
diff --git a/ACE/tests/Reactor_Notification_Queue_Test.cpp b/ACE/tests/Reactor_Notification_Queue_Test.cpp
new file mode 100644
index 00000000000..26dfb59e426
--- /dev/null
+++ b/ACE/tests/Reactor_Notification_Queue_Test.cpp
@@ -0,0 +1,223 @@
+/**
+ * @file Reactor_Notification_Queue_Test.cpp
+ *
+ * $Id$
+ *
+ * Verify that the notification queue can be used with large numbers
+ * of events.
+ *
+ * Normally the ACE_Reactor uses a pipe to implement the notify()
+ * methods. ACE can be compiled with
+ * ACE_HAS_REACTOR_NOTIFICATION_QUEUE, with this configuration flag
+ * the Reactor uses a user-space queue to contain the notifications.
+ * A single message is sent through the pipe to indicate "pipe not
+ * empty."
+ *
+ * @author Carlos O'Ryan <coryan@atdesk.com>
+ *
+ */
+
+#include "test_config.h"
+#include "ace/Reactor.h"
+#include "ace/TP_Reactor.h"
+#include "ace/Select_Reactor.h"
+#include "ace/WFMO_Reactor.h"
+
+ACE_RCSID(tests,
+ Reactor_Notification_Queue_Test, "$Id$")
+
+class Event_Handler : public ACE_Event_Handler
+{
+public:
+ Event_Handler(ACE_Reactor * reactor,
+ int max_notifications,
+ char const *test_name);
+
+ /// Run the test
+ void run(void);
+
+ /// Receive the notifications.
+ virtual int handle_exception(ACE_HANDLE);
+
+private:
+ /**
+ * @brief Implement a single iteration.
+ *
+ * Each iteration of the test consists of sending multiple
+ * notifications simultaneously.
+ */
+ void send_notifications (void);
+
+ /**
+ * @brief Return true if the test is finished.
+ */
+ bool done (void) const;
+
+private:
+ /**
+ * @brief The maximum number of notifications in any single
+ * iteration.
+ */
+ int max_notifications_;
+
+ /**
+ * @brief The name of the test
+ */
+ char const * test_name_;
+ /**
+ * @brief Number of notifications received
+ */
+ int notifications_sent_;
+ /**
+ * @brief Number of notifications sent
+ */
+ int notifications_recv_;
+
+ /**
+ * @brief Number of notifications sent on each iteration
+ */
+ int notifications_curr_;
+};
+
+int
+run_main (int, ACE_TCHAR *[])
+{
+ ACE_START_TEST (ACE_TEXT ("Reactor_Notification_Queue_Test"));
+
+#if !defined(ACE_HAS_REACTOR_NOTIFICATION_QUEUE)
+ ACE_DEBUG ((LM_INFO,
+ ACE_TEXT ("Notification queue disabled, ")
+ ACE_TEXT ("small test version, ")
+ ACE_TEXT ("which is of no practical use\n")));
+
+ int max_notifications = 16;
+#else
+ int max_notifications = 1024 * 1024;
+#endif /* ACE_HAS_THREADS */
+
+ {
+ ACE_Reactor select_reactor (
+ new ACE_Select_Reactor,
+ 1);
+
+ Event_Handler handler(&select_reactor,
+ max_notifications,
+ "Select_Reactor");
+
+ handler.run ();
+ }
+
+ {
+ ACE_Reactor tp_reactor (new ACE_TP_Reactor,
+ 1);
+ Event_Handler handler(&tp_reactor,
+ max_notifications,
+ "TP_Reactor");
+ handler.run();
+ }
+
+#if 0
+ /// @@todo: Need to talk to Irfan to see how the WFMO handles this.
+#if defined (ACE_WIN32)
+ {
+ ACE_Reactor wfmo_reactor (new ACE_WFMO_Reactor,
+ 1);
+
+ Event_Handler handler(&wfmo_reactor,
+ max_notifications,
+ "WFMO_Reactor");
+ handler.run();
+ }
+#endif /*ACE_WIN32*/
+#endif /*if 0 */
+ ACE_END_TEST;
+ return 0;
+}
+
+Event_Handler::Event_Handler (
+ ACE_Reactor * reactor,
+ int max_notifications,
+ char const * test_name)
+ : ACE_Event_Handler(reactor)
+ , max_notifications_(max_notifications)
+ , test_name_(test_name)
+ , notifications_sent_(0)
+ , notifications_recv_(0)
+ , notifications_curr_(1)
+{
+}
+
+void
+Event_Handler::run (void)
+{
+ send_notifications ();
+
+ // Run for 30 seconds or until the test is done.
+ for(int i = 0; i != 30 && !done(); ++i)
+ {
+ ACE_Time_Value tv (1,0);
+ reactor ()->run_reactor_event_loop(tv);
+ }
+
+ if(!done())
+ {
+ ACE_ERROR ((LM_ERROR,
+ ACE_TEXT ("Test %s failed due to timeout ")
+ ACE_TEXT (" sent=%d,recv=%d \n"),
+ test_name_,
+ notifications_sent_,
+ notifications_recv_));
+ }
+ else
+ {
+ ACE_DEBUG ((LM_INFO,
+ ACE_TEXT ("Test %s passed sent=%d, recv=%d\n"),
+ test_name_,
+ notifications_sent_,
+ notifications_recv_));
+ }
+}
+
+int
+Event_Handler::handle_exception (ACE_HANDLE)
+{
+ ++notifications_recv_;
+ if(notifications_recv_ == notifications_sent_)
+ {
+ if(notifications_curr_ >= max_notifications_)
+ {
+
+ return 0;
+ }
+ send_notifications();
+ }
+ return 0;
+}
+
+void
+Event_Handler::send_notifications (void)
+{
+ for (int i = 0; i != notifications_curr_; ++i)
+ {
+ if(reactor()->notify (this) == -1)
+ {
+ ACE_ERROR((LM_ERROR,
+ ACE_TEXT ("Cannot send notifications in %s test (%d/%d)\n"),
+ test_name_, i, notifications_curr_));
+ return;
+ }
+
+ ++notifications_sent_;
+ }
+ // ACE_ERROR((LM_ERROR,
+ // "Started iteration with %d notify() calls in test %s\n",
+ // notifications_curr_, test_name_));
+ notifications_curr_ *= 2;
+}
+
+bool
+Event_Handler::done (void) const
+{
+ return (notifications_curr_ >= max_notifications_)
+ && (notifications_sent_ == notifications_recv_);
+}