You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							123 lines
						
					
					
						
							2.5 KiB
						
					
					
				
			
		
		
	
	
							123 lines
						
					
					
						
							2.5 KiB
						
					
					
				| #ifndef BUFFERING_H
 | |
| #define BUFFERING_H
 | |
| 
 | |
| #include <stdbool.h>
 | |
| #include <pthread.h>
 | |
| 
 | |
| #ifdef __cplusplus
 | |
| extern "C" {
 | |
| #endif
 | |
| 
 | |
| // Tripple buffering helper
 | |
| 
 | |
| typedef struct TBuffer {
 | |
|     pthread_mutex_t lock;
 | |
|     pthread_cond_t cv;
 | |
|     int efd;
 | |
| 
 | |
|     bool* reading;
 | |
|     int pending_idx;
 | |
| 
 | |
|     int num_bufs;
 | |
|     const char* name;
 | |
| 
 | |
|     void (*release_cb)(void* c, int idx);
 | |
|     void *cb_cookie;
 | |
| 
 | |
|     bool stopped;
 | |
| } TBuffer;
 | |
| 
 | |
| // num_bufs must be at least the number of buffers that can be acquired simultaniously plus two
 | |
| void tbuffer_init(TBuffer *tb, int num_bufs, const char* name);
 | |
| 
 | |
| void tbuffer_init2(TBuffer *tb, int num_bufs, const char* name,
 | |
|   void (*release_cb)(void* c, int idx),
 | |
|   void* cb_cookie);
 | |
| 
 | |
| // returns an eventfd that signals if a buffer is ready and tbuffer_acquire shouldn't to block.
 | |
| // useful to polling on multiple tbuffers.
 | |
| int tbuffer_efd(TBuffer *tb);
 | |
| 
 | |
| // Chooses a buffer that's not reading or pending
 | |
| int tbuffer_select(TBuffer *tb);
 | |
| 
 | |
| // Called when the writer is done with their buffer
 | |
| //  - Wakes up the reader if it's waiting
 | |
| //  - releases the pending buffer if the reader's too slow
 | |
| void tbuffer_dispatch(TBuffer *tb, int idx);
 | |
| 
 | |
| // Called when the reader wants a new buffer, will return -1 when stopped
 | |
| int tbuffer_acquire(TBuffer *tb);
 | |
| 
 | |
| // Called when the reader is done with their buffer
 | |
| void tbuffer_release(TBuffer *tb, int idx);
 | |
| 
 | |
| void tbuffer_release_all(TBuffer *tb);
 | |
| 
 | |
| void tbuffer_stop(TBuffer *tb);
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| // pool: buffer pool + queue thing...
 | |
| 
 | |
| #define POOL_MAX_TBUFS 8
 | |
| #define POOL_MAX_QUEUES 8
 | |
| 
 | |
| typedef struct Pool Pool;
 | |
| 
 | |
| typedef struct PoolQueue {
 | |
|   pthread_mutex_t lock;
 | |
|   pthread_cond_t cv;
 | |
|   Pool* pool;
 | |
|   bool inited;
 | |
|   bool stopped;
 | |
|   int efd;
 | |
|   int num_bufs;
 | |
|   int num;
 | |
|   int head, tail;
 | |
|   int* idx;
 | |
| } PoolQueue;
 | |
| 
 | |
| int poolq_pop(PoolQueue *s);
 | |
| int poolq_efd(PoolQueue *s);
 | |
| void poolq_release(PoolQueue *c, int idx);
 | |
| 
 | |
| typedef struct Pool {
 | |
|   pthread_mutex_t lock;
 | |
|   bool stopped;
 | |
|   int num_bufs;
 | |
|   int counter;
 | |
| 
 | |
|   int* ts;
 | |
|   int* refcnt;
 | |
| 
 | |
|   void (*release_cb)(void* c, int idx);
 | |
|   void *cb_cookie;
 | |
| 
 | |
|   int num_tbufs;
 | |
|   TBuffer tbufs[POOL_MAX_TBUFS];
 | |
|   PoolQueue queues[POOL_MAX_QUEUES];
 | |
| } Pool;
 | |
| 
 | |
| void pool_init(Pool *s, int num_bufs);
 | |
| void pool_init2(Pool *s, int num_bufs,
 | |
|   void (*release_cb)(void* c, int idx), void* cb_cookie);
 | |
| 
 | |
| TBuffer* pool_get_tbuffer(Pool *s);
 | |
| 
 | |
| PoolQueue* pool_get_queue(Pool *s);
 | |
| void pool_release_queue(PoolQueue *q);
 | |
| 
 | |
| int pool_select(Pool *s);
 | |
| void pool_push(Pool *s, int idx);
 | |
| void pool_acquire(Pool *s, int idx);
 | |
| void pool_release(Pool *s, int idx);
 | |
| void pool_stop(Pool *s);
 | |
| 
 | |
| 
 | |
| #ifdef __cplusplus
 | |
| }  // extern "C"
 | |
| #endif
 | |
| 
 | |
| #endif
 | |
| 
 |