mirror of
https://gitlab.xiph.org/xiph/icecast-server.git
synced 2025-01-03 14:56:34 -05:00
Update: Reworked connection queues
This commit is contained in:
parent
b10e302c2a
commit
c34f715f0d
344
src/connection.c
344
src/connection.c
@ -93,13 +93,19 @@ typedef struct client_queue_tag {
|
|||||||
struct client_queue_tag *next;
|
struct client_queue_tag *next;
|
||||||
} client_queue_entry_t;
|
} client_queue_entry_t;
|
||||||
|
|
||||||
static spin_t _connection_lock; // protects _current_id, _con_queue, _con_queue_tail
|
typedef struct {
|
||||||
|
client_queue_entry_t *head;
|
||||||
|
client_queue_entry_t **tail;
|
||||||
|
mutex_t mutex;
|
||||||
|
} client_queue_t;
|
||||||
|
|
||||||
|
static spin_t _connection_lock; // protects _current_id
|
||||||
static volatile connection_id_t _current_id = 0;
|
static volatile connection_id_t _current_id = 0;
|
||||||
static int _initialized = 0;
|
static int _initialized = 0;
|
||||||
|
|
||||||
static volatile client_queue_entry_t *_req_queue = NULL, **_req_queue_tail = &_req_queue;
|
static client_queue_t _request_queue;
|
||||||
static volatile client_queue_entry_t *_con_queue = NULL, **_con_queue_tail = &_con_queue;
|
static client_queue_t _connection_queue;
|
||||||
static volatile client_queue_entry_t *_body_queue = NULL, **_body_queue_tail = &_body_queue;
|
static client_queue_t _body_queue;
|
||||||
static bool tls_ok = false;
|
static bool tls_ok = false;
|
||||||
static tls_ctx_t *tls_ctx;
|
static tls_ctx_t *tls_ctx;
|
||||||
|
|
||||||
@ -111,6 +117,54 @@ rwlock_t _source_shutdown_rwlock;
|
|||||||
static void _handle_connection(void);
|
static void _handle_connection(void);
|
||||||
static void get_tls_certificate(ice_config_t *config);
|
static void get_tls_certificate(ice_config_t *config);
|
||||||
|
|
||||||
|
static void client_queue_init(client_queue_t *queue)
|
||||||
|
{
|
||||||
|
memset(queue, 0, sizeof(*queue));
|
||||||
|
queue->tail = &(queue->head);
|
||||||
|
thread_mutex_create(&(queue->mutex));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void client_queue_destroy(client_queue_t *queue)
|
||||||
|
{
|
||||||
|
thread_mutex_destroy(&(queue->mutex));
|
||||||
|
}
|
||||||
|
|
||||||
|
static void client_queue_add(client_queue_t *queue, client_queue_entry_t *entry)
|
||||||
|
{
|
||||||
|
thread_mutex_lock(&(queue->mutex));
|
||||||
|
*(queue->tail) = entry;
|
||||||
|
queue->tail = &(entry->next);
|
||||||
|
thread_mutex_unlock(&(queue->mutex));
|
||||||
|
}
|
||||||
|
|
||||||
|
static client_queue_entry_t * client_queue_shift(client_queue_t *queue, client_queue_entry_t *stop)
|
||||||
|
{
|
||||||
|
client_queue_entry_t *ret;
|
||||||
|
|
||||||
|
thread_mutex_lock(&(queue->mutex));
|
||||||
|
ret = queue->head;
|
||||||
|
if (ret) {
|
||||||
|
if (ret == stop) {
|
||||||
|
ret = NULL;
|
||||||
|
} else {
|
||||||
|
queue->head = ret->next;
|
||||||
|
if (!queue->head) {
|
||||||
|
queue->tail = &(queue->head);
|
||||||
|
}
|
||||||
|
ret->next = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
thread_mutex_unlock(&(queue->mutex));
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool client_queue_empty(client_queue_t *queue)
|
||||||
|
{
|
||||||
|
/* No need to lock here as this is a point-in-time thing anyway */
|
||||||
|
return queue->head == NULL ? true : false;
|
||||||
|
}
|
||||||
|
|
||||||
void connection_initialize(void)
|
void connection_initialize(void)
|
||||||
{
|
{
|
||||||
if (_initialized)
|
if (_initialized)
|
||||||
@ -119,12 +173,9 @@ void connection_initialize(void)
|
|||||||
thread_spin_create (&_connection_lock);
|
thread_spin_create (&_connection_lock);
|
||||||
thread_mutex_create(&move_clients_mutex);
|
thread_mutex_create(&move_clients_mutex);
|
||||||
thread_rwlock_create(&_source_shutdown_rwlock);
|
thread_rwlock_create(&_source_shutdown_rwlock);
|
||||||
_req_queue = NULL;
|
client_queue_init(&_request_queue);
|
||||||
_req_queue_tail = &_req_queue;
|
client_queue_init(&_connection_queue);
|
||||||
_con_queue = NULL;
|
client_queue_init(&_body_queue);
|
||||||
_con_queue_tail = &_con_queue;
|
|
||||||
_body_queue = NULL;
|
|
||||||
_body_queue_tail = &_body_queue;
|
|
||||||
|
|
||||||
_initialized = 1;
|
_initialized = 1;
|
||||||
}
|
}
|
||||||
@ -141,6 +192,9 @@ void connection_shutdown(void)
|
|||||||
thread_rwlock_destroy(&_source_shutdown_rwlock);
|
thread_rwlock_destroy(&_source_shutdown_rwlock);
|
||||||
thread_spin_destroy (&_connection_lock);
|
thread_spin_destroy (&_connection_lock);
|
||||||
thread_mutex_destroy(&move_clients_mutex);
|
thread_mutex_destroy(&move_clients_mutex);
|
||||||
|
client_queue_destroy(&_request_queue);
|
||||||
|
client_queue_destroy(&_connection_queue);
|
||||||
|
client_queue_destroy(&_body_queue);
|
||||||
|
|
||||||
_initialized = 0;
|
_initialized = 0;
|
||||||
}
|
}
|
||||||
@ -391,161 +445,118 @@ int connection_read_put_back(connection_t *con, const void *buf, size_t len)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* add client to connection queue. At this point some header information
|
/* run along queue checking for any data that has come in or a timeout */
|
||||||
* has been collected, so we now pass it onto the connection thread for
|
static bool process_request_queue_one (client_queue_entry_t *node, int timeout)
|
||||||
* further processing
|
|
||||||
*/
|
|
||||||
static void _add_connection(client_queue_entry_t *node)
|
|
||||||
{
|
{
|
||||||
thread_spin_lock(&_connection_lock);
|
client_t *client = node->client;
|
||||||
*_con_queue_tail = node;
|
int len = PER_CLIENT_REFBUF_SIZE - 1 - node->offset;
|
||||||
_con_queue_tail = (volatile client_queue_entry_t **) &node->next;
|
char *buf = client->refbuf->data + node->offset;
|
||||||
thread_spin_unlock(&_connection_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
ICECAST_LOG_DDEBUG("Checking on client %p", client);
|
||||||
|
|
||||||
/* this returns queued clients for the connection thread. headers are
|
if (client->con->tlsmode == ICECAST_TLSMODE_AUTO || client->con->tlsmode == ICECAST_TLSMODE_AUTO_NO_PLAIN) {
|
||||||
* already provided, but need to be parsed.
|
char peak;
|
||||||
*/
|
if (recv(client->con->sock, &peak, 1, MSG_PEEK) == 1) {
|
||||||
static client_queue_entry_t *_get_connection(void)
|
if (peak == 0x16) { /* TLS Record Protocol Content type 0x16 == Handshake */
|
||||||
{
|
connection_uses_tls(client->con);
|
||||||
client_queue_entry_t *node = NULL;
|
}
|
||||||
|
}
|
||||||
thread_spin_lock(&_connection_lock);
|
|
||||||
|
|
||||||
if (_con_queue){
|
|
||||||
node = (client_queue_entry_t *)_con_queue;
|
|
||||||
_con_queue = node->next;
|
|
||||||
if (_con_queue == NULL)
|
|
||||||
_con_queue_tail = &_con_queue;
|
|
||||||
node->next = NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
thread_spin_unlock(&_connection_lock);
|
if (len > 0) {
|
||||||
return node;
|
if (client->con->con_time + timeout <= time(NULL)) {
|
||||||
|
len = 0;
|
||||||
|
} else {
|
||||||
|
len = client_read_bytes(client, buf, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (len > 0 || node->shoutcast > 1) {
|
||||||
|
ssize_t stream_offset = -1;
|
||||||
|
int pass_it = 1;
|
||||||
|
char *ptr;
|
||||||
|
|
||||||
|
if (len < 0 && node->shoutcast > 1)
|
||||||
|
len = 0;
|
||||||
|
|
||||||
|
/* handle \n, \r\n and nsvcap which for some strange reason has
|
||||||
|
* EOL as \r\r\n */
|
||||||
|
node->offset += len;
|
||||||
|
client->refbuf->data[node->offset] = '\000';
|
||||||
|
do {
|
||||||
|
if (node->shoutcast == 1) {
|
||||||
|
/* password line */
|
||||||
|
if (strstr (client->refbuf->data, "\r\r\n") != NULL)
|
||||||
|
break;
|
||||||
|
if (strstr (client->refbuf->data, "\r\n") != NULL)
|
||||||
|
break;
|
||||||
|
if (strstr (client->refbuf->data, "\n") != NULL)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* stream_offset refers to the start of any data sent after the
|
||||||
|
* http style headers, we don't want to lose those */
|
||||||
|
ptr = strstr(client->refbuf->data, "\r\r\n\r\r\n");
|
||||||
|
if (ptr) {
|
||||||
|
stream_offset = (ptr+6) - client->refbuf->data;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
ptr = strstr(client->refbuf->data, "\r\n\r\n");
|
||||||
|
if (ptr) {
|
||||||
|
stream_offset = (ptr+4) - client->refbuf->data;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
ptr = strstr(client->refbuf->data, "\n\n");
|
||||||
|
if (ptr) {
|
||||||
|
stream_offset = (ptr+2) - client->refbuf->data;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pass_it = 0;
|
||||||
|
} while (0);
|
||||||
|
|
||||||
|
ICECAST_LOG_DDEBUG("pass_it=%i, len=%i", pass_it, (int)len);
|
||||||
|
ICECAST_LOG_DDEBUG("Client %p has buffer: %H", client, client->refbuf->data);
|
||||||
|
|
||||||
|
if (pass_it) {
|
||||||
|
if (stream_offset != -1) {
|
||||||
|
connection_read_put_back(client->con, client->refbuf->data + stream_offset, node->offset - stream_offset);
|
||||||
|
node->offset = stream_offset;
|
||||||
|
}
|
||||||
|
client_queue_add(&_connection_queue, node);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (len == 0 || client->con->error) {
|
||||||
|
client_destroy(client);
|
||||||
|
free(node);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* run along queue checking for any data that has come in or a timeout */
|
|
||||||
static void process_request_queue (void)
|
static void process_request_queue (void)
|
||||||
{
|
{
|
||||||
client_queue_entry_t **node_ref = (client_queue_entry_t **)&_req_queue;
|
client_queue_entry_t *stop = NULL;
|
||||||
|
client_queue_entry_t *node;
|
||||||
ice_config_t *config;
|
ice_config_t *config;
|
||||||
int timeout;
|
int timeout;
|
||||||
char peak;
|
|
||||||
|
|
||||||
config = config_get_config();
|
config = config_get_config();
|
||||||
timeout = config->header_timeout;
|
timeout = config->header_timeout;
|
||||||
config_release_config();
|
config_release_config();
|
||||||
|
|
||||||
while (*node_ref) {
|
while ((node = client_queue_shift(&_request_queue, stop))) {
|
||||||
client_queue_entry_t *node = *node_ref;
|
if (!process_request_queue_one(node, timeout)) {
|
||||||
client_t *client = node->client;
|
client_queue_add(&_request_queue, node);
|
||||||
int len = PER_CLIENT_REFBUF_SIZE - 1 - node->offset;
|
if (!stop)
|
||||||
char *buf = client->refbuf->data + node->offset;
|
stop = node;
|
||||||
|
|
||||||
ICECAST_LOG_DDEBUG("Checking on client %p", client);
|
|
||||||
|
|
||||||
if (client->con->tlsmode == ICECAST_TLSMODE_AUTO || client->con->tlsmode == ICECAST_TLSMODE_AUTO_NO_PLAIN) {
|
|
||||||
if (recv(client->con->sock, &peak, 1, MSG_PEEK) == 1) {
|
|
||||||
if (peak == 0x16) { /* TLS Record Protocol Content type 0x16 == Handshake */
|
|
||||||
connection_uses_tls(client->con);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len > 0) {
|
|
||||||
if (client->con->con_time + timeout <= time(NULL)) {
|
|
||||||
len = 0;
|
|
||||||
} else {
|
|
||||||
len = client_read_bytes(client, buf, len);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (len > 0 || node->shoutcast > 1) {
|
|
||||||
ssize_t stream_offset = -1;
|
|
||||||
int pass_it = 1;
|
|
||||||
char *ptr;
|
|
||||||
|
|
||||||
if (len < 0 && node->shoutcast > 1)
|
|
||||||
len = 0;
|
|
||||||
|
|
||||||
/* handle \n, \r\n and nsvcap which for some strange reason has
|
|
||||||
* EOL as \r\r\n */
|
|
||||||
node->offset += len;
|
|
||||||
client->refbuf->data[node->offset] = '\000';
|
|
||||||
do {
|
|
||||||
if (node->shoutcast == 1) {
|
|
||||||
/* password line */
|
|
||||||
if (strstr (client->refbuf->data, "\r\r\n") != NULL)
|
|
||||||
break;
|
|
||||||
if (strstr (client->refbuf->data, "\r\n") != NULL)
|
|
||||||
break;
|
|
||||||
if (strstr (client->refbuf->data, "\n") != NULL)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
/* stream_offset refers to the start of any data sent after the
|
|
||||||
* http style headers, we don't want to lose those */
|
|
||||||
ptr = strstr(client->refbuf->data, "\r\r\n\r\r\n");
|
|
||||||
if (ptr) {
|
|
||||||
stream_offset = (ptr+6) - client->refbuf->data;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
ptr = strstr(client->refbuf->data, "\r\n\r\n");
|
|
||||||
if (ptr) {
|
|
||||||
stream_offset = (ptr+4) - client->refbuf->data;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
ptr = strstr(client->refbuf->data, "\n\n");
|
|
||||||
if (ptr) {
|
|
||||||
stream_offset = (ptr+2) - client->refbuf->data;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
pass_it = 0;
|
|
||||||
} while (0);
|
|
||||||
|
|
||||||
ICECAST_LOG_DDEBUG("pass_it=%i, len=%i", pass_it, (int)len);
|
|
||||||
ICECAST_LOG_DDEBUG("Client %p has buffer: %H", client, client->refbuf->data);
|
|
||||||
|
|
||||||
if (pass_it) {
|
|
||||||
if (stream_offset != -1) {
|
|
||||||
connection_read_put_back(client->con, client->refbuf->data + stream_offset, node->offset - stream_offset);
|
|
||||||
node->offset = stream_offset;
|
|
||||||
}
|
|
||||||
if ((client_queue_entry_t **)_req_queue_tail == &(node->next))
|
|
||||||
_req_queue_tail = (volatile client_queue_entry_t **)node_ref;
|
|
||||||
*node_ref = node->next;
|
|
||||||
node->next = NULL;
|
|
||||||
_add_connection(node);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (len == 0 || client->con->error) {
|
|
||||||
if ((client_queue_entry_t **)_req_queue_tail == &node->next)
|
|
||||||
_req_queue_tail = (volatile client_queue_entry_t **)node_ref;
|
|
||||||
*node_ref = node->next;
|
|
||||||
client_destroy(client);
|
|
||||||
free(node);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
node_ref = &node->next;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_handle_connection();
|
_handle_connection();
|
||||||
}
|
}
|
||||||
|
|
||||||
/* add client to body queue.
|
|
||||||
*/
|
|
||||||
static void _add_body_client(client_queue_entry_t *node)
|
|
||||||
{
|
|
||||||
ICECAST_LOG_DEBUG("Putting client %p in body queue.", node->client);
|
|
||||||
|
|
||||||
thread_spin_lock(&_connection_lock);
|
|
||||||
*_body_queue_tail = node;
|
|
||||||
_body_queue_tail = (volatile client_queue_entry_t **) &node->next;
|
|
||||||
thread_spin_unlock(&_connection_lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
static client_slurp_result_t process_request_body_queue_one(client_queue_entry_t *node, time_t timeout, size_t body_size_limit)
|
static client_slurp_result_t process_request_body_queue_one(client_queue_entry_t *node, time_t timeout, size_t body_size_limit)
|
||||||
{
|
{
|
||||||
client_t *client = node->client;
|
client_t *client = node->client;
|
||||||
@ -586,22 +597,20 @@ static client_slurp_result_t process_request_body_queue_one(client_queue_entry_t
|
|||||||
/* This queue reads data from the body of clients. */
|
/* This queue reads data from the body of clients. */
|
||||||
static void process_request_body_queue (void)
|
static void process_request_body_queue (void)
|
||||||
{
|
{
|
||||||
client_queue_entry_t **node_ref = (client_queue_entry_t **)&_body_queue;
|
client_queue_entry_t *stop = NULL;
|
||||||
|
client_queue_entry_t *node;
|
||||||
ice_config_t *config;
|
ice_config_t *config;
|
||||||
time_t timeout;
|
time_t timeout;
|
||||||
size_t body_size_limit;
|
size_t body_size_limit;
|
||||||
|
|
||||||
ICECAST_LOG_DDEBUG("Processing body queue.");
|
ICECAST_LOG_DDEBUG("Processing body queue.");
|
||||||
|
|
||||||
ICECAST_LOG_DDEBUG("_body_queue=%p, &_body_queue=%p, _body_queue_tail=%p", _body_queue, &_body_queue, _body_queue_tail);
|
|
||||||
|
|
||||||
config = config_get_config();
|
config = config_get_config();
|
||||||
timeout = time(NULL) - config->body_timeout;
|
timeout = time(NULL) - config->body_timeout;
|
||||||
body_size_limit = config->body_size_limit;
|
body_size_limit = config->body_size_limit;
|
||||||
config_release_config();
|
config_release_config();
|
||||||
|
|
||||||
while (*node_ref) {
|
while ((node = client_queue_shift(&_body_queue, stop))) {
|
||||||
client_queue_entry_t *node = *node_ref;
|
|
||||||
client_t *client = node->client;
|
client_t *client = node->client;
|
||||||
client_slurp_result_t res;
|
client_slurp_result_t res;
|
||||||
|
|
||||||
@ -611,29 +620,19 @@ static void process_request_body_queue (void)
|
|||||||
|
|
||||||
res = process_request_body_queue_one(node, timeout, body_size_limit);
|
res = process_request_body_queue_one(node, timeout, body_size_limit);
|
||||||
|
|
||||||
if (res != CLIENT_SLURP_NEEDS_MORE_DATA) {
|
if (res == CLIENT_SLURP_NEEDS_MORE_DATA) {
|
||||||
|
client_queue_add(&_body_queue, node);
|
||||||
|
if (!stop)
|
||||||
|
stop = node;
|
||||||
|
} else {
|
||||||
ICECAST_LOG_DEBUG("Putting client %p back in connection queue.", client);
|
ICECAST_LOG_DEBUG("Putting client %p back in connection queue.", client);
|
||||||
|
|
||||||
if ((client_queue_entry_t **)_body_queue_tail == &(node->next))
|
client_queue_add(&_connection_queue, node);
|
||||||
_body_queue_tail = (volatile client_queue_entry_t **)node_ref;
|
|
||||||
*node_ref = node->next;
|
|
||||||
node->next = NULL;
|
|
||||||
_add_connection(node);
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
node_ref = &node->next;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* add node to the queue of requests. This is where the clients are when
|
|
||||||
* initial http details are read.
|
|
||||||
*/
|
|
||||||
static void _add_request_queue(client_queue_entry_t *node)
|
|
||||||
{
|
|
||||||
*_req_queue_tail = node;
|
|
||||||
_req_queue_tail = (volatile client_queue_entry_t **)&node->next;
|
|
||||||
}
|
|
||||||
|
|
||||||
static client_queue_entry_t *create_client_node(client_t *client)
|
static client_queue_entry_t *create_client_node(client_t *client)
|
||||||
{
|
{
|
||||||
client_queue_entry_t *node = calloc (1, sizeof (client_queue_entry_t));
|
client_queue_entry_t *node = calloc (1, sizeof (client_queue_entry_t));
|
||||||
@ -692,7 +691,7 @@ void connection_queue(connection_t *con)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
_add_request_queue(node);
|
client_queue_add(&_request_queue, node);
|
||||||
stats_event_inc(NULL, "connections");
|
stats_event_inc(NULL, "connections");
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -713,7 +712,7 @@ void connection_accept_loop(void)
|
|||||||
connection_queue(con);
|
connection_queue(con);
|
||||||
duration = 5;
|
duration = 5;
|
||||||
} else {
|
} else {
|
||||||
if (_req_queue == NULL)
|
if (client_queue_empty(&_request_queue) && client_queue_empty(&_body_queue))
|
||||||
duration = 300; /* use longer timeouts when nothing waiting */
|
duration = 300; /* use longer timeouts when nothing waiting */
|
||||||
}
|
}
|
||||||
process_request_queue();
|
process_request_queue();
|
||||||
@ -863,7 +862,7 @@ static void _handle_shoutcast_compatible(client_queue_entry_t *node)
|
|||||||
memmove(client->refbuf->data, headers, node->offset+1);
|
memmove(client->refbuf->data, headers, node->offset+1);
|
||||||
node->shoutcast = 2;
|
node->shoutcast = 2;
|
||||||
/* we've checked the password, now send it back for reading headers */
|
/* we've checked the password, now send it back for reading headers */
|
||||||
_add_request_queue(node);
|
client_queue_add(&_request_queue, node);
|
||||||
ICECAST_LOG_DDEBUG("Client %p re-added to request queue", client);
|
ICECAST_LOG_DDEBUG("Client %p re-added to request queue", client);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -937,7 +936,7 @@ static void _handle_connection(void)
|
|||||||
client_queue_entry_t *node;
|
client_queue_entry_t *node;
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
node = _get_connection();
|
node = client_queue_shift(&_connection_queue, NULL);
|
||||||
if (node) {
|
if (node) {
|
||||||
client_t *client = node->client;
|
client_t *client = node->client;
|
||||||
int already_parsed = 0;
|
int already_parsed = 0;
|
||||||
@ -964,7 +963,7 @@ static void _handle_connection(void)
|
|||||||
/* early check if we need more data */
|
/* early check if we need more data */
|
||||||
client_complete(client);
|
client_complete(client);
|
||||||
if (_need_body(node)) {
|
if (_need_body(node)) {
|
||||||
/* Just calling _add_body_client() would do the job.
|
/* Just calling client_queue_add(&_body_queue, node) would do the job.
|
||||||
* However, if the client only has a small body this might work without moving it between queues.
|
* However, if the client only has a small body this might work without moving it between queues.
|
||||||
* -> much faster.
|
* -> much faster.
|
||||||
*/
|
*/
|
||||||
@ -980,7 +979,8 @@ static void _handle_connection(void)
|
|||||||
|
|
||||||
res = process_request_body_queue_one(node, timeout, body_size_limit);
|
res = process_request_body_queue_one(node, timeout, body_size_limit);
|
||||||
if (res != CLIENT_SLURP_SUCCESS) {
|
if (res != CLIENT_SLURP_SUCCESS) {
|
||||||
_add_body_client(node);
|
ICECAST_LOG_DEBUG("Putting client %p in body queue.", client);
|
||||||
|
client_queue_add(&_body_queue, node);
|
||||||
continue;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
ICECAST_LOG_DEBUG("Success on fast lane");
|
ICECAST_LOG_DEBUG("Success on fast lane");
|
||||||
@ -1080,5 +1080,5 @@ void connection_close(connection_t *con)
|
|||||||
void connection_queue_client(client_t *client)
|
void connection_queue_client(client_t *client)
|
||||||
{
|
{
|
||||||
client_queue_entry_t *node = create_client_node(client);
|
client_queue_entry_t *node = create_client_node(client);
|
||||||
_add_connection(node);
|
client_queue_add(&_connection_queue, node);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user