1#!/usr/bin/env python3
2#
3#   Copyright 2019 - The Android Open Source Project
4#
5#   Licensed under the Apache License, Version 2.0 (the 'License');
6#   you may not use this file except in compliance with the License.
7#   You may obtain a copy of the License at
8#
9#       http://www.apache.org/licenses/LICENSE-2.0
10#
11#   Unless required by applicable law or agreed to in writing, software
12#   distributed under the License is distributed on an 'AS IS' BASIS,
13#   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14#   See the License for the specific language governing permissions and
15#   limitations under the License.
16from acts import logger
17from acts.controllers import cellular_lib as sims
18
19
20class AbstractCellularSimulator:
21    """ A generic cellular simulator controller class that can be derived to
22    implement equipment specific classes and allows the tests to be implemented
23    without depending on a singular instrument model.
24
25    This class defines the interface that every cellular simulator controller
26    needs to implement and shouldn't be instantiated by itself. """
27
28    # Indicates if it is able to use 256 QAM as the downlink modulation for LTE
29    LTE_SUPPORTS_DL_256QAM = None
30
31    # Indicates if it is able to use 64 QAM as the uplink modulation for LTE
32    LTE_SUPPORTS_UL_64QAM = None
33
34    # Indicates if 4x4 MIMO is supported for LTE
35    LTE_SUPPORTS_4X4_MIMO = None
36
37    # The maximum number of carriers that this simulator can support for LTE
38    LTE_MAX_CARRIERS = None
39
40    # The maximum power that the equipment is able to transmit
41    MAX_DL_POWER = None
42
43    def __init__(self):
44        """ Initializes the cellular simulator. """
45        self.log = logger.create_tagged_trace_logger('CellularSimulator')
46
47    def destroy(self):
48        """ Sends finalization commands to the cellular equipment and closes
49        the connection. """
50        raise NotImplementedError()
51
52    def setup_lte_scenario(self):
53        """ Configures the equipment for an LTE simulation. """
54        raise NotImplementedError()
55
56    def setup_lte_ca_scenario(self):
57        """ Configures the equipment for an LTE with CA simulation. """
58        raise NotImplementedError()
59
60    def set_ca_combination(self, combination):
61        """ Prepares the test equipment for the indicated CA combination.
62
63        The reason why this is implemented in a separate method and not calling
64        LteSimulation.BtsConfig for each separate band is that configuring each
65        ssc cannot be done separately, as it is necessary to know which
66        carriers are on the same band in order to decide which RF outputs can
67        be shared in the test equipment.
68
69        Args:
70            combination: carrier aggregation configurations are indicated
71                with a list of strings consisting of the band number followed
72                by the CA class. For example, for 5 CA using 3C 7C and 28A
73                the parameter value should be [3c, 7c, 28a].
74        """
75        raise NotImplementedError()
76
77    def configure_bts(self, config, bts_index=0):
78        """ Commands the equipment to setup a base station with the required
79        configuration. This method applies configurations that are common to all
80        RATs.
81
82        Args:
83            config: a BaseSimulation.BtsConfig object.
84            bts_index: the base station number.
85        """
86
87        if config.output_power:
88            self.set_output_power(bts_index, config.output_power)
89
90        if config.input_power:
91            self.set_input_power(bts_index, config.input_power)
92
93        if isinstance(config, sims.LteSimulation.LteSimulation.BtsConfig):
94            self.configure_lte_bts(config, bts_index)
95
96    def configure_lte_bts(self, config, bts_index=0):
97        """ Commands the equipment to setup an LTE base station with the
98        required configuration.
99
100        Args:
101            config: an LteSimulation.BtsConfig object.
102            bts_index: the base station number.
103        """
104        if config.band:
105            self.set_band(bts_index, config.band)
106
107        if config.dlul_config:
108            self.set_tdd_config(bts_index, config.dlul_config)
109
110        if config.ssf_config:
111            self.set_ssf_config(bts_index, config.ssf_config)
112
113        if config.bandwidth:
114            self.set_bandwidth(bts_index, config.bandwidth)
115
116        if config.dl_channel:
117            self.set_downlink_channel_number(bts_index, config.dl_channel)
118
119        if config.mimo_mode:
120            self.set_mimo_mode(bts_index, config.mimo_mode)
121
122        if config.transmission_mode:
123            self.set_transmission_mode(bts_index, config.transmission_mode)
124
125        # Modulation order should be set before set_scheduling_mode being
126        # called.
127        if config.dl_modulation_order:
128            self.set_dl_modulation(bts_index, config.dl_modulation_order)
129
130        if config.ul_modulation_order:
131            self.set_ul_modulation(bts_index, config.ul_modulation_order)
132
133        if config.scheduling_mode:
134
135            if (config.scheduling_mode ==
136                    sims.LteSimulation.SchedulingMode.STATIC
137                    and not (config.dl_rbs and config.ul_rbs and config.dl_mcs
138                             and config.ul_mcs)):
139                raise ValueError('When the scheduling mode is set to manual, '
140                                 'the RB and MCS parameters are required.')
141
142            # If scheduling mode is set to Dynamic, the RB and MCS parameters
143            # will be ignored by set_scheduling_mode.
144            self.set_scheduling_mode(bts_index, config.scheduling_mode,
145                                     config.dl_mcs, config.ul_mcs,
146                                     config.dl_rbs, config.ul_rbs)
147
148        # This variable stores a boolean value so the following is needed to
149        # differentiate False from None
150        if config.tbs_pattern_on is not None:
151            self.set_tbs_pattern_on(bts_index, config.tbs_pattern_on)
152
153        if config.cfi:
154            self.set_cfi(bts_index, config.cfi)
155
156        if config.paging_cycle:
157            self.set_paging_cycle(bts_index, config.paging_cycle)
158
159        if config.phich:
160            self.set_phich_resource(bts_index, config.phich)
161
162        if config.drx_connected_mode:
163            self.set_drx_connected_mode(bts_index, config.drx_connected_mode)
164
165            if config.drx_on_duration_timer:
166                self.set_drx_on_duration_timer(bts_index,
167                                               config.drx_on_duration_timer)
168
169            if config.drx_inactivity_timer:
170                self.set_drx_inactivity_timer(bts_index,
171                                              config.drx_inactivity_timer)
172
173            if config.drx_retransmission_timer:
174                self.set_drx_retransmission_timer(
175                    bts_index, config.drx_retransmission_timer)
176
177            if config.drx_long_cycle:
178                self.set_drx_long_cycle(bts_index, config.drx_long_cycle)
179
180            if config.drx_long_cycle_offset is not None:
181                self.set_drx_long_cycle_offset(bts_index,
182                                               config.drx_long_cycle_offset)
183
184    def set_lte_rrc_state_change_timer(self, enabled, time=10):
185        """ Configures the LTE RRC state change timer.
186
187        Args:
188            enabled: a boolean indicating if the timer should be on or off.
189            time: time in seconds for the timer to expire
190        """
191        raise NotImplementedError()
192
193    def set_band(self, bts_index, band):
194        """ Sets the band for the indicated base station.
195
196        Args:
197            bts_index: the base station number
198            band: the new band
199        """
200        raise NotImplementedError()
201
202    def set_input_power(self, bts_index, input_power):
203        """ Sets the input power for the indicated base station.
204
205        Args:
206            bts_index: the base station number
207            input_power: the new input power
208        """
209        raise NotImplementedError()
210
211    def set_output_power(self, bts_index, output_power):
212        """ Sets the output power for the indicated base station.
213
214        Args:
215            bts_index: the base station number
216            output_power: the new output power
217        """
218        raise NotImplementedError()
219
220    def set_tdd_config(self, bts_index, tdd_config):
221        """ Sets the tdd configuration number for the indicated base station.
222
223        Args:
224            bts_index: the base station number
225            tdd_config: the new tdd configuration number
226        """
227        raise NotImplementedError()
228
229    def set_ssf_config(self, bts_index, ssf_config):
230        """ Sets the Special Sub-Frame config number for the indicated
231        base station.
232
233        Args:
234            bts_index: the base station number
235            ssf_config: the new ssf config number
236        """
237        raise NotImplementedError()
238
239    def set_bandwidth(self, bts_index, bandwidth):
240        """ Sets the bandwidth for the indicated base station.
241
242        Args:
243            bts_index: the base station number
244            bandwidth: the new bandwidth
245        """
246        raise NotImplementedError()
247
248    def set_downlink_channel_number(self, bts_index, channel_number):
249        """ Sets the downlink channel number for the indicated base station.
250
251        Args:
252            bts_index: the base station number
253            channel_number: the new channel number
254        """
255        raise NotImplementedError()
256
257    def set_mimo_mode(self, bts_index, mimo_mode):
258        """ Sets the mimo mode for the indicated base station.
259
260        Args:
261            bts_index: the base station number
262            mimo_mode: the new mimo mode
263        """
264        raise NotImplementedError()
265
266    def set_transmission_mode(self, bts_index, transmission_mode):
267        """ Sets the transmission mode for the indicated base station.
268
269        Args:
270            bts_index: the base station number
271            transmission_mode: the new transmission mode
272        """
273        raise NotImplementedError()
274
275    def set_scheduling_mode(self, bts_index, scheduling_mode, mcs_dl, mcs_ul,
276                            nrb_dl, nrb_ul):
277        """ Sets the scheduling mode for the indicated base station.
278
279        Args:
280            bts_index: the base station number
281            scheduling_mode: the new scheduling mode
282            mcs_dl: Downlink MCS (only for STATIC scheduling)
283            mcs_ul: Uplink MCS (only for STATIC scheduling)
284            nrb_dl: Number of RBs for downlink (only for STATIC scheduling)
285            nrb_ul: Number of RBs for uplink (only for STATIC scheduling)
286        """
287        raise NotImplementedError()
288
289    def set_dl_modulation(self, bts_index, modulation):
290        """ Sets the DL modulation for the indicated base station.
291
292        Args:
293            bts_index: the base station number
294            modulation: the new DL modulation
295        """
296        raise NotImplementedError()
297
298    def set_ul_modulation(self, bts_index, modulation):
299        """ Sets the UL modulation for the indicated base station.
300
301        Args:
302            bts_index: the base station number
303            modulation: the new UL modulation
304        """
305        raise NotImplementedError()
306
307    def set_tbs_pattern_on(self, bts_index, tbs_pattern_on):
308        """ Enables or disables TBS pattern in the indicated base station.
309
310        Args:
311            bts_index: the base station number
312            tbs_pattern_on: the new TBS pattern setting
313        """
314        raise NotImplementedError()
315
316    def set_cfi(self, bts_index, cfi):
317        """ Sets the Channel Format Indicator for the indicated base station.
318
319        Args:
320            bts_index: the base station number
321            cfi: the new CFI setting
322        """
323        raise NotImplementedError()
324
325    def set_paging_cycle(self, bts_index, cycle_duration):
326        """ Sets the paging cycle duration for the indicated base station.
327
328        Args:
329            bts_index: the base station number
330            cycle_duration: the new paging cycle duration in milliseconds
331        """
332        raise NotImplementedError()
333
334    def set_phich_resource(self, bts_index, phich):
335        """ Sets the PHICH Resource setting for the indicated base station.
336
337        Args:
338            bts_index: the base station number
339            phich: the new PHICH resource setting
340        """
341        raise NotImplementedError()
342
343    def set_drx_connected_mode(self, bts_index, active):
344        """ Sets the time interval to wait before entering DRX mode
345
346        Args:
347            bts_index: the base station number
348            active: Boolean indicating whether cDRX mode
349                is active
350        """
351        raise NotImplementedError()
352
353    def set_drx_on_duration_timer(self, bts_index, timer):
354        """ Sets the amount of PDCCH subframes to wait for data after
355            waking up from a DRX cycle
356
357        Args:
358            bts_index: the base station number
359            timer: Number of PDCCH subframes to wait and check for user data
360                after waking from the DRX cycle
361        """
362        raise NotImplementedError()
363
364    def set_drx_inactivity_timer(self, bts_index, timer):
365        """ Sets the number of PDCCH subframes to wait before entering DRX mode
366
367        Args:
368            bts_index: the base station number
369            timer: The amount of time to wait before entering DRX mode
370        """
371        raise NotImplementedError()
372
373    def set_drx_retransmission_timer(self, bts_index, timer):
374        """ Sets the number of consecutive PDCCH subframes to wait
375        for retransmission
376
377        Args:
378            bts_index: the base station number
379            timer: Number of PDCCH subframes to remain active
380
381        """
382        raise NotImplementedError()
383
384    def set_drx_long_cycle(self, bts_index, cycle):
385        """ Sets the amount of subframes representing a DRX long cycle.
386
387        Args:
388            bts_index: the base station number
389            cycle: The amount of subframes representing one long DRX cycle.
390                One cycle consists of DRX sleep + DRX on duration
391        """
392        raise NotImplementedError()
393
394    def set_drx_long_cycle_offset(self, bts_index, offset):
395        """ Sets the offset used to determine the subframe number
396        to begin the long drx cycle
397
398        Args:
399            bts_index: the base station number
400            offset: Number in range 0 to (long cycle - 1)
401        """
402        raise NotImplementedError()
403
404    def lte_attach_secondary_carriers(self, ue_capability_enquiry):
405        """ Activates the secondary carriers for CA. Requires the DUT to be
406        attached to the primary carrier first.
407
408        Args:
409            ue_capability_enquiry: UE capability enquiry message to be sent to
410        the UE before starting carrier aggregation.
411        """
412        raise NotImplementedError()
413
414    def wait_until_attached(self, timeout=120):
415        """ Waits until the DUT is attached to the primary carrier.
416
417        Args:
418            timeout: after this amount of time the method will raise a
419                CellularSimulatorError exception. Default is 120 seconds.
420        """
421        raise NotImplementedError()
422
423    def wait_until_communication_state(self, timeout=120):
424        """ Waits until the DUT is in Communication state.
425
426        Args:
427            timeout: after this amount of time the method will raise a
428                CellularSimulatorError exception. Default is 120 seconds.
429        """
430        raise NotImplementedError()
431
432    def wait_until_idle_state(self, timeout=120):
433        """ Waits until the DUT is in Idle state.
434
435        Args:
436            timeout: after this amount of time the method will raise a
437                CellularSimulatorError exception. Default is 120 seconds.
438        """
439        raise NotImplementedError()
440
441    def detach(self):
442        """ Turns off all the base stations so the DUT loose connection."""
443        raise NotImplementedError()
444
445    def stop(self):
446        """ Stops current simulation. After calling this method, the simulator
447        will need to be set up again. """
448        raise NotImplementedError()
449
450    def start_data_traffic(self):
451        """ Starts transmitting data from the instrument to the DUT. """
452        raise NotImplementedError()
453
454    def stop_data_traffic(self):
455        """ Stops transmitting data from the instrument to the DUT. """
456        raise NotImplementedError()
457
458    def get_measured_pusch_power(self):
459        """ Queries PUSCH power measured at the callbox.
460
461        Returns:
462            The PUSCH power in the primary input port.
463        """
464        raise NotImplementedError()
465
466
467class CellularSimulatorError(Exception):
468    """ Exceptions thrown when the cellular equipment is unreachable or it
469    returns an error after receiving a command. """
470    pass
471