diff options
Diffstat (limited to 'src/common/paging.c')
-rw-r--r-- | src/common/paging.c | 440 |
1 files changed, 440 insertions, 0 deletions
diff --git a/src/common/paging.c b/src/common/paging.c new file mode 100644 index 00000000..0f51d753 --- /dev/null +++ b/src/common/paging.c @@ -0,0 +1,440 @@ +/* Paging message encoding + queue management */ + +/* (C) 2011 by Harald Welte <laforge@gnumonks.org> + * + * All Rights Reserved + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU Affero General Public License as published by + * the Free Software Foundation; either version 3 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 Affero General Public License + * along with this program. If not, see <http://www.gnu.org/licenses/>. + * + */ + +/* TODO: + * eMLPP priprity + * add P1/P2/P3 rest octets + */ + +#include <stdlib.h> +#include <stdint.h> +#include <errno.h> +#include <string.h> +#include <time.h> + +#include <osmocom/core/talloc.h> +#include <osmocom/core/linuxlist.h> + +#include <osmocom/gsm/protocol/gsm_04_08.h> +#include <osmocom/gsm/gsm0502.h> + +#include <osmo-bts/bts.h> +#include <osmo-bts/rsl.h> +#include <osmo-bts/logging.h> +#include <osmo-bts/paging.h> +#include <osmo-bts/signal.h> + +#define MAX_PAGING_BLOCKS_CCCH 9 +#define MAX_BS_PA_MFRMS 9 + +struct paging_record { + struct llist_head list; + time_t expiration_time; + uint8_t chan_needed; + uint8_t identity_lv[9]; +}; + +struct paging_state { + /* parameters taken / interpreted from BCCH/CCCH configuration */ + struct gsm48_control_channel_descr chan_desc; + + /* configured otherwise */ + unsigned int paging_lifetime; /* in seconds */ + unsigned int num_paging_max; + + /* total number of currently active paging records in queue */ + unsigned int num_paging; + struct llist_head paging_queue[MAX_PAGING_BLOCKS_CCCH*MAX_BS_PA_MFRMS]; +}; + +static int tmsi_mi_to_uint(uint32_t *out, const uint8_t *tmsi_lv) +{ + if (tmsi_lv[0] < 5) + return -EINVAL; + if ((tmsi_lv[1] & 7) != GSM_MI_TYPE_TMSI) + return -EINVAL; + + *out = *((uint32_t *)(tmsi_lv+2)); + + return 0; +} + +/* paging block numbers in a simple non-combined CCCH */ +static const uint8_t block_by_tdma51[51] = { + 255, 255, /* FCCH, SCH */ + 255, 255, 255, 255, /* BCCH */ + 0, 0, 0, 0, /* B0(6..9) */ + 255, 255, /* FCCH, SCH */ + 1, 1, 1, 1, /* B1(12..15) */ + 2, 2, 2, 2, /* B2(16..19) */ + 255, 255, /* FCCH, SCH */ + 3, 3, 3, 3, /* B3(22..25) */ + 4, 4, 4, 4, /* B3(26..29) */ + 255, 255, /* FCCH, SCH */ + 5, 5, 5, 5, /* B3(32..35) */ + 6, 6, 6, 6, /* B3(36..39) */ + 255, 255, /* FCCH, SCH */ + 7, 7, 7, 7, /* B3(42..45) */ + 8, 8, 8, 8, /* B3(46..49) */ + 255, /* empty */ +}; + +/* get the paging block number _within_ current 51 multiframe */ +static int get_pag_idx_n(struct paging_state *ps, struct gsm_time *gt) +{ + int blk_n = block_by_tdma51[gt->t3]; + int blk_idx; + + if (blk_n == 255) + return -EINVAL; + + blk_idx = blk_n - ps->chan_desc.bs_ag_blks_res; + if (blk_idx < 0) + return -EINVAL; + + return blk_idx; +} + +/* get paging block index over multiple 51 multiframes */ +static int get_pag_subch_nr(struct paging_state *ps, struct gsm_time *gt) +{ + int pag_idx = get_pag_idx_n(ps, gt); + unsigned int n_pag_blks_51 = gsm0502_get_n_pag_blocks(&ps->chan_desc); + unsigned int mfrm_part; + + if (pag_idx < 0) + return pag_idx; + + mfrm_part = ((gt->fn / 51) % (ps->chan_desc.bs_pa_mfrms+2)) * n_pag_blks_51; + + return pag_idx + mfrm_part; +} + + +/* Add an identity to the paging queue */ +int paging_add_identity(struct paging_state *ps, uint8_t paging_group, + const uint8_t *identity_lv, uint8_t chan_needed) +{ + struct llist_head *group_q = &ps->paging_queue[paging_group]; + struct paging_record *pr; + + if (ps->num_paging >= ps->num_paging_max) { + LOGP(DPAG, LOGL_NOTICE, "Dropping paging, queue full (%u)\n", + ps->num_paging); + return -ENOSPC; + } + + /* Check if we already have this identity */ + llist_for_each_entry(pr, group_q, list) { + if (identity_lv[0] == pr->identity_lv[0] && + !memcmp(identity_lv+1, pr->identity_lv+1, identity_lv[0])) { + LOGP(DPAG, LOGL_INFO, "Ignoring duplicate paging\n"); + pr->expiration_time = time(NULL) + ps->paging_lifetime; + return -EEXIST; + } + } + + pr = talloc_zero(ps, struct paging_record); + if (!pr) + return -ENOMEM; + + if (*identity_lv + 1 > sizeof(pr->identity_lv)) { + talloc_free(pr); + return -E2BIG; + } + + LOGP(DPAG, LOGL_INFO, "Add paging to queue (group=%u, queue_len=%u)\n", + paging_group, ps->num_paging+1); + + pr->expiration_time = time(NULL) + ps->paging_lifetime; + pr->chan_needed = chan_needed; + memcpy(&pr->identity_lv, identity_lv, identity_lv[0]+1); + + /* enqueue the new identity to the HEAD of the queue, + * to ensure it will be paged quickly at least once. */ + llist_add(&pr->list, group_q); + ps->num_paging++; + + return 0; +} + +static int fill_paging_type_1(uint8_t *out_buf, const uint8_t *identity1_lv, + uint8_t chan1, const uint8_t *identity2_lv, + uint8_t chan2) +{ + struct gsm48_paging1 *pt1 = (struct gsm48_paging1 *) out_buf; + uint8_t *cur; + + memset(out_buf, 0, sizeof(*pt1)); + + pt1->proto_discr = GSM48_PDISC_RR; + pt1->msg_type = GSM48_MT_RR_PAG_REQ_1; + pt1->pag_mode = GSM48_PM_NORMAL; + pt1->cneed1 = chan1 & 3; + pt1->cneed2 = chan2 & 3; + cur = lv_put(pt1->data, identity1_lv[0], identity1_lv+1); + if (identity2_lv) + cur = lv_put(cur, identity2_lv[0], identity2_lv+1); + + pt1->l2_plen = cur - out_buf - 1; + + return cur - out_buf; +} + +static int fill_paging_type_2(uint8_t *out_buf, const uint8_t *tmsi1_lv, + uint8_t cneed1, const uint8_t *tmsi2_lv, + uint8_t cneed2, const uint8_t *identity3_lv) +{ + struct gsm48_paging2 *pt2 = (struct gsm48_paging2 *) out_buf; + uint8_t *cur; + + memset(out_buf, 0, sizeof(*pt2)); + + pt2->proto_discr = GSM48_PDISC_RR; + pt2->msg_type = GSM48_MT_RR_PAG_REQ_2; + pt2->pag_mode = GSM48_PM_NORMAL; + pt2->cneed1 = cneed1; + pt2->cneed2 = cneed2; + tmsi_mi_to_uint(&pt2->tmsi1, tmsi1_lv); + tmsi_mi_to_uint(&pt2->tmsi2, tmsi2_lv); + cur = out_buf + sizeof(*pt2); + + if (identity3_lv) + cur = lv_put(pt2->data, identity3_lv[0], identity3_lv+1); + + pt2->l2_plen = cur - out_buf -1; + + return cur - out_buf; +} + +static int fill_paging_type_3(uint8_t *out_buf, const uint8_t *tmsi1_lv, + uint8_t cneed1, const uint8_t *tmsi2_lv, + uint8_t cneed2, const uint8_t *tmsi3_lv, + const uint8_t *tmsi4_lv) +{ + struct gsm48_paging3 *pt3 = (struct gsm48_paging3 *) out_buf; + uint8_t *cur; + + memset(out_buf, 0, sizeof(*pt3)); + + pt3->proto_discr = GSM48_PDISC_RR; + pt3->msg_type = GSM48_MT_RR_PAG_REQ_3; + pt3->pag_mode = GSM48_PM_NORMAL; + pt3->cneed1 = cneed1; + pt3->cneed2 = cneed2; + tmsi_mi_to_uint(&pt3->tmsi1, tmsi1_lv); + tmsi_mi_to_uint(&pt3->tmsi2, tmsi2_lv); + tmsi_mi_to_uint(&pt3->tmsi3, tmsi3_lv); + tmsi_mi_to_uint(&pt3->tmsi4, tmsi4_lv); + + cur = out_buf + sizeof(*pt3); + + return cur - out_buf; +} + +static const uint8_t empty_id_lv[] = { 0x01, 0x00 }; + +static struct paging_record *dequeue_pr(struct llist_head *group_q) +{ + struct paging_record *pr; + + pr = llist_entry(group_q->next, struct paging_record, list); + llist_del(&pr->list); + + return pr; +} + +static int pr_is_imsi(struct paging_record *pr) +{ + if ((pr->identity_lv[1] & 7) == GSM_MI_TYPE_IMSI) + return 1; + else + return 0; +} + +static void sort_pr_tmsi_imsi(struct paging_record *pr[], unsigned int n) +{ + int i, j; + struct paging_record *t; + + if (n < 2) + return; + + /* simple bubble sort */ + for (i = n-2; i >= 0; i--) { + for (j=0; j<=i ; j++) { + if (pr_is_imsi(pr[j]) > pr_is_imsi(pr[j+1])) { + t = pr[j]; + pr[j] = pr[j+1]; + pr[j+1] = t; + } + } + } +} + +/* generate paging message for given gsm time */ +int paging_gen_msg(struct paging_state *ps, uint8_t *out_buf, struct gsm_time *gt) +{ + unsigned int group = get_pag_subch_nr(ps, gt); + struct llist_head *group_q = &ps->paging_queue[group]; + int len; + + /* There is nobody to be paged, send Type1 with two empty ID */ + if (llist_empty(group_q)) { + //DEBUGP(DPAG, "Tx PAGING TYPE 1 (empty)\n"); + len = fill_paging_type_1(out_buf, empty_id_lv, 0, + NULL, 0); + } else { + struct paging_record *pr[4]; + unsigned int num_pr = 0; + time_t now = time(NULL); + unsigned int i, num_imsi = 0; + + /* get (if we have) up to four paging records */ + for (i = 0; i < ARRAY_SIZE(pr); i++) { + if (llist_empty(group_q)) + break; + pr[i] = dequeue_pr(group_q); + num_pr++; + + /* count how many IMSIs are among them */ + if (pr_is_imsi(pr[i])) + num_imsi++; + } + + /* make sure the TMSIs are ahead of the IMSIs in the array */ + sort_pr_tmsi_imsi(pr, num_pr); + + if (num_pr == 4 && num_imsi == 0) { + /* No IMSI: easy case, can use TYPE 3 */ + DEBUGP(DPAG, "Tx PAGING TYPE 3 (4 TMSI)\n"); + len = fill_paging_type_3(out_buf, pr[0]->identity_lv, + pr[0]->chan_needed, + pr[1]->identity_lv, + pr[1]->chan_needed, + pr[2]->identity_lv, + pr[3]->identity_lv); + } else if (num_pr >= 3 && num_imsi <= 1) { + /* 3 or 4, of which only up to 1 is IMSI */ + DEBUGP(DPAG, "Tx PAGING TYPE 2 (2 TMSI,1 xMSI)\n"); + len = fill_paging_type_2(out_buf, + pr[0]->identity_lv, + pr[0]->chan_needed, + pr[1]->identity_lv, + pr[1]->chan_needed, + pr[2]->identity_lv); + if (num_pr == 4) { + /* re-add #4 for next time */ + llist_add(&pr[3]->list, group_q); + pr[3] = NULL; + } + } else if (num_pr == 1) { + DEBUGP(DPAG, "Tx PAGING TYPE 1 (1 xMSI,1 empty)\n"); + len = fill_paging_type_1(out_buf, pr[0]->identity_lv, + pr[0]->chan_needed, NULL, 0); + } else { + /* 2 (any type) or + * 3 or 4, of which only 2 will be sent */ + DEBUGP(DPAG, "Tx PAGING TYPE 1 (2 xMSI)\n"); + len = fill_paging_type_1(out_buf, pr[0]->identity_lv, + pr[0]->chan_needed, + pr[1]->identity_lv, + pr[1]->chan_needed); + if (num_pr >= 3) { + /* re-add #4 for next time */ + llist_add(&pr[2]->list, group_q); + pr[2] = NULL; + } + if (num_pr == 4) { + /* re-add #4 for next time */ + llist_add(&pr[3]->list, group_q); + pr[3] = NULL; + } + } + + for (i = 0; i < num_pr; i++) { + /* skip those that we might have re-added above */ + if (pr[i] == NULL) + continue; + /* check if we can expire the paging record, + * or if we need to re-queue it */ + if (pr[i]->expiration_time >= now) { + talloc_free(pr[i]); + ps->num_paging--; + LOGP(DPAG, LOGL_INFO, "Removed paging record, queue_len=%u\n", + ps->num_paging); + } else + llist_add_tail(&pr[i]->list, group_q); + } + } + memset(out_buf+len, 0x2B, GSM_MACBLOCK_LEN-len); + return len; +} + +int paging_si_update(struct paging_state *ps, struct gsm48_control_channel_descr *chan_desc) +{ + LOGP(DPAG, LOGL_INFO, "Paging SI update\n"); + + memcpy(&ps->chan_desc, chan_desc, sizeof(chan_desc)); + + /* FIXME: do we need to re-sort the old paging_records? */ + + return 0; +} + +static int paging_signal_cbfn(unsigned int subsys, unsigned int signal, void *hdlr_data, + void *signal_data) +{ + if (subsys == SS_GLOBAL && signal == S_NEW_SYSINFO) { + struct gsm_bts *bts = signal_data; + struct gsm_bts_role_bts *btsb = bts->role; + struct paging_state *ps = btsb->paging_state; + struct gsm48_system_information_type_3 *si3 = (void *) bts->si_buf[SYSINFO_TYPE_3]; + + paging_si_update(ps, &si3->control_channel_desc); + } + return 0; +} + +static int initialized = 0; + +struct paging_state *paging_init(void *ctx, unsigned int num_paging_max, + unsigned int paging_lifetime) +{ + struct paging_state *ps; + unsigned int i; + + ps = talloc_zero(ctx, struct paging_state); + if (!ps) + return NULL; + + ps->paging_lifetime = paging_lifetime; + ps->num_paging_max = num_paging_max; + + for (i = 0; i < ARRAY_SIZE(ps->paging_queue); i++) + INIT_LLIST_HEAD(&ps->paging_queue[i]); + + if (!initialized) { + osmo_signal_register_handler(SS_GLOBAL, paging_signal_cbfn, NULL); + initialized = 1; + } + return ps; +} |