mirror of
https://github.com/pikvm/ustreamer.git
synced 2026-02-18 02:55:46 +00:00
85 lines
3.1 KiB
C
85 lines
3.1 KiB
C
/*****************************************************************************
|
|
# #
|
|
# uStreamer - Lightweight and fast MJPEG-HTTP streamer. #
|
|
# #
|
|
# Copyright (C) 2018-2024 Maxim Devaev <mdevaev@gmail.com> #
|
|
# #
|
|
# This program is free software: you can redistribute it and/or modify #
|
|
# it under the terms of the GNU General Public License as published by #
|
|
# the Free Software Foundation, either version 3 of the License, or #
|
|
# (at your option) any later version. #
|
|
# #
|
|
# This program is distributed in the hope that it will be useful, #
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of #
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
|
|
# GNU General Public License for more details. #
|
|
# #
|
|
# You should have received a copy of the GNU General Public License #
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>. #
|
|
# #
|
|
*****************************************************************************/
|
|
|
|
|
|
#include "ring.h"
|
|
|
|
#include "types.h"
|
|
#include "tools.h"
|
|
#include "queue.h"
|
|
|
|
|
|
int _acquire(us_ring_s *ring, us_queue_s *q, ldf timeout);
|
|
void _release(us_ring_s *ring, us_queue_s *q, uint ri);
|
|
|
|
|
|
us_ring_s *us_ring_init(uint capacity) {
|
|
us_ring_s *ring;
|
|
US_CALLOC(ring, 1);
|
|
US_CALLOC(ring->items, capacity);
|
|
US_CALLOC(ring->places, capacity);
|
|
ring->capacity = capacity;
|
|
ring->producer = us_queue_init(capacity);
|
|
ring->consumer = us_queue_init(capacity);
|
|
for (uint ri = 0; ri < capacity; ++ri) {
|
|
ring->places[ri] = ri; // XXX: Just to avoid casting between pointer and uint
|
|
US_A(!us_queue_put(ring->producer, (void*)(ring->places + ri), 0));
|
|
}
|
|
return ring;
|
|
}
|
|
|
|
void us_ring_destroy(us_ring_s *ring) {
|
|
us_queue_destroy(ring->consumer);
|
|
us_queue_destroy(ring->producer);
|
|
free(ring->places);
|
|
free(ring->items);
|
|
free(ring);
|
|
}
|
|
|
|
int us_ring_producer_acquire(us_ring_s *ring, ldf timeout) {
|
|
return _acquire(ring, ring->producer, timeout);
|
|
}
|
|
|
|
void us_ring_producer_release(us_ring_s *ring, uint ri) {
|
|
_release(ring, ring->consumer, ri);
|
|
}
|
|
|
|
int us_ring_consumer_acquire(us_ring_s *ring, ldf timeout) {
|
|
return _acquire(ring, ring->consumer, timeout);
|
|
}
|
|
|
|
void us_ring_consumer_release(us_ring_s *ring, uint ri) {
|
|
_release(ring, ring->producer, ri);
|
|
}
|
|
|
|
int _acquire(us_ring_s *ring, us_queue_s *q, ldf timeout) {
|
|
(void)ring;
|
|
uint *place;
|
|
if (us_queue_get(q, (void**)&place, timeout) < 0) {
|
|
return -1;
|
|
}
|
|
return *place;
|
|
}
|
|
|
|
void _release(us_ring_s *ring, us_queue_s *q, uint ri) {
|
|
US_A(!us_queue_put(q, (void*)(ring->places + ri), 0));
|
|
}
|