/* (C) 2015 by Harald Welte * * 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 Affero 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 . * */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "virtual_um.h" /* TODO: check if dummy method is sufficient, else implement */ int bts_model_lchan_deactivate(struct gsm_lchan *lchan) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return -1; } /* TODO: check if dummy method is sufficient, else implement */ int osmo_amr_rtp_dec(const uint8_t *rtppayload, int payload_len, uint8_t *cmr, int8_t *cmi, enum osmo_amr_type *ft, enum osmo_amr_quality *bfi, int8_t *sti) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return -1; } void bts_model_trx_close(struct gsm_bts_trx *trx) { struct phy_instance *pinst = trx_phy_instance(trx); struct phy_link *plink = pinst->phy_link; if (phy_link_state_get(plink) != PHY_LINK_SHUTDOWN) { virt_um_destroy(plink->u.virt.virt_um); plink->u.virt.virt_um = NULL; phy_link_state_set(plink, PHY_LINK_SHUTDOWN); } bts_model_trx_close_cb(trx, 0); } int bts_model_adjst_ms_pwr(struct gsm_lchan *lchan) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return 0; } int bts_model_check_oml(struct gsm_bts *bts, uint8_t msg_type, struct tlv_parsed *old_attr, struct tlv_parsed *new_attr, void *obj) { return 0; } static uint8_t vbts_set_bts(struct gsm_bts *bts) { struct gsm_bts_trx *trx; llist_for_each_entry(trx, &bts->trx_list, list) { /* report availability of trx to the bts. this will trigger the rsl connection */ osmo_fsm_inst_dispatch(trx->mo.fi, NM_EV_SW_ACT, NULL); osmo_fsm_inst_dispatch(trx->bb_transc.mo.fi, NM_EV_SW_ACT, NULL); } return 0; } static uint8_t vbts_set_trx(struct gsm_bts_trx *trx) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return 0; } static uint8_t vbts_set_ts(struct gsm_bts_trx_ts *ts) { enum gsm_phys_chan_config pchan; /* For dynamic timeslots, pick the pchan type that should currently be * active. This should only be called during init, PDCH transitions * will call trx_set_ts_as_pchan() directly. */ switch (ts->pchan) { case GSM_PCHAN_TCH_F_PDCH: OSMO_ASSERT((ts->flags & TS_F_PDCH_PENDING_MASK) == 0); if (ts->flags & TS_F_PDCH_ACTIVE) pchan = GSM_PCHAN_PDCH; else pchan = GSM_PCHAN_TCH_F; break; case GSM_PCHAN_OSMO_DYN: OSMO_ASSERT(ts->dyn.pchan_is == ts->dyn.pchan_want); pchan = ts->dyn.pchan_is; break; default: pchan = ts->pchan; break; } if (trx_sched_set_pchan(ts, pchan) != 0) return NM_NACK_RES_NOTAVAIL; /* activate lchans for [CBCH/]BCCH/CCCH */ switch (pchan) { case GSM_PCHAN_SDCCH8_SACCH8C_CBCH: /* using RSL_CHAN_OSMO_CBCH4 is correct here, because the scheduler * does not distinguish between SDCCH/4+CBCH abd SDCCH/8+CBCH. */ trx_sched_set_lchan(&ts->lchan[CBCH_LCHAN], RSL_CHAN_OSMO_CBCH4, LID_DEDIC, true); break; case GSM_PCHAN_CCCH_SDCCH4_CBCH: trx_sched_set_lchan(&ts->lchan[CBCH_LCHAN], RSL_CHAN_OSMO_CBCH4, LID_DEDIC, true); /* fall-through */ case GSM_PCHAN_CCCH_SDCCH4: case GSM_PCHAN_CCCH: trx_sched_set_bcch_ccch(&ts->lchan[CCCH_LCHAN], true); ts->lchan[CCCH_LCHAN].rel_act_kind = LCHAN_REL_ACT_OML; lchan_set_state(&ts->lchan[CCCH_LCHAN], LCHAN_S_ACTIVE); break; default: break; } return 0; } int bts_model_apply_oml(struct gsm_bts *bts, const struct msgb *msg, struct gsm_abis_mo *mo, void *obj) { struct abis_om_fom_hdr *foh = msgb_l3(msg); int rc; switch (foh->msg_type) { case NM_MT_SET_BTS_ATTR: rc = vbts_set_bts(obj); break; case NM_MT_SET_RADIO_ATTR: rc = vbts_set_trx(obj); break; case NM_MT_SET_CHAN_ATTR: rc = vbts_set_ts(obj); break; default: rc = 0; break; } return rc; } /* MO: TS 12.21 Managed Object */ int bts_model_opstart(struct gsm_bts *bts, struct gsm_abis_mo *mo, void *obj) { int rc; switch (mo->obj_class) { case NM_OC_SITE_MANAGER: case NM_OC_BTS: case NM_OC_RADIO_CARRIER: case NM_OC_BASEB_TRANSC: case NM_OC_CHANNEL: case NM_OC_GPRS_NSE: case NM_OC_GPRS_CELL: case NM_OC_GPRS_NSVC: rc = osmo_fsm_inst_dispatch(mo->fi, NM_EV_OPSTART_ACK, NULL); break; default: rc = oml_mo_opstart_nack(mo, NM_NACK_OBJCLASS_NOTSUPP); } return rc; } int bts_model_chg_adm_state(struct gsm_bts *bts, struct gsm_abis_mo *mo, void *obj, uint8_t adm_state) { mo->nm_state.administrative = adm_state; return oml_mo_statechg_ack(mo); } int bts_model_trx_deact_rf(struct gsm_bts_trx *trx) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return 0; } int bts_model_change_power(struct gsm_bts_trx *trx, int p_trxout_mdBm) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); power_trx_change_compl(trx, p_trxout_mdBm); return 0; } int bts_model_ctrl_cmds_install(struct gsm_bts *bts) { LOGP(DL1C, LOGL_NOTICE, "Unimplemented %s\n", __func__); return 0; }