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.
kamailio/modules/dmq/worker.c

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(&current_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;
}