blob: 33fae347f34769de822c69cd3ee4e485377e055c [file] [log] [blame]
/* VTY code for the Cellmgr */
/*
* (C) 2010 by Holger Hans Peter Freyther <zecke@selfish.org>
* (C) 2010 by On-Waves
* All Rights Reserved
*
* 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.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*
*/
#include <bsc_data.h>
#include <cellmgr_debug.h>
#include <osmocore/talloc.h>
#include <osmocore/gsm48.h>
#include <osmocom/vty/command.h>
#include <osmocom/vty/vty.h>
#include <unistd.h>
#include <netdb.h>
#undef PACKAGE_NAME
#undef PACKAGE_VERSION
#undef PACKAGE_BUGREPORT
#undef PACKAGE_TARNAME
#undef PACKAGE_STRING
#include <cellmgr_config.h>
extern struct bsc_data bsc;
static struct vty_app_info vty_info = {
.name = "Cellmgr-ng",
.version = VERSION,
.go_parent_cb = NULL,
};
/* vty code */
enum cellmgr_node {
CELLMGR_NODE = _LAST_OSMOVTY_NODE,
};
static struct cmd_node cell_node = {
CELLMGR_NODE,
"%s(cellmgr)#",
1,
};
static int config_write_cell(struct vty *vty)
{
struct link_data *link;
vty_out(vty, "cellmgr%s", VTY_NEWLINE);
vty_out(vty, " mtp dpc %d%s", bsc.dpc, VTY_NEWLINE);
vty_out(vty, " mtp opc %d%s", bsc.opc, VTY_NEWLINE);
vty_out(vty, " mtp sccp-opc %d%s", bsc.sccp_opc, VTY_NEWLINE);
vty_out(vty, " mtp ni %d%s", bsc.ni_ni, VTY_NEWLINE);
vty_out(vty, " mtp spare %d%s", bsc.ni_spare, VTY_NEWLINE);
vty_out(vty, " mtp sltm once %d%s", bsc.once, VTY_NEWLINE);
vty_out(vty, " country-code %d%s", bsc.mcc, VTY_NEWLINE);
vty_out(vty, " network-code %d%s", bsc.mnc, VTY_NEWLINE);
vty_out(vty, " location-area-code %d%s", bsc.lac, VTY_NEWLINE);
llist_for_each_entry(link, &bsc.links, entry) {
vty_out(vty, " udp dest-ip %d %s%s",
link->link_index, link->udp.udp_ip, VTY_NEWLINE);
vty_out(vty, " udp dest-port %d %d%s",
link->link_index, link->udp.udp_port, VTY_NEWLINE);
}
vty_out(vty, " udp src port %d%s", bsc.src_port, VTY_NEWLINE);
vty_out(vty, " udp reset %d%s", bsc.udp_reset_timeout, VTY_NEWLINE);
vty_out(vty, " msc ip %s%s", bsc.msc_address, VTY_NEWLINE);
vty_out(vty, " msc ip-dscp %d%s", bsc.msc_ip_dscp, VTY_NEWLINE);
vty_out(vty, " msc token %s%s", bsc.token, VTY_NEWLINE);
return CMD_SUCCESS;
}
DEFUN(cfg_cell, cfg_cell_cmd,
"cellmgr", "Configure the Cellmgr")
{
vty->node = CELLMGR_NODE;
return CMD_SUCCESS;
}
DEFUN(cfg_net_dpc, cfg_net_dpc_cmd,
"mtp dpc DPC_NR",
"Set the DPC to be used.")
{
bsc.dpc = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_net_opc, cfg_net_opc_cmd,
"mtp opc OPC_NR",
"Set the OPC to be used.")
{
bsc.opc = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_net_sccp_opc, cfg_net_sccp_opc_cmd,
"mtp sccp-opc OPC_NR",
"Set the SCCP OPC to be used.")
{
bsc.sccp_opc = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_net_mtp_ni, cfg_net_mtp_ni_cmd,
"mtp ni NR",
"Set the MTP NI to be used.\n" "NR")
{
bsc.ni_ni = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_net_mtp_spare, cfg_net_mtp_spare_cmd,
"mtp spare NR",
"Set the MTP Spare to be used.\n" "NR")
{
bsc.ni_spare = atoi(argv[0]);
return CMD_SUCCESS;
}
static struct link_data *find_or_create(struct bsc_data *bsc, int index)
{
struct link_data *link = linkset_find_link(bsc, index);
if (link)
return link;
link = talloc_zero(NULL, struct link_data);
if (!link) {
LOGP(DINP, LOGL_ERROR, "Failed to allocate link.\n");
return NULL;
}
link->link_index = index;
llist_add(&link->entry, &bsc->links);
return link;
}
static int set_dest_ip(struct vty *vty, struct link_data *link, const char *name)
{
struct hostent *hosts;
struct in_addr *addr;
hosts = gethostbyname(name);
if (!hosts || hosts->h_length < 1 || hosts->h_addrtype != AF_INET) {
vty_out(vty, "Failed to resolve '%s'%s", name, VTY_NEWLINE);
return CMD_WARNING;
}
addr = (struct in_addr *) hosts->h_addr_list[0];
link->udp.udp_ip = talloc_strdup(NULL, inet_ntoa(*addr));
return CMD_SUCCESS;
}
DEFUN_DEPRECATED(cfg_udp_dst_ip, cfg_udp_dst_ip_cmd,
"udp dest ip IP",
"Set the IP when UDP mode is supposed to be used.")
{
struct link_data *link;
link = find_or_create(&bsc, 1);
if (!link) {
vty_out(vty, "Failed to create a link.\n");
return CMD_WARNING;
}
return set_dest_ip(vty, link, argv[0]);
}
DEFUN_DEPRECATED(cfg_udp_dst_port, cfg_udp_dst_port_cmd,
"udp dest port PORT_NR",
"If UDP mode is used specify the UDP dest port")
{
struct link_data *link;
link = find_or_create(&bsc, 1);
if (!link) {
vty_out(vty, "Failed to create a link.\n");
return CMD_WARNING;
}
link->udp.udp_port = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_link_dest_ip, cfg_link_dest_ip_cmd,
"udp dest-ip <0-64> IP",
"Set the dest-ip for a link.\n" "The link index\n" "IP or hostname\n")
{
struct link_data *link;
link = find_or_create(&bsc, atoi(argv[0]));
if (!link) {
vty_out(vty, "Failed to create a link.\n");
return CMD_WARNING;
}
return set_dest_ip(vty, link, argv[1]);
}
DEFUN(cfg_link_port_ip, cfg_link_port_ip_cmd,
"udp dest-port <0-64> PORT_NR",
"Set the dest-port for a link.\n" "The link index\n" "port number\n")
{
struct link_data *link;
link = find_or_create(&bsc, atoi(argv[0]));
if (!link) {
vty_out(vty, "Failed to create a link.\n");
return CMD_WARNING;
}
link->udp.udp_port = atoi(argv[1]);
return CMD_SUCCESS;
}
DEFUN(cfg_udp_src_port, cfg_udp_src_port_cmd,
"udp src port PORT_NR",
"Set the UDP source port to be used.")
{
bsc.src_port = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_udp_reset, cfg_udp_reset_cmd,
"udp reset TIMEOUT",
"Set the timeout to take the link down")
{
bsc.udp_reset_timeout = atoi(argv[0]);
link_set_reset_timeout(&bsc);
return CMD_SUCCESS;
}
DEFUN(cfg_sltm_once, cfg_sltm_once_cmd,
"mtp sltm once (0|1)",
"Send SLTMs until the link is established.")
{
bsc.once = !!atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_msc_ip, cfg_msc_ip_cmd,
"msc ip IP",
"Set the MSC IP")
{
struct hostent *hosts;
struct in_addr *addr;
hosts = gethostbyname(argv[0]);
if (!hosts || hosts->h_length < 1 || hosts->h_addrtype != AF_INET) {
vty_out(vty, "Failed to resolve '%s'%s", argv[0], VTY_NEWLINE);
return CMD_WARNING;
}
addr = (struct in_addr *) hosts->h_addr_list[0];
bsc.msc_address = talloc_strdup(NULL, inet_ntoa(*addr));
return CMD_SUCCESS;
}
DEFUN(cfg_msc_ip_dscp, cfg_msc_ip_dscp_cmd,
"msc ip-dscp <0-255>",
"Set the IP DSCP on the A-link\n"
"Set the DSCP in IP packets to the MSC")
{
bsc.msc_ip_dscp = atoi(argv[0]);
return CMD_SUCCESS;
}
ALIAS_DEPRECATED(cfg_msc_ip_dscp, cfg_msc_ip_tos_cmd,
"msc ip-tos <0-255>",
"Set the IP DSCP on the A-link\n"
"Set the DSCP in IP packets to the MSC")
DEFUN(cfg_msc_token, cfg_msc_token_cmd,
"msc token TOKEN",
"Set the Token to be used for the MSC")
{
bsc.token = talloc_strdup(NULL, argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_ping_time, cfg_ping_time_cmd,
"timeout ping NR",
"Set the PING interval. Negative to disable it")
{
bsc.ping_time = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_pong_time, cfg_pong_time_cmd,
"timeout pong NR",
"Set the PING interval. Negative to disable it")
{
bsc.pong_time = atoi(argv[0]);
return CMD_SUCCESS;
}
DEFUN(cfg_msc_time, cfg_msc_time_cmd,
"timeout msc NR",
"Set the MSC connect timeout")
{
bsc.msc_time = atoi(argv[0]);
return CMD_SUCCESS;
}
static void update_lai(struct bsc_data *bsc)
{
gsm48_generate_lai(&bsc->lai, bsc->mcc, bsc->mnc, bsc->lac);
}
DEFUN(cfg_mnc, cfg_mnc_cmd,
"network-code NR",
"Set the Mobile Network Code\n" "Number\n")
{
bsc.mnc = atoi(argv[0]);
update_lai(&bsc);
return CMD_SUCCESS;
}
DEFUN(cfg_mcc, cfg_mcc_cmd,
"country-code NR",
"Set the Mobile Country Code\n" "Number\n")
{
bsc.mcc = atoi(argv[0]);
update_lai(&bsc);
return CMD_SUCCESS;
}
DEFUN(cfg_lac, cfg_lac_cmd,
"location-area-code NR",
"Set the Location Area Code\n" "Number\n")
{
bsc.lac = atoi(argv[0]);
update_lai(&bsc);
return CMD_SUCCESS;
}
void cell_vty_init(void)
{
cmd_init(1);
vty_init(&vty_info);
install_element(CONFIG_NODE, &cfg_cell_cmd);
install_node(&cell_node, config_write_cell);
install_element(CELLMGR_NODE, &cfg_net_dpc_cmd);
install_element(CELLMGR_NODE, &cfg_net_opc_cmd);
install_element(CELLMGR_NODE, &cfg_net_sccp_opc_cmd);
install_element(CELLMGR_NODE, &cfg_net_mtp_ni_cmd);
install_element(CELLMGR_NODE, &cfg_net_mtp_spare_cmd);
install_element(CELLMGR_NODE, &cfg_udp_dst_ip_cmd);
install_element(CELLMGR_NODE, &cfg_udp_dst_port_cmd);
install_element(CELLMGR_NODE, &cfg_udp_src_port_cmd);
install_element(CELLMGR_NODE, &cfg_udp_reset_cmd);
install_element(CELLMGR_NODE, &cfg_sltm_once_cmd);
install_element(CELLMGR_NODE, &cfg_msc_ip_cmd);
install_element(CELLMGR_NODE, &cfg_msc_token_cmd);
install_element(CELLMGR_NODE, &cfg_msc_ip_dscp_cmd);
install_element(CELLMGR_NODE, &cfg_msc_ip_tos_cmd);
install_element(CELLMGR_NODE, &cfg_ping_time_cmd);
install_element(CELLMGR_NODE, &cfg_pong_time_cmd);
install_element(CELLMGR_NODE, &cfg_msc_time_cmd);
install_element(CELLMGR_NODE, &cfg_mcc_cmd);
install_element(CELLMGR_NODE, &cfg_mnc_cmd);
install_element(CELLMGR_NODE, &cfg_lac_cmd);
install_element(CELLMGR_NODE, &cfg_link_dest_ip_cmd);
install_element(CELLMGR_NODE, &cfg_link_port_ip_cmd);
}
const char *openbsc_copyright = "";