Training courses

Kernel and Embedded Linux

Bootlin training courses

Embedded Linux, kernel,
Yocto Project, Buildroot, real-time,
graphics, boot time, debugging...

Bootlin logo

Elixir Cross Referencer

/*	$NetBSD: select.c,v 1.3 2011/11/02 16:49:12 yamt Exp $	*/

/*-
 * Copyright (c)2008 YAMAMOTO Takashi,
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 */

#define	FD_SETSIZE	65536
#include <sys/select.h>
#include <sys/atomic.h>
#include <sys/time.h>

#include <assert.h>
#include <errno.h>
#include <fcntl.h>
#include <pthread.h>
#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>

#define	NPIPE	128
#define	NTHREAD	64
#define	NBALLS	5
#define	VERBOSE	0

#if !defined(RANDOM_MAX)
#define	RANDOM_MAX	((1UL << 31) - 1)
#endif

int fds[NPIPE][2];

volatile unsigned count;

pthread_barrier_t barrier;

static void
dowrite(void)
{
	char buf[1];
	int fd;
	int i;

	i = random() % NPIPE;
	fd = fds[i][1];
#if VERBOSE
	printf("[%p] write %d\n", (void *)pthread_self(), fd);
#endif
	if (write(fd, buf, sizeof(buf)) == -1) {
		perror("write");
		abort();
	}
}

static void *
f(void *dummy)
{

	pthread_barrier_wait(&barrier);

	for (;;) {
		struct timeval to;
		fd_set oset;
		fd_set set;
		int maxfd = -1;
		int nfd = 0;
		int ret;
		int fd;
		int i;

		FD_ZERO(&set);
		do {
			for (i = 0; i < NPIPE; i++) {
				fd = fds[i][0];
				if (fd > FD_SETSIZE) {
					fprintf(stderr,
					    "fd(%d) > FD_SETSIZE(%d)\n",
					    fd, FD_SETSIZE);
					abort();
				}
				if (random() & 1) {
					assert(!FD_ISSET(fd, &set));
					FD_SET(fd, &set);
					nfd++;
					if (fd > maxfd) {
						maxfd = fd;
					}
				}
			}
		} while (nfd == 0);
		memcpy(&oset, &set, sizeof(oset));
		memset(&to, 0, sizeof(to));
		to.tv_sec = random() % 10;
		to.tv_usec = random() % 1000000;
#if VERBOSE
		printf("[%p] select start to=%lu\n", (void *)pthread_self(),
		    (unsigned long)to.tv_sec);
#endif
		ret = select(maxfd + 1, &set, NULL, NULL, &to);
#if VERBOSE
		printf("[%p] select done ret=%d\n",
		    (void *)pthread_self(), ret);
#endif
		if (ret == -1) {
			perror("select");
			abort();
		}
		if (ret > nfd) {
			fprintf(stderr, "[%p] unexpected return value %d\n",
			    (void *)pthread_self(), ret);
			abort();
		}
		if (ret > NBALLS) {
			fprintf(stderr, "[%p] unexpected return value %d"
			    " > NBALLS\n",
			    (void *)pthread_self(), ret);
			abort();
		}
		nfd = 0;
		for (fd = 0; fd <= maxfd; fd++) {
			if (FD_ISSET(fd, &set)) {
				char buf[1];

#if VERBOSE
				printf("[%p] read %d\n",
				    (void *)pthread_self(), fd);
#endif
				if (!FD_ISSET(fd, &oset)) {
					fprintf(stderr, "[%p] unexpected\n",
					    (void *)pthread_self());
					abort();
				}
				if (read(fd, buf, sizeof(buf)) == -1) {
					if (errno != EAGAIN) {
						perror("read");
						abort();
					}
				} else {
					dowrite();
					atomic_inc_uint(&count);
				}
				nfd++;
			}
		}
		if (ret != nfd) {
			fprintf(stderr, "[%p] ret(%d) != nfd(%d)\n",
			    (void *)pthread_self(), ret, nfd);
			abort();
		}
	}
}

int
main(int argc, char *argv[])
{
	pthread_t pt[NTHREAD];
	int i;
	unsigned int secs;
	struct timeval start_tv;
	struct timeval end_tv;
	uint64_t usecs;
	unsigned int result;

	secs = atoi(argv[1]);

	for (i = 0; i < NPIPE; i++) {
		if (pipe(fds[i])) {
			perror("pipe");
			abort();
		}
		if (fcntl(fds[i][0], F_SETFL, O_NONBLOCK) == -1) {
			perror("fcntl");
			abort();
		}
	}
	pthread_barrier_init(&barrier, NULL, NTHREAD + 1);
	for (i = 0; i < NTHREAD; i++) {
		int error = pthread_create(&pt[i], NULL, f, NULL);
		if (error) {
			errno = error;
			perror("pthread_create");
			abort();
		}
	}
	pthread_barrier_wait(&barrier);
	gettimeofday(&start_tv, NULL);
	assert(count == 0);
	for (i = 0; i < NBALLS; i++) {
		dowrite();
	}
	sleep(secs);
	gettimeofday(&end_tv, NULL);
	result = count;
	usecs = (end_tv.tv_sec - start_tv.tv_sec) * 1000000
	    + end_tv.tv_usec - start_tv.tv_usec;
	printf("%u / %f = %f\n", result, (double)usecs / 1000000,
	    (double)result / usecs * 1000000);
	exit(EXIT_SUCCESS);
}