mirror of https://github.com/sipwise/kamailio.git
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.
272 lines
6.2 KiB
272 lines
6.2 KiB
/*
|
|
* $Id$
|
|
*
|
|
* dmq module - distributed message queue
|
|
*
|
|
* Copyright (C) 2011 Bucur Marius - Ovidiu
|
|
*
|
|
* This file is part of Kamailio, a free SIP server.
|
|
*
|
|
* Kamailio 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 2 of the License, or
|
|
* (at your option) any later version
|
|
*
|
|
* Kamailio 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*/
|
|
|
|
#include "dmq.h"
|
|
#include "peer.h"
|
|
#include "worker.h"
|
|
#include "../../data_lump_rpl.h"
|
|
#include "../../mod_fix.h"
|
|
|
|
/**
|
|
* @brief set the body of a response
|
|
*/
|
|
static int set_reply_body(struct sip_msg* msg, str* body, str* content_type)
|
|
{
|
|
char* buf;
|
|
int len;
|
|
int value_len;
|
|
str nb = *body;
|
|
str nc = *content_type;
|
|
|
|
/* add content-type */
|
|
value_len = nc.len;
|
|
len=sizeof("Content-Type: ") - 1 + value_len + CRLF_LEN;
|
|
buf=pkg_malloc(sizeof(char)*(len));
|
|
|
|
if (buf==0) {
|
|
LM_ERR("out of pkg memory\n");
|
|
return -1;
|
|
}
|
|
memcpy(buf, "Content-Type: ", sizeof("Content-Type: ") - 1);
|
|
memcpy(buf+sizeof("Content-Type: ") - 1, nc.s, value_len);
|
|
memcpy(buf+sizeof("Content-Type: ") - 1 + value_len, CRLF, CRLF_LEN);
|
|
if (add_lump_rpl(msg, buf, len, LUMP_RPL_HDR) == 0) {
|
|
LM_ERR("failed to insert content-type lump\n");
|
|
pkg_free(buf);
|
|
return -1;
|
|
}
|
|
pkg_free(buf);
|
|
|
|
/* add body */
|
|
if (add_lump_rpl(msg, nb.s, nb.len, LUMP_RPL_BODY) == 0) {
|
|
LM_ERR("cannot add body lump\n");
|
|
return -1;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* @brief dmq worker loop
|
|
*/
|
|
void worker_loop(int id)
|
|
{
|
|
dmq_worker_t* worker;
|
|
dmq_job_t* current_job;
|
|
peer_reponse_t peer_response;
|
|
int ret_value;
|
|
|
|
worker = &workers[id];
|
|
for(;;) {
|
|
LM_DBG("dmq_worker [%d %d] getting lock\n", id, my_pid());
|
|
lock_get(&worker->lock);
|
|
LM_DBG("dmq_worker [%d %d] lock acquired\n", id, my_pid());
|
|
/* multiple lock_release calls might be performed, so remove
|
|
* from queue until empty */
|
|
do {
|
|
/* fill the response with 0's */
|
|
memset(&peer_response, 0, sizeof(peer_response));
|
|
current_job = job_queue_pop(worker->queue);
|
|
/* job_queue_pop might return NULL if queue is empty */
|
|
if(current_job) {
|
|
ret_value = current_job->f(current_job->msg, &peer_response);
|
|
if(ret_value < 0) {
|
|
LM_ERR("running job failed\n");
|
|
continue;
|
|
}
|
|
/* add the body to the reply */
|
|
if(peer_response.body.s) {
|
|
if(set_reply_body(current_job->msg, &peer_response.body,
|
|
&peer_response.content_type) < 0) {
|
|
LM_ERR("error adding lumps\n");
|
|
continue;
|
|
}
|
|
}
|
|
/* send the reply */
|
|
if(slb.freply(current_job->msg, peer_response.resp_code,
|
|
&peer_response.reason) < 0)
|
|
{
|
|
LM_ERR("error sending reply\n");
|
|
}
|
|
|
|
/* if body given, free the lumps and free the body */
|
|
if(peer_response.body.s) {
|
|
del_nonshm_lump_rpl(¤t_job->msg->reply_lump);
|
|
pkg_free(peer_response.body.s);
|
|
}
|
|
LM_DBG("sent reply\n");
|
|
shm_free(current_job->msg);
|
|
shm_free(current_job);
|
|
worker->jobs_processed++;
|
|
}
|
|
} while(job_queue_size(worker->queue) > 0);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @brief add a dmq job
|
|
*/
|
|
int add_dmq_job(struct sip_msg* msg, dmq_peer_t* peer)
|
|
{
|
|
int i, found_available = 0;
|
|
int ret;
|
|
dmq_job_t new_job = { 0 };
|
|
dmq_worker_t* worker;
|
|
|
|
ret = 0;
|
|
new_job.f = peer->callback;
|
|
new_job.msg = msg;
|
|
new_job.orig_peer = peer;
|
|
if(!num_workers) {
|
|
LM_ERR("error in add_dmq_job: no workers spawned\n");
|
|
return -1;
|
|
}
|
|
/* initialize the worker with the first one */
|
|
worker = workers;
|
|
/* search for an available worker, or, if not possible,
|
|
* for the least busy one */
|
|
for(i = 0; i < num_workers; i++) {
|
|
if(job_queue_size(workers[i].queue) == 0) {
|
|
worker = &workers[i];
|
|
found_available = 1;
|
|
break;
|
|
} else if(job_queue_size(workers[i].queue)
|
|
< job_queue_size(worker->queue)) {
|
|
worker = &workers[i];
|
|
}
|
|
}
|
|
if(!found_available) {
|
|
LM_DBG("no available worker found, passing job"
|
|
" to the least busy one [%d %d]\n",
|
|
worker->pid, job_queue_size(worker->queue));
|
|
}
|
|
ret = job_queue_push(worker->queue, &new_job);
|
|
lock_release(&worker->lock);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief init dmq worker
|
|
*/
|
|
void init_worker(dmq_worker_t* worker)
|
|
{
|
|
memset(worker, 0, sizeof(*worker));
|
|
lock_init(&worker->lock);
|
|
// acquire the lock for the first time - so that dmq_worker_loop blocks
|
|
lock_get(&worker->lock);
|
|
worker->queue = alloc_job_queue();
|
|
}
|
|
|
|
/**
|
|
* @brief allog dmq job queue
|
|
*/
|
|
job_queue_t* alloc_job_queue()
|
|
{
|
|
job_queue_t* queue;
|
|
|
|
queue = shm_malloc(sizeof(job_queue_t));
|
|
if(queue==NULL) {
|
|
LM_ERR("no more shm\n");
|
|
return NULL;
|
|
}
|
|
memset(queue, 0, sizeof(job_queue_t));
|
|
atomic_set(&queue->count, 0);
|
|
lock_init(&queue->lock);
|
|
return queue;
|
|
}
|
|
|
|
/**
|
|
* @ brief destroy job queue
|
|
*/
|
|
void destroy_job_queue(job_queue_t* queue)
|
|
{
|
|
if(queue!=NULL)
|
|
shm_free(queue);
|
|
}
|
|
|
|
/**
|
|
* @brief return job queue size
|
|
*/
|
|
int job_queue_size(job_queue_t* queue)
|
|
{
|
|
return atomic_get(&queue->count);
|
|
}
|
|
|
|
/**
|
|
* @brief push to job queue
|
|
*/
|
|
int job_queue_push(job_queue_t* queue, dmq_job_t* job)
|
|
{
|
|
/* we need to copy the dmq_job into a newly created dmq_job in shm */
|
|
dmq_job_t* newjob;
|
|
|
|
newjob = shm_malloc(sizeof(dmq_job_t));
|
|
if(newjob==NULL) {
|
|
LM_ERR("no more shm\n");
|
|
return -1;
|
|
}
|
|
|
|
*newjob = *job;
|
|
|
|
lock_get(&queue->lock);
|
|
newjob->prev = NULL;
|
|
newjob->next = queue->back;
|
|
if(queue->back) {
|
|
queue->back->prev = newjob;
|
|
}
|
|
queue->back = newjob;
|
|
if(!queue->front) {
|
|
queue->front = newjob;
|
|
}
|
|
atomic_inc(&queue->count);
|
|
lock_release(&queue->lock);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief pop from job queue
|
|
*/
|
|
dmq_job_t* job_queue_pop(job_queue_t* queue)
|
|
{
|
|
dmq_job_t* front;
|
|
lock_get(&queue->lock);
|
|
if(!queue->front) {
|
|
lock_release(&queue->lock);
|
|
return NULL;
|
|
}
|
|
front = queue->front;
|
|
if(front->prev) {
|
|
queue->front = front->prev;
|
|
front->prev->next = NULL;
|
|
} else {
|
|
queue->front = NULL;
|
|
queue->back = NULL;
|
|
}
|
|
atomic_dec(&queue->count);
|
|
lock_release(&queue->lock);
|
|
return front;
|
|
}
|
|
|