blob: 0f3ef1c46aad39e43ce3bed66535740602aacca1 [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
Pau Espin Pedrol698ad4c2018-07-27 16:15:59 +020023import copy
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010024from abc import ABCMeta, abstractmethod
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020025from . import log, config, util, template, process, schema, pcu_osmo
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010026
27class Bts(log.Origin, metaclass=ABCMeta):
28 suite_run = None
29 conf = None
30 bsc = None
31 sgsn = None
32 lac = None
33 rac = None
34 cellid = None
35 bvci = None
Pau Espin Pedrole5194622018-05-07 13:36:58 +020036 defaults_cfg_name = None
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020037 _num_trx = 1
38 _max_trx = None
39 overlay_trx_list = []
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010040
41##############
42# PROTECTED
43##############
Pau Espin Pedrole5194622018-05-07 13:36:58 +020044 def __init__(self, suite_run, conf, name, defaults_cfg_name):
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +010045 super().__init__(log.C_RUN, name)
46 self.suite_run = suite_run
47 self.conf = conf
Pau Espin Pedrole5194622018-05-07 13:36:58 +020048 self.defaults_cfg_name = defaults_cfg_name
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020049 self._init_num_trx()
50
51 def _resolve_bts_cfg(self, cfg_name):
52 res = None
53 val = config.get_defaults('bsc_bts').get(cfg_name)
54 if val is not None:
55 res = val
56 val = config.get_defaults(self.defaults_cfg_name).get(cfg_name)
57 if val is not None:
58 res = val
59 val = self.conf.get(cfg_name)
60 if val is not None:
61 res = val
62 return res
63
64 def _init_num_trx(self):
65 self._num_trx = 1
66 self._max_trx = None
67 val = self._resolve_bts_cfg('num_trx')
68 if val is not None:
69 self._num_trx = int(val)
70 val = self._resolve_bts_cfg('max_trx')
71 if val is not None:
72 self._max_trx = int(val)
73 self._validate_new_num_trx(self._num_trx)
74 self.overlay_trx_list = [Bts._new_default_trx_cfg() for trx in range(self._num_trx)]
75
76 def _validate_new_num_trx(self, num_trx):
77 if self._max_trx is not None and num_trx > self._max_trx:
78 raise log.Error('Amount of TRX requested is too high for maximum allowed: %u > %u' %(num_trx, self._max_trx))
79
80 @staticmethod
81 def _new_default_trx_cfg():
82 return {'timeslot_list':[{} for ts in range(8)]}
83
84 @staticmethod
85 def _trx_list_recreate(trx_list, new_size):
86 curr_len = len(trx_list)
87 if new_size < curr_len:
88 trx_list = trx_list[0:new_size]
89 elif new_size > curr_len:
90 for i in range(new_size - curr_len):
91 trx_list.append(Bts._new_default_trx_cfg())
92 return trx_list
Pau Espin Pedrole6999122018-05-08 14:38:24 +020093
94 def conf_for_bsc_prepare(self):
95 values = config.get_defaults('bsc_bts')
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +020096 # Make sure the trx_list is adapted to num of trx configured at runtime
97 # to avoid overlay issues.
98 trx_list = values.get('trx_list')
99 if trx_list and len(trx_list) != self.num_trx():
100 values['trx_list'] = Bts._trx_list_recreate(trx_list, self.num_trx())
101
102 bts_defaults = config.get_defaults(self.defaults_cfg_name)
103 trx_list = bts_defaults.get('trx_list')
104 if trx_list and len(trx_list) != self.num_trx():
105 bts_defaults['trx_list'] = Bts._trx_list_recreate(trx_list, self.num_trx())
106
107 config.overlay(values, bts_defaults)
Pau Espin Pedrole6999122018-05-08 14:38:24 +0200108 if self.lac is not None:
109 config.overlay(values, { 'location_area_code': self.lac })
110 if self.rac is not None:
111 config.overlay(values, { 'routing_area_code': self.rac })
112 if self.cellid is not None:
113 config.overlay(values, { 'cell_identity': self.cellid })
114 if self.bvci is not None:
115 config.overlay(values, { 'bvci': self.bvci })
Pau Espin Pedrol698ad4c2018-07-27 16:15:59 +0200116
117 conf = copy.deepcopy(self.conf)
118 trx_list = conf.get('trx_list')
119 if trx_list and len(trx_list) != self.num_trx():
120 conf['trx_list'] = Bts._trx_list_recreate(trx_list, self.num_trx())
121 config.overlay(values, conf)
Pau Espin Pedrole6999122018-05-08 14:38:24 +0200122
123 sgsn_conf = {} if self.sgsn is None else self.sgsn.conf_for_client()
124 config.overlay(values, sgsn_conf)
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +0200125
126 config.overlay(values, { 'trx_list': self.overlay_trx_list })
Pau Espin Pedrole6999122018-05-08 14:38:24 +0200127 return values
128
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100129########################
130# PUBLIC - INTERNAL API
131########################
132 @abstractmethod
133 def conf_for_bsc(self):
134 'Used by bsc objects to get path to socket.'
135 pass
136
137 def remote_addr(self):
138 return self.conf.get('addr')
139
140 def cleanup(self):
141 'Nothing to do by default. Subclass can override if required.'
142 pass
143
144###################
145# PUBLIC (test API included)
146###################
147 @abstractmethod
Pau Espin Pedrolb1526b92018-05-22 20:32:30 +0200148 def start(self, keepalive=False):
149 '''Starts BTS. If keepalive is set, it will expect internal issues and
150 respawn related processes when detected'''
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100151 pass
152
153 @abstractmethod
154 def ready_for_pcu(self):
155 'True if the BTS is prepared to have a PCU connected, false otherwise'
156 pass
157
158 @abstractmethod
159 def pcu(self):
160 'Get the Pcu object associated with the BTS'
161 pass
162
163 def set_bsc(self, bsc):
164 self.bsc = bsc
165
166 def set_sgsn(self, sgsn):
167 self.sgsn = sgsn
168
169 def set_lac(self, lac):
170 self.lac = lac
171
172 def set_rac(self, rac):
173 self.rac = rac
174
175 def set_cellid(self, cellid):
176 self.cellid = cellid
177
178 def set_bvci(self, bvci):
179 self.bvci = bvci
180
Pau Espin Pedrol39df7f42018-05-07 13:49:33 +0200181 def set_num_trx(self, num_trx):
182 assert num_trx > 0
183 self._validate_new_num_trx(num_trx)
184 if num_trx == self._num_trx:
185 return
186 self._num_trx = num_trx
187 self.overlay_trx_list = Bts._trx_list_recreate(self.overlay_trx_list, num_trx)
188
189 def num_trx(self):
190 return self._num_trx
191
192 def set_trx_phy_channel(self, trx_idx, ts_idx, config):
193 assert trx_idx < self._num_trx
194 assert ts_idx < 8
195 schema.phy_channel_config(config) # validation
196 self.overlay_trx_list[trx_idx]['timeslot_list'][ts_idx]['phys_chan_config'] = config
197
Pau Espin Pedrol52ad3a62018-03-08 17:50:14 +0100198# vim: expandtab tabstop=4 shiftwidth=4