mirror of
				https://gitea.osmocom.org/cellular-infrastructure/osmo-mgw.git
				synced 2025-11-03 21:43:32 +00:00 
			
		
		
		
	Compare commits
	
		
			51 Commits
		
	
	
		
			on-waves/0
			...
			on-waves/0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 
						 | 
					6d17dd1314 | ||
| 
						 | 
					7cb6867ea3 | ||
| 
						 | 
					d8138c43a1 | ||
| 
						 | 
					46d9b94477 | ||
| 
						 | 
					4f705b9f99 | ||
| 
						 | 
					c592e697ce | ||
| 
						 | 
					ebb6b99c63 | ||
| 
						 | 
					e08253a3f7 | ||
| 
						 | 
					5e86095364 | ||
| 
						 | 
					a7c144888d | ||
| 
						 | 
					7897c4446b | ||
| 
						 | 
					ff9e09b2bc | ||
| 
						 | 
					ecf5cc294d | ||
| 
						 | 
					82126763a7 | ||
| 
						 | 
					a380c89a9c | ||
| 
						 | 
					bedaf5da64 | ||
| 
						 | 
					2b08aa35a6 | ||
| 
						 | 
					c24632930a | ||
| 
						 | 
					f140348eff | ||
| 
						 | 
					b5de1b0781 | ||
| 
						 | 
					b022cc3b8e | ||
| 
						 | 
					e8396c9663 | ||
| 
						 | 
					941839b300 | ||
| 
						 | 
					23a0e46f11 | ||
| 
						 | 
					cb8fd6e99e | ||
| 
						 | 
					e66bea8ad7 | ||
| 
						 | 
					e8a9f471ef | ||
| 
						 | 
					c2d66bdf5a | ||
| 
						 | 
					80b584bbe7 | ||
| 
						 | 
					15c21e8eec | ||
| 
						 | 
					c0a1fff064 | ||
| 
						 | 
					77fa4d2386 | ||
| 
						 | 
					9be8752541 | ||
| 
						 | 
					2b57b3cea4 | ||
| 
						 | 
					00c531709a | ||
| 
						 | 
					a094108f84 | ||
| 
						 | 
					61e73eec3f | ||
| 
						 | 
					1aa2798919 | ||
| 
						 | 
					b829eac9bc | ||
| 
						 | 
					7b1719327d | ||
| 
						 | 
					493645eda9 | ||
| 
						 | 
					8614cd0be7 | ||
| 
						 | 
					19bd74d093 | ||
| 
						 | 
					4821b5a847 | ||
| 
						 | 
					84df56d577 | ||
| 
						 | 
					4e23d5f87f | ||
| 
						 | 
					4346424987 | ||
| 
						 | 
					520656e004 | ||
| 
						 | 
					9dac231fe4 | ||
| 
						 | 
					2bb518a3bd | ||
| 
						 | 
					476940f747 | 
@@ -1,7 +1,7 @@
 | 
			
		||||
dnl Process this file with autoconf to produce a configure script
 | 
			
		||||
AC_INIT
 | 
			
		||||
 | 
			
		||||
AM_INIT_AUTOMAKE(openbsc, 0.3.95onwaves)
 | 
			
		||||
AM_INIT_AUTOMAKE(openbsc, 0.3.98onwaves)
 | 
			
		||||
 | 
			
		||||
dnl kernel style compile messages
 | 
			
		||||
m4_ifdef([AM_SILENT_RULES], [AM_SILENT_RULES([yes])])
 | 
			
		||||
 
 | 
			
		||||
@@ -6,7 +6,7 @@ noinst_HEADERS = abis_nm.h abis_rsl.h db.h gsm_04_08.h gsm_data.h \
 | 
			
		||||
		 bsc_rll.h mncc.h transaction.h ussd.h gsm_04_80.h \
 | 
			
		||||
		 silent_call.h mgcp.h meas_rep.h rest_octets.h \
 | 
			
		||||
		 system_information.h handover.h mgcp_internal.h \
 | 
			
		||||
		 vty.h bssap.h bsc_msc.h bsc_nat.h
 | 
			
		||||
		 vty.h bssap.h bsc_msc.h bsc_nat.h bsc_msc_rf.h
 | 
			
		||||
 | 
			
		||||
openbsc_HEADERS = gsm_04_08.h meas_rep.h bsc_api.h
 | 
			
		||||
openbscdir = $(includedir)/openbsc
 | 
			
		||||
 
 | 
			
		||||
@@ -29,6 +29,7 @@
 | 
			
		||||
struct bsc_msc_connection {
 | 
			
		||||
	struct write_queue write_queue;
 | 
			
		||||
	int is_connected;
 | 
			
		||||
	int is_authenticated;
 | 
			
		||||
	const char *ip;
 | 
			
		||||
	int port;
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										20
									
								
								openbsc/include/openbsc/bsc_msc_rf.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								openbsc/include/openbsc/bsc_msc_rf.h
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,20 @@
 | 
			
		||||
#ifndef BSC_MSC_RF
 | 
			
		||||
#define BSC_MSC_RF
 | 
			
		||||
 | 
			
		||||
#include <osmocore/write_queue.h>
 | 
			
		||||
 | 
			
		||||
struct gsm_network;
 | 
			
		||||
 | 
			
		||||
struct bsc_msc_rf {
 | 
			
		||||
	struct bsc_fd listen;
 | 
			
		||||
	struct gsm_network *gsm_network;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct bsc_msc_rf_conn {
 | 
			
		||||
	struct write_queue queue;
 | 
			
		||||
	struct gsm_network *gsm_network;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct bsc_msc_rf *bsc_msc_rf_create(const char *path, struct gsm_network *net);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
@@ -31,6 +31,9 @@
 | 
			
		||||
#include <osmocore/msgb.h>
 | 
			
		||||
#include <osmocore/timer.h>
 | 
			
		||||
#include <osmocore/write_queue.h>
 | 
			
		||||
#include <osmocore/statistics.h>
 | 
			
		||||
 | 
			
		||||
#include <regex.h>
 | 
			
		||||
 | 
			
		||||
#define DIR_BSC 1
 | 
			
		||||
#define DIR_MSC 2
 | 
			
		||||
@@ -113,6 +116,20 @@ struct sccp_connections {
 | 
			
		||||
	int bsc_timeslot;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Stats per BSC
 | 
			
		||||
 */
 | 
			
		||||
struct bsc_config_stats {
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *conn;
 | 
			
		||||
		struct counter *calls;
 | 
			
		||||
	} sccp;
 | 
			
		||||
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *reconn;
 | 
			
		||||
	} net;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * One BSC entry in the config
 | 
			
		||||
 */
 | 
			
		||||
@@ -123,7 +140,16 @@ struct bsc_config {
 | 
			
		||||
	unsigned int lac;
 | 
			
		||||
	int nr;
 | 
			
		||||
 | 
			
		||||
	/* imsi white and blacklist */
 | 
			
		||||
	char *imsi_allow;
 | 
			
		||||
	regex_t imsi_allow_re;
 | 
			
		||||
	char *imsi_deny;
 | 
			
		||||
	regex_t imsi_deny_re;
 | 
			
		||||
 | 
			
		||||
	/* backpointer */
 | 
			
		||||
	struct bsc_nat *nat;
 | 
			
		||||
 | 
			
		||||
	struct bsc_config_stats stats;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
@@ -138,6 +164,25 @@ struct bsc_endpoint {
 | 
			
		||||
	int pending_delete;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Statistic for the nat.
 | 
			
		||||
 */
 | 
			
		||||
struct bsc_nat_statistics {
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *conn;
 | 
			
		||||
		struct counter *calls;
 | 
			
		||||
	} sccp;
 | 
			
		||||
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *reconn;
 | 
			
		||||
                struct counter *auth_fail;
 | 
			
		||||
	} bsc;
 | 
			
		||||
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *reconn;
 | 
			
		||||
	} msc;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * the structure of the "nat" network
 | 
			
		||||
 */
 | 
			
		||||
@@ -159,9 +204,20 @@ struct bsc_nat {
 | 
			
		||||
	int mgcp_length;
 | 
			
		||||
 | 
			
		||||
	/* msc things */
 | 
			
		||||
	char *msc_ip;
 | 
			
		||||
	int msc_port;
 | 
			
		||||
	int first_contact;
 | 
			
		||||
 | 
			
		||||
	struct bsc_endpoint *bsc_endpoints;
 | 
			
		||||
 | 
			
		||||
	/* filter */
 | 
			
		||||
	char *imsi_allow;
 | 
			
		||||
	regex_t imsi_allow_re;
 | 
			
		||||
	char *imsi_deny;
 | 
			
		||||
	regex_t imsi_deny_re;
 | 
			
		||||
 | 
			
		||||
	/* statistics */
 | 
			
		||||
	struct bsc_nat_statistics stats;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/* create and init the structures */
 | 
			
		||||
@@ -169,6 +225,7 @@ struct bsc_config *bsc_config_alloc(struct bsc_nat *nat, const char *token, unsi
 | 
			
		||||
struct bsc_config *bsc_config_num(struct bsc_nat *nat, int num);
 | 
			
		||||
struct bsc_nat *bsc_nat_alloc(void);
 | 
			
		||||
struct bsc_connection *bsc_connection_alloc(struct bsc_nat *nat);
 | 
			
		||||
void bsc_nat_set_msc_ip(struct bsc_nat *bsc, const char *ip);
 | 
			
		||||
 | 
			
		||||
void sccp_connection_destroy(struct sccp_connections *);
 | 
			
		||||
 | 
			
		||||
@@ -182,7 +239,7 @@ struct bsc_nat_parsed *bsc_nat_parse(struct msgb *msg);
 | 
			
		||||
 */
 | 
			
		||||
int bsc_nat_filter_ipa(int direction, struct msgb *msg, struct bsc_nat_parsed *parsed);
 | 
			
		||||
int bsc_nat_vty_init(struct bsc_nat *nat);
 | 
			
		||||
struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg);
 | 
			
		||||
struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg, int *_lac);
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * SCCP patching and handling
 | 
			
		||||
@@ -203,7 +260,7 @@ void bsc_mgcp_free_endpoint(struct bsc_nat *nat, int);
 | 
			
		||||
void bsc_mgcp_free_endpoints(struct bsc_nat *nat);
 | 
			
		||||
int bsc_mgcp_init(struct bsc_nat *nat);
 | 
			
		||||
 | 
			
		||||
struct bsc_connection *bsc_mgcp_find_con(struct bsc_nat *, int endpoint_number);
 | 
			
		||||
struct sccp_connections *bsc_mgcp_find_con(struct bsc_nat *, int endpoint_number);
 | 
			
		||||
struct msgb *bsc_mgcp_rewrite(char *input, int length, const char *ip, int port);
 | 
			
		||||
void bsc_mgcp_forward(struct bsc_connection *bsc, struct msgb *msg);
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -330,6 +330,4 @@ void bts_send_queued(struct bss_sccp_connection_data*);
 | 
			
		||||
void bts_free_queued(struct bss_sccp_connection_data*);
 | 
			
		||||
void bts_unblock_queue(struct bss_sccp_connection_data*);
 | 
			
		||||
 | 
			
		||||
const struct tlv_definition *gsm0808_att_tlvdef();
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
@@ -63,10 +63,11 @@ struct gsm_lchan *lchan_find(struct gsm_bts *bts, struct gsm_subscriber *subscr)
 | 
			
		||||
struct gsm_lchan *lchan_for_subscr(struct gsm_subscriber *subscr);
 | 
			
		||||
 | 
			
		||||
/* Allocate a logical channel (SDCCH, TCH, ...) */
 | 
			
		||||
struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type);
 | 
			
		||||
struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type, int allow_bigger);
 | 
			
		||||
 | 
			
		||||
/* Free a logical channel (SDCCH, TCH, ...) */
 | 
			
		||||
void lchan_free(struct gsm_lchan *lchan);
 | 
			
		||||
void lchan_reset(struct gsm_lchan *lchan);
 | 
			
		||||
 | 
			
		||||
/* internal.. do not use */
 | 
			
		||||
int _lchan_release(struct gsm_lchan *lchan, u_int8_t release_reason);
 | 
			
		||||
 
 | 
			
		||||
@@ -562,6 +562,14 @@ struct gsmnet_stats {
 | 
			
		||||
		struct counter *alerted;	/* we alerted the other end */
 | 
			
		||||
		struct counter *connected;/* how many calls were accepted */
 | 
			
		||||
	} call;
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *rf_fail;
 | 
			
		||||
		struct counter *rll_err;
 | 
			
		||||
	} chan;
 | 
			
		||||
	struct {
 | 
			
		||||
		struct counter *oml_fail;
 | 
			
		||||
		struct counter *rsl_fail;
 | 
			
		||||
	} bts;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
enum gsm_auth_policy {
 | 
			
		||||
@@ -653,6 +661,8 @@ struct gsm_network {
 | 
			
		||||
 | 
			
		||||
	/* a simple token for this network... */
 | 
			
		||||
	char *bsc_token;
 | 
			
		||||
	char *msc_ip;
 | 
			
		||||
	int msc_port;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#define SMS_HDR_SIZE	128
 | 
			
		||||
 
 | 
			
		||||
@@ -53,6 +53,8 @@ int ipaccess_send_id_req(int fd);
 | 
			
		||||
 | 
			
		||||
int ipaccess_idtag_parse(struct tlv_parsed *dec, unsigned char *buf, int len);
 | 
			
		||||
 | 
			
		||||
int ipaccess_drop_oml(struct gsm_bts *bts);
 | 
			
		||||
int ipaccess_drop_rsl(struct gsm_bts_trx *trx);
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Firmware specific header
 | 
			
		||||
 
 | 
			
		||||
@@ -43,4 +43,7 @@ void paging_request_stop(struct gsm_bts *bts, struct gsm_subscriber *subscr,
 | 
			
		||||
/* update paging load */
 | 
			
		||||
void paging_update_buffer_space(struct gsm_bts *bts, u_int16_t);
 | 
			
		||||
 | 
			
		||||
/* pending paging requests */
 | 
			
		||||
unsigned int paging_pending_requests_nr(struct gsm_bts *bts);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,10 @@
 | 
			
		||||
#ifndef OPENBSC_VTY_H
 | 
			
		||||
#define OPENBSC_VTY_H
 | 
			
		||||
 | 
			
		||||
struct gsm_network;
 | 
			
		||||
struct vty;
 | 
			
		||||
 | 
			
		||||
void openbsc_vty_add_cmds(void);
 | 
			
		||||
void openbsc_vty_print_statistics(struct vty *vty, struct gsm_network *);
 | 
			
		||||
 | 
			
		||||
#endif
 | 
			
		||||
 
 | 
			
		||||
@@ -35,7 +35,7 @@ bsc_hack_LDADD = libmsc.a libbsc.a libmsc.a libvty.a -ldl -ldbi $(LIBCRYPT)
 | 
			
		||||
bs11_config_SOURCES = bs11_config.c abis_nm.c gsm_data.c debug.c \
 | 
			
		||||
		      rs232.c bts_siemens_bs11.c
 | 
			
		||||
bsc_msc_ip_SOURCES = bssap.c bsc_msc_ip.c bsc_init.c vty_interface.c vty_interface_bsc.c \
 | 
			
		||||
		bsc_msc.c
 | 
			
		||||
		bsc_msc.c bsc_msc_rf.c
 | 
			
		||||
bsc_msc_ip_LDADD = libbsc.a libvty.a libsccp.a
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -845,6 +845,7 @@ static int rsl_rx_conn_fail(struct msgb *msg)
 | 
			
		||||
 | 
			
		||||
	LOGPC(DRSL, LOGL_NOTICE, "\n");
 | 
			
		||||
	/* FIXME: only free it after channel release ACK */
 | 
			
		||||
	counter_inc(msg->lchan->ts->trx->bts->network->stats.chan.rf_fail);
 | 
			
		||||
	return rsl_rf_chan_release(msg->lchan, 1);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -1139,6 +1140,7 @@ static int rsl_rx_chan_rqd(struct msgb *msg)
 | 
			
		||||
	struct gsm_lchan *lchan;
 | 
			
		||||
	u_int8_t rqd_ta;
 | 
			
		||||
	int ret;
 | 
			
		||||
	int is_lu;
 | 
			
		||||
 | 
			
		||||
	u_int16_t arfcn;
 | 
			
		||||
	u_int8_t ts_number, subch;
 | 
			
		||||
@@ -1161,8 +1163,14 @@ static int rsl_rx_chan_rqd(struct msgb *msg)
 | 
			
		||||
 | 
			
		||||
	counter_inc(bts->network->stats.chreq.total);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * We want LOCATION UPDATES to succeed and will assign a TCH
 | 
			
		||||
	 * if we have no SDCCH available.
 | 
			
		||||
	 */
 | 
			
		||||
	is_lu = !!(chreq_reason == GSM_CHREQ_REASON_LOCATION_UPD);
 | 
			
		||||
 | 
			
		||||
	/* check availability / allocate channel */
 | 
			
		||||
	lchan = lchan_alloc(bts, lctype);
 | 
			
		||||
	lchan = lchan_alloc(bts, lctype, is_lu);
 | 
			
		||||
	if (!lchan) {
 | 
			
		||||
		LOGP(DRSL, LOGL_NOTICE, "BTS %d CHAN RQD: no resources for %s 0x%x\n",
 | 
			
		||||
		     msg->lchan->ts->trx->bts->nr, gsm_lchant_name(lctype), rqd_ref->ra);
 | 
			
		||||
@@ -1291,8 +1299,10 @@ static int rsl_rx_rll_err_ind(struct msgb *msg)
 | 
			
		||||
 | 
			
		||||
	rll_indication(msg->lchan, rllh->link_id, BSC_RLLR_IND_ERR_IND);
 | 
			
		||||
 | 
			
		||||
	if (rlm_cause[1] == RLL_CAUSE_T200_EXPIRED)
 | 
			
		||||
	if (rlm_cause[1] == RLL_CAUSE_T200_EXPIRED) {
 | 
			
		||||
		counter_inc(msg->lchan->ts->trx->bts->network->stats.chan.rll_err);
 | 
			
		||||
		return rsl_rf_chan_release(msg->lchan, 1);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -7,6 +7,7 @@ line vty
 | 
			
		||||
 no login
 | 
			
		||||
!
 | 
			
		||||
nat
 | 
			
		||||
 msc ip 10.0.0.23
 | 
			
		||||
 bsc 0
 | 
			
		||||
   token zecke
 | 
			
		||||
   location_area_code 3
 | 
			
		||||
 
 | 
			
		||||
@@ -31,6 +31,7 @@
 | 
			
		||||
#include <openbsc/system_information.h>
 | 
			
		||||
#include <openbsc/paging.h>
 | 
			
		||||
#include <openbsc/signal.h>
 | 
			
		||||
#include <openbsc/chan_alloc.h>
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
 | 
			
		||||
/* global pointer to the gsm network data structure */
 | 
			
		||||
@@ -901,6 +902,8 @@ static void bootstrap_rsl(struct gsm_bts_trx *trx)
 | 
			
		||||
 | 
			
		||||
void input_event(int event, enum e1inp_sign_type type, struct gsm_bts_trx *trx)
 | 
			
		||||
{
 | 
			
		||||
	int ts_no, lchan_no;
 | 
			
		||||
 | 
			
		||||
	switch (event) {
 | 
			
		||||
	case EVT_E1_TEI_UP:
 | 
			
		||||
		switch (type) {
 | 
			
		||||
@@ -915,8 +918,35 @@ void input_event(int event, enum e1inp_sign_type type, struct gsm_bts_trx *trx)
 | 
			
		||||
		}
 | 
			
		||||
		break;
 | 
			
		||||
	case EVT_E1_TEI_DN:
 | 
			
		||||
		LOGP(DMI, LOGL_NOTICE, "Lost some E1 TEI link\n");
 | 
			
		||||
		/* FIXME: deal with TEI or L1 link loss */
 | 
			
		||||
		LOGP(DMI, LOGL_ERROR, "Lost some E1 TEI link: %d %p\n", type, trx);
 | 
			
		||||
 | 
			
		||||
		if (type == E1INP_SIGN_OML)
 | 
			
		||||
			counter_inc(trx->bts->network->stats.bts.oml_fail);
 | 
			
		||||
		else if (type == E1INP_SIGN_RSL)
 | 
			
		||||
			counter_inc(trx->bts->network->stats.bts.rsl_fail);
 | 
			
		||||
 | 
			
		||||
		/*
 | 
			
		||||
		 * free all allocated channels. change the nm_state so the
 | 
			
		||||
		 * trx and trx_ts becomes unusable and chan_alloc.c can not
 | 
			
		||||
		 * allocate from it.
 | 
			
		||||
		 */
 | 
			
		||||
		for (ts_no = 0; ts_no < ARRAY_SIZE(trx->ts); ++ts_no) {
 | 
			
		||||
			struct gsm_bts_trx_ts *ts = &trx->ts[ts_no];
 | 
			
		||||
 | 
			
		||||
			for (lchan_no = 0; lchan_no < ARRAY_SIZE(ts->lchan); ++lchan_no) {
 | 
			
		||||
				if (ts->lchan[lchan_no].state != GSM_LCHAN_NONE)
 | 
			
		||||
					lchan_free(&ts->lchan[lchan_no]);
 | 
			
		||||
				lchan_reset(&ts->lchan[lchan_no]);
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			ts->nm_state.operational = 0;
 | 
			
		||||
			ts->nm_state.availability = 0;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		trx->nm_state.operational = 0;
 | 
			
		||||
		trx->nm_state.availability = 0;
 | 
			
		||||
		trx->bb_transc.nm_state.operational = 0;
 | 
			
		||||
		trx->bb_transc.nm_state.availability = 0;
 | 
			
		||||
		break;
 | 
			
		||||
	default:
 | 
			
		||||
		break;
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,4 @@
 | 
			
		||||
/* A hackish minimal BSC (+MSC +HLR) implementation */
 | 
			
		||||
/* The BSC Process to handle GSM08.08 (A-Interface) */
 | 
			
		||||
 | 
			
		||||
/* (C) 2008-2009 by Harald Welte <laforge@gnumonks.org>
 | 
			
		||||
 * (C) 2009 by Holger Hans Peter Freyther <zecke@selfish.org>
 | 
			
		||||
@@ -44,6 +44,7 @@
 | 
			
		||||
#include <openbsc/chan_alloc.h>
 | 
			
		||||
#include <openbsc/bsc_msc.h>
 | 
			
		||||
#include <openbsc/bsc_nat.h>
 | 
			
		||||
#include <openbsc/bsc_msc_rf.h>
 | 
			
		||||
 | 
			
		||||
#include <osmocore/select.h>
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
@@ -58,16 +59,23 @@
 | 
			
		||||
static struct log_target *stderr_target;
 | 
			
		||||
struct gsm_network *bsc_gsmnet = 0;
 | 
			
		||||
static const char *config_file = "openbsc.cfg";
 | 
			
		||||
static char *msc_address = "127.0.0.1";
 | 
			
		||||
static char *msc_address = NULL;
 | 
			
		||||
static struct bsc_msc_connection *msc_con;
 | 
			
		||||
static struct in_addr local_addr;
 | 
			
		||||
static LLIST_HEAD(active_connections);
 | 
			
		||||
static struct write_queue mgcp_agent;
 | 
			
		||||
static const char *rf_ctl = NULL;
 | 
			
		||||
extern int ipacc_rtp_direct;
 | 
			
		||||
 | 
			
		||||
extern int bsc_bootstrap_network(int (*layer4)(struct gsm_network *, int, void *), const char *cfg_file);
 | 
			
		||||
extern int bsc_shutdown_net(struct gsm_network *net);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
struct llist_head *bsc_sccp_connections()
 | 
			
		||||
{
 | 
			
		||||
	return &active_connections;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct bss_sccp_connection_data *bss_sccp_create_data()
 | 
			
		||||
{
 | 
			
		||||
	struct bss_sccp_connection_data *data;
 | 
			
		||||
@@ -217,7 +225,7 @@ static int open_sccp_connection(struct msgb *layer3)
 | 
			
		||||
	struct msgb *data;
 | 
			
		||||
 | 
			
		||||
	/* When not connected to a MSC. We will simply close things down. */
 | 
			
		||||
	if (!msc_con->is_connected) {
 | 
			
		||||
	if (!msc_con->is_authenticated) {
 | 
			
		||||
		LOGP(DMSC, LOGL_ERROR, "Not connected to a MSC. Not forwarding data.\n");
 | 
			
		||||
		use_subscr_con(&layer3->lchan->conn);
 | 
			
		||||
		put_subscr_con(&layer3->lchan->conn, 0);
 | 
			
		||||
@@ -568,7 +576,7 @@ static int handle_abisip_signal(unsigned int subsys, unsigned int signal,
 | 
			
		||||
 | 
			
		||||
static void print_usage()
 | 
			
		||||
{
 | 
			
		||||
	printf("Usage: bsc_hack\n");
 | 
			
		||||
	printf("Usage: bsc_msc_ip\n");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
@@ -762,18 +770,10 @@ static int msc_sccp_read(struct msgb *msgb, unsigned int length, void *data)
 | 
			
		||||
 */
 | 
			
		||||
static void initialize_if_needed(void)
 | 
			
		||||
{
 | 
			
		||||
	if (!bsc_gsmnet) {
 | 
			
		||||
		int rc;
 | 
			
		||||
	struct msgb *msg;
 | 
			
		||||
 | 
			
		||||
		fprintf(stderr, "Bootstraping the network. Sending GSM08.08 reset.\n");
 | 
			
		||||
		rc = bsc_bootstrap_network(NULL, config_file);
 | 
			
		||||
		if (rc < 0) {
 | 
			
		||||
			fprintf(stderr, "Bootstrapping the network failed. exiting.\n");
 | 
			
		||||
			exit(1);
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	if (!msc_con->is_authenticated) {
 | 
			
		||||
		/* send a gsm 08.08 reset message from here */
 | 
			
		||||
		msg = bssmap_create_reset();
 | 
			
		||||
		if (!msg) {
 | 
			
		||||
@@ -783,6 +783,7 @@ static void initialize_if_needed(void)
 | 
			
		||||
 | 
			
		||||
		sccp_write(msg, &sccp_ssn_bssap, &sccp_ssn_bssap, 0);
 | 
			
		||||
		msgb_free(msg);
 | 
			
		||||
		msc_con->is_authenticated = 1;
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -839,6 +840,7 @@ static void msc_connection_was_lost(struct bsc_msc_connection *msc)
 | 
			
		||||
		bss_sccp_free_data(bss);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	msc->is_authenticated = 0;
 | 
			
		||||
	bsc_msc_schedule_connect(msc);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -896,6 +898,7 @@ static void print_help()
 | 
			
		||||
	printf("  -m --msc=IP. The address of the MSC.\n");
 | 
			
		||||
	printf("  -l --local=IP. The local address of the MGCP.\n");
 | 
			
		||||
	printf("  -e --log-level number. Set a global loglevel.\n");
 | 
			
		||||
	printf("  -r --rf-ctl NAME. A unix domain socket to listen for cmds.\n");
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void handle_options(int argc, char** argv)
 | 
			
		||||
@@ -908,14 +911,14 @@ static void handle_options(int argc, char** argv)
 | 
			
		||||
			{"config-file", 1, 0, 'c'},
 | 
			
		||||
			{"disable-color", 0, 0, 's'},
 | 
			
		||||
			{"timestamp", 0, 0, 'T'},
 | 
			
		||||
			{"rtp-proxy", 0, 0, 'P'},
 | 
			
		||||
			{"msc", 1, 0, 'm'},
 | 
			
		||||
			{"local", 1, 0, 'l'},
 | 
			
		||||
			{"log-level", 1, 0, 'e'},
 | 
			
		||||
			{"rf-ctl", 1, 0, 'r'},
 | 
			
		||||
			{0, 0, 0, 0}
 | 
			
		||||
		};
 | 
			
		||||
 | 
			
		||||
		c = getopt_long(argc, argv, "hd:sTPc:m:l:e:",
 | 
			
		||||
		c = getopt_long(argc, argv, "hd:sTc:m:l:e:r:",
 | 
			
		||||
				long_options, &option_index);
 | 
			
		||||
		if (c == -1)
 | 
			
		||||
			break;
 | 
			
		||||
@@ -941,7 +944,7 @@ static void handle_options(int argc, char** argv)
 | 
			
		||||
			ipacc_rtp_direct = 0;
 | 
			
		||||
			break;
 | 
			
		||||
		case 'm':
 | 
			
		||||
			msc_address = strdup(optarg);
 | 
			
		||||
			msc_address = optarg;
 | 
			
		||||
			break;
 | 
			
		||||
		case 'l':
 | 
			
		||||
			inet_aton(optarg, &local_addr);
 | 
			
		||||
@@ -949,6 +952,9 @@ static void handle_options(int argc, char** argv)
 | 
			
		||||
		case 'e':
 | 
			
		||||
			log_set_log_level(stderr_target, atoi(optarg));
 | 
			
		||||
			break;
 | 
			
		||||
		case 'r':
 | 
			
		||||
			rf_ctl = optarg;
 | 
			
		||||
			break;
 | 
			
		||||
		default:
 | 
			
		||||
			/* ignore */
 | 
			
		||||
			break;
 | 
			
		||||
@@ -975,7 +981,7 @@ static void signal_handler(int signal)
 | 
			
		||||
		talloc_report_full(tall_bsc_ctx, stderr);
 | 
			
		||||
		break;
 | 
			
		||||
	case SIGUSR2:
 | 
			
		||||
		if (!msc_con->is_connected)
 | 
			
		||||
		if (!msc_con || !msc_con->is_connected)
 | 
			
		||||
			return;
 | 
			
		||||
		bsc_msc_lost(msc_con);
 | 
			
		||||
		break;
 | 
			
		||||
@@ -1030,6 +1036,9 @@ extern int bts_model_nanobts_init(void);
 | 
			
		||||
 | 
			
		||||
int main(int argc, char **argv)
 | 
			
		||||
{
 | 
			
		||||
	char *msc;
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	log_init(&log_info);
 | 
			
		||||
	tall_bsc_ctx = talloc_named_const(NULL, 1, "openbsc");
 | 
			
		||||
	stderr_target = log_target_create_stderr();
 | 
			
		||||
@@ -1048,6 +1057,12 @@ int main(int argc, char **argv)
 | 
			
		||||
	/* seed the PRNG */
 | 
			
		||||
	srand(time(NULL));
 | 
			
		||||
 | 
			
		||||
	signal(SIGINT, &signal_handler);
 | 
			
		||||
	signal(SIGABRT, &signal_handler);
 | 
			
		||||
	signal(SIGUSR1, &signal_handler);
 | 
			
		||||
	signal(SIGUSR2, &signal_handler);
 | 
			
		||||
	signal(SIGPIPE, SIG_IGN);
 | 
			
		||||
 | 
			
		||||
	/* attempt to register the local mgcp forward */
 | 
			
		||||
	if (mgcp_create_port() != 0) {
 | 
			
		||||
		fprintf(stderr, "Failed to bind local MGCP port\n");
 | 
			
		||||
@@ -1062,9 +1077,28 @@ int main(int argc, char **argv)
 | 
			
		||||
	/* initialize ipaccess handling */
 | 
			
		||||
	register_signal_handler(SS_ABISIP, handle_abisip_signal, NULL);
 | 
			
		||||
 | 
			
		||||
	fprintf(stderr, "Bootstraping the network. Sending GSM08.08 reset.\n");
 | 
			
		||||
	rc = bsc_bootstrap_network(NULL, config_file);
 | 
			
		||||
	if (rc < 0) {
 | 
			
		||||
		fprintf(stderr, "Bootstrapping the network failed. exiting.\n");
 | 
			
		||||
		exit(1);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (rf_ctl) {
 | 
			
		||||
		struct bsc_msc_rf *rf;
 | 
			
		||||
		rf = bsc_msc_rf_create(rf_ctl, bsc_gsmnet);
 | 
			
		||||
		if (!rf) {
 | 
			
		||||
			fprintf(stderr, "Failed to create the RF service.\n");
 | 
			
		||||
			exit(1);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* setup MSC Connection handling */
 | 
			
		||||
	msc_con = bsc_msc_create(msc_address, 5000);
 | 
			
		||||
	msc = bsc_gsmnet->msc_ip;
 | 
			
		||||
	if (msc_address)
 | 
			
		||||
		msc = msc_address;
 | 
			
		||||
 | 
			
		||||
	msc_con = bsc_msc_create(msc, bsc_gsmnet->msc_port);
 | 
			
		||||
	if (!msc_con) {
 | 
			
		||||
		fprintf(stderr, "Creating a bsc_msc_connection failed.\n");
 | 
			
		||||
		exit(1);
 | 
			
		||||
@@ -1076,11 +1110,6 @@ int main(int argc, char **argv)
 | 
			
		||||
	bsc_msc_connect(msc_con);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	signal(SIGINT, &signal_handler);
 | 
			
		||||
	signal(SIGABRT, &signal_handler);
 | 
			
		||||
	signal(SIGUSR1, &signal_handler);
 | 
			
		||||
	signal(SIGUSR2, &signal_handler);
 | 
			
		||||
	signal(SIGPIPE, SIG_IGN);
 | 
			
		||||
 | 
			
		||||
	while (1) {
 | 
			
		||||
		bsc_select_main(0);
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										249
									
								
								openbsc/src/bsc_msc_rf.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										249
									
								
								openbsc/src/bsc_msc_rf.c
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,249 @@
 | 
			
		||||
/* RF Ctl handling socket */
 | 
			
		||||
 | 
			
		||||
/* (C) 2010 by Harald Welte <laforge@gnumonks.org>
 | 
			
		||||
 * (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 <openbsc/bsc_msc_rf.h>
 | 
			
		||||
#include <openbsc/debug.h>
 | 
			
		||||
#include <openbsc/gsm_data.h>
 | 
			
		||||
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
#include <osmocore/protocol/gsm_12_21.h>
 | 
			
		||||
 | 
			
		||||
#include <sys/socket.h>
 | 
			
		||||
#include <sys/un.h>
 | 
			
		||||
 | 
			
		||||
#include <errno.h>
 | 
			
		||||
#include <unistd.h>
 | 
			
		||||
 | 
			
		||||
#define RF_CMD_QUERY '?'
 | 
			
		||||
#define RF_CMD_OFF   '0'
 | 
			
		||||
#define RF_CMD_ON    '1'
 | 
			
		||||
 | 
			
		||||
static int lock_each_trx(struct gsm_network *net, int lock)
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_bts *bts;
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(bts, &net->bts_list, list) {
 | 
			
		||||
		struct gsm_bts_trx *trx;
 | 
			
		||||
		llist_for_each_entry(trx, &bts->trx_list, list) {
 | 
			
		||||
			gsm_trx_lock_rf(trx, lock);
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Send a '1' when one TRX is online, otherwise send 0
 | 
			
		||||
 */
 | 
			
		||||
static void handle_query(struct bsc_msc_rf_conn *conn)
 | 
			
		||||
{
 | 
			
		||||
	struct msgb *msg;
 | 
			
		||||
	struct gsm_bts *bts;
 | 
			
		||||
	char send = '0';
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(bts, &conn->gsm_network->bts_list, list) {
 | 
			
		||||
		struct gsm_bts_trx *trx;
 | 
			
		||||
		llist_for_each_entry(trx, &bts->trx_list, list) {
 | 
			
		||||
			if (trx->nm_state.availability == NM_AVSTATE_OK &&
 | 
			
		||||
			    trx->nm_state.operational != NM_STATE_LOCKED) {
 | 
			
		||||
					send = '1';
 | 
			
		||||
					break;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	msg = msgb_alloc(10, "RF Query");
 | 
			
		||||
	if (!msg) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to allocate response msg.\n");
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	msg->l2h = msgb_put(msg, 1);
 | 
			
		||||
	msg->l2h[0] = send;
 | 
			
		||||
 | 
			
		||||
	if (write_queue_enqueue(&conn->queue, msg) != 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to enqueue the answer.\n");
 | 
			
		||||
		msgb_free(msg);
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int rf_read_cmd(struct bsc_fd *fd)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_msc_rf_conn *conn = fd->data;
 | 
			
		||||
	char buf[1];
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	rc = read(fd->fd, buf, sizeof(buf));
 | 
			
		||||
	if (rc != sizeof(buf)) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Short read %d/%s\n", errno, strerror(errno));
 | 
			
		||||
		bsc_unregister_fd(fd);
 | 
			
		||||
		close(fd->fd);
 | 
			
		||||
		write_queue_clear(&conn->queue);
 | 
			
		||||
		talloc_free(conn);
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	switch (buf[0]) {
 | 
			
		||||
	case RF_CMD_QUERY:
 | 
			
		||||
		handle_query(conn);
 | 
			
		||||
		break;
 | 
			
		||||
	case RF_CMD_OFF:
 | 
			
		||||
		lock_each_trx(conn->gsm_network, 1);
 | 
			
		||||
		break;
 | 
			
		||||
	case RF_CMD_ON:
 | 
			
		||||
		lock_each_trx(conn->gsm_network, 0);
 | 
			
		||||
		break;
 | 
			
		||||
	default:
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Unknown command %d\n", buf[0]);
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int rf_write_cmd(struct bsc_fd *fd, struct msgb *msg)
 | 
			
		||||
{
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	rc = write(fd->fd, msg->data, msg->len);
 | 
			
		||||
	if (rc != msg->len) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Short write %d/%s\n", errno, strerror(errno));
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int rf_ctl_accept(struct bsc_fd *bfd, unsigned int what)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_msc_rf_conn *conn;
 | 
			
		||||
	struct bsc_msc_rf *rf = bfd->data;
 | 
			
		||||
	struct sockaddr_un addr;
 | 
			
		||||
	socklen_t len = sizeof(addr);
 | 
			
		||||
	int fd;
 | 
			
		||||
 | 
			
		||||
	fd = accept(bfd->fd, (struct sockaddr *) &addr, &len);
 | 
			
		||||
	if (fd < 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to accept. errno: %d/%s\n",
 | 
			
		||||
		     errno, strerror(errno));
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	conn = talloc_zero(rf, struct bsc_msc_rf_conn);
 | 
			
		||||
	if (!conn) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to allocate mem.\n");
 | 
			
		||||
		close(fd);
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	write_queue_init(&conn->queue, 10);
 | 
			
		||||
	conn->queue.bfd.data = conn;
 | 
			
		||||
	conn->queue.bfd.fd = fd;
 | 
			
		||||
	conn->queue.bfd.when = BSC_FD_READ | BSC_FD_WRITE;
 | 
			
		||||
	conn->queue.read_cb = rf_read_cmd;
 | 
			
		||||
	conn->queue.write_cb = rf_write_cmd;
 | 
			
		||||
	conn->gsm_network = rf->gsm_network;
 | 
			
		||||
 | 
			
		||||
	if (bsc_register_fd(&conn->queue.bfd) != 0) {
 | 
			
		||||
		close(fd);
 | 
			
		||||
		talloc_free(conn);
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct bsc_msc_rf *bsc_msc_rf_create(const char *path, struct gsm_network *net)
 | 
			
		||||
{
 | 
			
		||||
	unsigned int namelen;
 | 
			
		||||
	struct sockaddr_un local;
 | 
			
		||||
	struct bsc_fd *bfd;
 | 
			
		||||
	struct bsc_msc_rf *rf;
 | 
			
		||||
	int rc;
 | 
			
		||||
 | 
			
		||||
	rf = talloc_zero(NULL, struct bsc_msc_rf);
 | 
			
		||||
	if (!rf) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to create bsc_msc_rf.\n");
 | 
			
		||||
		return NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	bfd = &rf->listen;
 | 
			
		||||
	bfd->fd = socket(AF_UNIX, SOCK_STREAM, 0);
 | 
			
		||||
	if (bfd->fd < 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Can not create socket. %d/%s\n",
 | 
			
		||||
		     errno, strerror(errno));
 | 
			
		||||
		return NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	local.sun_family = AF_UNIX;
 | 
			
		||||
	strncpy(local.sun_path, path, sizeof(local.sun_path));
 | 
			
		||||
	local.sun_path[sizeof(local.sun_path) - 1] = '\0';
 | 
			
		||||
	unlink(local.sun_path);
 | 
			
		||||
 | 
			
		||||
	/* we use the same magic that X11 uses in Xtranssock.c for
 | 
			
		||||
	 * calculating the proper length of the sockaddr */
 | 
			
		||||
#if defined(BSD44SOCKETS) || defined(__UNIXWARE__)
 | 
			
		||||
	local.sun_len = strlen(local.sun_path);
 | 
			
		||||
#endif
 | 
			
		||||
#if defined(BSD44SOCKETS) || defined(SUN_LEN)
 | 
			
		||||
	namelen = SUN_LEN(&local);
 | 
			
		||||
#else
 | 
			
		||||
	namelen = strlen(local.sun_path) +
 | 
			
		||||
		  offsetof(struct sockaddr_un, sun_path);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
	rc = bind(bfd->fd, (struct sockaddr *) &local, namelen);
 | 
			
		||||
	if (rc != 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to bind '%s' errno: %d/%s\n",
 | 
			
		||||
		     local.sun_path, errno, strerror(errno));
 | 
			
		||||
		close(bfd->fd);
 | 
			
		||||
		talloc_free(rf);
 | 
			
		||||
		return NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (listen(bfd->fd, 0) != 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to listen: %d/%s\n", errno, strerror(errno));
 | 
			
		||||
		close(bfd->fd);
 | 
			
		||||
		talloc_free(rf);
 | 
			
		||||
		return NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	bfd->when = BSC_FD_READ;
 | 
			
		||||
	bfd->cb = rf_ctl_accept;
 | 
			
		||||
	bfd->data = rf;
 | 
			
		||||
 | 
			
		||||
	if (bsc_register_fd(bfd) != 0) {
 | 
			
		||||
		LOGP(DINP, LOGL_ERROR, "Failed to register bfd.\n");
 | 
			
		||||
		close(bfd->fd);
 | 
			
		||||
		talloc_free(rf);
 | 
			
		||||
		return NULL;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	rf->gsm_network = net;
 | 
			
		||||
 | 
			
		||||
	return rf;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -29,7 +29,7 @@
 | 
			
		||||
#include <openbsc/paging.h>
 | 
			
		||||
#include <openbsc/chan_alloc.h>
 | 
			
		||||
 | 
			
		||||
#include <osmocore/tlv.h>
 | 
			
		||||
#include <osmocore/gsm0808.h>
 | 
			
		||||
 | 
			
		||||
#include <sccp/sccp.h>
 | 
			
		||||
 | 
			
		||||
@@ -44,34 +44,6 @@ static void bts_queue_send(struct msgb *msg, int link_id);
 | 
			
		||||
static void bssmap_free_secondary(struct bss_sccp_connection_data *data);
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static const struct tlv_definition bss_att_tlvdef = {
 | 
			
		||||
	.def = {
 | 
			
		||||
		[GSM0808_IE_IMSI]		    = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_TMSI]		    = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_CELL_IDENTIFIER_LIST]   = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_CHANNEL_NEEDED]	    = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_EMLPP_PRIORITY]	    = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_CHANNEL_TYPE]	    = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_PRIORITY]		    = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_CIRCUIT_IDENTITY_CODE]  = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_DOWNLINK_DTX_FLAG]	    = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_INTERFERENCE_BAND_TO_USE] = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_CLASSMARK_INFORMATION_T2] = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_GROUP_CALL_REFERENCE]   = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_TALKER_FLAG]	    = { TLV_TYPE_T },
 | 
			
		||||
		[GSM0808_IE_CONFIG_EVO_INDI]	    = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_LSA_ACCESS_CTRL_SUPPR]  = { TLV_TYPE_TV },
 | 
			
		||||
		[GSM0808_IE_SERVICE_HANDOVER]	    = { TLV_TYPE_TV},
 | 
			
		||||
		[GSM0808_IE_ENCRYPTION_INFORMATION] = { TLV_TYPE_TLV },
 | 
			
		||||
		[GSM0808_IE_CIPHER_RESPONSE_MODE]   = { TLV_TYPE_TV },
 | 
			
		||||
	},
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
const struct tlv_definition *gsm0808_att_tlvdef()
 | 
			
		||||
{
 | 
			
		||||
	return &bss_att_tlvdef;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static u_int16_t get_network_code_for_msc(struct gsm_network *net)
 | 
			
		||||
{
 | 
			
		||||
	if (net->core_network_code > 0)
 | 
			
		||||
@@ -88,7 +60,7 @@ static u_int16_t get_country_code_for_msc(struct gsm_network *net)
 | 
			
		||||
 | 
			
		||||
static int bssmap_paging_cb(unsigned int hooknum, unsigned int event, struct msgb *msg, void *data, void *param)
 | 
			
		||||
{
 | 
			
		||||
	LOGP(DMSC, LOGL_DEBUG, "Paging is complete.\n");
 | 
			
		||||
	LOGP(DPAG, LOGL_DEBUG, "Paging is complete.\n");
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -112,7 +84,7 @@ static int bssmap_handle_paging(struct gsm_network *net, struct msgb *msg, unsig
 | 
			
		||||
	u_int8_t chan_needed = RSL_CHANNEED_ANY;
 | 
			
		||||
	int paged;
 | 
			
		||||
 | 
			
		||||
	tlv_parse(&tp, &bss_att_tlvdef, msg->l4h + 1, payload_length - 1, 0, 0);
 | 
			
		||||
	tlv_parse(&tp, gsm0808_att_tlvdef(), msg->l4h + 1, payload_length - 1, 0, 0);
 | 
			
		||||
 | 
			
		||||
	if (!TLVP_PRESENT(&tp, GSM0808_IE_IMSI)) {
 | 
			
		||||
		LOGP(DMSC, LOGL_ERROR, "Mandantory IMSI not present.\n");
 | 
			
		||||
@@ -173,7 +145,7 @@ static int bssmap_handle_paging(struct gsm_network *net, struct msgb *msg, unsig
 | 
			
		||||
	subscr->tmsi = tmsi;
 | 
			
		||||
	subscr->lac = lac;
 | 
			
		||||
	paged = paging_request(net, subscr, chan_needed, bssmap_paging_cb, subscr);
 | 
			
		||||
	LOGP(DMSC, LOGL_DEBUG, "Paged IMSI: '%s' TMSI: '0x%x/%u' LAC: 0x%x on #bts: %d\n", mi_string, tmsi, tmsi, lac, paged);
 | 
			
		||||
	LOGP(DPAG, LOGL_DEBUG, "Paged IMSI: '%s' TMSI: '0x%x/%u' LAC: 0x%x on #bts: %d\n", mi_string, tmsi, tmsi, lac, paged);
 | 
			
		||||
 | 
			
		||||
	subscr_put(subscr);
 | 
			
		||||
	return -1;
 | 
			
		||||
@@ -242,7 +214,7 @@ static int bssmap_handle_cipher_mode(struct sccp_connection *conn,
 | 
			
		||||
	msg->lchan->msc_data->ciphering_handled = 1;
 | 
			
		||||
	msg->lchan->msc_data->block_gsm = 1;
 | 
			
		||||
 | 
			
		||||
	tlv_parse(&tp, &bss_att_tlvdef, msg->l4h + 1, payload_length - 1, 0, 0);
 | 
			
		||||
	tlv_parse(&tp, gsm0808_att_tlvdef(), msg->l4h + 1, payload_length - 1, 0, 0);
 | 
			
		||||
	if (!TLVP_PRESENT(&tp, GSM0808_IE_ENCRYPTION_INFORMATION)) {
 | 
			
		||||
		LOGP(DMSC, LOGL_ERROR, "IE Encryption Information missing.\n");
 | 
			
		||||
		goto reject;
 | 
			
		||||
@@ -431,7 +403,7 @@ static int handle_new_assignment(struct msgb *msg, int full_rate, int chan_mode)
 | 
			
		||||
	bts = msg->lchan->ts->trx->bts;
 | 
			
		||||
	chan_type = full_rate ? GSM_LCHAN_TCH_F : GSM_LCHAN_TCH_H;
 | 
			
		||||
 | 
			
		||||
	new_lchan = lchan_alloc(bts, chan_type);
 | 
			
		||||
	new_lchan = lchan_alloc(bts, chan_type, 0);
 | 
			
		||||
 | 
			
		||||
	if (!new_lchan) {
 | 
			
		||||
		LOGP(DMSC, LOGL_NOTICE, "No free channel.\n");
 | 
			
		||||
@@ -515,7 +487,7 @@ static int bssmap_handle_assignm_req(struct sccp_connection *conn,
 | 
			
		||||
 | 
			
		||||
	msc_data = msg->lchan->msc_data;
 | 
			
		||||
	network = msg->lchan->ts->trx->bts->network;
 | 
			
		||||
	tlv_parse(&tp, &bss_att_tlvdef, msg->l4h + 1, length - 1, 0, 0);
 | 
			
		||||
	tlv_parse(&tp, gsm0808_att_tlvdef(), msg->l4h + 1, length - 1, 0, 0);
 | 
			
		||||
 | 
			
		||||
	if (!TLVP_PRESENT(&tp, GSM0808_IE_CHANNEL_TYPE)) {
 | 
			
		||||
		LOGP(DMSC, LOGL_ERROR, "Mandantory channel type not present.\n");
 | 
			
		||||
 
 | 
			
		||||
@@ -223,7 +223,8 @@ _lc_find_bts(struct gsm_bts *bts, enum gsm_phys_chan_config pchan)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* Allocate a logical channel */
 | 
			
		||||
struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type)
 | 
			
		||||
struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type,
 | 
			
		||||
			      int allow_bigger)
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_lchan *lchan = NULL;
 | 
			
		||||
	enum gsm_phys_chan_config first, second;
 | 
			
		||||
@@ -241,6 +242,19 @@ struct gsm_lchan *lchan_alloc(struct gsm_bts *bts, enum gsm_chan_t type)
 | 
			
		||||
		lchan = _lc_find_bts(bts, first);
 | 
			
		||||
		if (lchan == NULL)
 | 
			
		||||
			lchan = _lc_find_bts(bts, second);
 | 
			
		||||
 | 
			
		||||
		/* allow to assign bigger channels */
 | 
			
		||||
		if (allow_bigger) {
 | 
			
		||||
			if (lchan == NULL) {
 | 
			
		||||
				lchan = _lc_find_bts(bts, GSM_PCHAN_TCH_H);
 | 
			
		||||
				type = GSM_LCHAN_TCH_H;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			if (lchan == NULL) {
 | 
			
		||||
				lchan = _lc_find_bts(bts, GSM_PCHAN_TCH_F);
 | 
			
		||||
				type = GSM_LCHAN_TCH_F;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
		break;
 | 
			
		||||
	case GSM_LCHAN_TCH_F:
 | 
			
		||||
		lchan = _lc_find_bts(bts, GSM_PCHAN_TCH_F);
 | 
			
		||||
@@ -322,6 +336,21 @@ void lchan_free(struct gsm_lchan *lchan)
 | 
			
		||||
	 * channel using it */
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * There was an error with the TRX and we need to forget
 | 
			
		||||
 * any state so that a lchan can be allocated again after
 | 
			
		||||
 * the trx is fully usable.
 | 
			
		||||
 */
 | 
			
		||||
void lchan_reset(struct gsm_lchan *lchan)
 | 
			
		||||
{
 | 
			
		||||
	bsc_del_timer(&lchan->T3101);
 | 
			
		||||
	bsc_del_timer(&lchan->T3111);
 | 
			
		||||
	bsc_del_timer(&lchan->error_timer);
 | 
			
		||||
 | 
			
		||||
	lchan->type = GSM_LCHAN_NONE;
 | 
			
		||||
	lchan->state = LCHAN_S_NONE;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int _lchan_release_next_sapi(struct gsm_lchan *lchan)
 | 
			
		||||
{
 | 
			
		||||
	int sapi;
 | 
			
		||||
 
 | 
			
		||||
@@ -420,7 +420,17 @@ e1inp_sign_link_create(struct e1inp_ts *ts, enum e1inp_sign_type type,
 | 
			
		||||
 | 
			
		||||
void e1inp_sign_link_destroy(struct e1inp_sign_link *link)
 | 
			
		||||
{
 | 
			
		||||
	struct msgb *msg;
 | 
			
		||||
 | 
			
		||||
	llist_del(&link->list);
 | 
			
		||||
	while (!llist_empty(&link->tx_list)) {
 | 
			
		||||
		msg = msgb_dequeue(&link->tx_list);
 | 
			
		||||
		msgb_free(msg);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (link->ts->type == E1INP_TS_TYPE_SIGN)
 | 
			
		||||
		bsc_del_timer(&link->ts->sign.tx_timer);
 | 
			
		||||
 | 
			
		||||
	talloc_free(link);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -281,6 +281,10 @@ struct gsm_network *gsm_network_init(u_int16_t country_code, u_int16_t network_c
 | 
			
		||||
	net->stats.call.dialled = counter_alloc("net.call.dialled");
 | 
			
		||||
	net->stats.call.alerted = counter_alloc("net.call.alerted");
 | 
			
		||||
	net->stats.call.connected = counter_alloc("net.call.connected");
 | 
			
		||||
	net->stats.chan.rf_fail = counter_alloc("net.chan.rf_fail");
 | 
			
		||||
	net->stats.chan.rll_err = counter_alloc("net.chan.rll_err");
 | 
			
		||||
	net->stats.bts.oml_fail = counter_alloc("net.bts.oml_fail");
 | 
			
		||||
	net->stats.bts.rsl_fail = counter_alloc("net.bts.rsl_fail");
 | 
			
		||||
 | 
			
		||||
	net->mncc_recv = mncc_recv;
 | 
			
		||||
 | 
			
		||||
@@ -290,6 +294,9 @@ struct gsm_network *gsm_network_init(u_int16_t country_code, u_int16_t network_c
 | 
			
		||||
	net->core_network_code = -1;
 | 
			
		||||
	net->rtp_base_port = 4000;
 | 
			
		||||
 | 
			
		||||
	net->msc_ip = talloc_strdup(net, "127.0.0.1");
 | 
			
		||||
	net->msc_port = 5000;
 | 
			
		||||
 | 
			
		||||
	return net;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -99,7 +99,7 @@ int bsc_handover_start(struct gsm_lchan *old_lchan, struct gsm_bts *bts)
 | 
			
		||||
 | 
			
		||||
	counter_inc(bts->network->stats.handover.attempted);
 | 
			
		||||
 | 
			
		||||
	new_lchan = lchan_alloc(bts, old_lchan->type);
 | 
			
		||||
	new_lchan = lchan_alloc(bts, old_lchan->type, 0);
 | 
			
		||||
	if (!new_lchan) {
 | 
			
		||||
		LOGP(DHO, LOGL_NOTICE, "No free channel\n");
 | 
			
		||||
		counter_inc(bts->network->stats.handover.no_channel);
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,8 @@
 | 
			
		||||
/* OpenBSC Abis input driver for ip.access */
 | 
			
		||||
 | 
			
		||||
/* (C) 2009 by Harald Welte <laforge@gnumonks.org>
 | 
			
		||||
 * (C) 2010 by Holger Hans Peter Freyther
 | 
			
		||||
 * (C) 2010 by On-Waves
 | 
			
		||||
 *
 | 
			
		||||
 * All Rights Reserved
 | 
			
		||||
 *
 | 
			
		||||
@@ -234,6 +236,8 @@ static int ipaccess_rcvmsg(struct e1inp_line *line, struct msgb *msg,
 | 
			
		||||
		}
 | 
			
		||||
		DEBUGP(DINP, "Identified BTS %u/%u/%u\n", site_id, bts_id, trx_id);
 | 
			
		||||
		if (bfd->priv_nr == PRIV_OML) {
 | 
			
		||||
			/* drop any old oml connection */
 | 
			
		||||
			ipaccess_drop_oml(bts);
 | 
			
		||||
			bts->oml_link = e1inp_sign_link_create(&line->ts[PRIV_OML - 1],
 | 
			
		||||
						  E1INP_SIGN_OML, bts->c0,
 | 
			
		||||
						  bts->oml_tei, 0);
 | 
			
		||||
@@ -242,6 +246,17 @@ static int ipaccess_rcvmsg(struct e1inp_line *line, struct msgb *msg,
 | 
			
		||||
			struct bsc_fd *newbfd;
 | 
			
		||||
			struct gsm_bts_trx *trx = gsm_bts_trx_num(bts, trx_id);
 | 
			
		||||
 | 
			
		||||
			/* drop any old rsl connection */
 | 
			
		||||
			ipaccess_drop_rsl(trx);
 | 
			
		||||
 | 
			
		||||
			if (!bts->oml_link) {
 | 
			
		||||
				bsc_unregister_fd(bfd);
 | 
			
		||||
				close(bfd->fd);
 | 
			
		||||
				bfd->fd = -1;
 | 
			
		||||
				talloc_free(bfd);
 | 
			
		||||
				return 0;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			bfd->data = line = bts->oml_link->ts->line;
 | 
			
		||||
			e1i_ts = &line->ts[PRIV_RSL + trx_id - 1];
 | 
			
		||||
			newbfd = &e1i_ts->driver.ipaccess.fd;
 | 
			
		||||
@@ -251,19 +266,13 @@ static int ipaccess_rcvmsg(struct e1inp_line *line, struct msgb *msg,
 | 
			
		||||
							E1INP_SIGN_RSL, trx,
 | 
			
		||||
							trx->rsl_tei, 0);
 | 
			
		||||
 | 
			
		||||
			if (newbfd->fd >= 0) {
 | 
			
		||||
				LOGP(DINP, LOGL_ERROR, "BTS is still registered. Closing old connection.\n");
 | 
			
		||||
				bsc_unregister_fd(newbfd);
 | 
			
		||||
				close(newbfd->fd);
 | 
			
		||||
				newbfd->fd = -1;
 | 
			
		||||
			}
 | 
			
		||||
 | 
			
		||||
			/* get rid of our old temporary bfd */
 | 
			
		||||
			memcpy(newbfd, bfd, sizeof(*newbfd));
 | 
			
		||||
			newbfd->priv_nr = PRIV_RSL + trx_id;
 | 
			
		||||
			bsc_unregister_fd(bfd);
 | 
			
		||||
			bsc_register_fd(newbfd);
 | 
			
		||||
			bfd->fd = -1;
 | 
			
		||||
			talloc_free(bfd);
 | 
			
		||||
			bsc_register_fd(newbfd);
 | 
			
		||||
		}
 | 
			
		||||
		break;
 | 
			
		||||
	}
 | 
			
		||||
@@ -328,6 +337,103 @@ struct msgb *ipaccess_read_msg(struct bsc_fd *bfd, int *error)
 | 
			
		||||
	return msg;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int ipaccess_drop_oml(struct gsm_bts *bts)
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_bts_trx *trx;
 | 
			
		||||
	struct e1inp_ts *ts;
 | 
			
		||||
	struct e1inp_line *line;
 | 
			
		||||
	struct bsc_fd *bfd;
 | 
			
		||||
 | 
			
		||||
	if (!bts || !bts->oml_link)
 | 
			
		||||
		return -1;
 | 
			
		||||
 | 
			
		||||
	/* send OML down */
 | 
			
		||||
	ts = bts->oml_link->ts;
 | 
			
		||||
	line = ts->line;
 | 
			
		||||
	e1inp_event(ts, EVT_E1_TEI_DN, bts->oml_link->tei, bts->oml_link->sapi);
 | 
			
		||||
 | 
			
		||||
	bfd = &ts->driver.ipaccess.fd;
 | 
			
		||||
	bsc_unregister_fd(bfd);
 | 
			
		||||
	close(bfd->fd);
 | 
			
		||||
	bfd->fd = -1;
 | 
			
		||||
 | 
			
		||||
	/* clean up OML and RSL */
 | 
			
		||||
	e1inp_sign_link_destroy(bts->oml_link);
 | 
			
		||||
	bts->oml_link = NULL;
 | 
			
		||||
	bts->ip_access.flags = 0;
 | 
			
		||||
 | 
			
		||||
	/* drop all RSL connections too */
 | 
			
		||||
	llist_for_each_entry(trx, &bts->trx_list, list)
 | 
			
		||||
		ipaccess_drop_rsl(trx);
 | 
			
		||||
 | 
			
		||||
	/* kill the E1 line now... as we have no one left to use it */
 | 
			
		||||
	talloc_free(line);
 | 
			
		||||
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int ipaccess_drop(struct e1inp_ts *ts, struct bsc_fd *bfd)
 | 
			
		||||
{
 | 
			
		||||
	struct e1inp_sign_link *link;
 | 
			
		||||
	int bts_nr;
 | 
			
		||||
 | 
			
		||||
	if (!ts) {
 | 
			
		||||
		/*
 | 
			
		||||
		 * If we don't have a TS this means that this is a RSL
 | 
			
		||||
		 * connection but we are not past the authentication
 | 
			
		||||
		 * handling yet. So we can safely delete this bfd and
 | 
			
		||||
		 * wait for a reconnect.
 | 
			
		||||
		 */
 | 
			
		||||
		bsc_unregister_fd(bfd);
 | 
			
		||||
		close(bfd->fd);
 | 
			
		||||
		bfd->fd = -1;
 | 
			
		||||
		talloc_free(bfd);
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* attempt to find a signalling link */
 | 
			
		||||
	if (ts->type == E1INP_TS_TYPE_SIGN) {
 | 
			
		||||
		llist_for_each_entry(link, &ts->sign.sign_links, list) {
 | 
			
		||||
			bts_nr = link->trx->bts->bts_nr;
 | 
			
		||||
			/* we have issues just reconnecting RLS so we drop OML */
 | 
			
		||||
			ipaccess_drop_oml(link->trx->bts);
 | 
			
		||||
			return bts_nr;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* error case */
 | 
			
		||||
	LOGP(DINP, LOGL_ERROR, "Failed to find a signalling link for ts: %p\n", ts);
 | 
			
		||||
	bsc_unregister_fd(bfd);
 | 
			
		||||
	close(bfd->fd);
 | 
			
		||||
	bfd->fd = -1;
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int ipaccess_drop_rsl(struct gsm_bts_trx *trx)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_fd *bfd;
 | 
			
		||||
	struct e1inp_ts *ts;
 | 
			
		||||
 | 
			
		||||
	if (!trx || !trx->rsl_link)
 | 
			
		||||
		return -1;
 | 
			
		||||
 | 
			
		||||
	/* send RSL down */
 | 
			
		||||
	ts = trx->rsl_link->ts;
 | 
			
		||||
	e1inp_event(ts, EVT_E1_TEI_DN, trx->rsl_link->tei, trx->rsl_link->sapi);
 | 
			
		||||
 | 
			
		||||
	/* close the socket */
 | 
			
		||||
	bfd = &ts->driver.ipaccess.fd;
 | 
			
		||||
	bsc_unregister_fd(bfd);
 | 
			
		||||
	close(bfd->fd);
 | 
			
		||||
	bfd->fd = -1;
 | 
			
		||||
 | 
			
		||||
	/* destroy */
 | 
			
		||||
	e1inp_sign_link_destroy(trx->rsl_link);
 | 
			
		||||
	trx->rsl_link = NULL;
 | 
			
		||||
 | 
			
		||||
	return -1;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int handle_ts1_read(struct bsc_fd *bfd)
 | 
			
		||||
{
 | 
			
		||||
	struct e1inp_line *line = bfd->data;
 | 
			
		||||
@@ -341,18 +447,12 @@ static int handle_ts1_read(struct bsc_fd *bfd)
 | 
			
		||||
	msg = ipaccess_read_msg(bfd, &error);
 | 
			
		||||
	if (!msg) {
 | 
			
		||||
		if (error == 0) {
 | 
			
		||||
			link = e1inp_lookup_sign_link(e1i_ts, IPAC_PROTO_OML, 0);
 | 
			
		||||
			if (link) {
 | 
			
		||||
				link->trx->bts->ip_access.flags = 0;
 | 
			
		||||
			int ret = ipaccess_drop(e1i_ts, bfd);
 | 
			
		||||
			if (ret >= 0)
 | 
			
		||||
				LOGP(DINP, LOGL_NOTICE, "BTS %u disappeared, dead socket\n",
 | 
			
		||||
					link->trx->bts->nr);
 | 
			
		||||
			} else
 | 
			
		||||
					ret);
 | 
			
		||||
			else
 | 
			
		||||
				LOGP(DINP, LOGL_NOTICE, "unknown BTS disappeared, dead socket\n");
 | 
			
		||||
			e1inp_event(e1i_ts, EVT_E1_TEI_DN, 0, IPAC_PROTO_RSL);
 | 
			
		||||
			e1inp_event(e1i_ts, EVT_E1_TEI_DN, 0, IPAC_PROTO_OML);
 | 
			
		||||
			bsc_unregister_fd(bfd);
 | 
			
		||||
			close(bfd->fd);
 | 
			
		||||
			bfd->fd = -1;
 | 
			
		||||
		}
 | 
			
		||||
		return error;
 | 
			
		||||
	}
 | 
			
		||||
@@ -362,13 +462,8 @@ static int handle_ts1_read(struct bsc_fd *bfd)
 | 
			
		||||
	hh = (struct ipaccess_head *) msg->data;
 | 
			
		||||
	if (hh->proto == IPAC_PROTO_IPACCESS) {
 | 
			
		||||
		ret = ipaccess_rcvmsg(line, msg, bfd);
 | 
			
		||||
		if (ret < 0) {
 | 
			
		||||
			e1inp_event(e1i_ts, EVT_E1_TEI_DN, 0, IPAC_PROTO_RSL);
 | 
			
		||||
			e1inp_event(e1i_ts, EVT_E1_TEI_DN, 0, IPAC_PROTO_OML);
 | 
			
		||||
			bsc_unregister_fd(bfd);
 | 
			
		||||
			close(bfd->fd);
 | 
			
		||||
			bfd->fd = -1;
 | 
			
		||||
		}
 | 
			
		||||
		if (ret < 0)
 | 
			
		||||
			ipaccess_drop(e1i_ts, bfd);
 | 
			
		||||
		msgb_free(msg);
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 
 | 
			
		||||
@@ -33,6 +33,8 @@
 | 
			
		||||
#include <vty/command.h>
 | 
			
		||||
#include <vty/vty.h>
 | 
			
		||||
 | 
			
		||||
#include <string.h>
 | 
			
		||||
 | 
			
		||||
static struct mgcp_config *g_cfg = NULL;
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
@@ -49,16 +51,18 @@ static int config_write_mgcp(struct vty *vty)
 | 
			
		||||
	vty_out(vty, "mgcp%s", VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->local_ip)
 | 
			
		||||
		vty_out(vty, "  local ip %s%s", g_cfg->local_ip, VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->bts_ip)
 | 
			
		||||
	if (g_cfg->bts_ip && strlen(g_cfg->bts_ip) != 0)
 | 
			
		||||
		vty_out(vty, "  bts ip %s%s", g_cfg->bts_ip, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  bind ip %s%s", g_cfg->source_addr, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  bind port %u%s", g_cfg->source_port, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  bind early %u%s", !!g_cfg->early_bind, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  rtp base %u%s", g_cfg->rtp_base_port, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  sdp audio payload number %u%s", g_cfg->audio_payload, VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->audio_payload != -1)
 | 
			
		||||
		vty_out(vty, "  sdp audio payload number %d%s", g_cfg->audio_payload, VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->audio_name)
 | 
			
		||||
		vty_out(vty, "  sdp audio payload name %s%s", g_cfg->audio_name, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  loop %u%s", !!g_cfg->audio_loop, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  endpoints %u%s", g_cfg->number_endpoints, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  number endpoints %u%s", g_cfg->number_endpoints - 1, VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->forward_ip)
 | 
			
		||||
		vty_out(vty, "  forward audio ip %s%s", g_cfg->forward_ip, VTY_NEWLINE);
 | 
			
		||||
	if (g_cfg->forward_port != 0)
 | 
			
		||||
 
 | 
			
		||||
@@ -27,6 +27,7 @@
 | 
			
		||||
#include <openbsc/mgcp_internal.h>
 | 
			
		||||
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
#include <osmocore/gsm0808.h>
 | 
			
		||||
 | 
			
		||||
#include <netinet/in.h>
 | 
			
		||||
#include <arpa/inet.h>
 | 
			
		||||
@@ -91,8 +92,9 @@ void bsc_mgcp_free_endpoints(struct bsc_nat *nat)
 | 
			
		||||
		bsc_mgcp_free_endpoint(nat, i);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct bsc_connection *bsc_mgcp_find_con(struct bsc_nat *nat, int endpoint)
 | 
			
		||||
struct sccp_connections *bsc_mgcp_find_con(struct bsc_nat *nat, int endpoint)
 | 
			
		||||
{
 | 
			
		||||
	struct sccp_connections *con = NULL;
 | 
			
		||||
	struct sccp_connections *sccp;
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(sccp, &nat->sccp_connections, list_entry) {
 | 
			
		||||
@@ -101,9 +103,12 @@ struct bsc_connection *bsc_mgcp_find_con(struct bsc_nat *nat, int endpoint)
 | 
			
		||||
		if (mgcp_timeslot_to_endpoint(0, sccp->msc_timeslot) != endpoint)
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		return sccp->bsc;
 | 
			
		||||
		con = sccp;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (con)
 | 
			
		||||
		return con;
 | 
			
		||||
 | 
			
		||||
	LOGP(DMGCP, LOGL_ERROR, "Failed to find the connection.\n");
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
@@ -112,7 +117,7 @@ int bsc_mgcp_policy_cb(struct mgcp_config *cfg, int endpoint, int state, const c
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_nat *nat;
 | 
			
		||||
	struct bsc_endpoint *bsc_endp;
 | 
			
		||||
	struct bsc_connection *bsc_con;
 | 
			
		||||
	struct sccp_connections *sccp;
 | 
			
		||||
	struct mgcp_endpoint *mgcp_endp;
 | 
			
		||||
	struct msgb *bsc_msg;
 | 
			
		||||
 | 
			
		||||
@@ -120,9 +125,9 @@ int bsc_mgcp_policy_cb(struct mgcp_config *cfg, int endpoint, int state, const c
 | 
			
		||||
	bsc_endp = &nat->bsc_endpoints[endpoint];
 | 
			
		||||
	mgcp_endp = &nat->mgcp_cfg->endpoints[endpoint];
 | 
			
		||||
 | 
			
		||||
	bsc_con = bsc_mgcp_find_con(nat, endpoint);
 | 
			
		||||
	sccp = bsc_mgcp_find_con(nat, endpoint);
 | 
			
		||||
 | 
			
		||||
	if (!bsc_con) {
 | 
			
		||||
	if (!sccp) {
 | 
			
		||||
		LOGP(DMGCP, LOGL_ERROR, "Did not find BSC for a new connection on 0x%x for %d\n", endpoint, state);
 | 
			
		||||
 | 
			
		||||
		switch (state) {
 | 
			
		||||
@@ -158,14 +163,14 @@ int bsc_mgcp_policy_cb(struct mgcp_config *cfg, int endpoint, int state, const c
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	bsc_endp->transaction_id = talloc_strdup(nat, transaction_id);
 | 
			
		||||
	bsc_endp->bsc = bsc_con;
 | 
			
		||||
	bsc_endp->bsc = sccp->bsc;
 | 
			
		||||
	bsc_endp->pending_delete = 0;
 | 
			
		||||
 | 
			
		||||
	/* we need to update some bits */
 | 
			
		||||
	if (state == MGCP_ENDP_CRCX) {
 | 
			
		||||
		struct sockaddr_in sock;
 | 
			
		||||
		socklen_t len = sizeof(sock);
 | 
			
		||||
		if (getpeername(bsc_con->write_queue.bfd.fd, (struct sockaddr *) &sock, &len) != 0) {
 | 
			
		||||
		if (getpeername(sccp->bsc->write_queue.bfd.fd, (struct sockaddr *) &sock, &len) != 0) {
 | 
			
		||||
			LOGP(DMGCP, LOGL_ERROR, "Can not get the peername...%d/%s\n",
 | 
			
		||||
			      errno, strerror(errno));
 | 
			
		||||
		} else {
 | 
			
		||||
@@ -173,11 +178,12 @@ int bsc_mgcp_policy_cb(struct mgcp_config *cfg, int endpoint, int state, const c
 | 
			
		||||
		}
 | 
			
		||||
	} else if (state == MGCP_ENDP_DLCX) {
 | 
			
		||||
		/* we will free the endpoint now in case the BSS does not respond */
 | 
			
		||||
		bsc_mgcp_clear(sccp);
 | 
			
		||||
		bsc_endp->pending_delete = 1;
 | 
			
		||||
		mgcp_free_endp(mgcp_endp);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	bsc_write(bsc_con, bsc_msg, NAT_IPAC_PROTO_MGCP);
 | 
			
		||||
	bsc_write(sccp->bsc, bsc_msg, NAT_IPAC_PROTO_MGCP);
 | 
			
		||||
	return MGCP_POLICY_DEFER;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -51,10 +51,10 @@
 | 
			
		||||
 | 
			
		||||
struct log_target *stderr_target;
 | 
			
		||||
static const char *config_file = "bsc-nat.cfg";
 | 
			
		||||
static char *msc_address = "127.0.0.1";
 | 
			
		||||
static struct in_addr local_addr;
 | 
			
		||||
static struct bsc_msc_connection *msc_con;
 | 
			
		||||
static struct bsc_fd bsc_listen;
 | 
			
		||||
static const char *msc_ip = NULL;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static struct bsc_nat *nat;
 | 
			
		||||
@@ -258,7 +258,10 @@ static int forward_sccp_to_bts(struct msgb *msg)
 | 
			
		||||
		case SCCP_MSG_TYPE_IT:
 | 
			
		||||
			con = patch_sccp_src_ref_to_bsc(msg, parsed, nat);
 | 
			
		||||
			if (parsed->gsm_type == BSS_MAP_MSG_ASSIGMENT_RQST) {
 | 
			
		||||
				counter_inc(nat->stats.sccp.calls);
 | 
			
		||||
 | 
			
		||||
				if (con) {
 | 
			
		||||
					counter_inc(con->bsc->cfg->stats.sccp.calls);
 | 
			
		||||
					if (bsc_mgcp_assign(con, msg) != 0)
 | 
			
		||||
						LOGP(DNAT, LOGL_ERROR, "Failed to assign...\n");
 | 
			
		||||
				} else
 | 
			
		||||
@@ -306,11 +309,13 @@ send_to_all:
 | 
			
		||||
	 * message and then send it to the authenticated messages...
 | 
			
		||||
	 */
 | 
			
		||||
	if (parsed->ipa_proto == IPAC_PROTO_SCCP && parsed->gsm_type == BSS_MAP_MSG_PAGING) {
 | 
			
		||||
		bsc = bsc_nat_find_bsc(nat, msg);
 | 
			
		||||
		int lac;
 | 
			
		||||
		bsc = bsc_nat_find_bsc(nat, msg, &lac);
 | 
			
		||||
		if (bsc)
 | 
			
		||||
			bsc_send_data(bsc, msg->l2h, msgb_l2len(msg), parsed->ipa_proto);
 | 
			
		||||
		else
 | 
			
		||||
			LOGP(DNAT, LOGL_ERROR, "Could not determine BSC for paging.\n");
 | 
			
		||||
			LOGP(DNAT, LOGL_ERROR, "Could not determine BSC for paging on lac: %d/0x%x\n",
 | 
			
		||||
			     lac, lac);
 | 
			
		||||
 | 
			
		||||
		goto exit;
 | 
			
		||||
	}
 | 
			
		||||
@@ -331,6 +336,8 @@ static void msc_connection_was_lost(struct bsc_msc_connection *con)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_connection *bsc, *tmp;
 | 
			
		||||
 | 
			
		||||
	counter_inc(nat->stats.msc.reconn);
 | 
			
		||||
 | 
			
		||||
	LOGP(DMSC, LOGL_ERROR, "Closing all connections downstream.\n");
 | 
			
		||||
	llist_for_each_entry_safe(bsc, tmp, &nat->bsc_connections, list_entry)
 | 
			
		||||
		remove_bsc_connection(bsc);
 | 
			
		||||
@@ -453,9 +460,14 @@ static void remove_bsc_connection(struct bsc_connection *connection)
 | 
			
		||||
 | 
			
		||||
static void ipaccess_close_bsc(void *data)
 | 
			
		||||
{
 | 
			
		||||
	struct sockaddr_in sock;
 | 
			
		||||
	socklen_t len = sizeof(sock);
 | 
			
		||||
	struct bsc_connection *conn = data;
 | 
			
		||||
 | 
			
		||||
	LOGP(DNAT, LOGL_ERROR, "BSC didn't respond to identity request. Closing.\n");
 | 
			
		||||
 | 
			
		||||
	getpeername(conn->write_queue.bfd.fd, (struct sockaddr *) &sock, &len);
 | 
			
		||||
	LOGP(DNAT, LOGL_ERROR, "BSC on %s didn't respond to identity request. Closing.\n",
 | 
			
		||||
	     inet_ntoa(sock.sin_addr));
 | 
			
		||||
	remove_bsc_connection(conn);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -466,13 +478,16 @@ static void ipaccess_auth_bsc(struct tlv_parsed *tvp, struct bsc_connection *bsc
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(conf, &bsc->nat->bsc_configs, entry) {
 | 
			
		||||
		if (strcmp(conf->token, token) == 0) {
 | 
			
		||||
			counter_inc(conf->stats.net.reconn);
 | 
			
		||||
			bsc->authenticated = 1;
 | 
			
		||||
			bsc->cfg = conf;
 | 
			
		||||
			bsc_del_timer(&bsc->id_timeout);
 | 
			
		||||
			LOGP(DNAT, LOGL_NOTICE, "Authenticated bsc nr: %d lac: %d\n", conf->nr, conf->lac);
 | 
			
		||||
			break;
 | 
			
		||||
			return;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	LOGP(DNAT, LOGL_ERROR, "No bsc found for token %s.\n", token);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int forward_sccp_to_msc(struct bsc_connection *bsc, struct msgb *msg)
 | 
			
		||||
@@ -632,6 +647,9 @@ static int ipaccess_listen_bsc_cb(struct bsc_fd *bfd, unsigned int what)
 | 
			
		||||
		return ret;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* count the reconnect */
 | 
			
		||||
	counter_inc(nat->stats.bsc.reconn);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * if we are not connected to a msc... just close the socket
 | 
			
		||||
	 */
 | 
			
		||||
@@ -773,7 +791,7 @@ static void handle_options(int argc, char** argv)
 | 
			
		||||
			log_set_print_timestamp(stderr_target, 1);
 | 
			
		||||
			break;
 | 
			
		||||
		case 'm':
 | 
			
		||||
			msc_address = strdup(optarg);
 | 
			
		||||
			msc_ip = optarg;
 | 
			
		||||
			break;
 | 
			
		||||
		case 'l':
 | 
			
		||||
			inet_aton(optarg, &local_addr);
 | 
			
		||||
@@ -806,10 +824,6 @@ int main(int argc, char** argv)
 | 
			
		||||
	log_add_target(stderr_target);
 | 
			
		||||
	log_set_all_filter(stderr_target, 1);
 | 
			
		||||
 | 
			
		||||
	/* parse options */
 | 
			
		||||
	local_addr.s_addr = INADDR_ANY;
 | 
			
		||||
	handle_options(argc, argv);
 | 
			
		||||
 | 
			
		||||
	nat = bsc_nat_alloc();
 | 
			
		||||
	if (!nat) {
 | 
			
		||||
		fprintf(stderr, "Failed to allocate the BSC nat.\n");
 | 
			
		||||
@@ -817,6 +831,14 @@ int main(int argc, char** argv)
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	nat->mgcp_cfg = talloc_zero(nat, struct mgcp_config);
 | 
			
		||||
	if (!nat->mgcp_cfg) {
 | 
			
		||||
		fprintf(stderr, "Failed to allocate MGCP cfg.\n");
 | 
			
		||||
		return -5;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* parse options */
 | 
			
		||||
	local_addr.s_addr = INADDR_ANY;
 | 
			
		||||
	handle_options(argc, argv);
 | 
			
		||||
 | 
			
		||||
	/* init vty and parse */
 | 
			
		||||
	bsc_nat_vty_init(nat);
 | 
			
		||||
@@ -826,6 +848,10 @@ int main(int argc, char** argv)
 | 
			
		||||
		return -3;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	/* over rule the VTY config */
 | 
			
		||||
	if (msc_ip)
 | 
			
		||||
		bsc_nat_set_msc_ip(nat, msc_ip);
 | 
			
		||||
 | 
			
		||||
	/* seed the PRNG */
 | 
			
		||||
	srand(time(NULL));
 | 
			
		||||
 | 
			
		||||
@@ -836,7 +862,7 @@ int main(int argc, char** argv)
 | 
			
		||||
		return -4;
 | 
			
		||||
 | 
			
		||||
	/* connect to the MSC */
 | 
			
		||||
	msc_con = bsc_msc_create(msc_address, 5000);
 | 
			
		||||
	msc_con = bsc_msc_create(nat->msc_ip, nat->msc_port);
 | 
			
		||||
	if (!msc_con) {
 | 
			
		||||
		fprintf(stderr, "Creating a bsc_msc_connection failed.\n");
 | 
			
		||||
		exit(1);
 | 
			
		||||
 
 | 
			
		||||
@@ -30,6 +30,7 @@
 | 
			
		||||
 | 
			
		||||
#include <osmocore/linuxlist.h>
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
#include <osmocore/gsm0808.h>
 | 
			
		||||
 | 
			
		||||
#include <sccp/sccp.h>
 | 
			
		||||
 | 
			
		||||
@@ -45,9 +46,23 @@ struct bsc_nat *bsc_nat_alloc(void)
 | 
			
		||||
	INIT_LLIST_HEAD(&nat->sccp_connections);
 | 
			
		||||
	INIT_LLIST_HEAD(&nat->bsc_connections);
 | 
			
		||||
	INIT_LLIST_HEAD(&nat->bsc_configs);
 | 
			
		||||
	nat->stats.sccp.conn = counter_alloc("nat.sccp.conn");
 | 
			
		||||
	nat->stats.sccp.calls = counter_alloc("nat.sccp.calls");
 | 
			
		||||
	nat->stats.bsc.reconn = counter_alloc("nat.bsc.conn");
 | 
			
		||||
	nat->stats.bsc.auth_fail = counter_alloc("nat.bsc.auth_fail");
 | 
			
		||||
	nat->stats.msc.reconn = counter_alloc("nat.msc.conn");
 | 
			
		||||
	nat->msc_ip = talloc_strdup(nat, "127.0.0.1");
 | 
			
		||||
	nat->msc_port = 5000;
 | 
			
		||||
	return nat;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void bsc_nat_set_msc_ip(struct bsc_nat *nat, const char *ip)
 | 
			
		||||
{
 | 
			
		||||
	if (nat->msc_ip)
 | 
			
		||||
		talloc_free(nat->msc_ip);
 | 
			
		||||
	nat->msc_ip = talloc_strdup(nat, ip);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct bsc_connection *bsc_connection_alloc(struct bsc_nat *nat)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_connection *con = talloc_zero(nat, struct bsc_connection);
 | 
			
		||||
@@ -69,9 +84,13 @@ struct bsc_config *bsc_config_alloc(struct bsc_nat *nat, const char *token, unsi
 | 
			
		||||
	conf->nr = nat->num_bsc;
 | 
			
		||||
	conf->nat = nat;
 | 
			
		||||
 | 
			
		||||
	llist_add(&conf->entry, &nat->bsc_configs);
 | 
			
		||||
	llist_add_tail(&conf->entry, &nat->bsc_configs);
 | 
			
		||||
	++nat->num_bsc;
 | 
			
		||||
 | 
			
		||||
	conf->stats.sccp.conn = counter_alloc("nat.bsc.sccp.conn");
 | 
			
		||||
	conf->stats.sccp.calls = counter_alloc("nat.bsc.sccp.calls");
 | 
			
		||||
	conf->stats.net.reconn = counter_alloc("nat.bsc.net.reconnects");
 | 
			
		||||
 | 
			
		||||
	return conf;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -85,7 +104,7 @@ void sccp_connection_destroy(struct sccp_connections *conn)
 | 
			
		||||
	talloc_free(conn);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg)
 | 
			
		||||
struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg, int *lac_out)
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_connection *bsc;
 | 
			
		||||
	int data_length;
 | 
			
		||||
@@ -93,6 +112,8 @@ struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg)
 | 
			
		||||
	struct tlv_parsed tp;
 | 
			
		||||
	int i = 0;
 | 
			
		||||
 | 
			
		||||
	*lac_out = -1;
 | 
			
		||||
 | 
			
		||||
	if (!msg->l3h || msgb_l3len(msg) < 3) {
 | 
			
		||||
		LOGP(DNAT, LOGL_ERROR, "Paging message is too short.\n");
 | 
			
		||||
		return NULL;
 | 
			
		||||
@@ -114,6 +135,7 @@ struct bsc_connection *bsc_nat_find_bsc(struct bsc_nat *nat, struct msgb *msg)
 | 
			
		||||
	/* Currently we only handle one BSC */
 | 
			
		||||
	for (i = 1; i < data_length - 1; i += 2) {
 | 
			
		||||
		unsigned int _lac = ntohs(*(unsigned int *) &data[i]);
 | 
			
		||||
		*lac_out = _lac;
 | 
			
		||||
		llist_for_each_entry(bsc, &nat->bsc_connections, list_entry) {
 | 
			
		||||
			if (!bsc->cfg)
 | 
			
		||||
				continue;
 | 
			
		||||
 
 | 
			
		||||
@@ -51,6 +51,12 @@ static struct cmd_node bsc_node = {
 | 
			
		||||
static int config_write_nat(struct vty *vty)
 | 
			
		||||
{
 | 
			
		||||
	vty_out(vty, "nat%s", VTY_NEWLINE);
 | 
			
		||||
	if (_nat->imsi_allow)
 | 
			
		||||
		vty_out(vty, " imsi allow %s%s", _nat->imsi_allow, VTY_NEWLINE);
 | 
			
		||||
	if (_nat->imsi_deny)
 | 
			
		||||
		vty_out(vty, " insi deny %s%s", _nat->imsi_deny, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " msc ip %s%s", _nat->msc_ip, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " msc port %d%s", _nat->msc_port, VTY_NEWLINE);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -58,7 +64,11 @@ static void config_write_bsc_single(struct vty *vty, struct bsc_config *bsc)
 | 
			
		||||
{
 | 
			
		||||
	vty_out(vty, " bsc %u%s", bsc->nr, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  token %s%s", bsc->token, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  lac %u%s", bsc->lac, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "  location_area_code %u%s", bsc->lac, VTY_NEWLINE);
 | 
			
		||||
	if (bsc->imsi_allow)
 | 
			
		||||
		vty_out(vty, "   imsi allow %s%s", bsc->imsi_allow, VTY_NEWLINE);
 | 
			
		||||
	if (bsc->imsi_deny)
 | 
			
		||||
		vty_out(vty, "   imsi deny %s%s", bsc->imsi_deny, VTY_NEWLINE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static int config_write_bsc(struct vty *vty)
 | 
			
		||||
@@ -71,7 +81,7 @@ static int config_write_bsc(struct vty *vty)
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
DEFUN(show_sccp, show_sccp_cmd, "show connections sccp",
 | 
			
		||||
DEFUN(show_sccp, show_sccp_cmd, "show sccp connections",
 | 
			
		||||
      SHOW_STR "Display information about current SCCP connections")
 | 
			
		||||
{
 | 
			
		||||
	struct sccp_connections *con;
 | 
			
		||||
@@ -88,7 +98,7 @@ DEFUN(show_sccp, show_sccp_cmd, "show connections sccp",
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_bsc, show_bsc_cmd, "show connections bsc",
 | 
			
		||||
DEFUN(show_bsc, show_bsc_cmd, "show bsc connections",
 | 
			
		||||
      SHOW_STR "Display information about current BSCs")
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_connection *con;
 | 
			
		||||
@@ -107,18 +117,51 @@ DEFUN(show_bsc, show_bsc_cmd, "show connections bsc",
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_bsc_cfg, show_bsc_cfg_cmd, "show bsc config",
 | 
			
		||||
      SHOW_STR "Display information about known BSC configs")
 | 
			
		||||
DEFUN(show_bsc_cfg, show_bsc_cfg_cmd, "bsc config show",
 | 
			
		||||
      "Display information about known BSC configs")
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_config *conf;
 | 
			
		||||
	llist_for_each_entry(conf, &_nat->bsc_configs, entry) {
 | 
			
		||||
		vty_out(vty, "BSC token: '%s' lac: %u nr: %u%s",
 | 
			
		||||
			conf->token, conf->lac, conf->nr, VTY_NEWLINE);
 | 
			
		||||
		vty_out(vty, " imsi_allow: '%s' imsi_deny: '%s'%s",
 | 
			
		||||
			conf->imsi_allow ? conf->imsi_allow: "any",
 | 
			
		||||
			conf->imsi_deny  ? conf->imsi_deny : "none",
 | 
			
		||||
			VTY_NEWLINE);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_stats,
 | 
			
		||||
      show_stats_cmd,
 | 
			
		||||
      "show statistics",
 | 
			
		||||
	SHOW_STR "Display network statistics\n")
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_config *conf;
 | 
			
		||||
 | 
			
		||||
	vty_out(vty, "NAT statistics%s", VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " SCCP Connections %lu total, %lu calls%s",
 | 
			
		||||
		counter_get(_nat->stats.sccp.conn),
 | 
			
		||||
		counter_get(_nat->stats.sccp.calls), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " MSC Connections %lu%s",
 | 
			
		||||
		counter_get(_nat->stats.msc.reconn), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " BSC Connections %lu total, %lu auth failed.%s",
 | 
			
		||||
		counter_get(_nat->stats.bsc.reconn),
 | 
			
		||||
		counter_get(_nat->stats.bsc.auth_fail), VTY_NEWLINE);
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(conf, &_nat->bsc_configs, entry) {
 | 
			
		||||
		vty_out(vty, " BSC lac: %d nr: %d%s",
 | 
			
		||||
			conf->lac, conf->nr, VTY_NEWLINE);
 | 
			
		||||
		vty_out(vty, "   SCCP Connnections %lu total, %lu calls%s",
 | 
			
		||||
			counter_get(conf->stats.sccp.conn),
 | 
			
		||||
			counter_get(conf->stats.sccp.calls), VTY_NEWLINE);
 | 
			
		||||
		vty_out(vty, "   BSC Connections %lu total%s",
 | 
			
		||||
			counter_get(conf->stats.net.reconn), VTY_NEWLINE);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_nat, cfg_nat_cmd, "nat", "Configute the NAT")
 | 
			
		||||
{
 | 
			
		||||
@@ -128,6 +171,58 @@ DEFUN(cfg_nat, cfg_nat_cmd, "nat", "Configute the NAT")
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void parse_reg(void *ctx, regex_t *reg, char **imsi, int argc, const char **argv)
 | 
			
		||||
{
 | 
			
		||||
	if (*imsi) {
 | 
			
		||||
		talloc_free(*imsi);
 | 
			
		||||
		*imsi = NULL;
 | 
			
		||||
	}
 | 
			
		||||
	regfree(reg);
 | 
			
		||||
 | 
			
		||||
	if (argc > 0) {
 | 
			
		||||
		*imsi = talloc_strdup(ctx, argv[0]);
 | 
			
		||||
		regcomp(reg, argv[0], 0);
 | 
			
		||||
	}
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_nat_imsi_allow,
 | 
			
		||||
      cfg_nat_imsi_allow_cmd,
 | 
			
		||||
      "imsi allow [REGEXP]",
 | 
			
		||||
      "Allow matching IMSIs to talk to the MSC. "
 | 
			
		||||
      "The defualt is to allow everyone.")
 | 
			
		||||
{
 | 
			
		||||
	parse_reg(_nat, &_nat->imsi_allow_re, &_nat->imsi_allow, argc, argv);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_nat_imsi_deny,
 | 
			
		||||
      cfg_nat_imsi_deny_cmd,
 | 
			
		||||
      "imsi deny [REGEXP]",
 | 
			
		||||
      "Deny matching IMSIs to talk to the MSC. "
 | 
			
		||||
      "The defualt is to not deny.")
 | 
			
		||||
{
 | 
			
		||||
	parse_reg(_nat, &_nat->imsi_deny_re, &_nat->imsi_deny, argc, argv);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_nat_msc_ip,
 | 
			
		||||
      cfg_nat_msc_ip_cmd,
 | 
			
		||||
      "msc ip IP",
 | 
			
		||||
      "Set the IP address of the MSC.")
 | 
			
		||||
{
 | 
			
		||||
	bsc_nat_set_msc_ip(_nat, argv[0]);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_nat_msc_port,
 | 
			
		||||
      cfg_nat_msc_port_cmd,
 | 
			
		||||
      "msc port <1-65500>",
 | 
			
		||||
      "Set the port of the MSC.")
 | 
			
		||||
{
 | 
			
		||||
	_nat->msc_port = atoi(argv[0]);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* per BSC configuration */
 | 
			
		||||
DEFUN(cfg_bsc, cfg_bsc_cmd, "bsc BSC_NR", "Select a BSC to configure\n")
 | 
			
		||||
{
 | 
			
		||||
@@ -196,6 +291,30 @@ DEFUN(cfg_bsc_lac, cfg_bsc_lac_cmd, "location_area_code <0-65535>",
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_bsc_imsi_allow,
 | 
			
		||||
      cfg_bsc_imsi_allow_cmd,
 | 
			
		||||
      "imsi allow [REGEXP]",
 | 
			
		||||
      "Allow IMSIs with the following network to talk to the MSC."
 | 
			
		||||
      "The default is to allow everyone)")
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_config *conf = vty->index;
 | 
			
		||||
 | 
			
		||||
	parse_reg(conf, &conf->imsi_allow_re, &conf->imsi_allow, argc, argv);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_bsc_imsi_deny,
 | 
			
		||||
      cfg_bsc_imsi_deny_cmd,
 | 
			
		||||
      "imsi deny [REGEXP]",
 | 
			
		||||
      "Deny IMSIs with the following network to talk to the MSC."
 | 
			
		||||
      "The default is to not deny anyone.)")
 | 
			
		||||
{
 | 
			
		||||
	struct bsc_config *conf = vty->index;
 | 
			
		||||
 | 
			
		||||
	parse_reg(conf, &conf->imsi_deny_re, &conf->imsi_deny, argc, argv);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int bsc_nat_vty_init(struct bsc_nat *nat)
 | 
			
		||||
{
 | 
			
		||||
	_nat = nat;
 | 
			
		||||
@@ -207,6 +326,7 @@ int bsc_nat_vty_init(struct bsc_nat *nat)
 | 
			
		||||
	install_element(VIEW_NODE, &show_sccp_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_bsc_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_bsc_cfg_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_stats_cmd);
 | 
			
		||||
 | 
			
		||||
	openbsc_vty_add_cmds();
 | 
			
		||||
 | 
			
		||||
@@ -214,6 +334,10 @@ int bsc_nat_vty_init(struct bsc_nat *nat)
 | 
			
		||||
	install_element(CONFIG_NODE, &cfg_nat_cmd);
 | 
			
		||||
	install_node(&nat_node, config_write_nat);
 | 
			
		||||
	install_default(NAT_NODE);
 | 
			
		||||
	install_element(NAT_NODE, &cfg_nat_imsi_allow_cmd);
 | 
			
		||||
	install_element(NAT_NODE, &cfg_nat_imsi_deny_cmd);
 | 
			
		||||
	install_element(NAT_NODE, &cfg_nat_msc_ip_cmd);
 | 
			
		||||
	install_element(NAT_NODE, &cfg_nat_msc_port_cmd);
 | 
			
		||||
 | 
			
		||||
	/* BSC subgroups */
 | 
			
		||||
	install_element(NAT_NODE, &cfg_bsc_cmd);
 | 
			
		||||
@@ -221,6 +345,8 @@ int bsc_nat_vty_init(struct bsc_nat *nat)
 | 
			
		||||
	install_default(BSC_NODE);
 | 
			
		||||
	install_element(BSC_NODE, &cfg_bsc_token_cmd);
 | 
			
		||||
	install_element(BSC_NODE, &cfg_bsc_lac_cmd);
 | 
			
		||||
	install_element(BSC_NODE, &cfg_bsc_imsi_allow_cmd);
 | 
			
		||||
	install_element(BSC_NODE, &cfg_bsc_imsi_deny_cmd);
 | 
			
		||||
 | 
			
		||||
	mgcp_vty_init();
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -99,7 +99,10 @@ int create_sccp_src_ref(struct bsc_connection *bsc, struct msgb *msg, struct bsc
 | 
			
		||||
		return -1;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	llist_add(&conn->list_entry, &bsc->nat->sccp_connections);
 | 
			
		||||
	bsc_mgcp_clear(conn);
 | 
			
		||||
	llist_add_tail(&conn->list_entry, &bsc->nat->sccp_connections);
 | 
			
		||||
	counter_inc(bsc->cfg->stats.sccp.conn);
 | 
			
		||||
	counter_inc(bsc->cfg->nat->stats.sccp.conn);
 | 
			
		||||
 | 
			
		||||
	LOGP(DNAT, LOGL_DEBUG, "Created 0x%x <-> 0x%x mapping for con %p\n",
 | 
			
		||||
	     sccp_src_ref_to_int(&conn->real_ref),
 | 
			
		||||
 
 | 
			
		||||
@@ -326,3 +326,15 @@ void paging_update_buffer_space(struct gsm_bts *bts, u_int16_t free_slots)
 | 
			
		||||
{
 | 
			
		||||
	bts->paging.available_slots = free_slots;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
unsigned int paging_pending_requests_nr(struct gsm_bts *bts)
 | 
			
		||||
{
 | 
			
		||||
	unsigned int requests = 0;
 | 
			
		||||
 | 
			
		||||
	struct gsm_paging_request *req;
 | 
			
		||||
 | 
			
		||||
	llist_for_each_entry(req, &bts->paging.pending_requests, entry)
 | 
			
		||||
		++requests;
 | 
			
		||||
 | 
			
		||||
	return requests;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -39,6 +39,8 @@
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
#include <openbsc/telnet_interface.h>
 | 
			
		||||
#include <openbsc/vty.h>
 | 
			
		||||
#include <openbsc/ipaccess.h>
 | 
			
		||||
#include <openbsc/paging.h>
 | 
			
		||||
 | 
			
		||||
static struct gsm_network *gsmnet;
 | 
			
		||||
 | 
			
		||||
@@ -195,7 +197,8 @@ static void bts_dump_vty(struct vty *vty, struct gsm_bts *bts)
 | 
			
		||||
	net_dump_nmstate(vty, &bts->nm_state);
 | 
			
		||||
	vty_out(vty, "  Site Mgr NM State: ");
 | 
			
		||||
	net_dump_nmstate(vty, &bts->site_mgr.nm_state);
 | 
			
		||||
	vty_out(vty, "  Paging: FIXME pending requests, %u free slots%s",
 | 
			
		||||
	vty_out(vty, "  Paging: %u pending requests, %u free slots%s",
 | 
			
		||||
		paging_pending_requests_nr(bts),
 | 
			
		||||
		bts->paging.available_slots, VTY_NEWLINE);
 | 
			
		||||
	if (!is_ipaccess_bts(bts)) {
 | 
			
		||||
		vty_out(vty, "  E1 Signalling Link:%s", VTY_NEWLINE);
 | 
			
		||||
@@ -233,6 +236,36 @@ DEFUN(show_bts, show_bts_cmd, "show bts [number]",
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(test_bts_lchan_alloc, test_bts_lchan_alloc_cmd, "test bts alloc (sdcch|tch_h|tch_f)",
 | 
			
		||||
      "Test command to allocate all channels. You will need to restart. To free these channels.\n")
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_network *net = gsmnet;
 | 
			
		||||
	int bts_nr;
 | 
			
		||||
 | 
			
		||||
	enum gsm_chan_t type = GSM_LCHAN_NONE;
 | 
			
		||||
 | 
			
		||||
	if (strcmp("sdcch", argv[0]) == 0)
 | 
			
		||||
		type = GSM_LCHAN_SDCCH;
 | 
			
		||||
	else if (strcmp("tch_h", argv[0]) == 0)
 | 
			
		||||
		type = GSM_LCHAN_TCH_H;
 | 
			
		||||
	else if (strcmp("tch_f", argv[0]) == 0)
 | 
			
		||||
		type = GSM_LCHAN_TCH_F;
 | 
			
		||||
	else {
 | 
			
		||||
		vty_out(vty, "Unknown mode for allocation.%s", VTY_NEWLINE);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	for (bts_nr = 0; bts_nr < net->num_bts; ++bts_nr) {
 | 
			
		||||
		struct gsm_bts *bts = gsm_bts_num(net, bts_nr);
 | 
			
		||||
		struct gsm_lchan *lchan;
 | 
			
		||||
 | 
			
		||||
		/* alloc the channel */
 | 
			
		||||
		while ((lchan = lchan_alloc(bts, type, 0)) != NULL)
 | 
			
		||||
			rsl_lchan_set_state(lchan, LCHAN_S_REL_ERR);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/* utility functions */
 | 
			
		||||
static void parse_e1_link(struct gsm_e1_subslot *e1_link, const char *line,
 | 
			
		||||
			  const char *ts, const char *ss)
 | 
			
		||||
@@ -275,6 +308,9 @@ static void config_write_trx_single(struct vty *vty, struct gsm_bts_trx *trx)
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	vty_out(vty, "  trx %u%s", trx->nr, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "   rf_locked %u%s",
 | 
			
		||||
		trx->nm_state.administrative == NM_STATE_LOCKED ? 1 : 0,
 | 
			
		||||
		VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "   arfcn %u%s", trx->arfcn, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "   nominal power %u%s", trx->nominal_power, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "   max_power_red %u%s", trx->max_power_red, VTY_NEWLINE);
 | 
			
		||||
@@ -429,6 +465,8 @@ static int config_write_net(struct vty *vty)
 | 
			
		||||
 | 
			
		||||
	if (gsmnet->bsc_token)
 | 
			
		||||
		vty_out(vty, " bsc_token %s%s", gsmnet->bsc_token, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " msc ip %s%s", gsmnet->msc_ip, VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, " msc port %d%s", gsmnet->msc_port, VTY_NEWLINE);
 | 
			
		||||
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
@@ -915,50 +953,10 @@ DEFUN(show_paging,
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_stats,
 | 
			
		||||
      show_stats_cmd,
 | 
			
		||||
      "show statistics",
 | 
			
		||||
	SHOW_STR "Display network statistics\n")
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_network *net = gsmnet;
 | 
			
		||||
 | 
			
		||||
	vty_out(vty, "Channel Requests        : %lu total, %lu no channel%s",
 | 
			
		||||
		counter_get(net->stats.chreq.total),
 | 
			
		||||
		counter_get(net->stats.chreq.no_channel), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Location Update         : %lu attach, %lu normal, %lu periodic%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.attach),
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.normal),
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.periodic), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "IMSI Detach Indications : %lu%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.detach), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Location Update Response: %lu accept, %lu reject%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_resp.accept),
 | 
			
		||||
		counter_get(net->stats.loc_upd_resp.reject), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Paging                  : %lu attempted, %lu complete, %lu expired%s",
 | 
			
		||||
		counter_get(net->stats.paging.attempted),
 | 
			
		||||
		counter_get(net->stats.paging.completed),
 | 
			
		||||
		counter_get(net->stats.paging.expired), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Handover                : %lu attempted, %lu no_channel, %lu timeout, "
 | 
			
		||||
		"%lu completed, %lu failed%s",
 | 
			
		||||
		counter_get(net->stats.handover.attempted),
 | 
			
		||||
		counter_get(net->stats.handover.no_channel),
 | 
			
		||||
		counter_get(net->stats.handover.timeout),
 | 
			
		||||
		counter_get(net->stats.handover.completed),
 | 
			
		||||
		counter_get(net->stats.handover.failed), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "SMS MO                  : %lu submitted, %lu no receiver%s",
 | 
			
		||||
		counter_get(net->stats.sms.submitted),
 | 
			
		||||
		counter_get(net->stats.sms.no_receiver), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "SMS MT                  : %lu delivered, %lu no memory, %lu other error%s",
 | 
			
		||||
		counter_get(net->stats.sms.delivered),
 | 
			
		||||
		counter_get(net->stats.sms.rp_err_mem),
 | 
			
		||||
		counter_get(net->stats.sms.rp_err_other), VTY_NEWLINE);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(drop_bts,
 | 
			
		||||
      drop_bts_cmd,
 | 
			
		||||
      "drop bts connection [nr] (oml|rsl)",
 | 
			
		||||
      SHOW_STR "Debug/Simulation command to drop ipaccess BTS\n")
 | 
			
		||||
      "drop bts connection <0-65535> (oml|rsl)",
 | 
			
		||||
      "Debug/Simulation command to drop ipaccess BTS\n")
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_bts_trx *trx;
 | 
			
		||||
	struct gsm_bts *bts;
 | 
			
		||||
@@ -984,12 +982,12 @@ DEFUN(drop_bts,
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
	/* close all connections */
 | 
			
		||||
	if (strcmp(argv[1], "oml") == 0) {
 | 
			
		||||
		close(bts->oml_link->ts->driver.ipaccess.fd.fd);
 | 
			
		||||
	} else if (strcmp(argv[1], "rsl") == 0) {
 | 
			
		||||
	if (strcmp(argv[1], "oml") == 0)
 | 
			
		||||
		ipaccess_drop_oml(bts);
 | 
			
		||||
	else if (strcmp(argv[1], "rsl") == 0) {
 | 
			
		||||
		/* close all rsl connections */
 | 
			
		||||
		llist_for_each_entry(trx, &bts->trx_list, list) {
 | 
			
		||||
			close(trx->rsl_link->ts->driver.ipaccess.fd.fd);
 | 
			
		||||
			ipaccess_drop_rsl(trx);
 | 
			
		||||
		}
 | 
			
		||||
	} else {
 | 
			
		||||
		vty_out(vty, "Argument must be 'oml# or 'rsl'.%s", VTY_NEWLINE);
 | 
			
		||||
@@ -1308,6 +1306,27 @@ DEFUN(cfg_net_pag_any_tch,
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_net_msc_ip,
 | 
			
		||||
      cfg_net_msc_ip_cmd,
 | 
			
		||||
      "msc ip IP",
 | 
			
		||||
      "Set the MSC/MUX IP address.")
 | 
			
		||||
{
 | 
			
		||||
	if (gsmnet->msc_ip)
 | 
			
		||||
		talloc_free(gsmnet->msc_ip);
 | 
			
		||||
	gsmnet->msc_ip = talloc_strdup(gsmnet, argv[0]);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(cfg_net_msc_port,
 | 
			
		||||
      cfg_net_msc_port_cmd,
 | 
			
		||||
      "msc port <1-65000>",
 | 
			
		||||
      "Set the MSC/MUX port.")
 | 
			
		||||
{
 | 
			
		||||
	gsmnet->msc_port = atoi(argv[0]);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#define DECLARE_TIMER(number, doc) \
 | 
			
		||||
    DEFUN(cfg_net_T##number,					\
 | 
			
		||||
      cfg_net_T##number##_cmd,					\
 | 
			
		||||
@@ -1964,9 +1983,9 @@ int bsc_vty_init(struct gsm_network *net)
 | 
			
		||||
	install_element(VIEW_NODE, &show_e1ts_cmd);
 | 
			
		||||
 | 
			
		||||
	install_element(VIEW_NODE, &show_paging_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_stats_cmd);
 | 
			
		||||
 | 
			
		||||
	install_element(VIEW_NODE, &drop_bts_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &test_bts_lchan_alloc_cmd);
 | 
			
		||||
 | 
			
		||||
	openbsc_vty_add_cmds();
 | 
			
		||||
        
 | 
			
		||||
@@ -2008,6 +2027,8 @@ int bsc_vty_init(struct gsm_network *net)
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_net_T3141_cmd);
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_net_bsc_token_cmd);
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_net_pag_any_tch_cmd);
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_net_msc_ip_cmd);
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_net_msc_port_cmd);
 | 
			
		||||
 | 
			
		||||
	install_element(GSMNET_NODE, &cfg_bts_cmd);
 | 
			
		||||
	install_node(&bts_node, config_write_bts);
 | 
			
		||||
 
 | 
			
		||||
@@ -27,13 +27,39 @@
 | 
			
		||||
#include <vty/vty.h>
 | 
			
		||||
 | 
			
		||||
#include <openbsc/gsm_data.h>
 | 
			
		||||
#include <openbsc/vty.h>
 | 
			
		||||
 | 
			
		||||
static struct gsmnet *gsmnet = NULL;
 | 
			
		||||
#include <sccp/sccp.h>
 | 
			
		||||
 | 
			
		||||
static struct gsm_network *gsmnet = NULL;
 | 
			
		||||
 | 
			
		||||
extern struct llist_head *bsc_sccp_connections();
 | 
			
		||||
 | 
			
		||||
DEFUN(show_bsc, show_bsc_cmd, "show bsc",
 | 
			
		||||
	SHOW_STR "Display information about the BSC\n")
 | 
			
		||||
{
 | 
			
		||||
	vty_out(vty, "BSC... not implemented yet%s", VTY_NEWLINE);
 | 
			
		||||
	struct bss_sccp_connection_data *con;
 | 
			
		||||
 | 
			
		||||
	vty_out(vty, "BSC Information%s", VTY_NEWLINE);
 | 
			
		||||
	llist_for_each_entry(con, bsc_sccp_connections(), active_connections) {
 | 
			
		||||
		vty_out(vty, " Connection: LCHAN: %p sec LCHAN: %p SCCP src: %d dest: %d%s",
 | 
			
		||||
			con->lchan, con->secondary_lchan,
 | 
			
		||||
			con->sccp ? (int) sccp_src_ref_to_int(&con->sccp->source_local_reference) : -1,
 | 
			
		||||
			con->sccp ? (int) sccp_src_ref_to_int(&con->sccp->destination_local_reference) : -1,
 | 
			
		||||
			VTY_NEWLINE);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_stats,
 | 
			
		||||
      show_stats_cmd,
 | 
			
		||||
      "show statistics",
 | 
			
		||||
	SHOW_STR "Display network statistics\n")
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_network *net = gsmnet;
 | 
			
		||||
 | 
			
		||||
	openbsc_vty_print_statistics(vty, net);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -43,6 +69,7 @@ int bsc_vty_init_extra(struct gsm_network *net)
 | 
			
		||||
 | 
			
		||||
	/* get runtime information */
 | 
			
		||||
	install_element(VIEW_NODE, &show_bsc_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_stats_cmd);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -228,6 +228,23 @@ DEFUN(diable_logging,
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void openbsc_vty_print_statistics(struct vty *vty, struct gsm_network *net)
 | 
			
		||||
{
 | 
			
		||||
	vty_out(vty, "Channel Requests        : %lu total, %lu no channel%s",
 | 
			
		||||
		counter_get(net->stats.chreq.total),
 | 
			
		||||
		counter_get(net->stats.chreq.no_channel), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Channel Failures        : %lu rf_failures, %lu rll failures%s",
 | 
			
		||||
		counter_get(net->stats.chan.rf_fail),
 | 
			
		||||
		counter_get(net->stats.chan.rll_err), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Paging                  : %lu attempted, %lu complete, %lu expired%s",
 | 
			
		||||
		counter_get(net->stats.paging.attempted),
 | 
			
		||||
		counter_get(net->stats.paging.completed),
 | 
			
		||||
		counter_get(net->stats.paging.expired), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "BTS failures            : %lu OML, %lu RSL%s",
 | 
			
		||||
		counter_get(net->stats.bts.oml_fail),
 | 
			
		||||
		counter_get(net->stats.bts.rsl_fail), VTY_NEWLINE);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void openbsc_vty_add_cmds()
 | 
			
		||||
{
 | 
			
		||||
	install_element(VIEW_NODE, &enable_logging_cmd);
 | 
			
		||||
 
 | 
			
		||||
@@ -41,6 +41,7 @@
 | 
			
		||||
#include <osmocore/talloc.h>
 | 
			
		||||
#include <openbsc/signal.h>
 | 
			
		||||
#include <openbsc/debug.h>
 | 
			
		||||
#include <openbsc/vty.h>
 | 
			
		||||
 | 
			
		||||
static struct gsm_network *gsmnet;
 | 
			
		||||
 | 
			
		||||
@@ -502,6 +503,41 @@ static int scall_cbfn(unsigned int subsys, unsigned int signal,
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
DEFUN(show_stats,
 | 
			
		||||
      show_stats_cmd,
 | 
			
		||||
      "show statistics",
 | 
			
		||||
	SHOW_STR "Display network statistics\n")
 | 
			
		||||
{
 | 
			
		||||
	struct gsm_network *net = gsmnet;
 | 
			
		||||
 | 
			
		||||
	openbsc_vty_print_statistics(vty, net);
 | 
			
		||||
	vty_out(vty, "Location Update         : %lu attach, %lu normal, %lu periodic%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.attach),
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.normal),
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.periodic), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "IMSI Detach Indications : %lu%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_type.detach), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Location Update Response: %lu accept, %lu reject%s",
 | 
			
		||||
		counter_get(net->stats.loc_upd_resp.accept),
 | 
			
		||||
		counter_get(net->stats.loc_upd_resp.reject), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "Handover                : %lu attempted, %lu no_channel, %lu timeout, "
 | 
			
		||||
		"%lu completed, %lu failed%s",
 | 
			
		||||
		counter_get(net->stats.handover.attempted),
 | 
			
		||||
		counter_get(net->stats.handover.no_channel),
 | 
			
		||||
		counter_get(net->stats.handover.timeout),
 | 
			
		||||
		counter_get(net->stats.handover.completed),
 | 
			
		||||
		counter_get(net->stats.handover.failed), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "SMS MO                  : %lu submitted, %lu no receiver%s",
 | 
			
		||||
		counter_get(net->stats.sms.submitted),
 | 
			
		||||
		counter_get(net->stats.sms.no_receiver), VTY_NEWLINE);
 | 
			
		||||
	vty_out(vty, "SMS MT                  : %lu delivered, %lu no memory, %lu other error%s",
 | 
			
		||||
		counter_get(net->stats.sms.delivered),
 | 
			
		||||
		counter_get(net->stats.sms.rp_err_mem),
 | 
			
		||||
		counter_get(net->stats.sms.rp_err_other), VTY_NEWLINE);
 | 
			
		||||
	return CMD_SUCCESS;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int bsc_vty_init_extra(struct gsm_network *net)
 | 
			
		||||
{
 | 
			
		||||
	gsmnet = net;
 | 
			
		||||
@@ -517,6 +553,7 @@ int bsc_vty_init_extra(struct gsm_network *net)
 | 
			
		||||
	install_element(VIEW_NODE, &subscriber_silent_sms_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &subscriber_silent_call_start_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &subscriber_silent_call_stop_cmd);
 | 
			
		||||
	install_element(VIEW_NODE, &show_stats_cmd);
 | 
			
		||||
 | 
			
		||||
	install_element(CONFIG_NODE, &cfg_subscr_cmd);
 | 
			
		||||
	install_node(&subscr_node, dummy_config_write);
 | 
			
		||||
 
 | 
			
		||||
@@ -242,6 +242,7 @@ static void test_contrack()
 | 
			
		||||
	fprintf(stderr, "Testing connection tracking.\n");
 | 
			
		||||
	nat = bsc_nat_alloc();
 | 
			
		||||
	con = bsc_connection_alloc(nat);
 | 
			
		||||
	con->cfg = bsc_config_alloc(nat, "foo", 23);
 | 
			
		||||
	msg = msgb_alloc(4096, "test");
 | 
			
		||||
 | 
			
		||||
	/* 1.) create a connection */
 | 
			
		||||
@@ -329,6 +330,7 @@ static void test_contrack()
 | 
			
		||||
 | 
			
		||||
static void test_paging(void)
 | 
			
		||||
{
 | 
			
		||||
	int lac;
 | 
			
		||||
	struct bsc_nat *nat;
 | 
			
		||||
	struct bsc_connection *con;
 | 
			
		||||
	struct bsc_nat_parsed *parsed;
 | 
			
		||||
@@ -347,7 +349,7 @@ static void test_paging(void)
 | 
			
		||||
 | 
			
		||||
	/* Test completely bad input */
 | 
			
		||||
	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd));
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg) != 0) {
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg, &lac) != 0) {
 | 
			
		||||
		fprintf(stderr, "Should have not found anything.\n");
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
@@ -355,7 +357,7 @@ static void test_paging(void)
 | 
			
		||||
	/* Test it by not finding it */
 | 
			
		||||
	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd));
 | 
			
		||||
	parsed = bsc_nat_parse(msg);
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg) != 0) {
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg, &lac) != 0) {
 | 
			
		||||
		fprintf(stderr, "Should have not found aynthing.\n");
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
@@ -365,7 +367,7 @@ static void test_paging(void)
 | 
			
		||||
	cfg.lac = 8213;
 | 
			
		||||
	copy_to_msg(msg, paging_by_lac_cmd, sizeof(paging_by_lac_cmd));
 | 
			
		||||
	parsed = bsc_nat_parse(msg);
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg) != con) {
 | 
			
		||||
	if (bsc_nat_find_bsc(nat, msg, &lac) != con) {
 | 
			
		||||
		fprintf(stderr, "Should have found it.\n");
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
@@ -431,14 +433,14 @@ static void test_mgcp_find(void)
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (bsc_mgcp_find_con(nat, 12) != con) {
 | 
			
		||||
	if (bsc_mgcp_find_con(nat, 12) != sccp_con) {
 | 
			
		||||
		fprintf(stderr, "Didn't find the connection\n");
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	sccp_con->msc_timeslot = 0;
 | 
			
		||||
	sccp_con->bsc_timeslot = 0;
 | 
			
		||||
	if (bsc_mgcp_find_con(nat, 1) != con) {
 | 
			
		||||
	if (bsc_mgcp_find_con(nat, 1) != sccp_con) {
 | 
			
		||||
		fprintf(stderr, "Didn't find the connection\n");
 | 
			
		||||
		abort();
 | 
			
		||||
	}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user