blob: a59b606b7dcd8e05325f275bf16baea10b02f5a8 [file] [log] [blame]
/* $Id$ */
/*
* Copyright (C) 2008-2009 Teluu Inc. (http://www.teluu.com)
*
* This program 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.
*
* This program 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 <pj/sock_qos.h>
#include <pj/assert.h>
#include <pj/errno.h>
#include <pj/log.h>
#include <pj/string.h>
#define THIS_FILE "sock_qos_common.c"
#define ALL_FLAGS (PJ_QOS_PARAM_HAS_DSCP | PJ_QOS_PARAM_HAS_802_1_P | \
PJ_QOS_PARAM_HAS_WMM)
/* "Standard" mapping between traffic type and QoS params */
static const pj_qos_params qos_map[] =
{
/* flags dscp prio wmm_prio */
{ALL_FLAGS, 0x00, 0, PJ_QOS_WMM_PRIO_BULK_EFFORT}, /* BE */
{ALL_FLAGS, 0x08, 2, PJ_QOS_WMM_PRIO_BULK}, /* BK */
{ALL_FLAGS, 0x28, 5, PJ_QOS_WMM_PRIO_VIDEO}, /* VI */
{ALL_FLAGS, 0x30, 6, PJ_QOS_WMM_PRIO_VOICE}, /* VO */
{ALL_FLAGS, 0x38, 7, PJ_QOS_WMM_PRIO_VOICE} /* CO */
};
/* Retrieve the mapping for the specified type */
PJ_DEF(pj_status_t) pj_qos_get_params(pj_qos_type type,
pj_qos_params *p_param)
{
PJ_ASSERT_RETURN(type<=PJ_QOS_TYPE_CONTROL && p_param, PJ_EINVAL);
pj_memcpy(p_param, &qos_map[type], sizeof(*p_param));
return PJ_SUCCESS;
}
/* Get the matching traffic type */
PJ_DEF(pj_status_t) pj_qos_get_type( const pj_qos_params *param,
pj_qos_type *p_type)
{
unsigned dscp_type = PJ_QOS_TYPE_BEST_EFFORT,
prio_type = PJ_QOS_TYPE_BEST_EFFORT,
wmm_type = PJ_QOS_TYPE_BEST_EFFORT;
unsigned i, count=0;
PJ_ASSERT_RETURN(param && p_type, PJ_EINVAL);
if (param->flags & PJ_QOS_PARAM_HAS_DSCP) {
for (i=0; i<=PJ_QOS_TYPE_CONTROL; ++i) {
if (param->dscp_val >= qos_map[i].dscp_val)
dscp_type = (pj_qos_type)i;
}
++count;
}
if (param->flags & PJ_QOS_PARAM_HAS_802_1_P) {
for (i=0; i<=PJ_QOS_TYPE_CONTROL; ++i) {
if (param->so_prio >= qos_map[i].so_prio)
prio_type = (pj_qos_type)i;
}
++count;
}
if (param->flags & PJ_QOS_PARAM_HAS_WMM) {
for (i=0; i<=PJ_QOS_TYPE_CONTROL; ++i) {
if (param->wmm_prio >= qos_map[i].wmm_prio)
wmm_type = (pj_qos_type)i;
}
++count;
}
if (count)
*p_type = (pj_qos_type)((dscp_type + prio_type + wmm_type) / count);
else
*p_type = PJ_QOS_TYPE_BEST_EFFORT;
return PJ_SUCCESS;
}
/* Apply QoS */
PJ_DEF(pj_status_t) pj_sock_apply_qos( pj_sock_t sock,
pj_qos_type qos_type,
pj_qos_params *qos_params,
unsigned log_level,
const char *log_sender,
const char *sock_name)
{
char fmt[60];
pj_status_t qos_type_rc = PJ_SUCCESS,
qos_params_rc = PJ_SUCCESS;
if (!log_sender)
log_sender = THIS_FILE;
if (!sock_name)
sock_name = "socket";
if (qos_type != PJ_QOS_TYPE_BEST_EFFORT) {
qos_type_rc = pj_sock_set_qos_type(sock, qos_type);
if (qos_type_rc != PJ_SUCCESS) {
pj_ansi_snprintf(fmt, sizeof(fmt),
"Error setting QoS type %d to %s",
qos_type, sock_name);
pj_perror(log_level, log_sender, qos_type_rc, fmt, 0);
}
}
if (qos_params && qos_params->flags) {
qos_params_rc = pj_sock_set_qos_params(sock, qos_params);
if (qos_params_rc != PJ_SUCCESS) {
pj_ansi_snprintf(fmt, sizeof(fmt),
"Error setting QoS params (flags=%d) to %s",
qos_params->flags, sock_name);
pj_perror(log_level, log_sender, qos_params_rc, fmt, 0);
if (qos_type_rc != PJ_SUCCESS)
return qos_params_rc;
}
} else if (qos_type_rc != PJ_SUCCESS)
return qos_type_rc;
return PJ_SUCCESS;
}
PJ_DEF(pj_status_t) pj_sock_apply_qos2( pj_sock_t sock,
pj_qos_type qos_type,
const pj_qos_params *qos_params,
unsigned log_level,
const char *log_sender,
const char *sock_name)
{
pj_qos_params qos_params_buf, *qos_params_copy = NULL;
if (qos_params) {
pj_memcpy(&qos_params_buf, qos_params, sizeof(*qos_params));
qos_params_copy = &qos_params_buf;
}
return pj_sock_apply_qos(sock, qos_type, qos_params_copy,
log_level, log_sender, sock_name);
}