xref: /xnu-12377.61.12/tools/tests/affinity/sets.c (revision 4d495c6e23c53686cf65f45067f79024cf5dcee8)
1*4d495c6eSApple OSS Distributions #include <AvailabilityMacros.h>
2*4d495c6eSApple OSS Distributions #include <mach/thread_policy.h>
3*4d495c6eSApple OSS Distributions #include <mach/mach.h>
4*4d495c6eSApple OSS Distributions #include <mach/mach_error.h>
5*4d495c6eSApple OSS Distributions #include <mach/mach_time.h>
6*4d495c6eSApple OSS Distributions #include <pthread.h>
7*4d495c6eSApple OSS Distributions #include <sys/queue.h>
8*4d495c6eSApple OSS Distributions #include <stdio.h>
9*4d495c6eSApple OSS Distributions #include <stdlib.h>
10*4d495c6eSApple OSS Distributions #include <string.h>
11*4d495c6eSApple OSS Distributions #include <unistd.h>
12*4d495c6eSApple OSS Distributions #include <err.h>
13*4d495c6eSApple OSS Distributions #include <errno.h>
14*4d495c6eSApple OSS Distributions #include <sys/sysctl.h>
15*4d495c6eSApple OSS Distributions 
16*4d495c6eSApple OSS Distributions /*
17*4d495c6eSApple OSS Distributions  * Sets is a multithreaded test/benchmarking program to evaluate
18*4d495c6eSApple OSS Distributions  * affinity set placement in Leopard.
19*4d495c6eSApple OSS Distributions  *
20*4d495c6eSApple OSS Distributions  * The picture here, for each set, is:
21*4d495c6eSApple OSS Distributions  *
22*4d495c6eSApple OSS Distributions  *       free                   work
23*4d495c6eSApple OSS Distributions  *    -> queue --> producer --> queue --> consumer --
24*4d495c6eSApple OSS Distributions  *   |                                               |
25*4d495c6eSApple OSS Distributions  *    -----------------------------------------------
26*4d495c6eSApple OSS Distributions  *
27*4d495c6eSApple OSS Distributions  *       <------ "stage" -----> <------ "stage" ----->
28*4d495c6eSApple OSS Distributions  *
29*4d495c6eSApple OSS Distributions  * We spin off sets of production line threads (2 sets by default).
30*4d495c6eSApple OSS Distributions  * All threads of each line sets the same affinity tag (unless disabled).
31*4d495c6eSApple OSS Distributions  * By default there are 2 stage (worker) threads per production line.
32*4d495c6eSApple OSS Distributions  * A worker thread removes a buffer from an input queue, processses it and
33*4d495c6eSApple OSS Distributions  * queues it on an output queue.  By default the initial stage (producer)
34*4d495c6eSApple OSS Distributions  * writes every byte in a buffer and the other (consumer) stages read every
35*4d495c6eSApple OSS Distributions  * byte. By default the buffers are 1MB (256 pages) in size but this can be
36*4d495c6eSApple OSS Distributions  * overidden.  By default there are 2 buffers per set (again overridable).
37*4d495c6eSApple OSS Distributions  * Worker threads process (iterate over) 10000 buffers by default.
38*4d495c6eSApple OSS Distributions  *
39*4d495c6eSApple OSS Distributions  * With affinity enabled, each producer and consumer thread sets its affinity
40*4d495c6eSApple OSS Distributions  * to the set number, 1 .. N. So the threads of each set share an L2 cache.
41*4d495c6eSApple OSS Distributions  *
42*4d495c6eSApple OSS Distributions  * Buffer management uses pthread mutex/condition variables. A thread blocks
43*4d495c6eSApple OSS Distributions  * when no buffer is available on a queue and it is signaled when a buffer
44*4d495c6eSApple OSS Distributions  * is placed on an empty queue. Queues are tailq'a a la <sys/queue.h>.
45*4d495c6eSApple OSS Distributions  * The queue management is centralized in a single routine: what queues to
46*4d495c6eSApple OSS Distributions  * use as input and output and what function to call for processing is
47*4d495c6eSApple OSS Distributions  * data-driven.
48*4d495c6eSApple OSS Distributions  */
49*4d495c6eSApple OSS Distributions 
50*4d495c6eSApple OSS Distributions pthread_mutex_t funnel;
51*4d495c6eSApple OSS Distributions pthread_cond_t  barrier;
52*4d495c6eSApple OSS Distributions 
53*4d495c6eSApple OSS Distributions uint64_t        timer;
54*4d495c6eSApple OSS Distributions int             threads;
55*4d495c6eSApple OSS Distributions int             threads_ready = 0;
56*4d495c6eSApple OSS Distributions 
57*4d495c6eSApple OSS Distributions int             iterations = 10000;
58*4d495c6eSApple OSS Distributions boolean_t       affinity = FALSE;
59*4d495c6eSApple OSS Distributions boolean_t       halting = FALSE;
60*4d495c6eSApple OSS Distributions boolean_t       cache_config = FALSE;
61*4d495c6eSApple OSS Distributions int             verbosity = 1;
62*4d495c6eSApple OSS Distributions 
63*4d495c6eSApple OSS Distributions typedef struct work {
64*4d495c6eSApple OSS Distributions 	TAILQ_ENTRY(work)       link;
65*4d495c6eSApple OSS Distributions 	int                     *data;
66*4d495c6eSApple OSS Distributions } work_t;
67*4d495c6eSApple OSS Distributions 
68*4d495c6eSApple OSS Distributions /*
69*4d495c6eSApple OSS Distributions  * A work queue, complete with pthread objects for its management
70*4d495c6eSApple OSS Distributions  */
71*4d495c6eSApple OSS Distributions typedef struct work_queue {
72*4d495c6eSApple OSS Distributions 	pthread_mutex_t         mtx;
73*4d495c6eSApple OSS Distributions 	pthread_cond_t          cnd;
74*4d495c6eSApple OSS Distributions 	TAILQ_HEAD(, work)      queue;
75*4d495c6eSApple OSS Distributions 	boolean_t               waiters;
76*4d495c6eSApple OSS Distributions } work_queue_t;
77*4d495c6eSApple OSS Distributions 
78*4d495c6eSApple OSS Distributions /* Worker functions take a integer array and size */
79*4d495c6eSApple OSS Distributions typedef void (worker_fn_t)(int *, int);
80*4d495c6eSApple OSS Distributions 
81*4d495c6eSApple OSS Distributions /* This struct controls the function of a thread */
82*4d495c6eSApple OSS Distributions typedef struct {
83*4d495c6eSApple OSS Distributions 	int                     stagenum;
84*4d495c6eSApple OSS Distributions 	char                    *name;
85*4d495c6eSApple OSS Distributions 	worker_fn_t             *fn;
86*4d495c6eSApple OSS Distributions 	work_queue_t            *input;
87*4d495c6eSApple OSS Distributions 	work_queue_t            *output;
88*4d495c6eSApple OSS Distributions 	struct line_info        *set;
89*4d495c6eSApple OSS Distributions 	pthread_t               thread;
90*4d495c6eSApple OSS Distributions 	work_queue_t            bufq;
91*4d495c6eSApple OSS Distributions } stage_info_t;
92*4d495c6eSApple OSS Distributions 
93*4d495c6eSApple OSS Distributions /* This defines a thread set */
94*4d495c6eSApple OSS Distributions #define WORKERS_MAX 10
95*4d495c6eSApple OSS Distributions typedef struct line_info {
96*4d495c6eSApple OSS Distributions 	int                     setnum;
97*4d495c6eSApple OSS Distributions 	int                     *data;
98*4d495c6eSApple OSS Distributions 	int                     isize;
99*4d495c6eSApple OSS Distributions 	stage_info_t            *stage[WORKERS_MAX];
100*4d495c6eSApple OSS Distributions } line_info_t;
101*4d495c6eSApple OSS Distributions 
102*4d495c6eSApple OSS Distributions #define DBG(x...) do {                          \
103*4d495c6eSApple OSS Distributions 	if (verbosity > 1) {                    \
104*4d495c6eSApple OSS Distributions 	        pthread_mutex_lock(&funnel);    \
105*4d495c6eSApple OSS Distributions 	        printf(x);                      \
106*4d495c6eSApple OSS Distributions 	        pthread_mutex_unlock(&funnel);  \
107*4d495c6eSApple OSS Distributions 	}                                       \
108*4d495c6eSApple OSS Distributions } while (0)
109*4d495c6eSApple OSS Distributions 
110*4d495c6eSApple OSS Distributions #define mutter(x...) do {                       \
111*4d495c6eSApple OSS Distributions 	if (verbosity > 0) {                    \
112*4d495c6eSApple OSS Distributions 	        printf(x);                      \
113*4d495c6eSApple OSS Distributions 	}                                       \
114*4d495c6eSApple OSS Distributions } while (0)
115*4d495c6eSApple OSS Distributions 
116*4d495c6eSApple OSS Distributions #define s_if_plural(x)  (((x) > 1) ? "s" : "")
117*4d495c6eSApple OSS Distributions 
118*4d495c6eSApple OSS Distributions static void
usage()119*4d495c6eSApple OSS Distributions usage()
120*4d495c6eSApple OSS Distributions {
121*4d495c6eSApple OSS Distributions 	fprintf(stderr,
122*4d495c6eSApple OSS Distributions 	    "usage: sets [-a]   Turn affinity on (off)\n"
123*4d495c6eSApple OSS Distributions 	    "            [-b B] Number of buffers per set/line (2)\n"
124*4d495c6eSApple OSS Distributions 	    "            [-c]   Configure for max cache performance\n"
125*4d495c6eSApple OSS Distributions 	    "            [-h]   Print this\n"
126*4d495c6eSApple OSS Distributions 	    "            [-i I] Number of items/buffers to process (1000)\n"
127*4d495c6eSApple OSS Distributions 	    "            [-s S] Number of stages per set/line (2)\n"
128*4d495c6eSApple OSS Distributions 	    "            [-t]   Halt for keyboard input to start\n"
129*4d495c6eSApple OSS Distributions 	    "            [-p P] Number of pages per buffer (256=1MB)]\n"
130*4d495c6eSApple OSS Distributions 	    "            [-w]   Consumer writes data\n"
131*4d495c6eSApple OSS Distributions 	    "            [-v V] Level of verbosity 0..2 (1)\n"
132*4d495c6eSApple OSS Distributions 	    "            [N]    Number of sets/lines (2)\n"
133*4d495c6eSApple OSS Distributions 	    );
134*4d495c6eSApple OSS Distributions 	exit(1);
135*4d495c6eSApple OSS Distributions }
136*4d495c6eSApple OSS Distributions 
137*4d495c6eSApple OSS Distributions /* Trivial producer: write to each byte */
138*4d495c6eSApple OSS Distributions void
writer_fn(int * data,int isize)139*4d495c6eSApple OSS Distributions writer_fn(int *data, int isize)
140*4d495c6eSApple OSS Distributions {
141*4d495c6eSApple OSS Distributions 	int     i;
142*4d495c6eSApple OSS Distributions 
143*4d495c6eSApple OSS Distributions 	for (i = 0; i < isize; i++) {
144*4d495c6eSApple OSS Distributions 		data[i] = i;
145*4d495c6eSApple OSS Distributions 	}
146*4d495c6eSApple OSS Distributions }
147*4d495c6eSApple OSS Distributions 
148*4d495c6eSApple OSS Distributions /* Trivial consumer: read each byte */
149*4d495c6eSApple OSS Distributions void
reader_fn(int * data,int isize)150*4d495c6eSApple OSS Distributions reader_fn(int *data, int isize)
151*4d495c6eSApple OSS Distributions {
152*4d495c6eSApple OSS Distributions 	int     i;
153*4d495c6eSApple OSS Distributions 	int     datum;
154*4d495c6eSApple OSS Distributions 
155*4d495c6eSApple OSS Distributions 	for (i = 0; i < isize; i++) {
156*4d495c6eSApple OSS Distributions 		datum = data[i];
157*4d495c6eSApple OSS Distributions 	}
158*4d495c6eSApple OSS Distributions }
159*4d495c6eSApple OSS Distributions 
160*4d495c6eSApple OSS Distributions /* Consumer reading and writing the buffer */
161*4d495c6eSApple OSS Distributions void
reader_writer_fn(int * data,int isize)162*4d495c6eSApple OSS Distributions reader_writer_fn(int *data, int isize)
163*4d495c6eSApple OSS Distributions {
164*4d495c6eSApple OSS Distributions 	int     i;
165*4d495c6eSApple OSS Distributions 
166*4d495c6eSApple OSS Distributions 	for (i = 0; i < isize; i++) {
167*4d495c6eSApple OSS Distributions 		data[i] += 1;
168*4d495c6eSApple OSS Distributions 	}
169*4d495c6eSApple OSS Distributions }
170*4d495c6eSApple OSS Distributions 
171*4d495c6eSApple OSS Distributions /*
172*4d495c6eSApple OSS Distributions  * This is the central function for every thread.
173*4d495c6eSApple OSS Distributions  * For each invocation, its role is ets by (a pointer to) a stage_info_t.
174*4d495c6eSApple OSS Distributions  */
175*4d495c6eSApple OSS Distributions void *
manager_fn(void * arg)176*4d495c6eSApple OSS Distributions manager_fn(void *arg)
177*4d495c6eSApple OSS Distributions {
178*4d495c6eSApple OSS Distributions 	stage_info_t                    *sp = (stage_info_t *) arg;
179*4d495c6eSApple OSS Distributions 	line_info_t                     *lp = sp->set;
180*4d495c6eSApple OSS Distributions 	kern_return_t                   ret;
181*4d495c6eSApple OSS Distributions 	long                            iteration = 0;
182*4d495c6eSApple OSS Distributions 
183*4d495c6eSApple OSS Distributions 	/*
184*4d495c6eSApple OSS Distributions 	 * If we're using affinity sets (we are by default)
185*4d495c6eSApple OSS Distributions 	 * set our tag to by our thread set number.
186*4d495c6eSApple OSS Distributions 	 */
187*4d495c6eSApple OSS Distributions 	thread_extended_policy_data_t   epolicy;
188*4d495c6eSApple OSS Distributions 	thread_affinity_policy_data_t   policy;
189*4d495c6eSApple OSS Distributions 
190*4d495c6eSApple OSS Distributions 	epolicy.timeshare = FALSE;
191*4d495c6eSApple OSS Distributions 	ret = thread_policy_set(
192*4d495c6eSApple OSS Distributions 		mach_thread_self(), THREAD_EXTENDED_POLICY,
193*4d495c6eSApple OSS Distributions 		(thread_policy_t) &epolicy,
194*4d495c6eSApple OSS Distributions 		THREAD_EXTENDED_POLICY_COUNT);
195*4d495c6eSApple OSS Distributions 	if (ret != KERN_SUCCESS) {
196*4d495c6eSApple OSS Distributions 		printf("thread_policy_set(THREAD_EXTENDED_POLICY) returned %d\n", ret);
197*4d495c6eSApple OSS Distributions 	}
198*4d495c6eSApple OSS Distributions 
199*4d495c6eSApple OSS Distributions 	if (affinity) {
200*4d495c6eSApple OSS Distributions 		policy.affinity_tag = lp->setnum;
201*4d495c6eSApple OSS Distributions 		ret = thread_policy_set(
202*4d495c6eSApple OSS Distributions 			mach_thread_self(), THREAD_AFFINITY_POLICY,
203*4d495c6eSApple OSS Distributions 			(thread_policy_t) &policy,
204*4d495c6eSApple OSS Distributions 			THREAD_AFFINITY_POLICY_COUNT);
205*4d495c6eSApple OSS Distributions 		if (ret != KERN_SUCCESS) {
206*4d495c6eSApple OSS Distributions 			printf("thread_policy_set(THREAD_AFFINITY_POLICY) returned %d\n", ret);
207*4d495c6eSApple OSS Distributions 		}
208*4d495c6eSApple OSS Distributions 	}
209*4d495c6eSApple OSS Distributions 
210*4d495c6eSApple OSS Distributions 	DBG("Starting %s set: %d stage: %d\n", sp->name, lp->setnum, sp->stagenum);
211*4d495c6eSApple OSS Distributions 
212*4d495c6eSApple OSS Distributions 	/*
213*4d495c6eSApple OSS Distributions 	 * Start barrier.
214*4d495c6eSApple OSS Distributions 	 * The tets thread to get here releases everyone and starts the timer.
215*4d495c6eSApple OSS Distributions 	 */
216*4d495c6eSApple OSS Distributions 	pthread_mutex_lock(&funnel);
217*4d495c6eSApple OSS Distributions 	threads_ready++;
218*4d495c6eSApple OSS Distributions 	if (threads_ready == threads) {
219*4d495c6eSApple OSS Distributions 		pthread_mutex_unlock(&funnel);
220*4d495c6eSApple OSS Distributions 		if (halting) {
221*4d495c6eSApple OSS Distributions 			printf("  all threads ready for process %d, "
222*4d495c6eSApple OSS Distributions 			    "hit any key to start", getpid());
223*4d495c6eSApple OSS Distributions 			fflush(stdout);
224*4d495c6eSApple OSS Distributions 			(void) getchar();
225*4d495c6eSApple OSS Distributions 		}
226*4d495c6eSApple OSS Distributions 		pthread_cond_broadcast(&barrier);
227*4d495c6eSApple OSS Distributions 		timer = mach_absolute_time();
228*4d495c6eSApple OSS Distributions 	} else {
229*4d495c6eSApple OSS Distributions 		pthread_cond_wait(&barrier, &funnel);
230*4d495c6eSApple OSS Distributions 		pthread_mutex_unlock(&funnel);
231*4d495c6eSApple OSS Distributions 	}
232*4d495c6eSApple OSS Distributions 
233*4d495c6eSApple OSS Distributions 	do {
234*4d495c6eSApple OSS Distributions 		int             i;
235*4d495c6eSApple OSS Distributions 		work_t          *workp;
236*4d495c6eSApple OSS Distributions 
237*4d495c6eSApple OSS Distributions 		/*
238*4d495c6eSApple OSS Distributions 		 * Get a buffer from the input queue.
239*4d495c6eSApple OSS Distributions 		 * Block if none.
240*4d495c6eSApple OSS Distributions 		 */
241*4d495c6eSApple OSS Distributions 		pthread_mutex_lock(&sp->input->mtx);
242*4d495c6eSApple OSS Distributions 		while (1) {
243*4d495c6eSApple OSS Distributions 			workp = TAILQ_FIRST(&(sp->input->queue));
244*4d495c6eSApple OSS Distributions 			if (workp != NULL) {
245*4d495c6eSApple OSS Distributions 				break;
246*4d495c6eSApple OSS Distributions 			}
247*4d495c6eSApple OSS Distributions 			DBG("    %s[%d,%d] iteration %d waiting for buffer\n",
248*4d495c6eSApple OSS Distributions 			    sp->name, lp->setnum, sp->stagenum, iteration);
249*4d495c6eSApple OSS Distributions 			sp->input->waiters = TRUE;
250*4d495c6eSApple OSS Distributions 			pthread_cond_wait(&sp->input->cnd, &sp->input->mtx);
251*4d495c6eSApple OSS Distributions 			sp->input->waiters = FALSE;
252*4d495c6eSApple OSS Distributions 		}
253*4d495c6eSApple OSS Distributions 		TAILQ_REMOVE(&(sp->input->queue), workp, link);
254*4d495c6eSApple OSS Distributions 		pthread_mutex_unlock(&sp->input->mtx);
255*4d495c6eSApple OSS Distributions 
256*4d495c6eSApple OSS Distributions 		DBG("  %s[%d,%d] iteration %d work %p data %p\n",
257*4d495c6eSApple OSS Distributions 		    sp->name, lp->setnum, sp->stagenum, iteration, workp, workp->data);
258*4d495c6eSApple OSS Distributions 
259*4d495c6eSApple OSS Distributions 		/* Do our stuff with the buffer */
260*4d495c6eSApple OSS Distributions 		(void) sp->fn(workp->data, lp->isize);
261*4d495c6eSApple OSS Distributions 
262*4d495c6eSApple OSS Distributions 		/*
263*4d495c6eSApple OSS Distributions 		 * Place the buffer on the input queue.
264*4d495c6eSApple OSS Distributions 		 * Signal  waiters if required.
265*4d495c6eSApple OSS Distributions 		 */
266*4d495c6eSApple OSS Distributions 		pthread_mutex_lock(&sp->output->mtx);
267*4d495c6eSApple OSS Distributions 		TAILQ_INSERT_TAIL(&(sp->output->queue), workp, link);
268*4d495c6eSApple OSS Distributions 		if (sp->output->waiters) {
269*4d495c6eSApple OSS Distributions 			DBG("    %s[%d,%d] iteration %d signaling work\n",
270*4d495c6eSApple OSS Distributions 			    sp->name, lp->setnum, sp->stagenum, iteration);
271*4d495c6eSApple OSS Distributions 			pthread_cond_signal(&sp->output->cnd);
272*4d495c6eSApple OSS Distributions 		}
273*4d495c6eSApple OSS Distributions 		pthread_mutex_unlock(&sp->output->mtx);
274*4d495c6eSApple OSS Distributions 	} while (++iteration < iterations);
275*4d495c6eSApple OSS Distributions 
276*4d495c6eSApple OSS Distributions 	DBG("Ending %s[%d,%d]\n", sp->name, lp->setnum, sp->stagenum);
277*4d495c6eSApple OSS Distributions 
278*4d495c6eSApple OSS Distributions 	return (void *) iteration;
279*4d495c6eSApple OSS Distributions }
280*4d495c6eSApple OSS Distributions 
281*4d495c6eSApple OSS Distributions #define MAX_CACHE_DEPTH 10
282*4d495c6eSApple OSS Distributions static void
auto_config(int npages,int * nbufs,int * nsets)283*4d495c6eSApple OSS Distributions auto_config(int npages, int *nbufs, int *nsets)
284*4d495c6eSApple OSS Distributions {
285*4d495c6eSApple OSS Distributions 	size_t  len;
286*4d495c6eSApple OSS Distributions 	int     ncpu;
287*4d495c6eSApple OSS Distributions 	int     llc;
288*4d495c6eSApple OSS Distributions 	int64_t cacheconfig[MAX_CACHE_DEPTH];
289*4d495c6eSApple OSS Distributions 	int64_t cachesize[MAX_CACHE_DEPTH];
290*4d495c6eSApple OSS Distributions 
291*4d495c6eSApple OSS Distributions 	mutter("Autoconfiguring...\n");
292*4d495c6eSApple OSS Distributions 
293*4d495c6eSApple OSS Distributions 	len = sizeof(cacheconfig);
294*4d495c6eSApple OSS Distributions 	if (sysctlbyname("hw.cacheconfig",
295*4d495c6eSApple OSS Distributions 	    &cacheconfig[0], &len, NULL, 0) != 0) {
296*4d495c6eSApple OSS Distributions 		printf("Unable to get hw.cacheconfig, %d\n", errno);
297*4d495c6eSApple OSS Distributions 		exit(1);
298*4d495c6eSApple OSS Distributions 	}
299*4d495c6eSApple OSS Distributions 	len = sizeof(cachesize);
300*4d495c6eSApple OSS Distributions 	if (sysctlbyname("hw.cachesize",
301*4d495c6eSApple OSS Distributions 	    &cachesize[0], &len, NULL, 0) != 0) {
302*4d495c6eSApple OSS Distributions 		printf("Unable to get hw.cachesize, %d\n", errno);
303*4d495c6eSApple OSS Distributions 		exit(1);
304*4d495c6eSApple OSS Distributions 	}
305*4d495c6eSApple OSS Distributions 
306*4d495c6eSApple OSS Distributions 	/*
307*4d495c6eSApple OSS Distributions 	 * Find LLC
308*4d495c6eSApple OSS Distributions 	 */
309*4d495c6eSApple OSS Distributions 	for (llc = MAX_CACHE_DEPTH - 1; llc > 0; llc--) {
310*4d495c6eSApple OSS Distributions 		if (cacheconfig[llc] != 0) {
311*4d495c6eSApple OSS Distributions 			break;
312*4d495c6eSApple OSS Distributions 		}
313*4d495c6eSApple OSS Distributions 	}
314*4d495c6eSApple OSS Distributions 
315*4d495c6eSApple OSS Distributions 	/*
316*4d495c6eSApple OSS Distributions 	 * Calculate number of buffers of size pages*4096 bytes
317*4d495c6eSApple OSS Distributions 	 * fit into 90% of an L2 cache.
318*4d495c6eSApple OSS Distributions 	 */
319*4d495c6eSApple OSS Distributions 	*nbufs = cachesize[llc] * 9 / (npages * 4096 * 10);
320*4d495c6eSApple OSS Distributions 	mutter("  L%d (LLC) cache %qd bytes: "
321*4d495c6eSApple OSS Distributions 	    "using %d buffers of size %d bytes\n",
322*4d495c6eSApple OSS Distributions 	    llc, cachesize[llc], *nbufs, (npages * 4096));
323*4d495c6eSApple OSS Distributions 
324*4d495c6eSApple OSS Distributions 	/*
325*4d495c6eSApple OSS Distributions 	 * Calcalute how many sets:
326*4d495c6eSApple OSS Distributions 	 */
327*4d495c6eSApple OSS Distributions 	*nsets = cacheconfig[0] / cacheconfig[llc];
328*4d495c6eSApple OSS Distributions 	mutter("  %qd cpus; %qd cpus per L%d cache: using %d sets\n",
329*4d495c6eSApple OSS Distributions 	    cacheconfig[0], cacheconfig[llc], llc, *nsets);
330*4d495c6eSApple OSS Distributions }
331*4d495c6eSApple OSS Distributions 
332*4d495c6eSApple OSS Distributions void (*producer_fnp)(int *data, int isize) = &writer_fn;
333*4d495c6eSApple OSS Distributions void (*consumer_fnp)(int *data, int isize) = &reader_fn;
334*4d495c6eSApple OSS Distributions 
335*4d495c6eSApple OSS Distributions int
main(int argc,char * argv[])336*4d495c6eSApple OSS Distributions main(int argc, char *argv[])
337*4d495c6eSApple OSS Distributions {
338*4d495c6eSApple OSS Distributions 	int                     i;
339*4d495c6eSApple OSS Distributions 	int                     j;
340*4d495c6eSApple OSS Distributions 	int                     pages = 256; /* 1MB */
341*4d495c6eSApple OSS Distributions 	int                     buffers = 2;
342*4d495c6eSApple OSS Distributions 	int                     sets = 2;
343*4d495c6eSApple OSS Distributions 	int                     stages = 2;
344*4d495c6eSApple OSS Distributions 	int                     *status;
345*4d495c6eSApple OSS Distributions 	line_info_t             *line_info;
346*4d495c6eSApple OSS Distributions 	line_info_t             *lp;
347*4d495c6eSApple OSS Distributions 	stage_info_t            *stage_info;
348*4d495c6eSApple OSS Distributions 	stage_info_t            *sp;
349*4d495c6eSApple OSS Distributions 	kern_return_t           ret;
350*4d495c6eSApple OSS Distributions 	int                     c;
351*4d495c6eSApple OSS Distributions 
352*4d495c6eSApple OSS Distributions 	/* Do switch parsing: */
353*4d495c6eSApple OSS Distributions 	while ((c = getopt(argc, argv, "ab:chi:p:s:twv:")) != -1) {
354*4d495c6eSApple OSS Distributions 		switch (c) {
355*4d495c6eSApple OSS Distributions 		case 'a':
356*4d495c6eSApple OSS Distributions 			affinity = !affinity;
357*4d495c6eSApple OSS Distributions 			break;
358*4d495c6eSApple OSS Distributions 		case 'b':
359*4d495c6eSApple OSS Distributions 			buffers = atoi(optarg);
360*4d495c6eSApple OSS Distributions 			break;
361*4d495c6eSApple OSS Distributions 		case 'c':
362*4d495c6eSApple OSS Distributions 			cache_config = TRUE;
363*4d495c6eSApple OSS Distributions 			break;
364*4d495c6eSApple OSS Distributions 		case 'i':
365*4d495c6eSApple OSS Distributions 			iterations = atoi(optarg);
366*4d495c6eSApple OSS Distributions 			break;
367*4d495c6eSApple OSS Distributions 		case 'p':
368*4d495c6eSApple OSS Distributions 			pages = atoi(optarg);
369*4d495c6eSApple OSS Distributions 			break;
370*4d495c6eSApple OSS Distributions 		case 's':
371*4d495c6eSApple OSS Distributions 			stages = atoi(optarg);
372*4d495c6eSApple OSS Distributions 			if (stages >= WORKERS_MAX) {
373*4d495c6eSApple OSS Distributions 				usage();
374*4d495c6eSApple OSS Distributions 			}
375*4d495c6eSApple OSS Distributions 			break;
376*4d495c6eSApple OSS Distributions 		case 't':
377*4d495c6eSApple OSS Distributions 			halting = TRUE;
378*4d495c6eSApple OSS Distributions 			break;
379*4d495c6eSApple OSS Distributions 		case 'w':
380*4d495c6eSApple OSS Distributions 			consumer_fnp = &reader_writer_fn;
381*4d495c6eSApple OSS Distributions 			break;
382*4d495c6eSApple OSS Distributions 		case 'v':
383*4d495c6eSApple OSS Distributions 			verbosity = atoi(optarg);
384*4d495c6eSApple OSS Distributions 			break;
385*4d495c6eSApple OSS Distributions 		case '?':
386*4d495c6eSApple OSS Distributions 		case 'h':
387*4d495c6eSApple OSS Distributions 		default:
388*4d495c6eSApple OSS Distributions 			usage();
389*4d495c6eSApple OSS Distributions 		}
390*4d495c6eSApple OSS Distributions 	}
391*4d495c6eSApple OSS Distributions 	argc -= optind; argv += optind;
392*4d495c6eSApple OSS Distributions 	if (argc > 0) {
393*4d495c6eSApple OSS Distributions 		sets = atoi(*argv);
394*4d495c6eSApple OSS Distributions 	}
395*4d495c6eSApple OSS Distributions 
396*4d495c6eSApple OSS Distributions 	if (cache_config) {
397*4d495c6eSApple OSS Distributions 		auto_config(pages, &buffers, &sets);
398*4d495c6eSApple OSS Distributions 	}
399*4d495c6eSApple OSS Distributions 
400*4d495c6eSApple OSS Distributions 	pthread_mutex_init(&funnel, NULL);
401*4d495c6eSApple OSS Distributions 	pthread_cond_init(&barrier, NULL);
402*4d495c6eSApple OSS Distributions 
403*4d495c6eSApple OSS Distributions 	/*
404*4d495c6eSApple OSS Distributions 	 * Fire up the worker threads.
405*4d495c6eSApple OSS Distributions 	 */
406*4d495c6eSApple OSS Distributions 	threads = sets * stages;
407*4d495c6eSApple OSS Distributions 	mutter("Launching %d set%s of %d threads with %saffinity, "
408*4d495c6eSApple OSS Distributions 	    "consumer reads%s data\n",
409*4d495c6eSApple OSS Distributions 	    sets, s_if_plural(sets), stages, affinity? "": "no ",
410*4d495c6eSApple OSS Distributions 	    (consumer_fnp == &reader_writer_fn)? " and writes" : "");
411*4d495c6eSApple OSS Distributions 	if (pages < 256) {
412*4d495c6eSApple OSS Distributions 		mutter("  %dkB bytes per buffer, ", pages * 4);
413*4d495c6eSApple OSS Distributions 	} else {
414*4d495c6eSApple OSS Distributions 		mutter("  %dMB bytes per buffer, ", pages / 256);
415*4d495c6eSApple OSS Distributions 	}
416*4d495c6eSApple OSS Distributions 	mutter("%d buffer%s per set ",
417*4d495c6eSApple OSS Distributions 	    buffers, s_if_plural(buffers));
418*4d495c6eSApple OSS Distributions 	if (buffers * pages < 256) {
419*4d495c6eSApple OSS Distributions 		mutter("(total %dkB)\n", buffers * pages * 4);
420*4d495c6eSApple OSS Distributions 	} else {
421*4d495c6eSApple OSS Distributions 		mutter("(total %dMB)\n", buffers * pages / 256);
422*4d495c6eSApple OSS Distributions 	}
423*4d495c6eSApple OSS Distributions 	mutter("  processing %d buffer%s...\n",
424*4d495c6eSApple OSS Distributions 	    iterations, s_if_plural(iterations));
425*4d495c6eSApple OSS Distributions 	line_info = (line_info_t *) malloc(sets * sizeof(line_info_t));
426*4d495c6eSApple OSS Distributions 	stage_info = (stage_info_t *) malloc(sets * stages * sizeof(stage_info_t));
427*4d495c6eSApple OSS Distributions 	for (i = 0; i < sets; i++) {
428*4d495c6eSApple OSS Distributions 		work_t  *work_array;
429*4d495c6eSApple OSS Distributions 
430*4d495c6eSApple OSS Distributions 		lp = &line_info[i];
431*4d495c6eSApple OSS Distributions 
432*4d495c6eSApple OSS Distributions 		lp->setnum = i + 1;
433*4d495c6eSApple OSS Distributions 		lp->isize = pages * 4096 / sizeof(int);
434*4d495c6eSApple OSS Distributions 		lp->data = (int *) malloc(buffers * pages * 4096);
435*4d495c6eSApple OSS Distributions 
436*4d495c6eSApple OSS Distributions 		/* Set up the queue for the workers of this thread set: */
437*4d495c6eSApple OSS Distributions 		for (j = 0; j < stages; j++) {
438*4d495c6eSApple OSS Distributions 			sp = &stage_info[(i * stages) + j];
439*4d495c6eSApple OSS Distributions 			sp->stagenum = j;
440*4d495c6eSApple OSS Distributions 			sp->set = lp;
441*4d495c6eSApple OSS Distributions 			lp->stage[j] = sp;
442*4d495c6eSApple OSS Distributions 			pthread_mutex_init(&sp->bufq.mtx, NULL);
443*4d495c6eSApple OSS Distributions 			pthread_cond_init(&sp->bufq.cnd, NULL);
444*4d495c6eSApple OSS Distributions 			TAILQ_INIT(&sp->bufq.queue);
445*4d495c6eSApple OSS Distributions 			sp->bufq.waiters = FALSE;
446*4d495c6eSApple OSS Distributions 		}
447*4d495c6eSApple OSS Distributions 
448*4d495c6eSApple OSS Distributions 		/*
449*4d495c6eSApple OSS Distributions 		 * Take a second pass through the stages
450*4d495c6eSApple OSS Distributions 		 * to define what the workers are and to interconnect their input/outputs
451*4d495c6eSApple OSS Distributions 		 */
452*4d495c6eSApple OSS Distributions 		for (j = 0; j < stages; j++) {
453*4d495c6eSApple OSS Distributions 			sp = lp->stage[j];
454*4d495c6eSApple OSS Distributions 			if (j == 0) {
455*4d495c6eSApple OSS Distributions 				sp->fn = producer_fnp;
456*4d495c6eSApple OSS Distributions 				sp->name = "producer";
457*4d495c6eSApple OSS Distributions 			} else {
458*4d495c6eSApple OSS Distributions 				sp->fn = consumer_fnp;
459*4d495c6eSApple OSS Distributions 				sp->name = "consumer";
460*4d495c6eSApple OSS Distributions 			}
461*4d495c6eSApple OSS Distributions 			sp->input = &lp->stage[j]->bufq;
462*4d495c6eSApple OSS Distributions 			sp->output = &lp->stage[(j + 1) % stages]->bufq;
463*4d495c6eSApple OSS Distributions 		}
464*4d495c6eSApple OSS Distributions 
465*4d495c6eSApple OSS Distributions 		/* Set up the buffers on the first worker of the set. */
466*4d495c6eSApple OSS Distributions 		work_array = (work_t *)  malloc(buffers * sizeof(work_t));
467*4d495c6eSApple OSS Distributions 		for (j = 0; j < buffers; j++) {
468*4d495c6eSApple OSS Distributions 			work_array[j].data = lp->data + (lp->isize * j);
469*4d495c6eSApple OSS Distributions 			TAILQ_INSERT_TAIL(&lp->stage[0]->bufq.queue, &work_array[j], link);
470*4d495c6eSApple OSS Distributions 			DBG("  empty work item %p for set %d data %p\n",
471*4d495c6eSApple OSS Distributions 			    &work_array[j], i, work_array[j].data);
472*4d495c6eSApple OSS Distributions 		}
473*4d495c6eSApple OSS Distributions 
474*4d495c6eSApple OSS Distributions 		/* Create this set of threads */
475*4d495c6eSApple OSS Distributions 		for (j = 0; j < stages; j++) {
476*4d495c6eSApple OSS Distributions 			if (ret = pthread_create(&lp->stage[j]->thread, NULL,
477*4d495c6eSApple OSS Distributions 			    &manager_fn,
478*4d495c6eSApple OSS Distributions 			    (void *) lp->stage[j])) {
479*4d495c6eSApple OSS Distributions 				err(1, "pthread_create %d,%d", i, j);
480*4d495c6eSApple OSS Distributions 			}
481*4d495c6eSApple OSS Distributions 		}
482*4d495c6eSApple OSS Distributions 	}
483*4d495c6eSApple OSS Distributions 
484*4d495c6eSApple OSS Distributions 	/*
485*4d495c6eSApple OSS Distributions 	 * We sit back anf wait for the slave to finish.
486*4d495c6eSApple OSS Distributions 	 */
487*4d495c6eSApple OSS Distributions 	for (i = 0; i < sets; i++) {
488*4d495c6eSApple OSS Distributions 		lp = &line_info[i];
489*4d495c6eSApple OSS Distributions 		for (j = 0; j < stages; j++) {
490*4d495c6eSApple OSS Distributions 			if (ret = pthread_join(lp->stage[j]->thread, (void **)&status)) {
491*4d495c6eSApple OSS Distributions 				err(1, "pthread_join %d,%d", i, j);
492*4d495c6eSApple OSS Distributions 			}
493*4d495c6eSApple OSS Distributions 			DBG("Thread %d,%d status %d\n", i, j, status);
494*4d495c6eSApple OSS Distributions 		}
495*4d495c6eSApple OSS Distributions 	}
496*4d495c6eSApple OSS Distributions 
497*4d495c6eSApple OSS Distributions 	/*
498*4d495c6eSApple OSS Distributions 	 * See how long the work took.
499*4d495c6eSApple OSS Distributions 	 */
500*4d495c6eSApple OSS Distributions 	timer = mach_absolute_time() - timer;
501*4d495c6eSApple OSS Distributions 	timer = timer / 1000000ULL;
502*4d495c6eSApple OSS Distributions 	printf("%d.%03d seconds elapsed.\n",
503*4d495c6eSApple OSS Distributions 	    (int) (timer / 1000ULL), (int) (timer % 1000ULL));
504*4d495c6eSApple OSS Distributions 
505*4d495c6eSApple OSS Distributions 	return 0;
506*4d495c6eSApple OSS Distributions }
507