/* Cell Broadcast Service Protocol (CBSP, 3GPP TS 48.049): Message encoding, decoding and reception */ /* * Copyright (C) 2019 Harald Welte <laforge@gnumonks.org> * * All Rights Reserved * * SPDX-License-Identifier: GPL-2.0+ * * 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. */ #include "config.h" #include <errno.h> #include <sys/types.h> #include <osmocom/core/linuxlist.h> #include <osmocom/core/msgb.h> #include <osmocom/gsm/tlv.h> #include <osmocom/gsm/cbsp.h> #include <osmocom/gsm/gsm0808_utils.h> __thread const char *osmo_cbsp_errstr; struct msgb *osmo_cbsp_msgb_alloc(void *ctx, const char *name) { /* make the messages rather large as the cell lists can be long! */ return msgb_alloc_headroom_c(ctx, 65535, 16, name); } /*********************************************************************** * IE Encoding ***********************************************************************/ /* 8.2.6 Cell List */ static void msgb_put_cbsp_cell_list(struct msgb *msg, const struct osmo_cbsp_cell_list *cl) { const struct osmo_cbsp_cell_ent *ent; uint8_t *lenptr; /* put tag; reserve space for length; put discriminator */ msgb_put_u8(msg, CBSP_IEI_CELL_LIST); lenptr = msgb_put(msg, sizeof(uint16_t)); msgb_put_u8(msg, cl->id_discr); /* put list elements */ llist_for_each_entry(ent, &cl->list, list) { gsm0808_msgb_put_cell_id_u(msg, cl->id_discr, &ent->cell_id); } /* update IE length */ osmo_store16be(msg->tail - (lenptr+2), lenptr); } /* 8.2.11 Failure List (discriminator per entry) */ static void msgb_put_cbsp_fail_list(struct msgb *msg, const struct llist_head *fl) { const struct osmo_cbsp_fail_ent *ent; uint8_t *lenptr; /* put tag; reserve space for length; put discriminator */ msgb_put_u8(msg, CBSP_IEI_FAILURE_LIST); lenptr = msgb_put(msg, sizeof(uint16_t)); /* put list elements */ llist_for_each_entry(ent, fl, list) { msgb_put_u8(msg, ent->id_discr); gsm0808_msgb_put_cell_id_u(msg, ent->id_discr, &ent->cell_id); msgb_put_u8(msg, ent->cause); } /* update IE length */ osmo_store16be(msg->tail - (lenptr+2), lenptr); } /* 8.2.12 Radio Resource Loading List */ static void msgb_put_cbsp_loading_list(struct msgb *msg, const struct osmo_cbsp_loading_list *ll) { const struct osmo_cbsp_loading_ent *ent; uint8_t *lenptr; /* put tag; reserve space for length; put discriminator */ msgb_put_u8(msg, CBSP_IEI_RR_LOADING_LIST); lenptr = msgb_put(msg, sizeof(uint16_t)); msgb_put_u8(msg, ll->id_discr); /* put list elements */ llist_for_each_entry(ent, &ll->list, list) { gsm0808_msgb_put_cell_id_u(msg, ll->id_discr, &ent->cell_id); msgb_put_u8(msg, ent->load[0]); msgb_put_u8(msg, ent->load[1]); } /* update IE length */ osmo_store16be(msg->tail - (lenptr+2), lenptr); } /* 8.2.10 Completed List */ static void msgb_put_cbsp_num_compl_list(struct msgb *msg, const struct osmo_cbsp_num_compl_list *cl) { const struct osmo_cbsp_num_compl_ent *ent; uint8_t *lenptr; /* put tag; reserve space for length; put discriminator */ msgb_put_u8(msg, CBSP_IEI_NUM_BCAST_COMPL_LIST); lenptr = msgb_put(msg, sizeof(uint16_t)); msgb_put_u8(msg, cl->id_discr); /* put list elements */ llist_for_each_entry(ent, &cl->list, list) { gsm0808_msgb_put_cell_id_u(msg, cl->id_discr, &ent->cell_id); msgb_put_u16(msg, ent->num_compl); msgb_put_u8(msg, ent->num_bcast_info); } /* update IE length */ osmo_store16be(msg->tail - (lenptr+2), lenptr); } static int encode_wperiod(uint32_t secs) { if (secs == 0xffffffff) return 0; /* infinite */ if (secs <= 10) return secs; if (secs <= 30) return 10 + (secs-10)/2; if (secs <= 120) return 30 + (secs-30)/5; if (secs <= 600) return 120 + (secs-120)/10; if (secs <= 60*60) return 600 + (secs-600)/30; osmo_cbsp_errstr = "warning period out of range"; return -1; } /*********************************************************************** * Message Encoding ***********************************************************************/ /* 8.1.3.1 WRITE REPLACE */ static int cbsp_enc_write_repl(struct msgb *msg, const struct osmo_cbsp_write_replace *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_NEW_SERIAL_NR, in->new_serial_nr); if (in->old_serial_nr) msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, *in->old_serial_nr); msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->is_cbs) { int num_of_pages = llist_count(&in->u.cbs.msg_content); struct osmo_cbsp_content *ce; if (num_of_pages == 0 || num_of_pages > 15) { osmo_cbsp_errstr = "invalid number of pages"; return -EINVAL; } msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->u.cbs.channel_ind); msgb_tv_put(msg, CBSP_IEI_CATEGORY, in->u.cbs.category); msgb_tv16_put(msg, CBSP_IEI_REP_PERIOD, in->u.cbs.rep_period); msgb_tv16_put(msg, CBSP_IEI_NUM_BCAST_REQ, in->u.cbs.num_bcast_req); msgb_tv_put(msg, CBSP_IEI_NUM_OF_PAGES, num_of_pages); msgb_tv_put(msg, CBSP_IEI_DCS, in->u.cbs.dcs); llist_for_each_entry(ce, &in->u.cbs.msg_content, list) { uint8_t *out; /* cannot use msgb_tlv_put() as 'len' isn't actually the length of * the data field */ msgb_put_u8(msg, CBSP_IEI_MSG_CONTENT); msgb_put_u8(msg, ce->user_len); out = msgb_put(msg, sizeof(ce->data)); memcpy(out, ce->data, sizeof(ce->data)); } } else { int wperiod = encode_wperiod(in->u.emergency.warning_period); uint8_t *cur; if (wperiod < 0) return -EINVAL; msgb_tv_put(msg, CBSP_IEI_EMERG_IND, in->u.emergency.indicator); msgb_tv16_put(msg, CBSP_IEI_WARN_TYPE, in->u.emergency.warning_type); /* Tag + fixed length value! */ msgb_put_u8(msg, CBSP_IEI_WARN_SEC_INFO); cur = msgb_put(msg, sizeof(in->u.emergency.warning_sec_info)); memcpy(cur, in->u.emergency.warning_sec_info, sizeof(in->u.emergency.warning_sec_info)); msgb_tv_put(msg, CBSP_IEI_WARNING_PERIOD, wperiod); } return 0; } /* 8.1.3.2 WRITE REPLACE COMPLETE*/ static int cbsp_enc_write_repl_compl(struct msgb *msg, const struct osmo_cbsp_write_replace_complete *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_NEW_SERIAL_NR, in->new_serial_nr); if (in->old_serial_nr) msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, *in->old_serial_nr); if (!llist_empty(&in->num_compl_list.list)) msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); if (!llist_empty(&in->cell_list.list)) msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /* 8.1.3.3 WRITE REPLACE FAILURE */ static int cbsp_enc_write_repl_fail(struct msgb *msg, const struct osmo_cbsp_write_replace_failure *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_NEW_SERIAL_NR, in->new_serial_nr); if (in->old_serial_nr) msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, *in->old_serial_nr); msgb_put_cbsp_fail_list(msg, &in->fail_list); if (!llist_empty(&in->num_compl_list.list)) msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); if (!llist_empty(&in->cell_list.list)) msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /* 8.1.3.4 KILL */ static int cbsp_enc_kill(struct msgb *msg, const struct osmo_cbsp_kill *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /* 8.1.3.5 KILL COMPLETE */ static int cbsp_enc_kill_compl(struct msgb *msg, const struct osmo_cbsp_kill_complete *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); if (!llist_empty(&in->num_compl_list.list)) msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); if (!llist_empty(&in->cell_list.list)) msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /* 8.1.3.6 KILL FAILURE */ static int cbsp_enc_kill_fail(struct msgb *msg, const struct osmo_cbsp_kill_failure *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); msgb_put_cbsp_fail_list(msg, &in->fail_list); if (!llist_empty(&in->num_compl_list.list)) msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); if (!llist_empty(&in->cell_list.list)) msgb_put_cbsp_cell_list(msg, &in->cell_list); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /* 8.1.3.7 LOAD QUERY */ static int cbsp_enc_load_query(struct msgb *msg, const struct osmo_cbsp_load_query *in) { msgb_put_cbsp_cell_list(msg, &in->cell_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); return 0; } /* 8.1.3.8 LOAD QUERY COMPLETE */ static int cbsp_enc_load_query_compl(struct msgb *msg, const struct osmo_cbsp_load_query_complete *in) { msgb_put_cbsp_loading_list(msg, &in->loading_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); return 0; } /* 8.1.3.9 LOAD QUERY FAILURE */ static int cbsp_enc_load_query_fail(struct msgb *msg, const struct osmo_cbsp_load_query_failure *in) { msgb_put_cbsp_fail_list(msg, &in->fail_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); if (!llist_empty(&in->loading_list.list)) msgb_put_cbsp_loading_list(msg, &in->loading_list); return 0; } /* 8.1.3.10 STATUS QUERY */ static int cbsp_enc_msg_status_query(struct msgb *msg, const struct osmo_cbsp_msg_status_query *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); msgb_put_cbsp_cell_list(msg, &in->cell_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); return 0; } /* 8.1.3.11 STATUS QUERY COMPLETE */ static int cbsp_enc_msg_status_query_compl(struct msgb *msg, const struct osmo_cbsp_msg_status_query_complete *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); return 0; } /* 8.1.3.12 STATUS QUERY FAILURE */ static int cbsp_enc_msg_status_query_fail(struct msgb *msg, const struct osmo_cbsp_msg_status_query_failure *in) { msgb_tv16_put(msg, CBSP_IEI_MSG_ID, in->msg_id); msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, in->old_serial_nr); msgb_put_cbsp_fail_list(msg, &in->fail_list); msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, in->channel_ind); if (!llist_empty(&in->num_compl_list.list)) msgb_put_cbsp_num_compl_list(msg, &in->num_compl_list); return 0; } /* 8.1.3.16 RESET */ static int cbsp_enc_reset(struct msgb *msg, const struct osmo_cbsp_reset *in) { msgb_put_cbsp_cell_list(msg, &in->cell_list); return 0; } /* 8.1.3.17 RESET COMPLETE */ static int cbsp_enc_reset_compl(struct msgb *msg, const struct osmo_cbsp_reset_complete *in) { msgb_put_cbsp_cell_list(msg, &in->cell_list); return 0; } /* 8.1.3.18 RESET FAILURE */ static int cbsp_enc_reset_fail(struct msgb *msg, const struct osmo_cbsp_reset_failure *in) { msgb_put_cbsp_fail_list(msg, &in->fail_list); if (!llist_empty(&in->cell_list.list)) msgb_put_cbsp_cell_list(msg, &in->cell_list); return 0; } /* 8.1.3.18a KEEP ALIVE */ static int cbsp_enc_keep_alive(struct msgb *msg, const struct osmo_cbsp_keep_alive *in) { int rperiod = encode_wperiod(in->repetition_period); if (in->repetition_period > 120) return -EINVAL; if (rperiod < 0) return -EINVAL; msgb_tv_put(msg, CBSP_IEI_KEEP_ALIVE_REP_PERIOD, rperiod); return 0; } /* 8.1.3.18b KEEP ALIVE COMPLETE */ static int cbsp_enc_keep_alive_compl(struct msgb *msg, const struct osmo_cbsp_keep_alive_complete *in) { return 0; } /* 8.1.3.19 RESTART */ static int cbsp_enc_restart(struct msgb *msg, const struct osmo_cbsp_restart *in) { msgb_put_cbsp_cell_list(msg, &in->cell_list); msgb_tv_put(msg, CBSP_IEI_BCAST_MSG_TYPE, in->bcast_msg_type); msgb_tv_put(msg, CBSP_IEI_RECOVERY_IND, in->recovery_ind); return 0; } /* 8.1.3.20 FAILURE */ static int cbsp_enc_failure(struct msgb *msg, const struct osmo_cbsp_failure *in) { msgb_put_cbsp_fail_list(msg, &in->fail_list); msgb_tv_put(msg, CBSP_IEI_BCAST_MSG_TYPE, in->bcast_msg_type); return 0; } /* 8.1.3.21 ERROR INDICATION */ static int cbsp_enc_error_ind(struct msgb *msg, const struct osmo_cbsp_error_ind *in) { msgb_tv_put(msg, CBSP_IEI_CAUSE, in->cause); if (in->msg_id) msgb_tv16_put(msg, CBSP_IEI_MSG_ID, *in->msg_id); if (in->new_serial_nr) msgb_tv16_put(msg, CBSP_IEI_NEW_SERIAL_NR, *in->new_serial_nr); if (in->old_serial_nr) msgb_tv16_put(msg, CBSP_IEI_OLD_SERIAL_NR, *in->old_serial_nr); if (in->channel_ind) msgb_tv_put(msg, CBSP_IEI_CHANNEL_IND, *in->channel_ind); return 0; } /*! Encode a CBSP message from the decoded/parsed structure representation to binary PDU. * \param[in] ctx talloc context from which to allocate returned msgb. * \param[in] in decoded CBSP message which is to be encoded. Ownership not transferred. * \return callee-allocated message buffer containing binary CBSP PDU; NULL on error */ struct msgb *osmo_cbsp_encode(void *ctx, const struct osmo_cbsp_decoded *in) { struct msgb *msg = osmo_cbsp_msgb_alloc(ctx, __func__); unsigned int len; int rc; osmo_cbsp_errstr = NULL; if (!msg) return NULL; switch (in->msg_type) { case CBSP_MSGT_WRITE_REPLACE: rc = cbsp_enc_write_repl(msg, &in->u.write_replace); break; case CBSP_MSGT_WRITE_REPLACE_COMPL: rc = cbsp_enc_write_repl_compl(msg, &in->u.write_replace_compl); break; case CBSP_MSGT_WRITE_REPLACE_FAIL: rc = cbsp_enc_write_repl_fail(msg, &in->u.write_replace_fail); break; case CBSP_MSGT_KILL: rc = cbsp_enc_kill(msg, &in->u.kill); break; case CBSP_MSGT_KILL_COMPL: rc = cbsp_enc_kill_compl(msg, &in->u.kill_compl); break; case CBSP_MSGT_KILL_FAIL: rc = cbsp_enc_kill_fail(msg, &in->u.kill_fail); break; case CBSP_MSGT_LOAD_QUERY: rc = cbsp_enc_load_query(msg, &in->u.load_query); break; case CBSP_MSGT_LOAD_QUERY_COMPL: rc = cbsp_enc_load_query_compl(msg, &in->u.load_query_compl); break; case CBSP_MSGT_LOAD_QUERY_FAIL: rc = cbsp_enc_load_query_fail(msg, &in->u.load_query_fail); break; case CBSP_MSGT_MSG_STATUS_QUERY: rc = cbsp_enc_msg_status_query(msg, &in->u.msg_status_query); break; case CBSP_MSGT_MSG_STATUS_QUERY_COMPL: rc = cbsp_enc_msg_status_query_compl(msg, &in->u.msg_status_query_compl); break; case CBSP_MSGT_MSG_STATUS_QUERY_FAIL: rc = cbsp_enc_msg_status_query_fail(msg, &in->u.msg_status_query_fail); break; case CBSP_MSGT_RESET: rc = cbsp_enc_reset(msg, &in->u.reset); break; case CBSP_MSGT_RESET_COMPL: rc = cbsp_enc_reset_compl(msg, &in->u.reset_compl); break; case CBSP_MSGT_RESET_FAIL: rc = cbsp_enc_reset_fail(msg, &in->u.reset_fail); break; case CBSP_MSGT_RESTART: rc = cbsp_enc_restart(msg, &in->u.restart); break; case CBSP_MSGT_FAILURE: rc = cbsp_enc_failure(msg, &in->u.failure); break; case CBSP_MSGT_ERROR_IND: rc = cbsp_enc_error_ind(msg, &in->u.error_ind); break; case CBSP_MSGT_KEEP_ALIVE: rc = cbsp_enc_keep_alive(msg, &in->u.keep_alive); break; case CBSP_MSGT_KEEP_ALIVE_COMPL: rc = cbsp_enc_keep_alive_compl(msg, &in->u.keep_alive_compl); break; case CBSP_MSGT_SET_DRX: case CBSP_MSGT_SET_DRX_COMPL: case CBSP_MSGT_SET_DRX_FAIL: osmo_cbsp_errstr = "message type not implemented"; rc = -1; break; default: osmo_cbsp_errstr = "message type not known in spec"; rc = -1; break; } if (rc < 0) { msgb_free(msg); return NULL; } /* push header in front */ len = msgb_length(msg); msgb_push_u8(msg, len & 0xff); msgb_push_u8(msg, (len >> 8) & 0xff); msgb_push_u8(msg, (len >> 16) & 0xff); msgb_push_u8(msg, in->msg_type); return msg; } /*********************************************************************** * IE Decoding ***********************************************************************/ /* 8.2.6 Cell List */ static int cbsp_decode_cell_list(struct osmo_cbsp_cell_list *cl, void *ctx, const uint8_t *buf, unsigned int len) { const uint8_t *cur = buf; int rc; cl->id_discr = *cur++; while (cur < buf + len) { struct osmo_cbsp_cell_ent *ent = talloc_zero(ctx, struct osmo_cbsp_cell_ent); unsigned int len_remain = len - (cur - buf); OSMO_ASSERT(ent); rc = gsm0808_decode_cell_id_u(&ent->cell_id, cl->id_discr, cur, len_remain); if (rc < 0) { osmo_cbsp_errstr = "cell list: error decoding cell_id_union"; return rc; } cur += gsm0808_cell_id_size(cl->id_discr); llist_add_tail(&ent->list, &cl->list); } return 0; } /* 8.2.11 Failure List (discriminator per entry) */ static int cbsp_decode_fail_list(struct llist_head *fl, void *ctx, const uint8_t *buf, unsigned int len) { const uint8_t *cur = buf; int rc; while (cur < buf + len) { struct osmo_cbsp_fail_ent *ent = talloc_zero(ctx, struct osmo_cbsp_fail_ent); unsigned int len_remain = len - (cur - buf); OSMO_ASSERT(ent); ent->id_discr = *cur++; rc = gsm0808_decode_cell_id_u(&ent->cell_id, ent->id_discr, cur, len_remain-1); if (rc < 0) { osmo_cbsp_errstr = "fail list: error decoding cell_id_union"; return rc; } cur += gsm0808_cell_id_size(ent->id_discr); ent->cause = *cur++; llist_add_tail(&ent->list, fl); } return 0; } /* 8.2.12 Radio Resource Loading List */ static int cbsp_decode_loading_list(struct osmo_cbsp_loading_list *ll, void *ctx, const uint8_t *buf, unsigned int len) { const uint8_t *cur = buf; int rc; ll->id_discr = *cur++; while (cur < buf + len) { struct osmo_cbsp_loading_ent *ent = talloc_zero(ctx, struct osmo_cbsp_loading_ent); unsigned int len_remain = len - (cur - buf); OSMO_ASSERT(ent); rc = gsm0808_decode_cell_id_u(&ent->cell_id, ll->id_discr, cur, len_remain); if (rc < 0) { osmo_cbsp_errstr = "load list: error decoding cell_id_union"; return rc; } cur += gsm0808_cell_id_size(ll->id_discr); if (cur + 2 > buf + len) { talloc_free(ent); osmo_cbsp_errstr = "load list: truncated IE"; return -EINVAL; } ent->load[0] = *cur++; ent->load[1] = *cur++; llist_add_tail(&ent->list, &ll->list); } return 0; } /* 8.2.10 Completed List */ static int cbsp_decode_num_compl_list(struct osmo_cbsp_num_compl_list *cl, void *ctx, const uint8_t *buf, unsigned int len) { const uint8_t *cur = buf; int rc; cl->id_discr = *cur++; while (cur < buf + len) { struct osmo_cbsp_num_compl_ent *ent = talloc_zero(ctx, struct osmo_cbsp_num_compl_ent); unsigned int len_remain = len - (cur - buf); OSMO_ASSERT(ent); rc = gsm0808_decode_cell_id_u(&ent->cell_id, cl->id_discr, cur, len_remain); if (rc < 0) { osmo_cbsp_errstr = "completed list: error decoding cell_id_union"; return rc; } cur += gsm0808_cell_id_size(cl->id_discr); if (cur + 3 > buf + len) { talloc_free(ent); osmo_cbsp_errstr = "completed list: truncated IE"; return -EINVAL; } ent->num_compl = osmo_load16be(cur); cur += 2; ent->num_bcast_info = *cur++; llist_add_tail(&ent->list, &cl->list); } return 0; } /* 8.2.25 */ static uint32_t decode_wperiod(uint8_t in) { if (in == 0x00) return 0xffffffff; /* infinite */ if (in <= 10) return in; if (in <= 20) return 10 + (in - 10)*2; if (in <= 38) return 30 + (in - 20)*5; if (in <= 86) return 120 + (in - 38)*10; if (in <= 186) return 600 + (in - 86)*30; else return 0; } /*********************************************************************** * Message Decoding ***********************************************************************/ /* 8.1.3.1 WRITE REPLACE */ static int cbsp_dec_write_repl(struct osmo_cbsp_write_replace *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { unsigned int i; int rc; /* check for mandatory IEs */ if (!TLVP_PRESENT(tp, CBSP_IEI_MSG_ID) || !TLVP_PRESENT(tp, CBSP_IEI_NEW_SERIAL_NR) || !TLVP_PRESENT(tp, CBSP_IEI_CELL_LIST)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->new_serial_nr = tlvp_val16be(tp, CBSP_IEI_NEW_SERIAL_NR); if (TLVP_PRESENT(tp, CBSP_IEI_OLD_SERIAL_NR)) { out->old_serial_nr = talloc(ctx, uint16_t); *out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; if (TLVP_PRESENT(tp, CBSP_IEI_CHANNEL_IND)) { uint8_t num_of_pages; INIT_LLIST_HEAD(&out->u.cbs.msg_content); if (TLVP_PRESENT(tp, CBSP_IEI_EMERG_IND)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } if (!TLVP_PRESENT(tp, CBSP_IEI_CATEGORY) || !TLVP_PRESENT(tp, CBSP_IEI_REP_PERIOD) || !TLVP_PRESENT(tp, CBSP_IEI_NUM_BCAST_REQ) || !TLVP_PRESENT(tp, CBSP_IEI_NUM_OF_PAGES) || !TLVP_PRESENT(tp, CBSP_IEI_DCS)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->is_cbs = true; out->u.cbs.channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); out->u.cbs.category = *TLVP_VAL(tp, CBSP_IEI_CATEGORY); out->u.cbs.rep_period = tlvp_val16be(tp, CBSP_IEI_REP_PERIOD); out->u.cbs.num_bcast_req = tlvp_val16be(tp, CBSP_IEI_NUM_BCAST_REQ); out->u.cbs.dcs = *TLVP_VAL(tp, CBSP_IEI_DCS); num_of_pages = *TLVP_VAL(tp, CBSP_IEI_NUM_OF_PAGES); if (num_of_pages < 1) return -EINVAL; /* parse pages */ for (i = 0; i < num_of_pages; i++) { const uint8_t *ie = TLVP_VAL(&tp[i], CBSP_IEI_MSG_CONTENT); struct osmo_cbsp_content *page; if (!ie) { osmo_cbsp_errstr = "insufficient message content IEs"; return -EINVAL; } page = talloc_zero(ctx, struct osmo_cbsp_content); OSMO_ASSERT(page); page->user_len = ie[0]; /* length byte before payload */ memcpy(page->data, ie+1, sizeof(page->data)); llist_add_tail(&page->list, &out->u.cbs.msg_content); } } else { if (!TLVP_PRES_LEN(tp, CBSP_IEI_EMERG_IND, 1) || !TLVP_PRES_LEN(tp, CBSP_IEI_WARN_TYPE, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_WARN_SEC_INFO, 50) || !TLVP_PRES_LEN(tp, CBSP_IEI_WARNING_PERIOD, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->u.emergency.indicator = *TLVP_VAL(tp, CBSP_IEI_EMERG_IND); out->u.emergency.warning_type = tlvp_val16be(tp, CBSP_IEI_WARN_TYPE); memcpy(&out->u.emergency.warning_sec_info, TLVP_VAL(tp, CBSP_IEI_WARN_SEC_INFO), sizeof(out->u.emergency.warning_sec_info)); out->u.emergency.warning_period = decode_wperiod(*TLVP_VAL(tp, CBSP_IEI_WARNING_PERIOD)); } return 0; } /* 8.1.3.2 WRITE REPLACE COMPLETE*/ static int cbsp_dec_write_repl_compl(struct osmo_cbsp_write_replace_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_NEW_SERIAL_NR, 2)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->new_serial_nr = tlvp_val16be(tp, CBSP_IEI_NEW_SERIAL_NR); if (TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2)) { out->old_serial_nr = talloc(ctx, uint16_t); *out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); } INIT_LLIST_HEAD(&out->num_compl_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7)) { rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /* 8.1.3.3 WRITE REPLACE FAILURE */ static int cbsp_dec_write_repl_fail(struct osmo_cbsp_write_replace_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_NEW_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->new_serial_nr = tlvp_val16be(tp, CBSP_IEI_NEW_SERIAL_NR); if (TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2)) { out->old_serial_nr = talloc(ctx, uint16_t); *out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); } INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; INIT_LLIST_HEAD(&out->num_compl_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7)) { rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; } INIT_LLIST_HEAD(&out->cell_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; } if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /* 8.1.3.4 KILL */ static int cbsp_dec_kill(struct osmo_cbsp_kill *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /* 8.1.3.5 KILL COMPLETE */ static int cbsp_dec_kill_compl(struct osmo_cbsp_kill_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->num_compl_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7)) { rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /* 8.1.3.6 KILL FAILURE */ static int cbsp_dec_kill_fail(struct osmo_cbsp_kill_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; INIT_LLIST_HEAD(&out->num_compl_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7)) { rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; } INIT_LLIST_HEAD(&out->cell_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; } if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /* 8.1.3.7 LOAD QUERY */ static int cbsp_dec_load_query(struct osmo_cbsp_load_query *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); return 0; } /* 8.1.3.8 LOAD QUERY COMPLETE */ static int cbsp_dec_load_query_compl(struct osmo_cbsp_load_query_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_RR_LOADING_LIST, 6) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->loading_list.list); rc = cbsp_decode_loading_list(&out->loading_list, ctx, TLVP_VAL(tp, CBSP_IEI_RR_LOADING_LIST), TLVP_LEN(tp, CBSP_IEI_RR_LOADING_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); return 0; } /* 8.1.3.9 LOAD QUERY FAILURE */ static int cbsp_dec_load_query_fail(struct osmo_cbsp_load_query_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); INIT_LLIST_HEAD(&out->loading_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_RR_LOADING_LIST, 6)) { rc = cbsp_decode_loading_list(&out->loading_list, ctx, TLVP_VAL(tp, CBSP_IEI_RR_LOADING_LIST), TLVP_LEN(tp, CBSP_IEI_RR_LOADING_LIST)); } return rc; } /* 8.1.3.10 STATUS QUERY */ static int cbsp_dec_msg_status_query(struct osmo_cbsp_msg_status_query *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); return 0; } /* 8.1.3.11 STATUS QUERY COMPLETE */ static int cbsp_dec_msg_status_query_compl(struct osmo_cbsp_msg_status_query_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->num_compl_list.list); rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); return 0; } /* 8.1.3.12 STATUS QUERY FAILURE */ static int cbsp_dec_msg_status_query_fail(struct osmo_cbsp_msg_status_query_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2) || !TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5) || !TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); INIT_LLIST_HEAD(&out->num_compl_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST, 7)) { rc = cbsp_decode_num_compl_list(&out->num_compl_list, ctx, TLVP_VAL(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST), TLVP_LEN(tp, CBSP_IEI_NUM_BCAST_COMPL_LIST)); if (rc < 0) return rc; } return 0; } /* 8.1.3.16 RESET */ static int cbsp_dec_reset(struct osmo_cbsp_reset *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; return 0; } /* 8.1.3.17 RESET COMPLETE */ static int cbsp_dec_reset_compl(struct osmo_cbsp_reset_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; return 0; } /* 8.1.3.18 RESET FAILURE */ static int cbsp_dec_reset_fail(struct osmo_cbsp_reset_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; INIT_LLIST_HEAD(&out->cell_list.list); if (TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1)) { rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; } return 0; } /* 8.1.3.18a KEEP ALIVE */ static int cbsp_dec_keep_alive(struct osmo_cbsp_keep_alive *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { uint8_t rperiod; if (!TLVP_PRES_LEN(tp, CBSP_IEI_KEEP_ALIVE_REP_PERIOD, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } rperiod = *TLVP_VAL(tp, CBSP_IEI_KEEP_ALIVE_REP_PERIOD); out->repetition_period = decode_wperiod(rperiod); return 0; } /* 8.1.3.18b KEEP ALIVE COMPLETE */ static int cbsp_dec_keep_alive_compl(struct osmo_cbsp_keep_alive_complete *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { return 0; } /* 8.1.3.19 RESTART */ static int cbsp_dec_restart(struct osmo_cbsp_restart *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_CELL_LIST, 1) || !TLVP_PRES_LEN(tp, CBSP_IEI_BCAST_MSG_TYPE, 1) || !TLVP_PRES_LEN(tp, CBSP_IEI_RECOVERY_IND, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->cell_list.list); rc = cbsp_decode_cell_list(&out->cell_list, ctx, TLVP_VAL(tp, CBSP_IEI_CELL_LIST), TLVP_LEN(tp, CBSP_IEI_CELL_LIST)); if (rc < 0) return rc; out->bcast_msg_type = *TLVP_VAL(tp, CBSP_IEI_BCAST_MSG_TYPE); out->recovery_ind = *TLVP_VAL(tp, CBSP_IEI_RECOVERY_IND); return 0; } /* 8.1.3.20 FAILURE */ static int cbsp_dec_failure(struct osmo_cbsp_failure *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { int rc; if (!TLVP_PRES_LEN(tp, CBSP_IEI_FAILURE_LIST, 5) || !TLVP_PRES_LEN(tp, CBSP_IEI_BCAST_MSG_TYPE, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } INIT_LLIST_HEAD(&out->fail_list); rc = cbsp_decode_fail_list(&out->fail_list, ctx, TLVP_VAL(tp, CBSP_IEI_FAILURE_LIST), TLVP_LEN(tp, CBSP_IEI_FAILURE_LIST)); if (rc < 0) return rc; out->bcast_msg_type = *TLVP_VAL(tp, CBSP_IEI_BCAST_MSG_TYPE); return 0; } /* 8.1.3.21 ERROR INDICATION */ static int cbsp_dec_error_ind(struct osmo_cbsp_error_ind *out, const struct tlv_parsed *tp, struct msgb *in, void *ctx) { if (!TLVP_PRES_LEN(tp, CBSP_IEI_CAUSE, 1)) { osmo_cbsp_errstr = "missing/short mandatory IE"; return -EINVAL; } out->cause = *TLVP_VAL(tp, CBSP_IEI_CAUSE); if (TLVP_PRES_LEN(tp, CBSP_IEI_MSG_ID, 2)) { out->msg_id = talloc(ctx, uint16_t); *out->msg_id = tlvp_val16be(tp, CBSP_IEI_MSG_ID); } if (TLVP_PRES_LEN(tp, CBSP_IEI_NEW_SERIAL_NR, 2)) { out->new_serial_nr = talloc(ctx, uint16_t); *out->new_serial_nr = tlvp_val16be(tp, CBSP_IEI_NEW_SERIAL_NR); } if (TLVP_PRES_LEN(tp, CBSP_IEI_OLD_SERIAL_NR, 2)) { out->old_serial_nr = talloc(ctx, uint16_t); *out->old_serial_nr = tlvp_val16be(tp, CBSP_IEI_OLD_SERIAL_NR); } if (TLVP_PRES_LEN(tp, CBSP_IEI_CHANNEL_IND, 1)) { out->channel_ind = talloc(ctx, enum cbsp_channel_ind); *out->channel_ind = *TLVP_VAL(tp, CBSP_IEI_CHANNEL_IND); } return 0; } /*! Decode a CBSP message from wire formwat to pased structure. * \param[in] ctx talloc context from which to allocate decoded output. * \param[in] in message buffer contiaining binary CBSP message. * \returns callee-allocated decoded representation of CBSP message; NULL on error */ struct osmo_cbsp_decoded *osmo_cbsp_decode(void *ctx, struct msgb *in) { OSMO_ASSERT(in->l1h != NULL && in->l2h != NULL); struct osmo_cbsp_decoded *out = talloc_zero(ctx, struct osmo_cbsp_decoded); const struct cbsp_header *h = msgb_l1(in); struct tlv_parsed tp[16]; /* max. number of pages in a given CBS message */ unsigned int len; int rc; osmo_cbsp_errstr = NULL; if (!out) return NULL; if (msgb_l1len(in) < sizeof(*h)) { goto out_err; } len = h->len[0] << 16 | h->len[1] << 8 | h->len[2]; /* discard messages where indicated length is more than we have */ if (len > msgb_l2len(in)) { goto out_err; } /* trim any messages with extra payload at the end */ if (len < msgb_l2len(in)) msgb_trim(in, (in->l2h - in->data) + msgb_l2len(in)); out->msg_type = h->msg_type; rc = tlv_parse2(tp, ARRAY_SIZE(tp), &cbsp_att_tlvdef, msgb_l2(in), msgb_l2len(in), 0, 0); if (rc < 0) { goto out_err; } switch (h->msg_type) { case CBSP_MSGT_WRITE_REPLACE: rc = cbsp_dec_write_repl(&out->u.write_replace, tp, in, out); break; case CBSP_MSGT_WRITE_REPLACE_COMPL: rc = cbsp_dec_write_repl_compl(&out->u.write_replace_compl, tp, in, out); break; case CBSP_MSGT_WRITE_REPLACE_FAIL: rc = cbsp_dec_write_repl_fail(&out->u.write_replace_fail, tp, in, out); break; case CBSP_MSGT_KILL: rc = cbsp_dec_kill(&out->u.kill, tp, in, out); break; case CBSP_MSGT_KILL_COMPL: rc = cbsp_dec_kill_compl(&out->u.kill_compl, tp, in, out); break; case CBSP_MSGT_KILL_FAIL: rc = cbsp_dec_kill_fail(&out->u.kill_fail, tp, in, out); break; case CBSP_MSGT_LOAD_QUERY: rc = cbsp_dec_load_query(&out->u.load_query, tp, in, out); break; case CBSP_MSGT_LOAD_QUERY_COMPL: rc = cbsp_dec_load_query_compl(&out->u.load_query_compl, tp, in, out); break; case CBSP_MSGT_LOAD_QUERY_FAIL: rc = cbsp_dec_load_query_fail(&out->u.load_query_fail, tp, in, out); break; case CBSP_MSGT_MSG_STATUS_QUERY: rc = cbsp_dec_msg_status_query(&out->u.msg_status_query, tp, in, out); break; case CBSP_MSGT_MSG_STATUS_QUERY_COMPL: rc = cbsp_dec_msg_status_query_compl(&out->u.msg_status_query_compl, tp, in, out); break; case CBSP_MSGT_MSG_STATUS_QUERY_FAIL: rc = cbsp_dec_msg_status_query_fail(&out->u.msg_status_query_fail, tp, in, out); break; case CBSP_MSGT_RESET: rc = cbsp_dec_reset(&out->u.reset, tp, in, out); break; case CBSP_MSGT_RESET_COMPL: rc = cbsp_dec_reset_compl(&out->u.reset_compl, tp, in, out); break; case CBSP_MSGT_RESET_FAIL: rc = cbsp_dec_reset_fail(&out->u.reset_fail, tp, in, out); break; case CBSP_MSGT_RESTART: rc = cbsp_dec_restart(&out->u.restart, tp, in, out); break; case CBSP_MSGT_FAILURE: rc = cbsp_dec_failure(&out->u.failure, tp, in, out); break; case CBSP_MSGT_ERROR_IND: rc = cbsp_dec_error_ind(&out->u.error_ind, tp, in, out); break; case CBSP_MSGT_KEEP_ALIVE: rc = cbsp_dec_keep_alive(&out->u.keep_alive, tp, in, out); break; case CBSP_MSGT_KEEP_ALIVE_COMPL: rc = cbsp_dec_keep_alive_compl(&out->u.keep_alive_compl, tp, in, out); break; case CBSP_MSGT_SET_DRX: case CBSP_MSGT_SET_DRX_COMPL: case CBSP_MSGT_SET_DRX_FAIL: osmo_cbsp_errstr = "message type not implemented"; rc = -1; break; default: osmo_cbsp_errstr = "message type not known in spec"; rc = -1; break; } if (rc < 0) { goto out_err; } return out; out_err: talloc_free(out); return NULL; } /* initialization of 'decoded' structure of given message type */ void osmo_cbsp_init_struct(struct osmo_cbsp_decoded *cbsp, enum cbsp_msg_type msg_type) { memset(cbsp, 0, sizeof(*cbsp)); cbsp->msg_type = msg_type; switch (msg_type) { case CBSP_MSGT_WRITE_REPLACE: INIT_LLIST_HEAD(&cbsp->u.write_replace.cell_list.list); break; case CBSP_MSGT_WRITE_REPLACE_COMPL: INIT_LLIST_HEAD(&cbsp->u.write_replace_compl.num_compl_list.list); INIT_LLIST_HEAD(&cbsp->u.write_replace_compl.cell_list.list); break; case CBSP_MSGT_WRITE_REPLACE_FAIL: INIT_LLIST_HEAD(&cbsp->u.write_replace_fail.fail_list); INIT_LLIST_HEAD(&cbsp->u.write_replace_fail.num_compl_list.list); INIT_LLIST_HEAD(&cbsp->u.write_replace_fail.cell_list.list); break; case CBSP_MSGT_KILL: INIT_LLIST_HEAD(&cbsp->u.kill.cell_list.list); break; case CBSP_MSGT_KILL_COMPL: INIT_LLIST_HEAD(&cbsp->u.kill_compl.num_compl_list.list); INIT_LLIST_HEAD(&cbsp->u.kill_compl.cell_list.list); break; case CBSP_MSGT_KILL_FAIL: INIT_LLIST_HEAD(&cbsp->u.kill_fail.fail_list); INIT_LLIST_HEAD(&cbsp->u.kill_fail.num_compl_list.list); INIT_LLIST_HEAD(&cbsp->u.kill_fail.cell_list.list); break; case CBSP_MSGT_LOAD_QUERY: INIT_LLIST_HEAD(&cbsp->u.load_query.cell_list.list); break; case CBSP_MSGT_LOAD_QUERY_COMPL: INIT_LLIST_HEAD(&cbsp->u.load_query_compl.loading_list.list); break; case CBSP_MSGT_LOAD_QUERY_FAIL: INIT_LLIST_HEAD(&cbsp->u.load_query_fail.fail_list); break; case CBSP_MSGT_MSG_STATUS_QUERY: INIT_LLIST_HEAD(&cbsp->u.msg_status_query.cell_list.list); break; case CBSP_MSGT_MSG_STATUS_QUERY_COMPL: INIT_LLIST_HEAD(&cbsp->u.msg_status_query_compl.num_compl_list.list); break; case CBSP_MSGT_MSG_STATUS_QUERY_FAIL: INIT_LLIST_HEAD(&cbsp->u.msg_status_query_fail.fail_list); INIT_LLIST_HEAD(&cbsp->u.msg_status_query_fail.num_compl_list.list); break; case CBSP_MSGT_RESET: INIT_LLIST_HEAD(&cbsp->u.reset.cell_list.list); break; case CBSP_MSGT_RESET_COMPL: INIT_LLIST_HEAD(&cbsp->u.reset_compl.cell_list.list); break; case CBSP_MSGT_RESET_FAIL: INIT_LLIST_HEAD(&cbsp->u.reset_fail.fail_list); INIT_LLIST_HEAD(&cbsp->u.reset_fail.cell_list.list); break; case CBSP_MSGT_RESTART: INIT_LLIST_HEAD(&cbsp->u.restart.cell_list.list); break; case CBSP_MSGT_FAILURE: INIT_LLIST_HEAD(&cbsp->u.failure.fail_list); break; default: break; } } /*! Dynamically allocate and initialize decoded CBSP structure. * \param[in] ctx talloc context from which to allocate * \param[in] msg_type CBSP message type for which to initialize result * \returns allocated + initialized decoded CBSP structure; NULL on talloc failure */ struct osmo_cbsp_decoded *osmo_cbsp_decoded_alloc(void *ctx, enum cbsp_msg_type msg_type) { struct osmo_cbsp_decoded *cbsp = talloc_zero(ctx, struct osmo_cbsp_decoded); if (!cbsp) return NULL; osmo_cbsp_init_struct(cbsp, msg_type); return cbsp; } /*********************************************************************** * Message Reception ***********************************************************************/ #ifdef HAVE_SYS_SOCKET_H #include <sys/socket.h> /*! Read one CBSP message from socket fd or store part if still not fully received. * \param[in] ctx talloc context from which to allocate new msgb. * \param[in] fd The fd for the socket to read from. * \param[out] rmsg internally allocated msgb containing a fully received CBSP message. * \param[inout] tmp_msg internally allocated msgb caching data for not yet fully received message. * * Function is designed just like ipa_msg_recv_buffered() */ int osmo_cbsp_recv_buffered(void *ctx, int fd, struct msgb **rmsg, struct msgb **tmp_msg) { struct msgb *msg = tmp_msg ? *tmp_msg : NULL; struct cbsp_header *h; int len, rc; int needed; if (!msg) { msg = osmo_cbsp_msgb_alloc(ctx, __func__); if (!msg) return -ENOMEM; msg->l1h = msg->tail; } if (msg->l2h == NULL) { /* first read the [missing part of the] header */ needed = sizeof(*h) - msg->len; rc = recv(fd, msg->tail, needed, 0); if (rc == 0) goto discard_msg; else if (rc < 0) { if (errno == EAGAIN || errno == EINTR) rc = 0; else { rc = -errno; goto discard_msg; } } msgb_put(msg, rc); if (rc < needed) { if (msg->len == 0) { rc = -EAGAIN; goto discard_msg; } if (!tmp_msg) { rc = -EIO; goto discard_msg; } *tmp_msg = msg; return -EAGAIN; } msg->l2h = msg->tail; } h = (struct cbsp_header *) msg->data; /* then read the length as specified in the header */ len = h->len[0] << 16 | h->len[1] << 8 | h->len[2]; needed = len - msgb_l2len(msg); if (needed > 0) { if (needed > msgb_tailroom(msg)) { rc = -ENOMEM; goto discard_msg; } rc = recv(fd, msg->tail, needed, 0); if (rc == 0) goto discard_msg; else if (rc < 0) { if (errno == EAGAIN || errno == EINTR) rc = 0; else { rc = -errno; goto discard_msg; } } msgb_put(msg, rc); /* still not all of payload received? */ if (rc < needed) { if (!tmp_msg) { rc = -EIO; goto discard_msg; } *tmp_msg = msg; return -EAGAIN; } } /* else: complete message received */ rc = msgb_length(msg); if (tmp_msg) *tmp_msg = NULL; *rmsg = msg; return rc; discard_msg: if (tmp_msg) *tmp_msg = NULL; msgb_free(msg); return rc; } /*! call-back function to segment the data at message boundaries. * Returns the size of the next message. If it returns -EAGAIN or a value larger than msgb_length() (message * is incomplete), the caller (e.g. osmo_io) has to wait for more data to be read. */ int osmo_cbsp_segmentation_cb(struct msgb *msg) { const struct cbsp_header *h; int len; if (msgb_length(msg) < sizeof(*h)) return -EAGAIN; h = (const struct cbsp_header *) msg->data; msg->l1h = msg->data; msg->l2h = msg->data + sizeof(*h); /* then read the length as specified in the header */ len = h->len[0] << 16 | h->len[1] << 8 | h->len[2]; return sizeof(*h) + len; } /*! value_string[] for enum osmo_cbsp_cause. */ const struct value_string osmo_cbsp_cause_names[] = { { OSMO_CBSP_CAUSE_PARAM_NOT_RECOGNISED, "Parameter-not-recognised" }, { OSMO_CBSP_CAUSE_PARAM_VALUE_INVALID, "Parameter-value-invalid" }, { OSMO_CBSP_CAUSE_MSG_REF_NOT_IDENTIFIED, "Message-reference-not-identified" }, { OSMO_CBSP_CAUSE_CELL_ID_NOT_VALID, "Cell-identity-not-valid" }, { OSMO_CBSP_CAUSE_UNRECOGNISED_MESSAGE, "Unrecognised-message" }, { OSMO_CBSP_CAUSE_MISSING_MANDATORY_ELEMENT, "Missing-mandatory-element" }, { OSMO_CBSP_CAUSE_BSC_CAPACITY_EXCEEDED, "BSC-capacity-exceeded" }, { OSMO_CBSP_CAUSE_CELL_MEMORY_EXCEEDED, "Cell-memory-exceeded" }, { OSMO_CBSP_CAUSE_BSC_MEMORY_EXCEEDED, "BSC-memory-exceeded" }, { OSMO_CBSP_CAUSE_CELL_BROADCAST_NOT_SUPPORTED, "Cell-broadcast-not-supported" }, { OSMO_CBSP_CAUSE_CELL_BROADCAST_NOT_OPERATIONAL, "Cell-broadcast-not-operational" }, { OSMO_CBSP_CAUSE_INCOMPATIBLE_DRX_PARAM, "Incompatible-DRX-parameter:"}, { OSMO_CBSP_CAUSE_EXT_CHAN_NOT_SUPPORTED, "Extended-channel-not-supported"}, { OSMO_CBSP_CAUSE_MSG_REF_ALREADY_USED, "Message-reference-already-used"}, { OSMO_CBSP_CAUSE_UNSPECIFIED_ERROR, "Unspecified-error"}, { OSMO_CBSP_CAUSE_LAI_OR_LAC_NOT_VALID, "LAI-or-LAC-not-valid"}, { 0, NULL } }; #endif /* HAVE_SYS_SOCKET_H */