1#   Copyright 2016 - The Android Open Source Project
2#
3#   Licensed under the Apache License, Version 2.0 (the "License");
4#   you may not use this file except in compliance with the License.
5#   You may obtain a copy of the License at
6#
7#       http://www.apache.org/licenses/LICENSE-2.0
8#
9#   Unless required by applicable law or agreed to in writing, software
10#   distributed under the License is distributed on an "AS IS" BASIS,
11#   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12#   See the License for the specific language governing permissions and
13#   limitations under the License.
14
15import enum
16import logging
17import os
18import collections
19import itertools
20
21from acts.controllers.ap_lib import hostapd_constants
22
23
24def ht40_plus_allowed(channel):
25    """Returns: True iff HT40+ is enabled for this configuration."""
26    channel_supported = (channel in hostapd_constants.HT40_ALLOW_MAP[
27        hostapd_constants.N_CAPABILITY_HT40_PLUS_CHANNELS])
28    return (channel_supported)
29
30
31def ht40_minus_allowed(channel):
32    """Returns: True iff HT40- is enabled for this configuration."""
33    channel_supported = (channel in hostapd_constants.HT40_ALLOW_MAP[
34        hostapd_constants.N_CAPABILITY_HT40_MINUS_CHANNELS])
35    return (channel_supported)
36
37
38def get_frequency_for_channel(channel):
39    """The frequency associated with a given channel number.
40
41    Args:
42        value: int channel number.
43
44    Returns:
45        int, frequency in MHz associated with the channel.
46
47    """
48    for frequency, channel_iter in \
49        hostapd_constants.CHANNEL_MAP.items():
50        if channel == channel_iter:
51            return frequency
52    else:
53        raise ValueError('Unknown channel value: %r.' % channel)
54
55
56def get_channel_for_frequency(frequency):
57    """The channel number associated with a given frequency.
58
59    Args:
60        value: int frequency in MHz.
61
62    Returns:
63        int, frequency associated with the channel.
64
65    """
66    return hostapd_constants.CHANNEL_MAP[frequency]
67
68
69class HostapdConfig(object):
70    """The root settings for the router.
71
72    All the settings for a router that are not part of an ssid.
73    """
74
75    def _get_11ac_center_channel_from_channel(self, channel):
76        """Returns the center channel of the selected channel band based
77           on the channel and channel bandwidth provided.
78        """
79        channel = int(channel)
80        center_channel_delta = hostapd_constants.CENTER_CHANNEL_MAP[
81            self._vht_oper_chwidth]['delta']
82
83        for channel_map in hostapd_constants.CENTER_CHANNEL_MAP[
84                self._vht_oper_chwidth]['channels']:
85            lower_channel_bound, upper_channel_bound = channel_map
86            if lower_channel_bound <= channel <= upper_channel_bound:
87                return lower_channel_bound + center_channel_delta
88        raise ValueError('Invalid channel for {channel_width}.'.format(
89            channel_width=self._vht_oper_chwidth))
90
91    @property
92    def _get_default_config(self):
93        """Returns: dict of default options for hostapd."""
94        return collections.OrderedDict([
95            ('logger_syslog', '-1'),
96            ('logger_syslog_level', '0'),
97            # default RTS and frag threshold to ``off''
98            ('rts_threshold', '2347'),
99            ('fragm_threshold', '2346'),
100            ('driver', hostapd_constants.DRIVER_NAME)
101        ])
102
103    @property
104    def _hostapd_ht_capabilities(self):
105        """Returns: string suitable for the ht_capab= line in a hostapd config.
106        """
107        ret = []
108        for cap in hostapd_constants.N_CAPABILITIES_MAPPING.keys():
109            if cap in self._n_capabilities:
110                ret.append(hostapd_constants.N_CAPABILITIES_MAPPING[cap])
111        return ''.join(ret)
112
113    @property
114    def _hostapd_vht_capabilities(self):
115        """Returns: string suitable for the vht_capab= line in a hostapd config.
116        """
117        ret = []
118        for cap in hostapd_constants.AC_CAPABILITIES_MAPPING.keys():
119            if cap in self._ac_capabilities:
120                ret.append(hostapd_constants.AC_CAPABILITIES_MAPPING[cap])
121        return ''.join(ret)
122
123    @property
124    def _require_ht(self):
125        """Returns: True iff clients should be required to support HT."""
126        # TODO(wiley) Why? (crbug.com/237370)
127        # DOES THIS APPLY TO US?
128        logging.warning('Not enforcing pure N mode because Snow does '
129                        'not seem to support it...')
130        return False
131
132    @property
133    def _require_vht(self):
134        """Returns: True if clients should be required to support VHT."""
135        return self._mode == hostapd_constants.MODE_11AC_PURE
136
137    @property
138    def hw_mode(self):
139        """Returns: string hardware mode understood by hostapd."""
140        if self._mode == hostapd_constants.MODE_11A:
141            return hostapd_constants.MODE_11A
142        if self._mode == hostapd_constants.MODE_11B:
143            return hostapd_constants.MODE_11B
144        if self._mode == hostapd_constants.MODE_11G:
145            return hostapd_constants.MODE_11G
146        if self.is_11n or self.is_11ac:
147            # For their own historical reasons, hostapd wants it this way.
148            if self._frequency > 5000:
149                return hostapd_constants.MODE_11A
150            return hostapd_constants.MODE_11G
151        raise ValueError('Invalid mode.')
152
153    @property
154    def is_11n(self):
155        """Returns: True if we're trying to host an 802.11n network."""
156        return self._mode in (hostapd_constants.MODE_11N_MIXED,
157                              hostapd_constants.MODE_11N_PURE)
158
159    @property
160    def is_11ac(self):
161        """Returns: True if we're trying to host an 802.11ac network."""
162        return self._mode in (hostapd_constants.MODE_11AC_MIXED,
163                              hostapd_constants.MODE_11AC_PURE)
164
165    @property
166    def channel(self):
167        """Returns: int channel number for self.frequency."""
168        return get_channel_for_frequency(self.frequency)
169
170    @channel.setter
171    def channel(self, value):
172        """Sets the channel number to configure hostapd to listen on.
173
174        Args:
175            value: int, channel number.
176
177        """
178        self.frequency = get_frequency_for_channel(value)
179
180    @property
181    def bssid(self):
182        return self._bssid
183
184    @bssid.setter
185    def bssid(self, value):
186        self._bssid = value
187
188    @property
189    def frequency(self):
190        """Returns: int, frequency for hostapd to listen on."""
191        return self._frequency
192
193    @frequency.setter
194    def frequency(self, value):
195        """Sets the frequency for hostapd to listen on.
196
197        Args:
198            value: int, frequency in MHz.
199
200        """
201        if value not in hostapd_constants.CHANNEL_MAP:
202            raise ValueError('Tried to set an invalid frequency: %r.' % value)
203
204        self._frequency = value
205
206    @property
207    def bss_lookup(self):
208        return self._bss_lookup
209
210    @property
211    def ssid(self):
212        """Returns: SsidSettings, The root Ssid settings being used."""
213        return self._ssid
214
215    @ssid.setter
216    def ssid(self, value):
217        """Sets the ssid for the hostapd.
218
219        Args:
220            value: SsidSettings, new ssid settings to use.
221
222        """
223        self._ssid = value
224
225    @property
226    def hidden(self):
227        """Returns: bool, True if the ssid is hidden, false otherwise."""
228        return self._hidden
229
230    @hidden.setter
231    def hidden(self, value):
232        """Sets if this ssid is hidden.
233
234        Args:
235            value: bool, If true the ssid will be hidden.
236        """
237        self.hidden = value
238
239    @property
240    def security(self):
241        """Returns: The security type being used."""
242        return self._security
243
244    @security.setter
245    def security(self, value):
246        """Sets the security options to use.
247
248        Args:
249            value: Security, The type of security to use.
250        """
251        self._security = value
252
253    @property
254    def ht_packet_capture_mode(self):
255        """Get an appropriate packet capture HT parameter.
256
257        When we go to configure a raw monitor we need to configure
258        the phy to listen on the correct channel.  Part of doing
259        so is to specify the channel width for HT channels.  In the
260        case that the AP is configured to be either HT40+ or HT40-,
261        we could return the wrong parameter because we don't know which
262        configuration will be chosen by hostap.
263
264        Returns:
265            string, HT parameter for frequency configuration.
266
267        """
268        if not self.is_11n:
269            return None
270
271        if ht40_plus_allowed(self.channel):
272            return 'HT40+'
273
274        if ht40_minus_allowed(self.channel):
275            return 'HT40-'
276
277        return 'HT20'
278
279    @property
280    def beacon_footer(self):
281        """Returns: bool _beacon_footer value."""
282        return self._beacon_footer
283
284    def beacon_footer(self, value):
285        """Changes the beacon footer.
286
287        Args:
288            value: bool, The beacon footer vlaue.
289        """
290        self._beacon_footer = value
291
292    @property
293    def scenario_name(self):
294        """Returns: string _scenario_name value, or None."""
295        return self._scenario_name
296
297    @property
298    def min_streams(self):
299        """Returns: int, _min_streams value, or None."""
300        return self._min_streams
301
302    def __init__(self,
303                 interface=None,
304                 mode=None,
305                 channel=None,
306                 frequency=None,
307                 n_capabilities=[],
308                 beacon_interval=None,
309                 dtim_period=None,
310                 frag_threshold=None,
311                 short_preamble=None,
312                 ssid=None,
313                 hidden=False,
314                 security=None,
315                 bssid=None,
316                 force_wmm=None,
317                 pmf_support=hostapd_constants.PMF_SUPPORT_DISABLED,
318                 obss_interval=None,
319                 vht_channel_width=None,
320                 vht_center_channel=None,
321                 ac_capabilities=[],
322                 beacon_footer='',
323                 spectrum_mgmt_required=None,
324                 scenario_name=None,
325                 min_streams=None,
326                 bss_settings=[],
327                 set_ap_defaults_model=None):
328        """Construct a HostapdConfig.
329
330        You may specify channel or frequency, but not both.  Both options
331        are checked for validity (i.e. you can't specify an invalid channel
332        or a frequency that will not be accepted).
333
334        Args:
335            interface: string, The name of the interface to use.
336            mode: string, MODE_11x defined above.
337            channel: int, channel number.
338            frequency: int, frequency of channel.
339            n_capabilities: list of N_CAPABILITY_x defined above.
340            beacon_interval: int, beacon interval of AP.
341            dtim_period: int, include a DTIM every |dtim_period| beacons.
342            frag_threshold: int, maximum outgoing data frame size.
343            short_preamble: Whether to use a short preamble.
344            ssid: string, The name of the ssid to brodcast.
345            hidden: bool, Should the ssid be hidden.
346            security: Security, the secuirty settings to use.
347            bssid: string, a MAC address like string for the BSSID.
348            force_wmm: True if we should force WMM on, False if we should
349                force it off, None if we shouldn't force anything.
350            pmf_support: one of PMF_SUPPORT_* above.  Controls whether the
351                client supports/must support 802.11w.
352            obss_interval: int, interval in seconds that client should be
353                required to do background scans for overlapping BSSes.
354            vht_channel_width: object channel width
355            vht_center_channel: int, center channel of segment 0.
356            ac_capabilities: list of AC_CAPABILITY_x defined above.
357            beacon_footer: string, containing (unvalidated) IE data to be
358                placed at the end of the beacon.
359            spectrum_mgmt_required: True if we require the DUT to support
360                spectrum management.
361            scenario_name: string to be included in file names, instead
362                of the interface name.
363            min_streams: int, number of spatial streams required.
364            control_interface: The file name to use as the control interface.
365            bss_settings: The settings for all bss.
366        """
367        self._interface = interface
368        if channel is not None and frequency is not None:
369            raise ValueError('Specify either frequency or channel '
370                             'but not both.')
371
372        self._wmm_enabled = False
373        unknown_caps = [
374            cap for cap in n_capabilities
375            if cap not in hostapd_constants.N_CAPABILITIES_MAPPING
376        ]
377        if unknown_caps:
378            raise ValueError('Unknown capabilities: %r' % unknown_caps)
379
380        self._frequency = None
381        if channel:
382            self.channel = channel
383        elif frequency:
384            self.frequency = frequency
385        else:
386            raise ValueError('Specify either frequency or channel.')
387        '''
388        if set_ap_defaults_model:
389            ap_default_config = hostapd_ap_default_configs.APDefaultConfig(
390                profile_name=set_ap_defaults_model, frequency=self.frequency)
391            force_wmm = ap_default_config.force_wmm
392            beacon_interval = ap_default_config.beacon_interval
393            dtim_period = ap_default_config.dtim_period
394            short_preamble = ap_default_config.short_preamble
395            self._interface = ap_default_config.interface
396            mode = ap_default_config.mode
397            if ap_default_config.n_capabilities:
398                n_capabilities = ap_default_config.n_capabilities
399            if ap_default_config.ac_capabilities:
400                ap_default_config = ap_default_config.ac_capabilities
401        '''
402
403        self._n_capabilities = set(n_capabilities)
404        if self._n_capabilities:
405            self._wmm_enabled = True
406        if self._n_capabilities and mode is None:
407            mode = hostapd_constants.MODE_11N_PURE
408        self._mode = mode
409
410        if not self.supports_frequency(self.frequency):
411            raise ValueError('Configured a mode %s that does not support '
412                             'frequency %d' % (self._mode, self.frequency))
413
414        self._beacon_interval = beacon_interval
415        self._dtim_period = dtim_period
416        self._frag_threshold = frag_threshold
417        self._short_preamble = short_preamble
418
419        self._ssid = ssid
420        self._hidden = hidden
421        self._security = security
422        self._bssid = bssid
423        if force_wmm is not None:
424            self._wmm_enabled = force_wmm
425        if pmf_support not in hostapd_constants.PMF_SUPPORT_VALUES:
426            raise ValueError('Invalid value for pmf_support: %r' % pmf_support)
427
428        self._pmf_support = pmf_support
429        self._obss_interval = obss_interval
430        if self.is_11ac:
431            if str(vht_channel_width) == '40' or str(
432                    vht_channel_width) == '20':
433                self._vht_oper_chwidth = hostapd_constants.VHT_CHANNEL_WIDTH_40
434            elif str(vht_channel_width) == '80':
435                self._vht_oper_chwidth = hostapd_constants.VHT_CHANNEL_WIDTH_80
436            elif str(vht_channel_width) == '160':
437                self._vht_oper_chwidth = hostapd_constants.VHT_CHANNEL_WIDTH_160
438            elif str(vht_channel_width) == '80+80':
439                self._vht_oper_chwidth = hostapd_constants.VHT_CHANNEL_WIDTH_80_80
440            elif vht_channel_width is not None:
441                raise ValueError('Invalid channel width')
442            else:
443                logging.warning(
444                    'No channel bandwidth specified.  Using 80MHz for 11ac.')
445                self._vht_oper_chwidth = 1
446            if not vht_channel_width == 20:
447                if not vht_center_channel:
448                    self._vht_oper_centr_freq_seg0_idx = self._get_11ac_center_channel_from_channel(
449                        self.channel)
450            else:
451                self._vht_oper_centr_freq_seg0_idx = vht_center_channel
452            self._ac_capabilities = set(ac_capabilities)
453        self._beacon_footer = beacon_footer
454        self._spectrum_mgmt_required = spectrum_mgmt_required
455        self._scenario_name = scenario_name
456        self._min_streams = min_streams
457
458        self._bss_lookup = {}
459        for bss in bss_settings:
460            if bss.name in self._bss_lookup:
461                raise ValueError('Cannot have multiple bss settings with the'
462                                 ' same name.')
463            self._bss_lookup[bss.name] = bss
464
465    def __repr__(self):
466        return ('%s(mode=%r, channel=%r, frequency=%r, '
467                'n_capabilities=%r, beacon_interval=%r, '
468                'dtim_period=%r, frag_threshold=%r, ssid=%r, bssid=%r, '
469                'wmm_enabled=%r, security_config=%r, '
470                'spectrum_mgmt_required=%r)' %
471                (self.__class__.__name__, self._mode, self.channel,
472                 self.frequency, self._n_capabilities, self._beacon_interval,
473                 self._dtim_period, self._frag_threshold, self._ssid,
474                 self._bssid, self._wmm_enabled, self._security,
475                 self._spectrum_mgmt_required))
476
477    def supports_channel(self, value):
478        """Check whether channel is supported by the current hardware mode.
479
480        @param value: int channel to check.
481        @return True iff the current mode supports the band of the channel.
482
483        """
484        for freq, channel in hostapd_constants.CHANNEL_MAP.iteritems():
485            if channel == value:
486                return self.supports_frequency(freq)
487
488        return False
489
490    def supports_frequency(self, frequency):
491        """Check whether frequency is supported by the current hardware mode.
492
493        @param frequency: int frequency to check.
494        @return True iff the current mode supports the band of the frequency.
495
496        """
497        if self._mode == hostapd_constants.MODE_11A and frequency < 5000:
498            return False
499
500        if self._mode in (hostapd_constants.MODE_11B,
501                          hostapd_constants.MODE_11G) and frequency > 5000:
502            return False
503
504        if frequency not in hostapd_constants.CHANNEL_MAP:
505            return False
506
507        channel = hostapd_constants.CHANNEL_MAP[frequency]
508        supports_plus = (channel in hostapd_constants.HT40_ALLOW_MAP[
509            hostapd_constants.N_CAPABILITY_HT40_PLUS_CHANNELS])
510        supports_minus = (channel in hostapd_constants.HT40_ALLOW_MAP[
511            hostapd_constants.N_CAPABILITY_HT40_MINUS_CHANNELS])
512        if (hostapd_constants.N_CAPABILITY_HT40_PLUS in self._n_capabilities
513                and not supports_plus):
514            return False
515
516        if (hostapd_constants.N_CAPABILITY_HT40_MINUS in self._n_capabilities
517                and not supports_minus):
518            return False
519
520        return True
521
522    def add_bss(self, bss):
523        """Adds a new bss setting.
524
525        Args:
526            bss: The bss settings to add.
527        """
528        if bss.name in self._bss_lookup:
529            raise ValueError('A bss with the same name already exists.')
530
531        self._bss_lookup[bss.name] = bss
532
533    def remove_bss(self, bss_name):
534        """Removes a bss setting from the config."""
535        del self._bss_lookup[bss_name]
536
537    def package_configs(self):
538        """Package the configs.
539
540        Returns:
541            A list of dictionaries, one dictionary for each section of the
542            config.
543        """
544        # Start with the default config parameters.
545        conf = self._get_default_config
546        if self._interface:
547            conf['interface'] = self._interface
548        if self._bssid:
549            conf['bssid'] = self._bssid
550        if self._ssid:
551            conf['ssid'] = self._ssid
552            conf['ignore_broadcast_ssid'] = 1 if self._hidden else 0
553        conf['channel'] = self.channel
554        conf['hw_mode'] = self.hw_mode
555        if self.is_11n or self.is_11ac:
556            conf['ieee80211n'] = 1
557            conf['ht_capab'] = self._hostapd_ht_capabilities
558        if self.is_11ac:
559            conf['ieee80211ac'] = 1
560            conf['vht_oper_chwidth'] = self._vht_oper_chwidth
561            conf['vht_oper_centr_freq_seg0_idx'] = \
562                    self._vht_oper_centr_freq_seg0_idx
563            conf['vht_capab'] = self._hostapd_vht_capabilities
564        if self._wmm_enabled:
565            conf['wmm_enabled'] = 1
566        if self._require_ht:
567            conf['require_ht'] = 1
568        if self._require_vht:
569            conf['require_vht'] = 1
570        if self._beacon_interval:
571            conf['beacon_int'] = self._beacon_interval
572        if self._dtim_period:
573            conf['dtim_period'] = self._dtim_period
574        if self._frag_threshold:
575            conf['fragm_threshold'] = self._frag_threshold
576        if self._pmf_support:
577            conf['ieee80211w'] = self._pmf_support
578        if self._obss_interval:
579            conf['obss_interval'] = self._obss_interval
580        if self._short_preamble:
581            conf['preamble'] = 1
582        if self._spectrum_mgmt_required:
583            # To set spectrum_mgmt_required, we must first set
584            # local_pwr_constraint. And to set local_pwr_constraint,
585            # we must first set ieee80211d. And to set ieee80211d, ...
586            # Point being: order matters here.
587            conf['country_code'] = 'US'  # Required for local_pwr_constraint
588            conf['ieee80211d'] = 1  # Required for local_pwr_constraint
589            conf['local_pwr_constraint'] = 0  # No local constraint
590            conf['spectrum_mgmt_required'] = 1  # Requires local_pwr_constraint
591
592        if self._security:
593            for k, v in self._security.generate_dict().items():
594                conf[k] = v
595
596        all_conf = [conf]
597
598        for bss in self._bss_lookup.values():
599            bss_conf = collections.OrderedDict()
600            for k, v in (bss.generate_dict()).items():
601                bss_conf[k] = v
602            all_conf.append(bss_conf)
603
604        return all_conf
605