pipewire/spa/tests/stress-ringbuffer.c

115 lines
2.4 KiB
C
Raw Normal View History

2017-04-25 19:22:06 +02:00
#define _GNU_SOURCE
#include <unistd.h>
#include <pthread.h>
#include <stdio.h>
#include <sched.h>
#include <spa/utils/ringbuffer.h>
2017-04-25 19:22:06 +02:00
#define ARRAY_SIZE 64
#define MAX_VALUE 0x10000
struct spa_ringbuffer rb;
2017-04-25 19:22:06 +02:00
uint8_t *data;
2017-05-26 08:05:01 +02:00
static int fill_int_array(int *array, int start, int count)
2017-04-25 19:22:06 +02:00
{
2017-05-26 08:05:01 +02:00
int i, j = start;
for (i = 0; i < count; i++) {
array[i] = j;
j = (j + 1) % MAX_VALUE;
}
return j;
2017-04-25 19:22:06 +02:00
}
2017-05-26 08:05:01 +02:00
static int cmp_array(int *array1, int *array2, int count)
2017-04-25 19:22:06 +02:00
{
2017-05-26 08:05:01 +02:00
int i;
for (i = 0; i < count; i++)
if (array1[i] != array2[i]) {
printf("%d != %d at offset %d\n", array1[i], array2[i], i);
return 0;
}
return 1;
2017-04-25 19:22:06 +02:00
}
2017-05-26 08:05:01 +02:00
static void *reader_start(void *arg)
2017-04-25 19:22:06 +02:00
{
2017-05-26 08:05:01 +02:00
int i = 0, a[ARRAY_SIZE], b[ARRAY_SIZE];
unsigned long j = 0, nfailures = 0;
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
printf("reader started on cpu: %d\n", sched_getcpu());
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
i = fill_int_array(a, i, ARRAY_SIZE);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
while (1) {
uint32_t index;
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
if (spa_ringbuffer_get_read_index(&rb, &index) >= ARRAY_SIZE * sizeof(int)) {
spa_ringbuffer_read_data(&rb, data, index & rb.mask, b,
ARRAY_SIZE * sizeof(int));
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
if (!cmp_array(a, b, ARRAY_SIZE)) {
nfailures++;
printf
("failure in chunk %lu - probability: %lu/%lu = %.3f per million\n",
j, nfailures, j, (float) nfailures / (j + 1) * 1000000);
i = (b[0] + ARRAY_SIZE) % MAX_VALUE;
}
i = fill_int_array(a, i, ARRAY_SIZE);
j++;
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
spa_ringbuffer_read_update(&rb, index + ARRAY_SIZE * sizeof(int));
}
}
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
return NULL;
2017-04-25 19:22:06 +02:00
}
2017-05-26 08:05:01 +02:00
static void *writer_start(void *arg)
2017-04-25 19:22:06 +02:00
{
2017-05-26 08:05:01 +02:00
int i = 0, a[ARRAY_SIZE];
printf("writer started on cpu: %d\n", sched_getcpu());
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
i = fill_int_array(a, i, ARRAY_SIZE);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
while (1) {
uint32_t index;
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
if (spa_ringbuffer_get_write_index(&rb, &index) >= ARRAY_SIZE * sizeof(int)) {
spa_ringbuffer_write_data(&rb, data, index & rb.mask, a,
ARRAY_SIZE * sizeof(int));
spa_ringbuffer_write_update(&rb, index + ARRAY_SIZE * sizeof(int));
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
i = fill_int_array(a, i, ARRAY_SIZE);
}
}
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
return NULL;
2017-04-25 19:22:06 +02:00
}
int main(int argc, char *argv[])
{
2017-05-26 08:05:01 +02:00
int size;
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
printf("starting ringbuffer stress test\n");
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
sscanf(argv[1], "%d", &size);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
printf("buffer size (bytes): %d\n", size);
printf("array size (bytes): %ld\n", sizeof(int) * ARRAY_SIZE);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
spa_ringbuffer_init(&rb, size);
data = malloc(size);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
pthread_t reader_thread, writer_thread;
pthread_create(&reader_thread, NULL, reader_start, NULL);
pthread_create(&writer_thread, NULL, writer_start, NULL);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
while (1)
sleep(1);
2017-04-25 19:22:06 +02:00
2017-05-26 08:05:01 +02:00
return 0;
2017-04-25 19:22:06 +02:00
}