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