Lecture Notes: 18 Condvar
··3 mins
// sequential stack
#include <pthread.h>
#include <stdio.h>
#include <assert.h>
#include <unistd.h>
int stack[5];
int stptr = 0;
void
stack_push(int xx)
{
stack[stptr++] = xx;
}
int
stack_pop()
{
return stack[stptr--];
}
int
main(int _ac, char* _av[])
{
for (int ii = 0; ii < 5; ++ii) {
stack_push(ii);
}
for (int ii = 0; ii < 5; ++ii) {
int yy = stack_pop();
printf("%d\n", yy);
}
return 0;
}
Parallel stack with cond vars:
#include <pthread.h>
#include <stdio.h>
#include <assert.h>
#include <unistd.h>
#include <stdlib.h>
#define STACK_SIZE 5
int stack[STACK_SIZE];
int stptr = 0;
pthread_mutex_t mutex;
pthread_cond_t condv;
void
stack_push(int xx)
{
pthread_mutex_lock(&mutex);
while (stptr >= STACK_SIZE) {
pthread_cond_wait(&condv, &mutex);
}
stack[++stptr] = xx;
pthread_cond_broadcast(&condv);
pthread_mutex_unlock(&mutex);
}
int
stack_pop()
{
pthread_mutex_lock(&mutex);
while (stptr <= 0) {
pthread_cond_wait(&condv, &mutex);
}
int yy = stack[stptr--];
pthread_cond_broadcast(&condv);
pthread_mutex_unlock(&mutex);
return yy;
}
void*
producer_thread(void* arg)
{
int nn = *((int*) arg);
free(arg);
for (int ii = 0; ii < nn; ++ii) {
stack_push(ii);
}
}
int
main(int _ac, char* _av[])
{
pthread_t threads[2];
pthread_mutex_init(&mutex, 0);
pthread_cond_init(&condv, 0);
for (int ii = 0; ii < 2; ++ii) {
int* nn = malloc(sizeof(int));
*nn = 1000;
int rv = pthread_create(&(threads[ii]), 0, producer_thread, nn);
assert(rv == 0);
}
while (1) {
int yy = stack_pop();
printf("%d\n", yy);
usleep(10000);
}
return 0;
}
Here’s the crazy semaphore queue:
#include <pthread.h>
#include <semaphore.h>
#include <stdatomic.h>
#include <assert.h>
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/mman.h>
#define QUEUE_SIZE 8
#define NN 1024
typedef struct shq {
int queue[QUEUE_SIZE];
unsigned int qii; // Input index
unsigned int qjj; // Output index
sem_t isem; // Space
sem_t osem; // Items
} shq;
shq* shared = 0;
int
queue_get()
{
int rv;
rv = sem_wait(&(shared->osem));
assert(rv == 0);
unsigned int ii = atomic_fetch_add(&(shared->qii), 1);
int yy = shared->queue[ii % QUEUE_SIZE];
rv = sem_post(&(shared->isem));
assert(rv == 0);
return yy;
}
void
queue_put(int xx)
{
int rv;
rv = sem_wait(&(shared->isem));
assert(rv == 0);
unsigned int jj = atomic_fetch_add(&(shared->qjj), 1);
shared->queue[jj % QUEUE_SIZE] = xx;
rv = sem_post(&(shared->osem));
assert(rv == 0);
}
void
producer_proc()
{
for (int ii = 0; ii < NN; ++ii) {
queue_put(ii);
}
}
int
main(int _ac, char* _av[])
{
int kids[2];
shared = mmap(0, 1024 * 1024, PROT_READ | PROT_WRITE,
MAP_SHARED | MAP_ANONYMOUS, -1, 0);
shared->qii = 0;
shared->qjj = 0;
sem_init(&(shared->isem), 1, QUEUE_SIZE);
sem_init(&(shared->osem), 1, 0);
for (int ii = 0; ii < 2; ++ii) {
kids[ii] = fork();
if (kids[ii] == 0) {
producer_proc();
return 0;
}
}
for (int ii = 0; ii < 2 * NN; ++ii) {
int yy = queue_get();
printf("%d\n", yy);
usleep(1000);
}
for (int ii = 0; ii < 2; ++ii) {
waitpid(kids[ii], 0, 0);
}
return 0;
}
Too much time? Let’s do more slides.