xref: /aosp_15_r20/external/openthread/tests/toranj/cli/test-701-multi-radio-probe.py (revision cfb92d1480a9e65faed56933e9c12405f45898b4)
1*cfb92d14SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*cfb92d14SAndroid Build Coastguard Worker#
3*cfb92d14SAndroid Build Coastguard Worker#  Copyright (c) 2022, 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 Workerfrom cli import verify
30*cfb92d14SAndroid Build Coastguard Workerfrom cli import verify_within
31*cfb92d14SAndroid Build Coastguard Workerimport cli
32*cfb92d14SAndroid Build Coastguard Workerimport time
33*cfb92d14SAndroid Build Coastguard Worker
34*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
35*cfb92d14SAndroid Build Coastguard Worker# Test description: This test covers behavior of device after trel network is temporarily disabled
36*cfb92d14SAndroid Build Coastguard Worker# and rediscovery of trel radio using probe mechanism.
37*cfb92d14SAndroid Build Coastguard Worker#
38*cfb92d14SAndroid Build Coastguard Worker#   r1  ---------- r2
39*cfb92d14SAndroid Build Coastguard Worker# (15.4+trel)   (15.4+trel)
40*cfb92d14SAndroid Build Coastguard Worker#
41*cfb92d14SAndroid Build Coastguard Worker#  On r2 we disable trel temporarily.
42*cfb92d14SAndroid Build Coastguard Worker#
43*cfb92d14SAndroid Build Coastguard Worker
44*cfb92d14SAndroid Build Coastguard Workertest_name = __file__[:-3] if __file__.endswith('.py') else __file__
45*cfb92d14SAndroid Build Coastguard Workerprint('-' * 120)
46*cfb92d14SAndroid Build Coastguard Workerprint('Starting \'{}\''.format(test_name))
47*cfb92d14SAndroid Build Coastguard Worker
48*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
49*cfb92d14SAndroid Build Coastguard Worker# Creating `cli.Node` instances
50*cfb92d14SAndroid Build Coastguard Worker
51*cfb92d14SAndroid Build Coastguard Workerspeedup = 10
52*cfb92d14SAndroid Build Coastguard Workercli.Node.set_time_speedup_factor(speedup)
53*cfb92d14SAndroid Build Coastguard Worker
54*cfb92d14SAndroid Build Coastguard Workerr1 = cli.Node(cli.RADIO_15_4_TREL)
55*cfb92d14SAndroid Build Coastguard Workerr2 = cli.Node(cli.RADIO_15_4_TREL)
56*cfb92d14SAndroid Build Coastguard Worker
57*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
58*cfb92d14SAndroid Build Coastguard Worker# Build network topology
59*cfb92d14SAndroid Build Coastguard Worker
60*cfb92d14SAndroid Build Coastguard Workerr1.form("prove-discover")
61*cfb92d14SAndroid Build Coastguard Workerr2.join(r1)
62*cfb92d14SAndroid Build Coastguard Worker
63*cfb92d14SAndroid Build Coastguard Workerverify(r1.get_state() == 'leader')
64*cfb92d14SAndroid Build Coastguard Workerverify(r2.get_state() == 'router')
65*cfb92d14SAndroid Build Coastguard Worker
66*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
67*cfb92d14SAndroid Build Coastguard Worker# Test Implementation
68*cfb92d14SAndroid Build Coastguard Worker
69*cfb92d14SAndroid Build Coastguard Workerhigh_preference_threshold = 220
70*cfb92d14SAndroid Build Coastguard Workermin_preference_threshold = 0
71*cfb92d14SAndroid Build Coastguard Worker
72*cfb92d14SAndroid Build Coastguard Workerverify(r1.multiradio_get_radios() == '[15.4, TREL]')
73*cfb92d14SAndroid Build Coastguard Workerverify(r2.multiradio_get_radios() == '[15.4, TREL]')
74*cfb92d14SAndroid Build Coastguard Worker
75*cfb92d14SAndroid Build Coastguard Workerr1_rloc = int(r1.get_rloc16(), 16)
76*cfb92d14SAndroid Build Coastguard Workerr2_rloc = int(r2.get_rloc16(), 16)
77*cfb92d14SAndroid Build Coastguard Worker
78*cfb92d14SAndroid Build Coastguard Workerr1_ml_addr = r1.get_mleid_ip_addr()
79*cfb92d14SAndroid Build Coastguard Workerr2_ml_addr = r2.get_mleid_ip_addr()
80*cfb92d14SAndroid Build Coastguard Worker
81*cfb92d14SAndroid Build Coastguard Worker# Wait till routes are discovered.
82*cfb92d14SAndroid Build Coastguard Worker
83*cfb92d14SAndroid Build Coastguard Worker
84*cfb92d14SAndroid Build Coastguard Workerdef check_r1_router_table():
85*cfb92d14SAndroid Build Coastguard Worker    table = r1.get_router_table()
86*cfb92d14SAndroid Build Coastguard Worker    verify(len(table) == 2)
87*cfb92d14SAndroid Build Coastguard Worker    for entry in table:
88*cfb92d14SAndroid Build Coastguard Worker        verify(int(entry['RLOC16'], 0) == r1_rloc or int(entry['Link']) == 1)
89*cfb92d14SAndroid Build Coastguard Worker
90*cfb92d14SAndroid Build Coastguard Worker
91*cfb92d14SAndroid Build Coastguard Workerverify_within(check_r1_router_table, 120)
92*cfb92d14SAndroid Build Coastguard Worker
93*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
94*cfb92d14SAndroid Build Coastguard Worker# Verify that r1 detected both TREL & 15.4 as supported radios by r2
95*cfb92d14SAndroid Build Coastguard Worker
96*cfb92d14SAndroid Build Coastguard Worker
97*cfb92d14SAndroid Build Coastguard Workerdef check_r1_sees_r2_has_two_radio_links():
98*cfb92d14SAndroid Build Coastguard Worker    neighbor_radios = r1.multiradio_get_neighbor_list()
99*cfb92d14SAndroid Build Coastguard Worker    verify(len(neighbor_radios) == 1)
100*cfb92d14SAndroid Build Coastguard Worker    info = cli.Node.parse_multiradio_neighbor_entry(neighbor_radios[0])
101*cfb92d14SAndroid Build Coastguard Worker    verify(int(info['RLOC16'], 16) == r2_rloc)
102*cfb92d14SAndroid Build Coastguard Worker    radios = info['Radios']
103*cfb92d14SAndroid Build Coastguard Worker    verify(len(radios) == 2)
104*cfb92d14SAndroid Build Coastguard Worker    verify('15.4' in radios)
105*cfb92d14SAndroid Build Coastguard Worker    verify('TREL' in radios)
106*cfb92d14SAndroid Build Coastguard Worker
107*cfb92d14SAndroid Build Coastguard Worker
108*cfb92d14SAndroid Build Coastguard Workercli.verify_within(check_r1_sees_r2_has_two_radio_links, 10)
109*cfb92d14SAndroid Build Coastguard Worker
110*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
111*cfb92d14SAndroid Build Coastguard Worker# Ping r2 from r1 and verify that r1 prefers trel radio link for
112*cfb92d14SAndroid Build Coastguard Worker# sending to r2.
113*cfb92d14SAndroid Build Coastguard Worker
114*cfb92d14SAndroid Build Coastguard Workerr1.ping(r2_ml_addr, count=5)
115*cfb92d14SAndroid Build Coastguard Worker
116*cfb92d14SAndroid Build Coastguard Workerneighbor_radios = r1.multiradio_get_neighbor_list()
117*cfb92d14SAndroid Build Coastguard Workerverify(len(neighbor_radios) == 1)
118*cfb92d14SAndroid Build Coastguard Workerinfo = cli.Node.parse_multiradio_neighbor_entry(neighbor_radios[0])
119*cfb92d14SAndroid Build Coastguard Workerradios = info['Radios']
120*cfb92d14SAndroid Build Coastguard Workerverify(radios['TREL'] >= high_preference_threshold)
121*cfb92d14SAndroid Build Coastguard Worker
122*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
123*cfb92d14SAndroid Build Coastguard Worker# Now temporary filter trel link on r2 and ping again. We expect that
124*cfb92d14SAndroid Build Coastguard Worker# r1 to quickly detect that trel is no longer supported by r2 and
125*cfb92d14SAndroid Build Coastguard Worker# prefer 15.4 for tx to r2.
126*cfb92d14SAndroid Build Coastguard Worker
127*cfb92d14SAndroid Build Coastguard Workerr2.cli('trel filter enable')
128*cfb92d14SAndroid Build Coastguard Workerverify(r2.cli('trel filter')[0] == 'Enabled')
129*cfb92d14SAndroid Build Coastguard Worker
130*cfb92d14SAndroid Build Coastguard Workerr1.udp_open()
131*cfb92d14SAndroid Build Coastguard Workerfor count in range(5):
132*cfb92d14SAndroid Build Coastguard Worker    r1.udp_send(r2_ml_addr, 12345, 'hi_r2_from_r1')
133*cfb92d14SAndroid Build Coastguard Worker
134*cfb92d14SAndroid Build Coastguard Worker
135*cfb92d14SAndroid Build Coastguard Workerdef check_r1_does_not_prefer_trel_for_r2():
136*cfb92d14SAndroid Build Coastguard Worker    neighbor_radios = r1.multiradio_get_neighbor_list()
137*cfb92d14SAndroid Build Coastguard Worker    verify(len(neighbor_radios) == 1)
138*cfb92d14SAndroid Build Coastguard Worker    info = cli.Node.parse_multiradio_neighbor_entry(neighbor_radios[0])
139*cfb92d14SAndroid Build Coastguard Worker    radios = info['Radios']
140*cfb92d14SAndroid Build Coastguard Worker    verify(radios['TREL'] <= min_preference_threshold)
141*cfb92d14SAndroid Build Coastguard Worker
142*cfb92d14SAndroid Build Coastguard Worker
143*cfb92d14SAndroid Build Coastguard Workerverify_within(check_r1_does_not_prefer_trel_for_r2, 10)
144*cfb92d14SAndroid Build Coastguard Worker
145*cfb92d14SAndroid Build Coastguard Worker# Check that we can send between r1 and r2 (now all tx should use 15.4)
146*cfb92d14SAndroid Build Coastguard Worker
147*cfb92d14SAndroid Build Coastguard Workerr1.ping(r2_ml_addr, count=5, verify_success=False)
148*cfb92d14SAndroid Build Coastguard Workerr1.ping(r2_ml_addr, count=5)
149*cfb92d14SAndroid Build Coastguard Worker
150*cfb92d14SAndroid Build Coastguard Workerneighbor_radios = r1.multiradio_get_neighbor_list()
151*cfb92d14SAndroid Build Coastguard Workerverify(len(neighbor_radios) == 1)
152*cfb92d14SAndroid Build Coastguard Workerinfo = cli.Node.parse_multiradio_neighbor_entry(neighbor_radios[0])
153*cfb92d14SAndroid Build Coastguard Workerradios = info['Radios']
154*cfb92d14SAndroid Build Coastguard Workerverify(radios['TREL'] <= min_preference_threshold)
155*cfb92d14SAndroid Build Coastguard Workerverify(radios['15.4'] >= high_preference_threshold)
156*cfb92d14SAndroid Build Coastguard Worker
157*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
158*cfb92d14SAndroid Build Coastguard Worker# Enable trel back on r2, start sending traffic from r1 to r2
159*cfb92d14SAndroid Build Coastguard Worker# The probe mechanism should kick and detect that r2 has trel again.
160*cfb92d14SAndroid Build Coastguard Worker
161*cfb92d14SAndroid Build Coastguard Workerr2.cli('trel filter disable')
162*cfb92d14SAndroid Build Coastguard Workerverify(r2.cli('trel filter')[0] == 'Disabled')
163*cfb92d14SAndroid Build Coastguard Worker
164*cfb92d14SAndroid Build Coastguard Workerr2.udp_open()
165*cfb92d14SAndroid Build Coastguard Workerfor count in range(80):
166*cfb92d14SAndroid Build Coastguard Worker    r2.udp_send(r1_ml_addr, 12345, 'hi_r1_from_r2')
167*cfb92d14SAndroid Build Coastguard Worker
168*cfb92d14SAndroid Build Coastguard Worker
169*cfb92d14SAndroid Build Coastguard Workerdef check_r1_again_prefers_trel_for_r2():
170*cfb92d14SAndroid Build Coastguard Worker    neighbor_radios = r1.multiradio_get_neighbor_list()
171*cfb92d14SAndroid Build Coastguard Worker    verify(len(neighbor_radios) == 1)
172*cfb92d14SAndroid Build Coastguard Worker    info = cli.Node.parse_multiradio_neighbor_entry(neighbor_radios[0])
173*cfb92d14SAndroid Build Coastguard Worker    radios = info['Radios']
174*cfb92d14SAndroid Build Coastguard Worker    verify(radios['TREL'] >= high_preference_threshold)
175*cfb92d14SAndroid Build Coastguard Worker
176*cfb92d14SAndroid Build Coastguard Worker
177*cfb92d14SAndroid Build Coastguard Workerverify_within(check_r1_again_prefers_trel_for_r2, 10)
178*cfb92d14SAndroid Build Coastguard Worker
179*cfb92d14SAndroid Build Coastguard Worker# -----------------------------------------------------------------------------------------------------------------------
180*cfb92d14SAndroid Build Coastguard Worker# Test finished
181*cfb92d14SAndroid Build Coastguard Worker
182*cfb92d14SAndroid Build Coastguard Workercli.Node.finalize_all_nodes()
183*cfb92d14SAndroid Build Coastguard Worker
184*cfb92d14SAndroid Build Coastguard Workerprint('\'{}\' passed.'.format(test_name))
185