1*cfb92d14SAndroid Build Coastguard Worker#!/usr/bin/env python3 2*cfb92d14SAndroid Build Coastguard Worker# 3*cfb92d14SAndroid Build Coastguard Worker# Copyright (c) 2018, The OpenThread Authors. 4*cfb92d14SAndroid Build Coastguard Worker# All rights reserved. 5*cfb92d14SAndroid Build Coastguard Worker# 6*cfb92d14SAndroid Build Coastguard Worker# Redistribution and use in source and binary forms, with or without 7*cfb92d14SAndroid Build Coastguard Worker# modification, are permitted provided that the following conditions are met: 8*cfb92d14SAndroid Build Coastguard Worker# 1. Redistributions of source code must retain the above copyright 9*cfb92d14SAndroid Build Coastguard Worker# notice, this list of conditions and the following disclaimer. 10*cfb92d14SAndroid Build Coastguard Worker# 2. Redistributions in binary form must reproduce the above copyright 11*cfb92d14SAndroid Build Coastguard Worker# notice, this list of conditions and the following disclaimer in the 12*cfb92d14SAndroid Build Coastguard Worker# documentation and/or other materials provided with the distribution. 13*cfb92d14SAndroid Build Coastguard Worker# 3. Neither the name of the copyright holder nor the 14*cfb92d14SAndroid Build Coastguard Worker# names of its contributors may be used to endorse or promote products 15*cfb92d14SAndroid Build Coastguard Worker# derived from this software without specific prior written permission. 16*cfb92d14SAndroid Build Coastguard Worker# 17*cfb92d14SAndroid Build Coastguard Worker# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 18*cfb92d14SAndroid Build Coastguard Worker# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19*cfb92d14SAndroid Build Coastguard Worker# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20*cfb92d14SAndroid Build Coastguard Worker# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 21*cfb92d14SAndroid Build Coastguard Worker# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 22*cfb92d14SAndroid Build Coastguard Worker# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 23*cfb92d14SAndroid Build Coastguard Worker# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 24*cfb92d14SAndroid Build Coastguard Worker# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 25*cfb92d14SAndroid Build Coastguard Worker# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 26*cfb92d14SAndroid Build Coastguard Worker# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 27*cfb92d14SAndroid Build Coastguard Worker# POSSIBILITY OF SUCH DAMAGE. 28*cfb92d14SAndroid Build Coastguard Worker 29*cfb92d14SAndroid Build Coastguard Workerimport time 30*cfb92d14SAndroid Build Coastguard Workerimport wpan 31*cfb92d14SAndroid Build Coastguard Workerfrom wpan import verify 32*cfb92d14SAndroid Build Coastguard Worker 33*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 34*cfb92d14SAndroid Build Coastguard Worker# Test description: verifies `ChannelManager` channel selection procedure 35*cfb92d14SAndroid Build Coastguard Worker 36*cfb92d14SAndroid Build Coastguard Workertest_name = __file__[:-3] if __file__.endswith('.py') else __file__ 37*cfb92d14SAndroid Build Coastguard Workerprint('-' * 120) 38*cfb92d14SAndroid Build Coastguard Workerprint('Starting \'{}\''.format(test_name)) 39*cfb92d14SAndroid Build Coastguard Worker 40*cfb92d14SAndroid Build Coastguard Worker 41*cfb92d14SAndroid Build Coastguard Workerdef verify_channel(nodes, new_channel, wait_time=20): 42*cfb92d14SAndroid Build Coastguard Worker """ 43*cfb92d14SAndroid Build Coastguard Worker This function checks the channel on a given list of `nodes` and verifies that all nodes 44*cfb92d14SAndroid Build Coastguard Worker switch to a given `new_channel` (as int) within certain `wait_time` (int and in seconds) 45*cfb92d14SAndroid Build Coastguard Worker """ 46*cfb92d14SAndroid Build Coastguard Worker start_time = time.time() 47*cfb92d14SAndroid Build Coastguard Worker 48*cfb92d14SAndroid Build Coastguard Worker while not all([(new_channel == int(node.get(wpan.WPAN_CHANNEL), 0)) for node in nodes]): 49*cfb92d14SAndroid Build Coastguard Worker if time.time() - start_time > wait_time: 50*cfb92d14SAndroid Build Coastguard Worker print('Took too long to switch to channel {} ({}>{} sec)'.format(new_channel, 51*cfb92d14SAndroid Build Coastguard Worker time.time() - start_time, wait_time)) 52*cfb92d14SAndroid Build Coastguard Worker exit(1) 53*cfb92d14SAndroid Build Coastguard Worker time.sleep(0.1) 54*cfb92d14SAndroid Build Coastguard Worker 55*cfb92d14SAndroid Build Coastguard Worker 56*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 57*cfb92d14SAndroid Build Coastguard Worker# Creating `wpan.Nodes` instances 58*cfb92d14SAndroid Build Coastguard Worker 59*cfb92d14SAndroid Build Coastguard Worker# Run the test with 10,000 time speedup factor 60*cfb92d14SAndroid Build Coastguard Workerwpan.Node.set_time_speedup_factor(10000) 61*cfb92d14SAndroid Build Coastguard Worker 62*cfb92d14SAndroid Build Coastguard Workernode = wpan.Node() 63*cfb92d14SAndroid Build Coastguard Worker 64*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 65*cfb92d14SAndroid Build Coastguard Worker# Init all nodes 66*cfb92d14SAndroid Build Coastguard Worker 67*cfb92d14SAndroid Build Coastguard Workerwpan.Node.init_all_nodes() 68*cfb92d14SAndroid Build Coastguard Worker 69*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 70*cfb92d14SAndroid Build Coastguard Worker# Build network topology 71*cfb92d14SAndroid Build Coastguard Worker 72*cfb92d14SAndroid Build Coastguard Workernode.form('channel-manager', channel=24) 73*cfb92d14SAndroid Build Coastguard Worker 74*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 75*cfb92d14SAndroid Build Coastguard Worker# Test implementation 76*cfb92d14SAndroid Build Coastguard Worker 77*cfb92d14SAndroid Build Coastguard Workerall_channels_mask = int('0x7fff800', 0) 78*cfb92d14SAndroid Build Coastguard Workerchan_12_to_15_mask = int('0x000f000', 0) 79*cfb92d14SAndroid Build Coastguard Workerchan_15_to_17_mask = int('0x0038000', 0) 80*cfb92d14SAndroid Build Coastguard Worker 81*cfb92d14SAndroid Build Coastguard Worker# Set supported channel mask to be all channels 82*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_SUPPORTED_CHANNEL_MASK, str(all_channels_mask)) 83*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_SUPPORTED_CHANNEL_MASK), 0) == all_channels_mask) 84*cfb92d14SAndroid Build Coastguard Worker 85*cfb92d14SAndroid Build Coastguard WorkerWAIT_TIME = 15 86*cfb92d14SAndroid Build Coastguard WorkerEXPECTED_SAMEPLE_COUNT = 970 87*cfb92d14SAndroid Build Coastguard Worker 88*cfb92d14SAndroid Build Coastguard Worker# Sleep for 4.5 second with speedup factor of 10,000 this is more than 12 89*cfb92d14SAndroid Build Coastguard Worker# hours. We sleep instead of immediately checking the sample counter in 90*cfb92d14SAndroid Build Coastguard Worker# order to not add more actions/events into simulation (specially since 91*cfb92d14SAndroid Build Coastguard Worker# we are running at very high speedup). 92*cfb92d14SAndroid Build Coastguard Workertime.sleep(4.5) 93*cfb92d14SAndroid Build Coastguard Worker 94*cfb92d14SAndroid Build Coastguard Worker 95*cfb92d14SAndroid Build Coastguard Workerdef check_sample_count(): 96*cfb92d14SAndroid Build Coastguard Worker verify(int(node.get(wpan.WPAN_CHANNEL_MONITOR_SAMPLE_COUNT), 0) > EXPECTED_SAMEPLE_COUNT) 97*cfb92d14SAndroid Build Coastguard Worker 98*cfb92d14SAndroid Build Coastguard Worker 99*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_sample_count, WAIT_TIME) 100*cfb92d14SAndroid Build Coastguard Worker 101*cfb92d14SAndroid Build Coastguard Worker# Verify the initial value of `NEW_CHANNEL` (should be zero if there has 102*cfb92d14SAndroid Build Coastguard Worker# been no channel change so far). 103*cfb92d14SAndroid Build Coastguard Worker 104*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 0) 105*cfb92d14SAndroid Build Coastguard Worker 106*cfb92d14SAndroid Build Coastguard Worker# Issue a channel-select with quality check enabled, and verify that no 107*cfb92d14SAndroid Build Coastguard Worker# action is taken. 108*cfb92d14SAndroid Build Coastguard Worker 109*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'false') 110*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 0) 111*cfb92d14SAndroid Build Coastguard Worker 112*cfb92d14SAndroid Build Coastguard Worker# Issue a channel-select with quality check disabled, verify that channel 113*cfb92d14SAndroid Build Coastguard Worker# is switched to channel 11. 114*cfb92d14SAndroid Build Coastguard Worker 115*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'true') 116*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 11) 117*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 11) 118*cfb92d14SAndroid Build Coastguard Worker 119*cfb92d14SAndroid Build Coastguard Worker# Set channels 12-15 as favorable and request a channel select, verify that channel is switched to 12. 120*cfb92d14SAndroid Build Coastguard Worker# 121*cfb92d14SAndroid Build Coastguard Worker# Even though 11 would be best, quality difference between 11 and 12 is not high enough for selection 122*cfb92d14SAndroid Build Coastguard Worker# algorithm to pick an unfavored channel. 123*cfb92d14SAndroid Build Coastguard Worker 124*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_FAVORED_CHANNEL_MASK, str(chan_12_to_15_mask)) 125*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL, '25') # request a channel change to 25 126*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 25) 127*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'true') 128*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 12) 129*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 12) 130*cfb92d14SAndroid Build Coastguard Worker 131*cfb92d14SAndroid Build Coastguard Worker# Set channels 15-17 as favorables and request a channel select, verify that channel is switched to 11. 132*cfb92d14SAndroid Build Coastguard Worker# 133*cfb92d14SAndroid Build Coastguard Worker# This time the quality difference between 11 and 15 should be high enough for selection 134*cfb92d14SAndroid Build Coastguard Worker# algorithm to pick the best though unfavored channel (i.e., channel 11). 135*cfb92d14SAndroid Build Coastguard Worker 136*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL, '25') # request a channel change to 25 137*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 25) 138*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_FAVORED_CHANNEL_MASK, str(chan_15_to_17_mask)) 139*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'true') 140*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 11) 141*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 11) 142*cfb92d14SAndroid Build Coastguard Worker 143*cfb92d14SAndroid Build Coastguard Worker# Set channels 12-15 as favorable and request a channel select, verify 144*cfb92d14SAndroid Build Coastguard Worker# that channel is not switched. 145*cfb92d14SAndroid Build Coastguard Worker 146*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_FAVORED_CHANNEL_MASK, str(chan_12_to_15_mask)) 147*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'true') 148*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 11) 149*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 11) 150*cfb92d14SAndroid Build Coastguard Worker 151*cfb92d14SAndroid Build Coastguard Worker# Starting from channel 12 and issuing a channel select (which would pick 11 as best channel). 152*cfb92d14SAndroid Build Coastguard Worker# However, since quality difference between current channel 12 and new best channel 11 is not large 153*cfb92d14SAndroid Build Coastguard Worker# enough, no action should be taken. 154*cfb92d14SAndroid Build Coastguard Worker 155*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL, '12') # request a channel change to 12 156*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 12) 157*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 12) 158*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_FAVORED_CHANNEL_MASK, str(all_channels_mask)) 159*cfb92d14SAndroid Build Coastguard Workernode.set(wpan.WPAN_CHANNEL_MANAGER_CHANNEL_SELECT, 'true') 160*cfb92d14SAndroid Build Coastguard Workerverify(int(node.get(wpan.WPAN_CHANNEL_MANAGER_NEW_CHANNEL), 0) == 12) 161*cfb92d14SAndroid Build Coastguard Workerverify_channel([node], 12) 162*cfb92d14SAndroid Build Coastguard Worker 163*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 164*cfb92d14SAndroid Build Coastguard Worker# Test finished 165*cfb92d14SAndroid Build Coastguard Worker 166*cfb92d14SAndroid Build Coastguard Workerwpan.Node.finalize_all_nodes() 167*cfb92d14SAndroid Build Coastguard Worker 168*cfb92d14SAndroid Build Coastguard Workerprint('\'{}\' passed.'.format(test_name)) 169