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.
282 lines
5.8 KiB
282 lines
5.8 KiB
/*
|
|
* Copyright (C) 2010 Daniel-Constantin Mierla (asipto.com)
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
/*!
|
|
* \file
|
|
* \brief KEX :: Kamailio private memory pool statistics
|
|
* \ingroup kex
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
|
|
#include "../../dprint.h"
|
|
#include "../../ut.h"
|
|
#include "../../pt.h"
|
|
#include "../../sr_module.h"
|
|
#include "../../events.h"
|
|
#include "../../mem/mem.h"
|
|
#include "../../mem/shm_mem.h"
|
|
#include "../../rpc.h"
|
|
#include "../../rpc_lookup.h"
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
typedef struct pkg_proc_stats {
|
|
int rank;
|
|
unsigned int pid;
|
|
unsigned long used;
|
|
unsigned long available;
|
|
unsigned long real_used;
|
|
unsigned long total_frags;
|
|
unsigned long total_size;
|
|
} pkg_proc_stats_t;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static pkg_proc_stats_t *_pkg_proc_stats_list = NULL;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static int _pkg_proc_stats_no = 0;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int pkg_proc_stats_init(void)
|
|
{
|
|
_pkg_proc_stats_no = get_max_procs();
|
|
|
|
if(_pkg_proc_stats_no<=0)
|
|
return -1;
|
|
if(_pkg_proc_stats_list!=NULL)
|
|
return -1;
|
|
_pkg_proc_stats_list = (pkg_proc_stats_t*)shm_malloc(
|
|
_pkg_proc_stats_no*sizeof(pkg_proc_stats_t));
|
|
if(_pkg_proc_stats_list==NULL)
|
|
return -1;
|
|
memset(_pkg_proc_stats_list, 0,
|
|
_pkg_proc_stats_no*sizeof(pkg_proc_stats_t));
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int pkg_proc_stats_myinit(int rank)
|
|
{
|
|
struct mem_info info;
|
|
if(_pkg_proc_stats_list==NULL)
|
|
return -1;
|
|
if(process_no>=_pkg_proc_stats_no)
|
|
return -1;
|
|
_pkg_proc_stats_list[process_no].pid = (unsigned int)my_pid();
|
|
_pkg_proc_stats_list[process_no].rank = rank;
|
|
|
|
/* init pkg usage values */
|
|
pkg_info(&info);
|
|
_pkg_proc_stats_list[process_no].available = info.free;
|
|
_pkg_proc_stats_list[process_no].used = info.used;
|
|
_pkg_proc_stats_list[process_no].real_used = info.real_used;
|
|
_pkg_proc_stats_list[process_no].total_size = info.total_size;
|
|
_pkg_proc_stats_list[process_no].total_frags = info.total_frags;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int pkg_proc_stats_destroy(void)
|
|
{
|
|
if(_pkg_proc_stats_list==NULL)
|
|
return -1;
|
|
shm_free(_pkg_proc_stats_list);
|
|
_pkg_proc_stats_list = 0;
|
|
_pkg_proc_stats_no = 0;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static int pkg_proc_update_stats(void *data)
|
|
{
|
|
struct mem_info info;
|
|
if(unlikely(_pkg_proc_stats_list==NULL))
|
|
return -1;
|
|
if(unlikely(process_no>=_pkg_proc_stats_no))
|
|
return -1;
|
|
pkg_info(&info);
|
|
_pkg_proc_stats_list[process_no].available = info.free;
|
|
_pkg_proc_stats_list[process_no].used = info.used;
|
|
_pkg_proc_stats_list[process_no].real_used = info.real_used;
|
|
_pkg_proc_stats_list[process_no].total_frags = info.total_frags;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int register_pkg_proc_stats(void)
|
|
{
|
|
sr_event_register_cb(SREV_PKG_UPDATE_STATS, pkg_proc_update_stats);
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static const char* rpc_pkg_stats_doc[2] = {
|
|
"Private memory (pkg) statistics per process",
|
|
0
|
|
};
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int pkg_proc_get_pid_index(unsigned int pid)
|
|
{
|
|
int i;
|
|
for(i=0; i<_pkg_proc_stats_no; i++)
|
|
{
|
|
if(_pkg_proc_stats_list[i].pid == pid)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
static void rpc_pkg_stats(rpc_t* rpc, void* ctx)
|
|
{
|
|
int i;
|
|
int limit;
|
|
int cval;
|
|
str cname;
|
|
void* th;
|
|
int mode;
|
|
|
|
if(_pkg_proc_stats_list==NULL)
|
|
{
|
|
rpc->fault(ctx, 500, "Not initialized");
|
|
return;
|
|
}
|
|
i = 0;
|
|
mode = 0;
|
|
cval = 0;
|
|
limit = _pkg_proc_stats_no;
|
|
if (rpc->scan(ctx, "*S", &cname) == 1)
|
|
{
|
|
if(cname.len==3 && strncmp(cname.s, "pid", 3)==0)
|
|
mode = 1;
|
|
else if(cname.len==4 && strncmp(cname.s, "rank", 4)==0)
|
|
mode = 2;
|
|
else if(cname.len==5 && strncmp(cname.s, "index", 5)==0)
|
|
mode = 3;
|
|
else {
|
|
rpc->fault(ctx, 500, "Invalid filter type");
|
|
return;
|
|
}
|
|
|
|
if (rpc->scan(ctx, "d", &cval) < 1)
|
|
{
|
|
rpc->fault(ctx, 500, "One more parameter expected");
|
|
return;
|
|
}
|
|
if(mode==1)
|
|
{
|
|
i = pkg_proc_get_pid_index((unsigned int)cval);
|
|
if(i<0)
|
|
{
|
|
rpc->fault(ctx, 500, "No such pid");
|
|
return;
|
|
}
|
|
limit = i + 1;
|
|
} else if(mode==3) {
|
|
i=cval;
|
|
limit = i + 1;
|
|
}
|
|
}
|
|
|
|
for(; i<limit; i++)
|
|
{
|
|
/* add entry node */
|
|
if(mode!=2 || _pkg_proc_stats_list[i].rank==cval)
|
|
{
|
|
if (rpc->add(ctx, "{", &th) < 0)
|
|
{
|
|
rpc->fault(ctx, 500, "Internal error creating rpc");
|
|
return;
|
|
}
|
|
if(_pkg_proc_stats_list[i].pid==0) {
|
|
_pkg_proc_stats_list[i].pid = pt[i].pid;
|
|
_pkg_proc_stats_list[i].total_size = _pkg_proc_stats_list[0].total_size;
|
|
_pkg_proc_stats_list[i].rank = PROC_NOCHLDINIT;
|
|
}
|
|
if(rpc->struct_add(th, "dddddddd",
|
|
"entry", i,
|
|
"pid", _pkg_proc_stats_list[i].pid,
|
|
"rank", _pkg_proc_stats_list[i].rank,
|
|
"used", _pkg_proc_stats_list[i].used,
|
|
"free", _pkg_proc_stats_list[i].available,
|
|
"real_used", _pkg_proc_stats_list[i].real_used,
|
|
"total_size", _pkg_proc_stats_list[i].total_size,
|
|
"total_frags", _pkg_proc_stats_list[i].total_frags
|
|
)<0)
|
|
{
|
|
rpc->fault(ctx, 500, "Internal error creating rpc");
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
*
|
|
*/
|
|
rpc_export_t kex_pkg_rpc[] = {
|
|
{"pkg.stats", rpc_pkg_stats, rpc_pkg_stats_doc, RET_ARRAY},
|
|
{0, 0, 0, 0}
|
|
};
|
|
|
|
/**
|
|
*
|
|
*/
|
|
int pkg_proc_stats_init_rpc(void)
|
|
{
|
|
if (rpc_register_array(kex_pkg_rpc)!=0)
|
|
{
|
|
LM_ERR("failed to register RPC commands\n");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|