1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
|
// $Id: Semaphores_2.cpp 91671 2010-09-08 18:39:23Z johnnyw $
// Illustrates the use of the ACE_SV_Semaphore_Complex class and the
// ACE_Malloc class using the ACE_Shared_Memory_Pool (which uses
// System V shared memory). Note that it doesn't matter whether the
// parent or the child creates the semaphore since Semaphore_Complex
// will correctly serialize the intialization of the mutex and synch
// objects.
#include "ace/Malloc_T.h"
#include "ace/Shared_Memory_Pool.h"
#include "ace/SV_Semaphore_Complex.h"
#include "ace/OS_NS_unistd.h"
#if defined (ACE_HAS_SYSV_IPC) && !defined(ACE_LACKS_SYSV_SHMEM)
// Shared memory allocator (note that this chews up the
// ACE_DEFAULT_SEM_KEY).
static ACE_Malloc<ACE_SHARED_MEMORY_POOL, ACE_SV_Semaphore_Simple> my_alloc;
const int SEM_KEY_1 = ACE_DEFAULT_SEM_KEY + 1;
const int SEM_KEY_2 = ACE_DEFAULT_SEM_KEY + 2;
static int
parent (char *shm)
{
char *s = shm;
// Both semaphores are initially created with a count of 0, i.e.,
// they are "locked."
ACE_SV_Semaphore_Complex mutex (SEM_KEY_1, ACE_SV_Semaphore_Complex::ACE_CREATE, 0);
ACE_SV_Semaphore_Complex synch (SEM_KEY_2, ACE_SV_Semaphore_Complex::ACE_CREATE, 0);
// This is a critical section, which is protected by the mutex
// semaphore.
for (char c = 'a'; c <= 'z'; c++)
*s++ = c;
*s = '\0';
if (mutex.release () == -1)
ACE_ERROR ((LM_ERROR, "%p", "parent mutex.release"));
else if (synch.acquire () == -1)
ACE_ERROR ((LM_ERROR, "%p", "parent synch.acquire"));
if (my_alloc.remove () == -1)
ACE_ERROR ((LM_ERROR, "%p\n", "my_alloc.remove"));
if (mutex.remove () == -1)
ACE_ERROR ((LM_ERROR, "%p\n", "mutex.remove"));
if (synch.remove () == -1)
ACE_ERROR ((LM_ERROR, "%p\n", "synch.remove"));
return 0;
}
static int
child (char *shm)
{
// Both semaphores are initially created with a count of 0, i.e.,
// they are "locked."
ACE_SV_Semaphore_Complex mutex (SEM_KEY_1, ACE_SV_Semaphore_Complex::ACE_CREATE, 0);
ACE_SV_Semaphore_Complex synch (SEM_KEY_2, ACE_SV_Semaphore_Complex::ACE_CREATE, 0);
// Perform "busy waiting" here until we acquire the semaphore. This
// isn't really a good design -- it's just to illustrate that you
// can do non-blocking acquire() calls with the ACE System V
// semaphore wrappers.
while (mutex.tryacquire () == -1)
if (errno == EAGAIN)
ACE_DEBUG ((LM_DEBUG, "spinning in child!\n"));
else
ACE_ERROR_RETURN ((LM_ERROR, "child mutex.tryacquire"), 1);
for (char *s = (char *) shm; *s != '\0'; s++)
ACE_DEBUG ((LM_DEBUG, "%c", *s));
ACE_DEBUG ((LM_DEBUG, "\n"));
if (synch.release () == -1)
ACE_ERROR_RETURN ((LM_ERROR, "child synch.release"), 1);
return 0;
}
int
ACE_TMAIN (int, ACE_TCHAR *[])
{
char *shm = (char *) my_alloc.malloc (27);
switch (ACE_OS::fork ())
{
case -1:
ACE_ERROR_RETURN ((LM_ERROR, "fork failed\n"), -1);
/* NOTREACHED */
case 0:
// Child.
return child (shm);
default:
return parent (shm);
}
}
#else
int ACE_TMAIN (int, ACE_TCHAR *[])
{
ACE_ERROR ((LM_ERROR,
"SYSV IPC, or SYSV SHMEM is not supported on this platform\n"));
return 0;
}
#endif /* ACE_HAS_SYSV_IPC */
|