ec7bb6e2c6e10ee5e9b7c40e3437068b5d8dc396
[platform/upstream/nspr.git] / nspr / pr / tests / sem.c
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
2 /* This Source Code Form is subject to the terms of the Mozilla Public
3  * License, v. 2.0. If a copy of the MPL was not distributed with this
4  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
5
6 /***********************************************************************
7 **
8 ** Name: sem.c
9 **
10 ** Description: Tests Semaphonre functions.
11 **
12 ** Modification History:
13 ** 20-May-97 AGarcia- Converted the test to accomodate the debug_mode flag.
14 **               The debug mode will print all of the printfs associated with this test.
15 **                       The regress mode will be the default mode. Since the regress tool limits
16 **           the output to a one line status:PASS or FAIL,all of the printf statements
17 **                       have been handled with an if (debug_mode) statement.
18 ** 04-June-97 AGarcia removed the Test_Result function. Regress tool has been updated to
19 **                      recognize the return code from tha main program.
20 ***********************************************************************/
21
22 /***********************************************************************
23 ** Includes
24 ***********************************************************************/
25 /* Used to get the command line option */
26 #include "plgetopt.h"
27
28 #include "nspr.h"
29 #include "prpriv.h"
30
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34
35 PRIntn failed_already=0;
36 PRIntn debug_mode;
37
38 /* 
39         Since we don't have stdin, stdout everywhere, we will fake 
40         it with our in-memory buffers called stdin and stdout.
41 */
42
43 #define SBSIZE 1024
44
45 #include "obsolete/prsem.h"
46
47 static char stdinBuf[SBSIZE];
48 static char stdoutBuf[SBSIZE];
49
50 static PRUintn stdinBufIdx = 0;
51 static PRUintn stdoutBufIdx = 0;
52 static PRStatus finalResult = PR_SUCCESS;
53
54
55 static size_t dread (PRUintn device, char *buf, size_t bufSize)
56 {
57         PRUintn i;
58         
59         /* during first read call, initialize the stdinBuf buffer*/
60         if (stdinBufIdx == 0) {
61                 for (i=0; i<SBSIZE; i++)
62                         stdinBuf[i] = i;
63         }
64
65         /* now copy data from stdinBuf to the given buffer upto bufSize */
66         for (i=0; i<bufSize; i++) {
67                 if (stdinBufIdx == SBSIZE)
68                         break;
69                 buf[i] = stdinBuf[stdinBufIdx++];
70         }
71
72         return i;
73 }
74
75 static size_t dwrite (PRUintn device, char *buf, size_t bufSize)
76 {
77         PRUintn i, j;
78         
79         /* copy data from the given buffer upto bufSize to stdoutBuf */
80         for (i=0; i<bufSize; i++) {
81                 if (stdoutBufIdx == SBSIZE)
82                         break;
83                 stdoutBuf[stdoutBufIdx++] = buf[i];
84         }
85
86         /* during last write call, compare the two buffers */
87         if (stdoutBufIdx == SBSIZE)
88                 for (j=0; j<SBSIZE; j++)
89                         if (stdinBuf[j] != stdoutBuf[j]) {
90                                 if (debug_mode) printf("data mismatch for index= %d \n", j);
91                                 finalResult = PR_FAILURE;
92                         }
93
94         return i;
95 }
96
97 /*------------------ Following is the real test program ---------*/
98 /*
99         Program to copy standard input to standard output.  The program
100         uses two threads.  One reads the input and puts the data in a 
101         double buffer.  The other reads the buffer contents and writes 
102         it to standard output.
103 */
104
105 PRSemaphore     *emptyBufs;     /* number of empty buffers */
106 PRSemaphore *fullBufs;  /* number of buffers that are full */
107
108 #define BSIZE   100
109
110 struct {
111         char data[BSIZE];
112         PRUintn nbytes;         /* number of bytes in this buffer */
113 } buf[2];
114
115 static void PR_CALLBACK reader(void *arg)
116 {
117         PRUintn i = 0;
118         size_t  nbytes;
119         
120         do {
121                 (void) PR_WaitSem(emptyBufs);
122                 nbytes = dread(0, buf[i].data, BSIZE);
123                 buf[i].nbytes = nbytes;
124                 PR_PostSem(fullBufs);
125                 i = (i + 1) % 2;
126         } while (nbytes > 0);
127 }
128
129 static void writer(void)
130 {
131         PRUintn i = 0;
132         size_t  nbytes;
133         
134         do {
135                 (void) PR_WaitSem(fullBufs);
136                 nbytes = buf[i].nbytes;
137                 if (nbytes > 0) {
138                         nbytes = dwrite(1, buf[i].data, nbytes);
139                         PR_PostSem(emptyBufs);
140                         i = (i + 1) % 2;
141                 }
142         } while (nbytes > 0);
143 }
144
145 int main(int argc, char **argv)
146 {
147         PRThread *r;
148
149     PR_STDIO_INIT();
150     PR_Init(PR_USER_THREAD, PR_PRIORITY_NORMAL, 0);
151
152     {
153         /* The command line argument: -d is used to determine if the test is being run
154         in debug mode. The regress tool requires only one line output:PASS or FAIL.
155         All of the printfs associated with this test has been handled with a if (debug_mode)
156         test.
157         Usage: test_name -d
158         */
159         PLOptStatus os;
160         PLOptState *opt = PL_CreateOptState(argc, argv, "d:");
161         while (PL_OPT_EOL != (os = PL_GetNextOpt(opt)))
162         {
163                 if (PL_OPT_BAD == os) continue;
164             switch (opt->option)
165             {
166             case 'd':  /* debug mode */
167                         debug_mode = 1;
168                 break;
169              default:
170                 break;
171             }
172         }
173         PL_DestroyOptState(opt);
174     }        
175
176  /* main test */
177
178     emptyBufs = PR_NewSem(2);   /* two empty buffers */
179
180     fullBufs = PR_NewSem(0);    /* zero full buffers */
181
182         /* create the reader thread */
183         
184         r = PR_CreateThread(PR_USER_THREAD,
185                                       reader, 0, 
186                                       PR_PRIORITY_NORMAL,
187                                       PR_LOCAL_THREAD,
188                                   PR_UNJOINABLE_THREAD,
189                                       0);
190
191         /* Do the writer operation in this thread */
192         writer();
193
194         PR_DestroySem(emptyBufs);
195         PR_DestroySem(fullBufs);
196
197         if (finalResult == PR_SUCCESS) {
198                 if (debug_mode) printf("sem Test Passed.\n");
199         }
200         else{
201                 if (debug_mode) printf("sem Test Failed.\n");
202                 failed_already=1;
203         }
204     PR_Cleanup();
205         if(failed_already)      
206                 return 1;
207         else
208                 return 0;
209 }