blob: cdb23bf97d294b00d9fcedbcc3141707334d025f [file] [log] [blame]
Jacob Erlbecke04e0b02015-05-06 18:30:48 +02001/* gprs_ms.h
2 *
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +01003 * Copyright (C) 2015-2020 by Sysmocom s.f.m.c. GmbH
Jacob Erlbecke04e0b02015-05-06 18:30:48 +02004 * Author: Jacob Erlbeck <jerlbeck@sysmocom.de>
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020015 */
16
17#pragma once
18
Jacob Erlbeckd4ad7312015-07-17 16:39:09 +020019struct gprs_codel;
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020020
Jacob Erlbeck489a2b32015-05-28 19:07:01 +020021#include "llc.h"
Jacob Erlbeckac89a552015-06-29 14:18:46 +020022#include "tbf.h"
Pau Espin Pedrol442198c2020-10-23 22:30:04 +020023#include "tbf_ul.h"
Pau Espin Pedrol9d1cdb12019-09-25 17:47:02 +020024#include "tbf_dl.h"
Jacob Erlbecke4bcb622015-06-08 11:26:38 +020025#include "pcu_l1_if.h"
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020026
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010027#ifdef __cplusplus
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020028extern "C" {
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010029#endif
Pau Espin Pedrol2ae83372020-05-18 11:35:35 +020030
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010031#include <osmocom/core/timer.h>
32#include <osmocom/core/linuxlist.h>
Pau Espin Pedrolbed48cc2021-01-11 17:32:18 +010033#include <osmocom/core/rate_ctr.h>
Pau Espin Pedrol403e0482023-04-17 20:28:10 +020034#include <osmocom/core/use_count.h>
Pau Espin Pedrol43f0bce2020-06-26 13:09:44 +020035
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010036#include <osmocom/gsm/protocol/gsm_23_003.h>
37#include <osmocom/gsm/gsm48.h>
38
39#include "coding_scheme.h"
Pau Espin Pedrolc0a250d2021-01-21 18:46:13 +010040#include <gsm_rlcmac.h>
Jacob Erlbeckd9e10242015-05-28 15:43:53 +020041
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020042#include <stdint.h>
43#include <stddef.h>
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +020044#include <inttypes.h>
Jacob Erlbecke04e0b02015-05-06 18:30:48 +020045
Pau Espin Pedrolbed48cc2021-01-11 17:32:18 +010046enum ms_counter_id {
47 MS_CTR_DL_CTRL_MSG_SCHED,
48};
49
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010050struct gprs_rlcmac_bts;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +020051struct gprs_rlcmac_trx;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010052struct GprsMs;
Jacob Erlbeck17214bb2015-06-02 14:06:12 +020053
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010054struct GprsMs {
55 struct llist_head list; /* list of all GprsMs */
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010056 bool app_info_pending;
57
Pau Espin Pedrol2182e622021-01-14 16:48:38 +010058 struct gprs_rlcmac_bts *bts;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010059 struct gprs_rlcmac_ul_tbf *ul_tbf;
60 struct gprs_rlcmac_dl_tbf *dl_tbf;
61 struct llist_head old_tbfs; /* list of gprs_rlcmac_tbf */
Pau Espin Pedrol9935d0d2022-12-13 18:29:25 +010062 /* First common timeslot used both in UL and DL, or NULL if not set: */
63 struct gprs_rlcmac_pdch *first_common_ts;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010064
65 uint32_t tlli;
66 uint32_t new_ul_tlli;
67 uint32_t new_dl_tlli;
68
69 /* store IMSI for look-up and PCH retransmission */
70 char imsi[OSMO_IMSI_BUF_SIZE];
71 uint8_t ta;
72 uint8_t ms_class;
73 uint8_t egprs_ms_class;
74 /* current coding scheme */
75 enum CodingScheme current_cs_ul;
76 enum CodingScheme current_cs_dl;
77
78 struct gprs_llc_queue llc_queue;
Pau Espin Pedrol14beef62022-10-26 19:44:07 +020079 struct osmo_timer_list llc_timer;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010080
Pau Espin Pedrol403e0482023-04-17 20:28:10 +020081 struct osmo_use_count use_count;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010082 struct osmo_timer_list timer;
83 unsigned delay;
84
85 int64_t last_cs_not_low;
86
87 struct pcu_l1_meas l1_meas;
88 unsigned nack_rate_dl;
89 uint8_t reserved_dl_slots;
90 uint8_t reserved_ul_slots;
91 struct gprs_rlcmac_trx *current_trx;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010092 enum mcs_kind mode;
93
Pau Espin Pedrolbed48cc2021-01-11 17:32:18 +010094 struct rate_ctr_group *ctrs;
Pau Espin Pedrolc0a250d2021-01-21 18:46:13 +010095 struct nacc_fsm_ctx *nacc;
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010096};
97
Pau Espin Pedrolf80cc5b2023-04-17 14:25:51 +020098struct GprsMs *ms_alloc(struct gprs_rlcmac_bts *bts);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +010099
Pau Espin Pedrol9935d0d2022-12-13 18:29:25 +0100100struct gprs_rlcmac_pdch *ms_first_common_ts(const struct GprsMs *ms);
101void ms_set_first_common_ts(struct GprsMs *ms, struct gprs_rlcmac_pdch *pdch);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100102void ms_set_reserved_slots(struct GprsMs *ms, struct gprs_rlcmac_trx *trx,
103 uint8_t ul_slots, uint8_t dl_slots);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100104void ms_set_mode(struct GprsMs *ms, enum mcs_kind mode);
105void ms_set_ms_class(struct GprsMs *ms, uint8_t ms_class_);
106void ms_set_egprs_ms_class(struct GprsMs *ms, uint8_t ms_class_);
107void ms_set_ta(struct GprsMs *ms, uint8_t ta_);
108
Pau Espin Pedrolfc464932021-01-25 12:05:32 +0100109enum CodingScheme ms_current_cs_dl(const struct GprsMs *ms, enum mcs_kind req_mcs_kind);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100110enum CodingScheme ms_max_cs_ul(const struct GprsMs *ms);
111enum CodingScheme ms_max_cs_dl(const struct GprsMs *ms);
112void ms_set_current_cs_dl(struct GprsMs *ms, enum CodingScheme scheme);
113
114void ms_update_error_rate(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf, int error_rate);
115uint8_t ms_current_pacch_slots(const struct GprsMs *ms);
116
Pau Espin Pedrol8abe13c2022-10-21 18:49:48 +0200117void ms_update_announced_tlli(struct GprsMs *ms, uint32_t tlli);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100118void ms_merge_and_clear_ms(struct GprsMs *ms, struct GprsMs *old_ms);
119
120void ms_attach_tbf(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf);
121void ms_detach_tbf(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf);
122
123void ms_set_tlli(struct GprsMs *ms, uint32_t tlli);
124bool ms_confirm_tlli(struct GprsMs *ms, uint32_t tlli);
125void ms_set_imsi(struct GprsMs *ms, const char *imsi);
126
127void ms_update_l1_meas(struct GprsMs *ms, const struct pcu_l1_meas *meas);
128
129struct gprs_rlcmac_tbf *ms_tbf(const struct GprsMs *ms, enum gprs_rlcmac_tbf_direction dir);
130static inline struct gprs_rlcmac_ul_tbf *ms_ul_tbf(const struct GprsMs *ms) {return ms->ul_tbf;}
131static inline struct gprs_rlcmac_dl_tbf *ms_dl_tbf(const struct GprsMs *ms) {return ms->dl_tbf;}
132
Pau Espin Pedrol3547d642022-10-21 15:00:08 +0200133const char *ms_name(const struct GprsMs *ms);
134char *ms_name_buf(const struct GprsMs *ms, char *buf, unsigned int buf_size);
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100135
Pau Espin Pedrolc0a250d2021-01-21 18:46:13 +0100136int ms_nacc_start(struct GprsMs *ms, Packet_Cell_Change_Notification_t *notif);
137bool ms_nacc_rts(const struct GprsMs *ms);
Pau Espin Pedrol5ba3ef92022-12-12 18:02:25 +0100138struct msgb *ms_nacc_create_rlcmac_msg(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf,
139 const struct gprs_rlcmac_pdch *pdch, uint32_t fn);
Pau Espin Pedrolc0a250d2021-01-21 18:46:13 +0100140
Pau Espin Pedrol94386132022-10-28 19:50:09 +0200141struct gprs_rlcmac_ul_tbf *ms_new_ul_tbf_assigned_pacch(struct GprsMs *ms, int8_t use_trx);
142struct gprs_rlcmac_ul_tbf *ms_new_ul_tbf_assigned_agch(struct GprsMs *ms);
Pau Espin Pedrola621d592022-12-13 17:35:04 +0100143struct gprs_rlcmac_ul_tbf *ms_new_ul_tbf_rejected_pacch(struct GprsMs *ms, struct gprs_rlcmac_pdch *pdch);
Pau Espin Pedrol091642a2022-10-28 18:01:31 +0200144int ms_new_dl_tbf_assigned_on_pacch(struct GprsMs *ms, struct gprs_rlcmac_tbf *tbf);
145int ms_new_dl_tbf_assigned_on_pch(struct GprsMs *ms);
Pau Espin Pedrol14beef62022-10-26 19:44:07 +0200146int ms_append_llc_dl_data(struct GprsMs *ms, uint16_t pdu_delay_csec, const uint8_t *data, uint16_t len);
147
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100148static inline bool ms_is_idle(const struct GprsMs *ms)
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200149{
Pau Espin Pedrol403e0482023-04-17 20:28:10 +0200150 return !ms->ul_tbf && !ms->dl_tbf &&
Pau Espin Pedrolac4d4a62023-04-18 19:02:55 +0200151 llist_empty(&ms->old_tbfs);
Jacob Erlbeck6835cea2015-08-21 15:24:02 +0200152}
153
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100154static inline struct gprs_llc_queue *ms_llc_queue(struct GprsMs *ms)
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200155{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100156 return &ms->llc_queue;
157}
158
159static inline bool ms_need_dl_tbf(struct GprsMs *ms)
160{
161 if (ms_dl_tbf(ms) != NULL &&
Pau Espin Pedroldc2aaac2021-05-14 12:50:46 +0200162 tbf_state((const struct gprs_rlcmac_tbf *)ms_dl_tbf(ms)) != TBF_ST_WAIT_RELEASE)
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200163 return false;
164
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100165 return llc_queue_size(ms_llc_queue(ms)) > 0;
Jacob Erlbeckc8cbfc22015-09-01 11:38:40 +0200166}
167
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100168static inline uint32_t ms_tlli(const struct GprsMs *ms)
Jacob Erlbeck93990462015-05-15 15:50:43 +0200169{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100170 if (ms->new_ul_tlli != GSM_RESERVED_TMSI)
171 return ms->new_ul_tlli;
172 if (ms->tlli != GSM_RESERVED_TMSI)
173 return ms->tlli;
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700174
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100175 return ms->new_dl_tlli;
Jacob Erlbeck93990462015-05-15 15:50:43 +0200176}
177
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100178static inline bool ms_check_tlli(struct GprsMs *ms, uint32_t tlli)
Jacob Erlbeck93990462015-05-15 15:50:43 +0200179{
Vadim Yanitskiycb988942020-11-08 13:27:35 +0700180 return tlli != GSM_RESERVED_TMSI &&
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100181 (tlli == ms->tlli || tlli == ms->new_ul_tlli || tlli == ms->new_dl_tlli);
Jacob Erlbeck93990462015-05-15 15:50:43 +0200182}
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200183
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100184static inline const char *ms_imsi(const struct GprsMs *ms)
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200185{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100186 return ms->imsi;
Jacob Erlbeckb0e5eaf2015-05-21 11:07:16 +0200187}
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200188
Pau Espin Pedrol5deac142021-11-12 18:01:50 +0100189static inline bool ms_imsi_is_valid(const struct GprsMs *ms)
190{
191 return ms->imsi[0] != '\0';
192}
193
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100194static inline uint8_t ms_ta(const struct GprsMs *ms)
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200195{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100196 return ms->ta;
Jacob Erlbeck9200ce62015-05-22 17:48:04 +0200197}
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200198
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100199static inline uint8_t ms_ms_class(const struct GprsMs *ms)
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200200{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100201 return ms->ms_class;
Jacob Erlbeckbefc7602015-06-02 12:33:30 +0200202}
203
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100204static inline uint8_t ms_egprs_ms_class(const struct GprsMs *ms)
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200205{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100206 return ms->egprs_ms_class;
Jacob Erlbeckc3c58042015-09-28 17:55:32 +0200207}
208
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100209static inline enum CodingScheme ms_current_cs_ul(const struct GprsMs *ms)
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200210{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100211 return ms->current_cs_ul;
Jacob Erlbecka700dd92015-06-02 16:00:41 +0200212}
213
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100214static inline enum mcs_kind ms_mode(const struct GprsMs *ms)
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100215{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100216 return ms->mode;
Jacob Erlbeckcb728902016-01-05 15:33:03 +0100217}
218
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100219static inline void ms_set_timeout(struct GprsMs *ms, unsigned secs)
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200220{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100221 ms->delay = secs;
Jacob Erlbeckd9e10242015-05-28 15:43:53 +0200222}
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200223
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100224static inline unsigned ms_nack_rate_dl(const struct GprsMs *ms)
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200225{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100226 return ms->nack_rate_dl;
Jacob Erlbeck489a2b32015-05-28 19:07:01 +0200227}
228
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100229static inline uint8_t ms_reserved_dl_slots(const struct GprsMs *ms)
sivasankarida7250a2016-12-16 12:57:18 +0530230{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100231 return ms->reserved_dl_slots;
sivasankarida7250a2016-12-16 12:57:18 +0530232}
233
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100234static inline uint8_t ms_reserved_ul_slots(const struct GprsMs *ms)
sivasankarida7250a2016-12-16 12:57:18 +0530235{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100236 return ms->reserved_ul_slots;
sivasankarida7250a2016-12-16 12:57:18 +0530237}
238
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100239static inline struct gprs_rlcmac_trx *ms_current_trx(const struct GprsMs *ms)
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200240{
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100241 return ms->current_trx;
Jacob Erlbeck23f93a12015-06-30 08:52:54 +0200242}
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200243
Pau Espin Pedrolac4d4a62023-04-18 19:02:55 +0200244#define MS_USE_TBF "tbf"
Pau Espin Pedrol403e0482023-04-17 20:28:10 +0200245#define ms_ref(ms, use) \
246 OSMO_ASSERT(osmo_use_count_get_put(&(ms)->use_count, use, 1) == 0)
247#define ms_unref(ms, use) \
248 OSMO_ASSERT(osmo_use_count_get_put(&(ms)->use_count, use, -1) == 0)
249
Pau Espin Pedrolf2dad592020-08-18 20:26:25 +0200250#define LOGPMS(ms, category, level, fmt, args...) \
Pau Espin Pedrol3547d642022-10-21 15:00:08 +0200251 LOGP(category, level, "%s " fmt, ms_name(ms), ## args)
Pau Espin Pedrolda971ee2020-12-16 15:59:45 +0100252
253#ifdef __cplusplus
254}
255#endif