/* (C) 2010 by Dieter Spaar * (C) 2010,2017 by Harald Welte * (C) 2016 by Sebastian Stumpf * * 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 #include #include #include #include #include #include #include #include #include #include #include static uint16_t sync_count = 0; /** * @brief Handler for received L1CTL_FBSB_REQ from L23. * * -- frequency burst synchronisation burst request -- * * @param [in] msg the received message. * * Transmit frequency control and synchronisation bursts on FCCH and SCH to calibrate transceiver and search for base stations. * Sync to a given arfcn. * * Note: ms will start receiving msgs on virtual um only after this req was received. * Note: virt bts does not broadcast freq and sync bursts. * */ void l1ctl_rx_fbsb_req(struct l1_model_ms *ms, struct msgb *msg) { struct l1_state_ms *l1s = &ms->state; struct l1ctl_hdr *l1h = (struct l1ctl_hdr *) msg->data; struct l1ctl_fbsb_req *sync_req = (struct l1ctl_fbsb_req *) l1h->data; LOGPMS(DL1C, LOGL_INFO, ms, "Rx L1CTL_FBSB_REQ (arfcn=%u, flags=0x%x)\n", ntohs(sync_req->band_arfcn), sync_req->flags); l1s->state = MS_STATE_IDLE_SYNCING; l1s->fbsb.arfcn = ntohs(sync_req->band_arfcn); } /** * @brief A msg was received on l1 that can be used for synchronization. * * Note: for virtual layer 1 this can be a random downlink message, as we can parse the fn from the gsmtap header. */ void prim_fbsb_sync(struct l1_model_ms *ms, struct msgb *msg) { struct l1_state_ms *l1s = &ms->state; struct gsmtap_hdr *gh = msgb_l1(msg); uint32_t fn = ntohl(gh->frame_number); /* frame number of the rcv msg */ uint16_t arfcn = ntohs(gh->arfcn); /* arfcn of the received msg */ /* ignore messages from other arfcns as the one requested to sync to by l23 */ if (l1s->fbsb.arfcn != arfcn) { /* cancel sync if we did not receive a msg on dl from * the requested arfcn that we can sync to */ if (sync_count++ > 20) { sync_count = 0; l1s->state = MS_STATE_IDLE_SEARCHING; l1ctl_tx_fbsb_conf(ms, 1, (l1s->fbsb.arfcn)); } return; } l1s->serving_cell.arfcn = arfcn; l1s->state = MS_STATE_IDLE_CAMPING; /* Not needed in virtual phy */ l1s->serving_cell.fn_offset = 0; l1s->serving_cell.time_alignment = 0; l1s->serving_cell.bsic = 0; /* Update current gsm time each time we receive a message on the virt um */ gsm_fn2gsmtime(&l1s->downlink_time, fn); /* Restart scheduler */ virt_l1_sched_restart(ms, l1s->downlink_time); l1ctl_tx_fbsb_conf(ms, 0, arfcn); } /** * @brief Transmit L1CTL_FBSB_CONF to l23. * * -- frequency burst synchronisation burst confirm -- * * @param [in] res 0 -> success, 255 -> error. * @param [in] arfcn the arfcn we are synced to. * * No calculation needed for virtual pyh -> uses dummy values for a good link quality. */ void l1ctl_tx_fbsb_conf(struct l1_model_ms *ms, uint8_t res, uint16_t arfcn) { struct msgb *msg; struct l1ctl_fbsb_conf *resp; uint32_t fn = 0; /* 0 should be okay here */ uint16_t snr = 40; /* signal noise ratio > 40db is best signal (unused in virt)*/ int16_t initial_freq_err = 0; /* 0 means no error (unused in virt) */ uint8_t bsic = 0; /* BSIC can be read from sync burst (unused in virt) */ msg = l1ctl_create_l2_msg(L1CTL_FBSB_CONF, fn, snr, arfcn); resp = (struct l1ctl_fbsb_conf *)msgb_put(msg, sizeof(*resp)); resp->initial_freq_err = htons(initial_freq_err); resp->result = res; resp->bsic = bsic; LOGPMS(DL1C, LOGL_INFO, ms, "Tx L1CTL_FBSB_CONF (res: %u)\n", res); l1ctl_sap_tx_to_l23_inst(ms, msg); }