[PATCH 3/3] layer23 cell_log: log CBCH cell info

Alex Badea vamposdecampos at gmail.com
Sun Nov 28 15:07:29 CET 2010


If System Information contained CBCH channel description,
tune to it and log SMSCB messages with a Geographical Scope
of Cell Wide (Immediate) -- these are typically used for
"Cell Info Display".

The CBCH timeout is set to a generous 8 seconds; this is because
we need 4 frames for reassembly, we usually miss the first, and we
might hit a burst of Null SMSCB messages.

Signed-off-by: Alex Badea <vamposdecampos at gmail.com>
---
 src/host/layer23/src/misc/cell_log.c |  107 +++++++++++++++++++++++++++++++++-
 1 files changed, 105 insertions(+), 2 deletions(-)

diff --git a/src/host/layer23/src/misc/cell_log.c b/src/host/layer23/src/misc/cell_log.c
index e6c5771..37aa271 100644
--- a/src/host/layer23/src/misc/cell_log.c
+++ b/src/host/layer23/src/misc/cell_log.c
@@ -26,6 +26,7 @@
 #include <stdlib.h>
 #include <time.h>
 #include <errno.h>
+#include <netinet/in.h>
 
 #include <l1ctl_proto.h>
 
@@ -34,7 +35,9 @@
 #include <osmocore/signal.h>
 #include <osmocore/msgb.h>
 #include <osmocore/gsm_utils.h>
+#include <osmocore/protocol/gsm_03_41.h>
 #include <osmocore/protocol/gsm_04_08.h>
+#include <osmocore/protocol/gsm_08_58.h>
 #include <osmocore/rsl.h>
 
 #include <osmocom/bb/common/l1ctl.h>
@@ -49,6 +52,7 @@
 #define READ_WAIT	2, 0
 #define RACH_MAX	2
 #define RACH_WAIT	0, 900000
+#define CBCH_WAIT	8, 0
 #define MIN_RXLEV	-106
 #define MAX_DIST	2000
 
@@ -57,6 +61,7 @@ enum {
 	SCAN_STATE_SYNC,
 	SCAN_STATE_READ,
 	SCAN_STATE_RACH,
+	SCAN_STATE_CBCH,
 };
 
 /* ranges of bands */
@@ -114,6 +119,7 @@ struct rach_ref {
 
 static void start_sync(void);
 static void start_rach(void);
+static void start_cbch(void);
 static void start_pm(void);
 
 static void log_gps(void)
@@ -227,6 +233,11 @@ static void timeout_cb(void *arg)
 		rach_count++;
 		start_rach();
 		break;
+	case SCAN_STATE_CBCH:
+		LOGP(DRR, LOGL_INFO, "Timeout reading CBCH\n");
+		l1ctl_tx_dm_rel_req(ms);
+		start_sync();
+		break;
 	}
 }
 
@@ -244,6 +255,33 @@ static void start_timer(int sec, int micro)
 	bsc_schedule_timer(&timer, sec, micro);
 }
 
+static void start_cbch(void)
+{
+	struct gsm48_sysinfo *s = &sysinfo;
+	int res;
+
+	if (!s->chan_nr) {
+		LOGP(DRR, LOGL_DEBUG, "No CBCH description in sysinfo\n");
+		res = -1;
+	} else if (s->h) {
+		res = l1ctl_tx_dm_est_req_h1(ms,
+			s->maio, s->hsn, s->hopping, s->hopp_len,
+			s->chan_nr, s->tsc,
+			GSM48_CMODE_SIGN);
+	} else {
+		res = l1ctl_tx_dm_est_req_h0(ms, s->arfcn,
+			s->chan_nr, s->tsc, GSM48_CMODE_SIGN);
+	}
+
+	if (res < 0) {
+		start_sync();
+		return;
+	}
+
+	state = SCAN_STATE_CBCH;
+	start_timer(CBCH_WAIT);
+}
+
 static void start_rach(void)
 {
 	struct gsm48_sysinfo *s = &sysinfo;
@@ -253,7 +291,7 @@ static void start_rach(void)
 
 	if (rach_count == RACH_MAX) {
 		log_sysinfo();
-		start_sync();
+		start_cbch();
 		return;
 	}
 
@@ -426,7 +464,7 @@ static int ta_result(uint8_t ta)
 	}
 
 	log_sysinfo();
-	start_sync();
+	start_cbch();
 
 	return 0;
 }
@@ -730,6 +768,66 @@ int chan_conf(struct osmocom_ms *ms, struct msgb *msg)
 	return 0;
 }
 
+static int sms_cb_cmd(struct osmocom_ms *ms, struct msgb *msg)
+{
+	struct abis_rsl_cchan_hdr *ch = msgb_l2(msg);
+	struct tlv_parsed tv;
+	struct rsl_ie_cb_cmd_type *cmd_type;
+	struct gsm341_ms_message *cbmsg;
+	unsigned int cbmsglen;
+	char buf[256], *p;
+
+	rsl_tlv_parse(&tv, ch->data, msgb_l2len(msg) - sizeof(*ch));
+	if (!TLVP_PRESENT(&tv, RSL_IE_SMSCB_MSG) ||
+	    !TLVP_PRESENT(&tv, RSL_IE_CB_CMD_TYPE)) {
+		LOGP(DRR, LOGL_ERROR, "SMS_BC_CMD missing mandatory IEs\n");
+		return -EINVAL;
+	}
+
+	cmd_type = (struct rsl_ie_cb_cmd_type *) TLVP_VAL(&tv, RSL_IE_CB_CMD_TYPE);
+	cbmsg = (struct gsm341_ms_message *) TLVP_VAL(&tv, RSL_IE_SMSCB_MSG);
+	cbmsglen = TLVP_LEN(&tv, RSL_IE_SMSCB_MSG);
+	if (cbmsglen < sizeof(*cbmsg)) {
+		LOGP(DRR, LOGL_ERROR, "RSL_IE_SMSCB_MSG too short\n");
+		return -EINVAL;
+	}
+
+	if (cmd_type->command != RSL_CB_CMD_TYPE_NORMAL)
+		return 0;
+	if (cbmsg->serial.gs != GSM341_GS_CELL_WIDE_IMMED)
+		return 0;
+
+	gsm_7bit_decode(buf, cbmsg->data, cbmsglen - sizeof(*cbmsg));
+	if ((p = strchr(buf, '\r')))
+		*p = 0;
+
+	LOGP(DSUM, LOGL_INFO, "Cell info: code=%d update=%d id=%d text=\"%s\"\n",
+		cbmsg->serial.code_lo | (cbmsg->serial.code_hi << 6),
+		cbmsg->serial.update,
+		ntohs(cbmsg->msg_id),
+		buf);
+
+	LOGFILE("[cbch]\n");
+	LOGFILE("arfcn %d\n", sysinfo.arfcn);
+	LOGFILE("gs %d code %d update %d msg_id %d page %d numpages %d\n",
+		cbmsg->serial.gs,
+		cbmsg->serial.code_lo | (cbmsg->serial.code_hi << 6),
+		cbmsg->serial.update,
+		ntohs(cbmsg->msg_id),
+		cbmsg->page.current, cbmsg->page.total);
+	LOGFILE("text \"%s\"\n", buf);
+	LOGFILE("\n");
+	LOGFLUSH();
+
+	l1ctl_tx_dm_rel_req(ms);
+	stop_timer();
+
+	start_sync();
+
+	return 0;
+
+}
+
 static int rcv_cch(struct osmocom_ms *ms, struct msgb *msg)
 {
 	struct abis_rsl_cchan_hdr *ch = msgb_l2(msg);
@@ -744,6 +842,11 @@ static int rcv_cch(struct osmocom_ms *ms, struct msgb *msg)
 		msgb_free(msg);
 		return rc;
 	}
+	if (state == SCAN_STATE_CBCH && msg_type == RSL_MT_SMS_BC_CMD) {
+		rc = sms_cb_cmd(ms, msg);
+		msgb_free(msg);
+		return rc;
+	}
 
 	LOGP(DRSL, LOGL_NOTICE, "RSLms message unhandled\n");
 	msgb_free(msg);
-- 
1.7.0.4




More information about the baseband-devel mailing list