1*cfb92d14SAndroid Build Coastguard Worker#!/usr/bin/env python3 2*cfb92d14SAndroid Build Coastguard Worker# 3*cfb92d14SAndroid Build Coastguard Worker# Copyright (c) 2020, 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: This test covers behavior of device after trel network is temporarily disabled 35*cfb92d14SAndroid Build Coastguard Worker# and rediscovery of trel radio using probe mechanism. 36*cfb92d14SAndroid Build Coastguard Worker# 37*cfb92d14SAndroid Build Coastguard Worker# r1 ---------- r2 38*cfb92d14SAndroid Build Coastguard Worker# (15.4+trel) (15.4+trel) 39*cfb92d14SAndroid Build Coastguard Worker# 40*cfb92d14SAndroid Build Coastguard Worker# On r2 we disable trel temporarily. 41*cfb92d14SAndroid Build Coastguard Worker# 42*cfb92d14SAndroid Build Coastguard Worker 43*cfb92d14SAndroid Build Coastguard Workertest_name = __file__[:-3] if __file__.endswith('.py') else __file__ 44*cfb92d14SAndroid Build Coastguard Workerprint('-' * 120) 45*cfb92d14SAndroid Build Coastguard Workerprint('Starting \'{}\''.format(test_name)) 46*cfb92d14SAndroid Build Coastguard Worker 47*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 48*cfb92d14SAndroid Build Coastguard Worker# Creating `wpan.Nodes` instances 49*cfb92d14SAndroid Build Coastguard Worker 50*cfb92d14SAndroid Build Coastguard Workerspeedup = 4 51*cfb92d14SAndroid Build Coastguard Workerwpan.Node.set_time_speedup_factor(speedup) 52*cfb92d14SAndroid Build Coastguard Worker 53*cfb92d14SAndroid Build Coastguard Workerr1 = wpan.Node(wpan.NODE_15_4_TREL) 54*cfb92d14SAndroid Build Coastguard Workerr2 = wpan.Node(wpan.NODE_15_4_TREL) 55*cfb92d14SAndroid Build Coastguard Workerc2 = wpan.Node(wpan.NODE_15_4) 56*cfb92d14SAndroid Build Coastguard Worker 57*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 58*cfb92d14SAndroid Build Coastguard Worker# Init all nodes 59*cfb92d14SAndroid Build Coastguard Worker 60*cfb92d14SAndroid Build Coastguard Workerwpan.Node.init_all_nodes() 61*cfb92d14SAndroid Build Coastguard Worker 62*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 63*cfb92d14SAndroid Build Coastguard Worker# Build network topology 64*cfb92d14SAndroid Build Coastguard Worker# 65*cfb92d14SAndroid Build Coastguard Worker 66*cfb92d14SAndroid Build Coastguard Workerr1.allowlist_node(r2) 67*cfb92d14SAndroid Build Coastguard Workerr2.allowlist_node(r1) 68*cfb92d14SAndroid Build Coastguard Worker 69*cfb92d14SAndroid Build Coastguard Workerr2.allowlist_node(c2) 70*cfb92d14SAndroid Build Coastguard Workerc2.allowlist_node(r2) 71*cfb92d14SAndroid Build Coastguard Worker 72*cfb92d14SAndroid Build Coastguard Workerr1.form("discover-by-rx") 73*cfb92d14SAndroid Build Coastguard Worker 74*cfb92d14SAndroid Build Coastguard Workerr2.join_node(r1, wpan.JOIN_TYPE_ROUTER) 75*cfb92d14SAndroid Build Coastguard Workerc2.join_node(r2, wpan.JOIN_TYPE_SLEEPY_END_DEVICE) 76*cfb92d14SAndroid Build Coastguard Worker 77*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 78*cfb92d14SAndroid Build Coastguard Worker# Test implementation 79*cfb92d14SAndroid Build Coastguard Worker 80*cfb92d14SAndroid Build Coastguard WorkerWAIT_TIME = 5 81*cfb92d14SAndroid Build Coastguard WorkerHIGH_PREFERENCE_THRESHOLD = 220 82*cfb92d14SAndroid Build Coastguard WorkerMIN_PREFERENCE_THRESHOLD = 0 83*cfb92d14SAndroid Build Coastguard Worker 84*cfb92d14SAndroid Build Coastguard Workerr1_ext_address = r1.get(wpan.WPAN_EXT_ADDRESS)[1:-1] 85*cfb92d14SAndroid Build Coastguard Workerr1_rloc = int(r1.get(wpan.WPAN_THREAD_RLOC16), 16) 86*cfb92d14SAndroid Build Coastguard Workerr1_ml_address = r1.get(wpan.WPAN_IP6_MESH_LOCAL_ADDRESS)[1:-1] 87*cfb92d14SAndroid Build Coastguard Workerr2_ext_address = r2.get(wpan.WPAN_EXT_ADDRESS)[1:-1] 88*cfb92d14SAndroid Build Coastguard Workerr2_rloc = int(r2.get(wpan.WPAN_THREAD_RLOC16), 16) 89*cfb92d14SAndroid Build Coastguard Workerr2_ml_address = r2.get(wpan.WPAN_IP6_MESH_LOCAL_ADDRESS)[1:-1] 90*cfb92d14SAndroid Build Coastguard Worker 91*cfb92d14SAndroid Build Coastguard Worker# Wait for r2 to become router and r1 establishes a link with it 92*cfb92d14SAndroid Build Coastguard Worker 93*cfb92d14SAndroid Build Coastguard Worker 94*cfb92d14SAndroid Build Coastguard Workerdef check_r1_router_table(): 95*cfb92d14SAndroid Build Coastguard Worker router_table = wpan.parse_router_table_result(r1.get(wpan.WPAN_THREAD_ROUTER_TABLE)) 96*cfb92d14SAndroid Build Coastguard Worker verify(len(router_table) == 2) 97*cfb92d14SAndroid Build Coastguard Worker for entry in router_table: 98*cfb92d14SAndroid Build Coastguard Worker verify(entry.rloc16 == r1_rloc or entry.is_link_established()) 99*cfb92d14SAndroid Build Coastguard Worker 100*cfb92d14SAndroid Build Coastguard Worker 101*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_r1_router_table, WAIT_TIME) 102*cfb92d14SAndroid Build Coastguard Worker 103*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 104*cfb92d14SAndroid Build Coastguard Worker# Check that r1 detected both trel and 15.4 as supported radio by r2 105*cfb92d14SAndroid Build Coastguard Worker 106*cfb92d14SAndroid Build Coastguard Workerr1_radios = wpan.parse_list(r1.get(wpan.WPAN_OT_SUPPORTED_RADIO_LINKS)) 107*cfb92d14SAndroid Build Coastguard Workerverify( 108*cfb92d14SAndroid Build Coastguard Worker len(r1_radios) == 2 and (wpan.RADIO_LINK_IEEE_802_15_4 in r1_radios) and (wpan.RADIO_LINK_TREL_UDP6 in r1_radios)) 109*cfb92d14SAndroid Build Coastguard Worker 110*cfb92d14SAndroid Build Coastguard Workerr2_radios = wpan.parse_list(r2.get(wpan.WPAN_OT_SUPPORTED_RADIO_LINKS)) 111*cfb92d14SAndroid Build Coastguard Workerverify( 112*cfb92d14SAndroid Build Coastguard Worker len(r2_radios) == 2 and (wpan.RADIO_LINK_IEEE_802_15_4 in r2_radios) and (wpan.RADIO_LINK_TREL_UDP6 in r2_radios)) 113*cfb92d14SAndroid Build Coastguard Worker 114*cfb92d14SAndroid Build Coastguard Worker 115*cfb92d14SAndroid Build Coastguard Workerdef check_r1_sees_r2_has_two_radio_links(): 116*cfb92d14SAndroid Build Coastguard Worker r1_neighbor_radios = wpan.parse_multi_radio_result(r1.get(wpan.WPAN_OT_NEIGHBOR_TABLE_MULTI_RADIO_INFO)) 117*cfb92d14SAndroid Build Coastguard Worker verify(len(r1_neighbor_radios) == 1) 118*cfb92d14SAndroid Build Coastguard Worker verify(len(r1_neighbor_radios[0].radios) == 2) 119*cfb92d14SAndroid Build Coastguard Worker 120*cfb92d14SAndroid Build Coastguard Worker 121*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_r1_sees_r2_has_two_radio_links, WAIT_TIME) 122*cfb92d14SAndroid Build Coastguard Worker 123*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 124*cfb92d14SAndroid Build Coastguard Worker# Send from r1 to r2 and verify that r1 prefers trel radio link for sending to r2. 125*cfb92d14SAndroid Build Coastguard Worker 126*cfb92d14SAndroid Build Coastguard Workersender = r1.prepare_tx(r1_ml_address, r2_ml_address, "Hi r2 from r1", 5) 127*cfb92d14SAndroid Build Coastguard Workerrecver = r2.prepare_rx(sender) 128*cfb92d14SAndroid Build Coastguard Workerwpan.Node.perform_async_tx_rx() 129*cfb92d14SAndroid Build Coastguard Workerverify(sender.was_successful) 130*cfb92d14SAndroid Build Coastguard Workerverify(recver.was_successful) 131*cfb92d14SAndroid Build Coastguard Worker 132*cfb92d14SAndroid Build Coastguard Workerr1_neighbor_radios = wpan.parse_multi_radio_result(r1.get(wpan.WPAN_OT_NEIGHBOR_TABLE_MULTI_RADIO_INFO)) 133*cfb92d14SAndroid Build Coastguard Workerverify(len(r1_neighbor_radios) == 1) 134*cfb92d14SAndroid Build Coastguard Workerr2_radio_info = r1_neighbor_radios[0] 135*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.supports(wpan.RADIO_LINK_TREL_UDP6)) 136*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.preference(wpan.RADIO_LINK_TREL_UDP6) >= HIGH_PREFERENCE_THRESHOLD) 137*cfb92d14SAndroid Build Coastguard Worker 138*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 139*cfb92d14SAndroid Build Coastguard Worker# Now disable trel link on r2 and send again. We expect that r1 would quickly learn that trel is 140*cfb92d14SAndroid Build Coastguard Worker# no longer supported by r2 and prefer 15.4 for tx to r2. 141*cfb92d14SAndroid Build Coastguard Worker 142*cfb92d14SAndroid Build Coastguard Workerr2.set(wpan.WPAN_OT_TREL_TEST_MODE_ENABLE, 'false') 143*cfb92d14SAndroid Build Coastguard Workerverify(r2.get(wpan.WPAN_OT_TREL_TEST_MODE_ENABLE) == 'false') 144*cfb92d14SAndroid Build Coastguard Worker 145*cfb92d14SAndroid Build Coastguard Workersender = r1.prepare_tx(r1_ml_address, r2_ml_address, "Hi again r2 from r1", 5) 146*cfb92d14SAndroid Build Coastguard Workerwpan.Node.perform_async_tx_rx() 147*cfb92d14SAndroid Build Coastguard Workerverify(sender.was_successful) 148*cfb92d14SAndroid Build Coastguard Worker 149*cfb92d14SAndroid Build Coastguard Worker 150*cfb92d14SAndroid Build Coastguard Workerdef check_r1_does_not_prefer_trel_for_r2(): 151*cfb92d14SAndroid Build Coastguard Worker r1_neighbor_radios = wpan.parse_multi_radio_result(r1.get(wpan.WPAN_OT_NEIGHBOR_TABLE_MULTI_RADIO_INFO)) 152*cfb92d14SAndroid Build Coastguard Worker verify(len(r1_neighbor_radios) == 1) 153*cfb92d14SAndroid Build Coastguard Worker r2_radio_info = r1_neighbor_radios[0] 154*cfb92d14SAndroid Build Coastguard Worker verify(r2_radio_info.supports(wpan.RADIO_LINK_TREL_UDP6)) 155*cfb92d14SAndroid Build Coastguard Worker verify(r2_radio_info.preference(wpan.RADIO_LINK_TREL_UDP6) <= MIN_PREFERENCE_THRESHOLD) 156*cfb92d14SAndroid Build Coastguard Worker 157*cfb92d14SAndroid Build Coastguard Worker 158*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_r1_does_not_prefer_trel_for_r2, WAIT_TIME) 159*cfb92d14SAndroid Build Coastguard Worker 160*cfb92d14SAndroid Build Coastguard Worker# Check that we can send between r1 and r2 (now all tx should use 15.4) 161*cfb92d14SAndroid Build Coastguard Worker 162*cfb92d14SAndroid Build Coastguard Workersender = r1.prepare_tx(r1_ml_address, r2_ml_address, "Hi on 15.4 r2 from r1", 5) 163*cfb92d14SAndroid Build Coastguard Workerrecver = r2.prepare_rx(sender) 164*cfb92d14SAndroid Build Coastguard Workerwpan.Node.perform_async_tx_rx() 165*cfb92d14SAndroid Build Coastguard Workerverify(sender.was_successful) 166*cfb92d14SAndroid Build Coastguard Workerverify(recver.was_successful) 167*cfb92d14SAndroid Build Coastguard Worker 168*cfb92d14SAndroid Build Coastguard Workerr1_neighbor_radios = wpan.parse_multi_radio_result(r1.get(wpan.WPAN_OT_NEIGHBOR_TABLE_MULTI_RADIO_INFO)) 169*cfb92d14SAndroid Build Coastguard Workerverify(len(r1_neighbor_radios) == 1) 170*cfb92d14SAndroid Build Coastguard Workerr2_radio_info = r1_neighbor_radios[0] 171*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.supports(wpan.RADIO_LINK_TREL_UDP6)) 172*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.preference(wpan.RADIO_LINK_TREL_UDP6) <= MIN_PREFERENCE_THRESHOLD) 173*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.supports(wpan.RADIO_LINK_IEEE_802_15_4)) 174*cfb92d14SAndroid Build Coastguard Workerverify(r2_radio_info.preference(wpan.RADIO_LINK_IEEE_802_15_4) >= HIGH_PREFERENCE_THRESHOLD) 175*cfb92d14SAndroid Build Coastguard Worker 176*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 177*cfb92d14SAndroid Build Coastguard Worker# Enable trel back on r2, start sending traffic from r1 to r2 178*cfb92d14SAndroid Build Coastguard Worker# The probe mechanism should kick and detect that r2 has trel again. 179*cfb92d14SAndroid Build Coastguard Worker 180*cfb92d14SAndroid Build Coastguard Workerr2.set(wpan.WPAN_OT_TREL_TEST_MODE_ENABLE, 'true') 181*cfb92d14SAndroid Build Coastguard Workerverify(r2.get(wpan.WPAN_OT_TREL_TEST_MODE_ENABLE) == 'true') 182*cfb92d14SAndroid Build Coastguard Worker 183*cfb92d14SAndroid Build Coastguard Workersender = r1.prepare_tx(r1_ml_address, r2_ml_address, "Probing r2 from r1", 80) 184*cfb92d14SAndroid Build Coastguard Workerrecver = r2.prepare_rx(sender) 185*cfb92d14SAndroid Build Coastguard Workerwpan.Node.perform_async_tx_rx() 186*cfb92d14SAndroid Build Coastguard Workerverify(sender.was_successful) 187*cfb92d14SAndroid Build Coastguard Workerverify(recver.was_successful) 188*cfb92d14SAndroid Build Coastguard Worker 189*cfb92d14SAndroid Build Coastguard Worker 190*cfb92d14SAndroid Build Coastguard Workerdef check_r1_again_prefers_trel_for_r2(): 191*cfb92d14SAndroid Build Coastguard Worker r1_neighbor_radios = wpan.parse_multi_radio_result(r1.get(wpan.WPAN_OT_NEIGHBOR_TABLE_MULTI_RADIO_INFO)) 192*cfb92d14SAndroid Build Coastguard Worker verify(len(r1_neighbor_radios) == 1) 193*cfb92d14SAndroid Build Coastguard Worker r2_radio_info = r1_neighbor_radios[0] 194*cfb92d14SAndroid Build Coastguard Worker verify(r2_radio_info.supports(wpan.RADIO_LINK_TREL_UDP6)) 195*cfb92d14SAndroid Build Coastguard Worker verify(r2_radio_info.preference(wpan.RADIO_LINK_TREL_UDP6) >= HIGH_PREFERENCE_THRESHOLD) 196*cfb92d14SAndroid Build Coastguard Worker 197*cfb92d14SAndroid Build Coastguard Worker 198*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_r1_again_prefers_trel_for_r2, WAIT_TIME) 199*cfb92d14SAndroid Build Coastguard Worker 200*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 201*cfb92d14SAndroid Build Coastguard Worker# Test finished 202*cfb92d14SAndroid Build Coastguard Worker 203*cfb92d14SAndroid Build Coastguard Workerwpan.Node.finalize_all_nodes() 204*cfb92d14SAndroid Build Coastguard Worker 205*cfb92d14SAndroid Build Coastguard Workerprint('\'{}\' passed.'.format(test_name)) 206