summaryrefslogtreecommitdiff
path: root/tests
diff options
context:
space:
mode:
authorschmidt <douglascraigschmidt@users.noreply.github.com>2001-08-29 12:07:23 +0000
committerschmidt <douglascraigschmidt@users.noreply.github.com>2001-08-29 12:07:23 +0000
commitbaa09a735f3daa789b5fea13c4c30762735ca820 (patch)
tree2b46c869dbfd7b37619004c5ceaf59f4d187921a /tests
parent4b2d72cf9c33d4c55114b1fb4ca8c72c4c021cbe (diff)
downloadATCD-baa09a735f3daa789b5fea13c4c30762735ca820.tar.gz
ChangeLogTag:Tue Aug 28 17:51:19 2001 Steve Huston <shuston@riverace.com>
Diffstat (limited to 'tests')
-rw-r--r--tests/Bound_Ptr_Test.cpp556
1 files changed, 556 insertions, 0 deletions
diff --git a/tests/Bound_Ptr_Test.cpp b/tests/Bound_Ptr_Test.cpp
new file mode 100644
index 00000000000..22a2749dee4
--- /dev/null
+++ b/tests/Bound_Ptr_Test.cpp
@@ -0,0 +1,556 @@
+// $Id$
+
+//=============================================================================
+/**
+ * @file Bound_Ptr_Test.cpp
+ *
+ * $Id$
+ *
+ * This example tests the <ACE_Strong_Bound_Ptr> and
+ * <ACE_Weak_Bound_Ptr> and illustrates how they may be dispersed
+ * between multiple threads using an implementation of the Active
+ * Object pattern, which is available in the POSA2 book
+ * <http://www.cs.wustl.edu/~schmidt/POSA>.
+ *
+ * @author Christopher Kohlhoff <chris@kohlhoff.com> */
+ *
+//=============================================================================
+
+#include "test_config.h"
+#include "ace/ACE.h"
+#include "ace/Task.h"
+#include "ace/Synch.h"
+#include "ace/Message_Queue.h"
+#include "ace/Method_Request.h"
+#include "ace/Activation_Queue.h"
+#include "ace/Bound_Ptr.h"
+
+ACE_RCSID (tests,
+ Bound_Ptr_Test,
+ "Bound_Ptr_Test.cpp,v 4.8 2000/04/23 04:43:58 brunsch Exp")
+
+// The following Parent and Child classes illustrate how you might use the
+// ACE_Strong_Bound_Ptr and ACE_Weak_Bound_Ptr together.
+
+struct Child;
+
+// This class should only be created on the heap. Normally it would be an
+// abstract class, and the implementation would be elsewhere.
+struct Parent
+{
+ Parent (void);
+ ~Parent (void);
+
+ // Weak pointer to this object used to hand out new references. Must be
+ // weak since it can't own itself!
+ ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> weak_self_;
+
+ // The parent owns the child. When the parent is destroyed the child will
+ // be automatically deleted.
+ ACE_Strong_Bound_Ptr<Child, ACE_Null_Mutex> child_;
+
+ // Called by the child to perform some operation.
+ void do_something (void);
+
+ static size_t instance_count_;
+};
+
+// This class should only be created on the heap. Normally it would be an
+// abstract class, and the implementation would be elsewhere.
+struct Child
+{
+ Child (ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> parent);
+ ~Child (void);
+
+ // Back pointer to the parent. The child does not own the parent so has no
+ // effect on its lifetime.
+ ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> parent_;
+
+ // Perform some operation. Delegates the work to the parent.
+ void do_something (void);
+
+ static size_t instance_count_;
+};
+
+size_t Parent::instance_count_ = 0;
+
+Parent::Parent (void)
+ : weak_self_(this),
+ child_(new Child(weak_self_))
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Creating Parent object\n")));
+ ++Parent::instance_count_;
+}
+
+Parent::~Parent (void)
+{
+ --Parent::instance_count_;
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Deleting Parent object\n")));
+}
+
+void Parent::do_something (void)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Parent doing something\n")));
+}
+
+size_t Child::instance_count_ = 0;
+
+Child::Child (ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> parent)
+ : parent_(parent)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Creating Child object\n")));
+ ++Child::instance_count_;
+}
+
+Child::~Child (void)
+{
+ --Child::instance_count_;
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Deleting Child object\n")));
+}
+
+void Child::do_something (void)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Child doing something\n")));
+
+ // Using operator-> on a weak pointer will automatically create a strong
+ // pointer as a temporary. This ensures that the object exists for the
+ // lifetime of the call (although it does not check for null).
+ parent_->do_something ();
+
+ // In cases where we may need to call operations on the weak pointer
+ // many times, we can reduce the overhead by explicitly converting to a
+ // strong pointer first.
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> strong_parent (parent_);
+
+ // You can check for null to see if the parent object still exists (in this
+ // case it is not strictly necessary since the child will only exist if the
+ // parent still exists).
+ if (strong_parent == 0)
+ return;
+
+ for (int i = 0; i < 5; ++i)
+ strong_parent->do_something ();
+}
+
+#if defined (ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION)
+template class ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex>;
+template class ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex>;
+template class ACE_Strong_Bound_Ptr<Child, ACE_Null_Mutex>;
+template class ACE_Bound_Ptr_Counter<ACE_Null_Mutex>;
+#elif defined (ACE_HAS_TEMPLATE_INSTANTIATION_PRAGMA)
+#pragma instantiate ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex>
+#pragma instantiate ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex>
+#pragma instantiate ACE_Strong_Bound_Ptr<Child, ACE_Null_Mutex>
+#pragma instantiate ACE_Bound_Ptr_Counter<ACE_Null_Mutex>
+#endif /* ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION */
+
+struct Printer
+{
+ Printer (const char *message);
+ ~Printer (void) ;
+
+ void print (void);
+
+ const char *message_;
+ static size_t instance_count_;
+};
+
+size_t Printer::instance_count_ = 0;
+
+Printer::Printer (const char *message)
+ : message_ (message)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Creating Printer object\n")));
+ ++Printer::instance_count_;
+}
+
+Printer::~Printer (void)
+{
+ --Printer::instance_count_;
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Deleting Printer object\n")));
+}
+
+void
+Printer::print (void)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) %s\n"),
+ this->message_));
+}
+
+#if defined (ACE_HAS_THREADS)
+
+typedef ACE_Strong_Bound_Ptr<Printer, ACE_Thread_Mutex> Printer_var;
+
+/**
+ * @class Scheduler
+ *
+ * @brief The scheduler for the Active Object.
+ *
+ * This class also plays the role of the Proxy and the Servant
+ * in the Active Object pattern. Naturally, these roles could
+ * be split apart from the Scheduler.
+ */
+class Scheduler : public ACE_Task<ACE_SYNCH>
+{
+
+ friend class Method_Request_print;
+ friend class Method_Request_end;
+public:
+ // = Initialization and termination methods.
+ /// Constructor.
+ Scheduler (Scheduler * = 0);
+
+ /// Initializer.
+ virtual int open (void *args = 0);
+
+ /// Terminator.
+ virtual int close (u_long flags = 0);
+
+ /// Destructor.
+ virtual ~Scheduler (void);
+
+ // = These methods are part of the Active Object Proxy interface.
+ void print (Printer_var &printer);
+ void end (void);
+
+protected:
+ /// Runs the Scheduler's event loop, which dequeues <Method_Requests>
+ /// and dispatches them.
+ virtual int svc (void);
+
+private:
+ // = These are the <Scheduler> implementation details.
+ ACE_Activation_Queue activation_queue_;
+ Scheduler *scheduler_;
+};
+
+/**
+ * @class Method_Request_print
+ *
+ * @brief Reification of the <print> method.
+ */
+class Method_Request_print : public ACE_Method_Request
+{
+public:
+ Method_Request_print (Scheduler *,
+ Printer_var &printer);
+ virtual ~Method_Request_print (void);
+
+ /// This is the entry point into the Active Object method.
+ virtual int call (void);
+
+private:
+ Scheduler *scheduler_;
+ Printer_var printer_;
+};
+
+Method_Request_print::Method_Request_print (Scheduler *new_scheduler,
+ Printer_var &printer)
+ : scheduler_ (new_scheduler),
+ printer_ (printer)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Method_Request_print created\n")));
+}
+
+Method_Request_print::~Method_Request_print (void)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Method_Request_print will be deleted.\n")));
+}
+
+int
+Method_Request_print::call (void)
+{
+ // Dispatch the Servant's operation and store the result into the
+ // Future.
+ Printer_var temp = printer_;
+
+ temp->print ();
+
+ return 0;
+}
+
+/**
+ * @class Method_Request_end
+ *
+ * @brief Reification of the <end> method.
+ */
+class Method_Request_end : public ACE_Method_Request
+{
+public:
+ Method_Request_end (Scheduler *new_Prime_Scheduler);
+ virtual ~Method_Request_end (void);
+ virtual int call (void);
+
+private:
+ Scheduler *scheduler_;
+};
+
+Method_Request_end::Method_Request_end (Scheduler *scheduler)
+ : scheduler_ (scheduler)
+{
+}
+
+Method_Request_end::~Method_Request_end (void)
+{
+}
+
+int
+Method_Request_end::call (void)
+{
+ // Shut down the scheduler by deactivating the activation queue's
+ // underlying message queue - should pop all worker threads off their
+ // wait and they'll exit.
+ this->scheduler_->msg_queue ()->deactivate ();
+ return -1;
+}
+
+// Constructor
+// Associates the activation queue with this task's message queue,
+// allowing easy access to the message queue for shutting it down
+// when it's time to stop this object's service threads.
+Scheduler::Scheduler (Scheduler *new_scheduler)
+ : activation_queue_ (msg_queue ()), scheduler_ (new_scheduler)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Scheduler created\n")));
+}
+
+// Destructor
+
+Scheduler::~Scheduler (void)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Scheduler will be destroyed\n")));
+}
+
+// open
+
+int
+Scheduler::open (void *)
+{
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Scheduler open\n")));
+ // Become an Active Object.
+ int num_threads = 3;
+ return this->activate (THR_BOUND | THR_JOINABLE, num_threads);
+}
+
+// close
+
+int
+Scheduler::close (u_long)
+{
+ ACE_DEBUG ((LM_DEBUG, ACE_TEXT ("(%t) rundown\n")));
+ return 0;
+}
+
+// Service..
+
+int
+Scheduler::svc (void)
+{
+ for (;;)
+ {
+ // Dequeue the next method request (we use an strong pointer in
+ // case an exception is thrown in the <call>).
+ ACE_Strong_Bound_Ptr<ACE_Method_Request, ACE_Null_Mutex> mo (this->activation_queue_.dequeue ());
+ if (mo == 0)
+ {
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) activation queue shut down\n")));
+ break;
+ }
+
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) calling method request\n")));
+ // Call it.
+ if (mo->call () == -1)
+ break;
+
+ // Destructor automatically deletes it.
+ }
+
+ return 0;
+}
+
+void
+Scheduler::end (void)
+{
+ this->activation_queue_.enqueue (new Method_Request_end (this));
+}
+
+// Here's where the work takes place.
+
+void
+Scheduler::print (Printer_var &printer)
+{
+ this->activation_queue_.enqueue
+ (new Method_Request_print (this,
+ printer));
+}
+
+// Total number of loops.
+static int n_loops = 10;
+
+#if defined (ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION)
+
+template class ACE_Strong_Bound_Ptr<Printer, ACE_Thread_Mutex>;
+template class ACE_Weak_Bound_Ptr<Printer, ACE_Thread_Mutex>;
+template class ACE_Bound_Ptr_Counter<ACE_Thread_Mutex>;
+template class ACE_Auto_Basic_Ptr<Printer>;
+template class auto_ptr<Printer>;
+template class ACE_Strong_Bound_Ptr<Scheduler, ACE_Null_Mutex>;
+template class ACE_Strong_Bound_Ptr<ACE_Method_Request, ACE_Null_Mutex>;
+
+#elif defined (ACE_HAS_TEMPLATE_INSTANTIATION_PRAGMA)
+
+#pragma instantiate ACE_Strong_Bound_Ptr<Printer, ACE_Thread_Mutex>
+#pragma instantiate ACE_Weak_Bound_Ptr<Printer, ACE_Thread_Mutex>
+#pragma instantiate ACE_Bound_Ptr_Counter<ACE_Thread_Mutex>
+#pragma instantiate ACE_Auto_Basic_Ptr<Printer>
+#pragma instantiate auto_ptr<Printer>
+#pragma instantiate ACE_Strong_Bound_Ptr<Scheduler, ACE_Null_Mutex>
+#pragma instantiate ACE_Strong_Bound_Ptr<ACE_Method_Request, ACE_Null_Mutex>
+
+#endif /* ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION */
+
+#endif /* ACE_HAS_THREADS */
+
+#if defined (ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION)
+template class ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex>;
+template class ACE_Weak_Bound_Ptr<Printer, ACE_Null_Mutex>;
+#elif defined (ACE_HAS_TEMPLATE_INSTANTIATION_PRAGMA)
+#pragma instantiate ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex>
+#pragma instantiate ACE_Weak_Bound_Ptr<Printer, ACE_Null_Mutex>
+#endif /* ACE_HAS_EXPLICIT_TEMPLATE_INSTANTIATION */
+
+
+int
+main (int, ACE_TCHAR *[])
+{
+ ACE_START_TEST (ACE_TEXT ("Bound_Ptr_Test"));
+
+
+ // =========================================================================
+ // The following test uses the ACE_Strong_Bound_Ptr in a single
+ // thread of control, hence we use the ACE_Null_Mutex
+
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) performing synchronous test...\n")));
+
+ Parent *parent1;
+ ACE_NEW_RETURN (parent1,
+ Parent,
+ -1);
+ ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> p8;
+ {
+ // Must get the pointer from the parent object's weak_self_ member.
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p(parent1->weak_self_);
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p1(p);
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p2(p);
+ ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> p3(p);
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p4(p);
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p5 = p2;
+ ACE_Strong_Bound_Ptr<Parent, ACE_Null_Mutex> p6 = p3;
+ ACE_Weak_Bound_Ptr<Parent, ACE_Null_Mutex> p7(p1);
+ p8 = p2;
+ p->child_->do_something ();
+ }
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Parent instance count is %d, expecting 0\n"),
+ Parent::instance_count_));
+ ACE_ASSERT (Parent::instance_count_ == 0);
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Child instance count is %d, expecting 0\n"),
+ Child::instance_count_));
+ ACE_ASSERT (Child::instance_count_ == 0);
+ // Weak pointer should now be set to null.
+ ACE_ASSERT (p8.null ());
+
+ Printer *printer1;
+ ACE_NEW_RETURN (printer1,
+ Printer ("I am printer 1"),
+ -1);
+ ACE_Weak_Bound_Ptr<Printer, ACE_Null_Mutex> r9;
+ {
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r(printer1);
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r1(r);
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r2(r);
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r3(r);
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r4(r);
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r5 = r2;
+ ACE_Strong_Bound_Ptr<Printer, ACE_Null_Mutex> r6 = r1;
+ ACE_Weak_Bound_Ptr<Printer, ACE_Null_Mutex> r7(r1);
+ ACE_Weak_Bound_Ptr<Printer, ACE_Null_Mutex> r8 = r2;
+ r9 = r3;
+ r9->print ();
+ }
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Printer instance count is %d, expecting 0\n"),
+ Printer::instance_count_));
+ ACE_ASSERT (Printer::instance_count_ == 0);
+ // Weak pointer should now be set to null.
+ ACE_ASSERT (r9.null ());
+
+#if defined (ACE_HAS_THREADS)
+
+ // =========================================================================
+ // The following test uses the ACE_Strong_Bound_Ptr in multiple
+ // threads of control.
+
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) performing asynchronous test...\n")));
+
+ Scheduler *scheduler_ptr;
+
+ // Create active objects..
+ ACE_NEW_RETURN (scheduler_ptr,
+ Scheduler (),
+ -1);
+
+ ACE_Strong_Bound_Ptr<Scheduler, ACE_Null_Mutex> scheduler(scheduler_ptr);
+
+ ACE_ASSERT (scheduler->open () != -1);
+
+ {
+ Printer *printer2;
+ ACE_NEW_RETURN (printer2,
+ Printer ("I am printer 2"),
+ -1);
+
+ // Ownership is transferred from the auto_ptr to the strong pointer.
+ auto_ptr<Printer> a (printer2);
+ Printer_var r (a);
+
+ for (int i = 0; i < n_loops; i++)
+ // Spawn off the methods, which run in a separate thread as
+ // active object invocations.
+ scheduler->print (r);
+ }
+
+ // Close things down.
+ scheduler->end ();
+
+ scheduler->wait ();
+
+ ACE_DEBUG ((LM_DEBUG,
+ ACE_TEXT ("(%t) Printer instance count is %d, expecting 0\n"),
+ Printer::instance_count_));
+ ACE_ASSERT (Printer::instance_count_ == 0);
+
+#endif /* ACE_HAS_THREADS */
+ ACE_END_TEST;
+
+ return 0;
+}