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.
245 lines
5.8 KiB
245 lines
5.8 KiB
/*
|
|
* $Id$
|
|
*
|
|
* The initial version of this code was written by Dragos Vingarzan
|
|
* (dragos(dot)vingarzan(at)fokus(dot)fraunhofer(dot)de and the
|
|
* Fruanhofer Institute. It was and still is maintained in a separate
|
|
* branch of the original SER. We are therefore migrating it to
|
|
* Kamailio/SR and look forward to maintaining it from here on out.
|
|
* 2011/2012 Smile Communications, Pty. Ltd.
|
|
* ported/maintained/improved by
|
|
* Jason Penton (jason(dot)penton(at)smilecoms.com and
|
|
* Richard Good (richard(dot)good(at)smilecoms.com) as part of an
|
|
* effort to add full IMS support to Kamailio/SR using a new and
|
|
* improved architecture
|
|
*
|
|
* NB: Alot of this code was originally part of OpenIMSCore,
|
|
* FhG Focus. Thanks for great work! This is an effort to
|
|
* break apart the various CSCF functions into logically separate
|
|
* components. We hope this will drive wider use. We also feel
|
|
* that in this way the architecture is more complete and thereby easier
|
|
* to manage in the Kamailio/SR environment
|
|
*
|
|
* 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 "avp_get_base_data_format.h"
|
|
|
|
extern struct cdp_binds *cdp;
|
|
|
|
/*
|
|
* RFC 3588 Basic AVP Data Types
|
|
*
|
|
* http://tools.ietf.org/html/rfc3588#section-4.2
|
|
*
|
|
*/
|
|
|
|
inline int cdp_avp_get_OctetString(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|
|
inline int cdp_avp_get_Integer32(AAA_AVP *avp,int32_t *data)
|
|
{
|
|
int32_t x=0;
|
|
if (avp->data.len<4) {
|
|
LOG(L_ERR,"Error decoding Integer32 from data len < 4 bytes!\n");
|
|
return 0;
|
|
}
|
|
x = get_4bytes(avp->data.s);
|
|
if (data) *data = x;
|
|
return 1;
|
|
}
|
|
|
|
inline int cdp_avp_get_Integer64(AAA_AVP *avp,int64_t *data)
|
|
{
|
|
int64_t x=0;
|
|
int i;
|
|
if (avp->data.len<8) {
|
|
LOG(L_ERR,"Error decoding Integer64 from data len < 8 bytes!\n");
|
|
return 0;
|
|
}
|
|
for(i=0;i<8;i++)
|
|
x = (x*256) + avp->data.s[i];
|
|
if (data) *data = x;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_Unsigned32(AAA_AVP *avp,uint32_t *data)
|
|
{
|
|
if (avp->data.len<4) {
|
|
LOG(L_ERR,"Error decoding Unsigned32 from data len < 4 bytes!\n");
|
|
return 0;
|
|
}
|
|
if (data) *data = ntohl(*((uint32_t*)avp->data.s));
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_Unsigned64(AAA_AVP *avp,uint64_t *data)
|
|
{
|
|
uint64_t x=0;
|
|
int i;
|
|
if (avp->data.len<8) {
|
|
LOG(L_ERR,"Error decoding Unsigned64 from data len < 8 bytes!\n");
|
|
return 0;
|
|
}
|
|
for(i=0;i<8;i++)
|
|
x = (x*256) + avp->data.s[i];
|
|
if (data) *data = x;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_Float32(AAA_AVP *avp,float *data)
|
|
{
|
|
float x;
|
|
uint32_t y;
|
|
if (avp->data.len<4) {
|
|
LOG(L_ERR,"Error decoding Float32 from data len < 4 bytes!\n");
|
|
return 0;
|
|
}
|
|
y = cdp_avp_get_Unsigned32(avp,(uint32_t*)data);//TODO - check if this is correct
|
|
memcpy(&x,&y,sizeof(uint32_t));
|
|
//*((uint32_t*)&x)=y;
|
|
if (data) *data = y;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_Float64(AAA_AVP *avp,double *data)
|
|
{
|
|
double x;
|
|
uint64_t y;
|
|
if (avp->data.len<8) {
|
|
LOG(L_ERR,"Error decoding Float64 from data len < 8 bytes!\n");
|
|
return 0;
|
|
}
|
|
y = cdp_avp_get_Unsigned64(avp,(uint64_t*)data);//TODO - check if this is correct
|
|
memcpy(&x,&y,sizeof(uint64_t));
|
|
if (data) *data = y;
|
|
return 1;
|
|
}
|
|
|
|
inline int cdp_avp_get_Grouped(AAA_AVP *avp,AAA_AVP_LIST *data)
|
|
{
|
|
if (data) *data = cdp->AAAUngroupAVPS(avp->data);
|
|
return 1;
|
|
}
|
|
|
|
inline void cdp_avp_free_Grouped(AAA_AVP_LIST *data)
|
|
{
|
|
if (data) cdp->AAAFreeAVPList(data);
|
|
}
|
|
|
|
/*
|
|
* RFC 3588 Derived AVP Data Formats
|
|
*
|
|
* http://tools.ietf.org/html/rfc3588#section-4.3
|
|
*
|
|
*/
|
|
|
|
inline int cdp_avp_get_Address(AAA_AVP *avp,ip_address *data)
|
|
{
|
|
ip_address x;
|
|
bzero(&x,sizeof(ip_address));
|
|
if (avp->data.len<6){
|
|
LOG(L_ERR,"Error decoding Address from AVP data of length %d < 6",avp->data.len);
|
|
goto error;
|
|
}
|
|
x.ai_family = (avp->data.s[0]<<8) | avp->data.s[1];
|
|
switch (x.ai_family){
|
|
case 1:
|
|
x.ai_family = AF_INET;
|
|
memcpy(&(x.ip.v4.s_addr), avp->data.s+2, 4);
|
|
break;
|
|
case 2:
|
|
x.ai_family = AF_INET6;
|
|
if (avp->data.len<18){
|
|
LOG(L_ERR,"Error decoding Address (AF_INET6) from AVP data of length %d < 16",avp->data.len);
|
|
goto error;
|
|
}
|
|
memcpy(x.ip.v6.s6_addr,avp->data.s+2,16);
|
|
}
|
|
if (data) *data = x;
|
|
return 1;
|
|
error:
|
|
bzero(&x,sizeof(ip_address));
|
|
if (data) *data = x;
|
|
return 0;
|
|
}
|
|
|
|
inline int cdp_avp_get_Time(AAA_AVP *avp,time_t *data)
|
|
{
|
|
if (avp->data.len<4) {
|
|
LOG(L_ERR,"Error decoding Time from data len < 4 bytes!\n");
|
|
return 0;
|
|
}
|
|
if (data) *data = ntohl(*((uint32_t*)avp->data.s))-EPOCH_UNIX_TO_EPOCH_NTP;
|
|
return 1;
|
|
}
|
|
|
|
inline int cdp_avp_get_UTF8String(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|
|
inline int cdp_avp_get_DiameterIdentity(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_DiameterURI(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_Enumerated(AAA_AVP *avp,int32_t *data)
|
|
{
|
|
int32_t x=0;
|
|
if (!avp || avp->data.len<4) {
|
|
LOG(L_ERR,"Error decoding Enumerated from data len < 4 bytes!\n");
|
|
return 0;
|
|
}
|
|
x = get_4bytes(avp->data.s);
|
|
if (data) *data = x;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_IPFilterRule(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|
|
|
|
inline int cdp_avp_get_QoSFilterRule(AAA_AVP *avp,str *data)
|
|
{
|
|
if (data) *data = avp->data;
|
|
return 1;
|
|
}
|
|
|