blob: ca33eb437595bdf5d830f6351932ee755cf574fa [file] [log] [blame]
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +01001# osmo_gsm_tester: base classes to share code among BTS subclasses.
2#
3# Copyright (C) 2018 by sysmocom - s.f.m.c. GmbH
4#
5# Author: Pau Espin Pedrol <pespin@sysmocom.de>
6#
7# This program is free software: you can redistribute it and/or modify
8# it under the terms of the GNU General Public License as
9# published by the Free Software Foundation, either version 3 of the
10# License, or (at your option) any later version.
11#
12# This program is distributed in the hope that it will be useful,
13# but WITHOUT ANY WARRANTY; without even the implied warranty of
14# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15# GNU General Public License for more details.
16#
17# You should have received a copy of the GNU General Public License
18# along with this program. If not, see <http://www.gnu.org/licenses/>.
19
20import os
21import pprint
22import tempfile
23from abc import ABCMeta, abstractmethod
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020024from . import log, config, util, template, process, schema, pcu_osmo
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010025
26class Bts(log.Origin, metaclass=ABCMeta):
27 suite_run = None
28 conf = None
29 bsc = None
30 sgsn = None
31 lac = None
32 rac = None
33 cellid = None
34 bvci = None
Pau Espin Pedrole5194622018-05-07 13:36:58 +020035 defaults_cfg_name = None
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020036 _num_trx = 1
37 _max_trx = None
38 overlay_trx_list = []
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010039
40##############
41# PROTECTED
42##############
Pau Espin Pedrole5194622018-05-07 13:36:58 +020043 def __init__(self, suite_run, conf, name, defaults_cfg_name):
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010044 super().__init__(log.C_RUN, name)
45 self.suite_run = suite_run
46 self.conf = conf
Pau Espin Pedrole5194622018-05-07 13:36:58 +020047 self.defaults_cfg_name = defaults_cfg_name
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020048 self._init_num_trx()
49
50 def _resolve_bts_cfg(self, cfg_name):
51 res = None
52 val = config.get_defaults('bsc_bts').get(cfg_name)
53 if val is not None:
54 res = val
55 val = config.get_defaults(self.defaults_cfg_name).get(cfg_name)
56 if val is not None:
57 res = val
58 val = self.conf.get(cfg_name)
59 if val is not None:
60 res = val
61 return res
62
63 def _init_num_trx(self):
64 self._num_trx = 1
65 self._max_trx = None
66 val = self._resolve_bts_cfg('num_trx')
67 if val is not None:
68 self._num_trx = int(val)
69 val = self._resolve_bts_cfg('max_trx')
70 if val is not None:
71 self._max_trx = int(val)
72 self._validate_new_num_trx(self._num_trx)
73 self.overlay_trx_list = [Bts._new_default_trx_cfg() for trx in range(self._num_trx)]
74
75 def _validate_new_num_trx(self, num_trx):
76 if self._max_trx is not None and num_trx > self._max_trx:
77 raise log.Error('Amount of TRX requested is too high for maximum allowed: %u > %u' %(num_trx, self._max_trx))
78
79 @staticmethod
80 def _new_default_trx_cfg():
81 return {'timeslot_list':[{} for ts in range(8)]}
82
83 @staticmethod
84 def _trx_list_recreate(trx_list, new_size):
85 curr_len = len(trx_list)
86 if new_size < curr_len:
87 trx_list = trx_list[0:new_size]
88 elif new_size > curr_len:
89 for i in range(new_size - curr_len):
90 trx_list.append(Bts._new_default_trx_cfg())
91 return trx_list
Pau Espin Pedrole6999122018-05-08 14:38:24 +020092
93 def conf_for_bsc_prepare(self):
94 values = config.get_defaults('bsc_bts')
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020095 # Make sure the trx_list is adapted to num of trx configured at runtime
96 # to avoid overlay issues.
97 trx_list = values.get('trx_list')
98 if trx_list and len(trx_list) != self.num_trx():
99 values['trx_list'] = Bts._trx_list_recreate(trx_list, self.num_trx())
100
101 bts_defaults = config.get_defaults(self.defaults_cfg_name)
102 trx_list = bts_defaults.get('trx_list')
103 if trx_list and len(trx_list) != self.num_trx():
104 bts_defaults['trx_list'] = Bts._trx_list_recreate(trx_list, self.num_trx())
105
106 config.overlay(values, bts_defaults)
Pau Espin Pedrole6999122018-05-08 14:38:24 +0200107 if self.lac is not None:
108 config.overlay(values, { 'location_area_code': self.lac })
109 if self.rac is not None:
110 config.overlay(values, { 'routing_area_code': self.rac })
111 if self.cellid is not None:
112 config.overlay(values, { 'cell_identity': self.cellid })
113 if self.bvci is not None:
114 config.overlay(values, { 'bvci': self.bvci })
115 config.overlay(values, self.conf)
116
117 sgsn_conf = {} if self.sgsn is None else self.sgsn.conf_for_client()
118 config.overlay(values, sgsn_conf)
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +0200119
120 config.overlay(values, { 'trx_list': self.overlay_trx_list })
Pau Espin Pedrole6999122018-05-08 14:38:24 +0200121 return values
122
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100123########################
124# PUBLIC - INTERNAL API
125########################
126 @abstractmethod
127 def conf_for_bsc(self):
128 'Used by bsc objects to get path to socket.'
129 pass
130
131 def remote_addr(self):
132 return self.conf.get('addr')
133
134 def cleanup(self):
135 'Nothing to do by default. Subclass can override if required.'
136 pass
137
138###################
139# PUBLIC (test API included)
140###################
141 @abstractmethod
Pau Espin Pedrolb1526b92018-05-22 20:32:30 +0200142 def start(self, keepalive=False):
143 '''Starts BTS. If keepalive is set, it will expect internal issues and
144 respawn related processes when detected'''
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100145 pass
146
147 @abstractmethod
148 def ready_for_pcu(self):
149 'True if the BTS is prepared to have a PCU connected, false otherwise'
150 pass
151
152 @abstractmethod
153 def pcu(self):
154 'Get the Pcu object associated with the BTS'
155 pass
156
157 def set_bsc(self, bsc):
158 self.bsc = bsc
159
160 def set_sgsn(self, sgsn):
161 self.sgsn = sgsn
162
163 def set_lac(self, lac):
164 self.lac = lac
165
166 def set_rac(self, rac):
167 self.rac = rac
168
169 def set_cellid(self, cellid):
170 self.cellid = cellid
171
172 def set_bvci(self, bvci):
173 self.bvci = bvci
174
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +0200175 def set_num_trx(self, num_trx):
176 assert num_trx > 0
177 self._validate_new_num_trx(num_trx)
178 if num_trx == self._num_trx:
179 return
180 self._num_trx = num_trx
181 self.overlay_trx_list = Bts._trx_list_recreate(self.overlay_trx_list, num_trx)
182
183 def num_trx(self):
184 return self._num_trx
185
186 def set_trx_phy_channel(self, trx_idx, ts_idx, config):
187 assert trx_idx < self._num_trx
188 assert ts_idx < 8
189 schema.phy_channel_config(config) # validation
190 self.overlay_trx_list[trx_idx]['timeslot_list'][ts_idx]['phys_chan_config'] = config
191
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100192# vim: expandtab tabstop=4 shiftwidth=4