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: Address Cache Table 35*cfb92d14SAndroid Build Coastguard Worker# 36*cfb92d14SAndroid Build Coastguard Worker# This test verifies the behavior of `AddressResolver` and how the cache 37*cfb92d14SAndroid Build Coastguard Worker# table is managed. In particular it verifies behavior query timeout and 38*cfb92d14SAndroid Build Coastguard Worker# query retry and snoop optimization. 39*cfb92d14SAndroid Build Coastguard Worker 40*cfb92d14SAndroid Build Coastguard Workertest_name = __file__[:-3] if __file__.endswith('.py') else __file__ 41*cfb92d14SAndroid Build Coastguard Workerprint('-' * 120) 42*cfb92d14SAndroid Build Coastguard Workerprint('Starting \'{}\''.format(test_name)) 43*cfb92d14SAndroid Build Coastguard Worker 44*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 45*cfb92d14SAndroid Build Coastguard Worker# Creating `wpan.Nodes` instances 46*cfb92d14SAndroid Build Coastguard Worker 47*cfb92d14SAndroid Build Coastguard Workerspeedup = 4 48*cfb92d14SAndroid Build Coastguard Workerwpan.Node.set_time_speedup_factor(speedup) 49*cfb92d14SAndroid Build Coastguard Worker 50*cfb92d14SAndroid Build Coastguard Workerr1 = wpan.Node() 51*cfb92d14SAndroid Build Coastguard Workerr2 = wpan.Node() 52*cfb92d14SAndroid Build Coastguard Workerr3 = wpan.Node() 53*cfb92d14SAndroid Build Coastguard Worker 54*cfb92d14SAndroid Build Coastguard Workerc2 = wpan.Node() 55*cfb92d14SAndroid Build Coastguard Workerc3 = wpan.Node() 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# r3 ---- r1 ---- r2 66*cfb92d14SAndroid Build Coastguard Worker# | | 67*cfb92d14SAndroid Build Coastguard Worker# | | 68*cfb92d14SAndroid Build Coastguard Worker# c3 c2 69*cfb92d14SAndroid Build Coastguard Worker# 70*cfb92d14SAndroid Build Coastguard Worker 71*cfb92d14SAndroid Build Coastguard WorkerPREFIX = "fd00:1234::" 72*cfb92d14SAndroid Build Coastguard WorkerPOLL_INTERVAL = 200 73*cfb92d14SAndroid Build Coastguard Worker 74*cfb92d14SAndroid Build Coastguard WorkerMAX_SNOOPED_NON_EVICTABLE = 2 75*cfb92d14SAndroid Build Coastguard WorkerINITIAL_RETRY_DELAY = 4 76*cfb92d14SAndroid Build Coastguard WorkerMAX_CACHE_ENTRIES = 16 77*cfb92d14SAndroid Build Coastguard Worker 78*cfb92d14SAndroid Build Coastguard Workerr1.form("sekiro") # shadows die twice! 79*cfb92d14SAndroid Build Coastguard Worker 80*cfb92d14SAndroid Build Coastguard Workerr1.add_prefix(PREFIX, stable=True, on_mesh=True, slaac=False, preferred=True) 81*cfb92d14SAndroid Build Coastguard Worker 82*cfb92d14SAndroid Build Coastguard Workerr1.allowlist_node(r2) 83*cfb92d14SAndroid Build Coastguard Workerr2.allowlist_node(r1) 84*cfb92d14SAndroid Build Coastguard Workerr2.join_node(r1, wpan.JOIN_TYPE_ROUTER) 85*cfb92d14SAndroid Build Coastguard Worker 86*cfb92d14SAndroid Build Coastguard Workerc2.allowlist_node(r2) 87*cfb92d14SAndroid Build Coastguard Workerr2.allowlist_node(c2) 88*cfb92d14SAndroid Build Coastguard Workerc2.join_node(r2, wpan.JOIN_TYPE_END_DEVICE) 89*cfb92d14SAndroid Build Coastguard Worker 90*cfb92d14SAndroid Build Coastguard Workerr1.allowlist_node(r3) 91*cfb92d14SAndroid Build Coastguard Workerr3.allowlist_node(r1) 92*cfb92d14SAndroid Build Coastguard Workerr3.join_node(r1, wpan.JOIN_TYPE_ROUTER) 93*cfb92d14SAndroid Build Coastguard Worker 94*cfb92d14SAndroid Build Coastguard Workerc3.allowlist_node(r3) 95*cfb92d14SAndroid Build Coastguard Workerr3.allowlist_node(c3) 96*cfb92d14SAndroid Build Coastguard Workerc3.join_node(r3, wpan.JOIN_TYPE_SLEEPY_END_DEVICE) 97*cfb92d14SAndroid Build Coastguard Workerc3.set(wpan.WPAN_POLL_INTERVAL, str(POLL_INTERVAL)) 98*cfb92d14SAndroid Build Coastguard Worker 99*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 100*cfb92d14SAndroid Build Coastguard Worker# Test implementation 101*cfb92d14SAndroid Build Coastguard Worker# 102*cfb92d14SAndroid Build Coastguard Worker 103*cfb92d14SAndroid Build Coastguard Worker# Add IPv6 addresses on different nodes. 104*cfb92d14SAndroid Build Coastguard Worker 105*cfb92d14SAndroid Build Coastguard Workerr1_address = PREFIX + "1" 106*cfb92d14SAndroid Build Coastguard Workerr1.add_ip6_address_on_interface(r1_address) 107*cfb92d14SAndroid Build Coastguard Worker 108*cfb92d14SAndroid Build Coastguard WorkerWAIT_TIME = 10 109*cfb92d14SAndroid Build Coastguard WorkerPORT = 1234 110*cfb92d14SAndroid Build Coastguard Worker 111*cfb92d14SAndroid Build Coastguard WorkerNUM_ADDRESSES = 4 # Number of addresses to add on r2, r3, c2, and c3 112*cfb92d14SAndroid Build Coastguard Worker 113*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 114*cfb92d14SAndroid Build Coastguard Worker r2.add_ip6_address_on_interface(PREFIX + "2:" + str(num)) 115*cfb92d14SAndroid Build Coastguard Worker r3.add_ip6_address_on_interface(PREFIX + "3:" + str(num)) 116*cfb92d14SAndroid Build Coastguard Worker c2.add_ip6_address_on_interface(PREFIX + "c2:" + str(num)) 117*cfb92d14SAndroid Build Coastguard Worker c3.add_ip6_address_on_interface(PREFIX + "c3:" + str(num)) 118*cfb92d14SAndroid Build Coastguard Worker 119*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 120*cfb92d14SAndroid Build Coastguard Worker 121*cfb92d14SAndroid Build Coastguard Worker# From r1 send msg to a group of addresses (not provided by any nodes in network). 122*cfb92d14SAndroid Build Coastguard Worker 123*cfb92d14SAndroid Build Coastguard WorkerNUM_QUERY_ADDRS = 5 124*cfb92d14SAndroid Build Coastguard WorkerMAX_STAGGER_INTERVAL = 2.5 125*cfb92d14SAndroid Build Coastguard Worker 126*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_QUERY_ADDRS): 127*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "800:" + str(num), PORT), "hi nobody!", 1) 128*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 129*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 130*cfb92d14SAndroid Build Coastguard Worker # Wait before next tx to stagger the address queries 131*cfb92d14SAndroid Build Coastguard Worker # request ensuring different timeouts 132*cfb92d14SAndroid Build Coastguard Worker time.sleep(MAX_STAGGER_INTERVAL / (NUM_QUERY_ADDRS * speedup)) 133*cfb92d14SAndroid Build Coastguard Worker 134*cfb92d14SAndroid Build Coastguard Workerr2_rloc = int(r2.get(wpan.WPAN_THREAD_RLOC16), 16) 135*cfb92d14SAndroid Build Coastguard Workerc2_rloc = int(c2.get(wpan.WPAN_THREAD_RLOC16), 16) 136*cfb92d14SAndroid Build Coastguard Workerr3_rloc = int(r3.get(wpan.WPAN_THREAD_RLOC16), 16) 137*cfb92d14SAndroid Build Coastguard Worker 138*cfb92d14SAndroid Build Coastguard Worker# Verify that we do see entries in cache table for all the addresses and all are in "query" state 139*cfb92d14SAndroid Build Coastguard Worker 140*cfb92d14SAndroid Build Coastguard Workeraddr_cache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 141*cfb92d14SAndroid Build Coastguard Worker 142*cfb92d14SAndroid Build Coastguard Workerverify(len(addr_cache_table) == NUM_QUERY_ADDRS) 143*cfb92d14SAndroid Build Coastguard Workerfor entry in addr_cache_table: 144*cfb92d14SAndroid Build Coastguard Worker verify(entry.state == wpan.ADDRESS_CACHE_ENTRY_STATE_QUERY) 145*cfb92d14SAndroid Build Coastguard Worker verify(not entry.can_evict()) 146*cfb92d14SAndroid Build Coastguard Worker verify(entry.timeout > 0) 147*cfb92d14SAndroid Build Coastguard Worker verify(entry.retry_delay == INITIAL_RETRY_DELAY) 148*cfb92d14SAndroid Build Coastguard Worker 149*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 150*cfb92d14SAndroid Build Coastguard Worker 151*cfb92d14SAndroid Build Coastguard Worker# Check the retry-query behavior 152*cfb92d14SAndroid Build Coastguard Worker 153*cfb92d14SAndroid Build Coastguard Worker# Wait till all the address queries time out and verify they enter "retry-query" state. 154*cfb92d14SAndroid Build Coastguard Worker 155*cfb92d14SAndroid Build Coastguard Worker 156*cfb92d14SAndroid Build Coastguard Workerdef check_cache_entry_switch_to_retry_state(): 157*cfb92d14SAndroid Build Coastguard Worker cache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 158*cfb92d14SAndroid Build Coastguard Worker for index in range(NUM_QUERY_ADDRS): 159*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_RETRY_QUERY) 160*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].retry_delay == 2 * INITIAL_RETRY_DELAY) 161*cfb92d14SAndroid Build Coastguard Worker 162*cfb92d14SAndroid Build Coastguard Worker 163*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_cache_entry_switch_to_retry_state, WAIT_TIME) 164*cfb92d14SAndroid Build Coastguard Worker 165*cfb92d14SAndroid Build Coastguard Worker# Try sending again to same addresses which are in "retry-delay" state. 166*cfb92d14SAndroid Build Coastguard Worker 167*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_QUERY_ADDRS): 168*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "800:" + str(num), PORT), "hi nobody!", 1) 169*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 170*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 171*cfb92d14SAndroid Build Coastguard Worker 172*cfb92d14SAndroid Build Coastguard Worker# Make sure the entries stayed in retry-delay as before. 173*cfb92d14SAndroid Build Coastguard Worker 174*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_cache_entry_switch_to_retry_state, WAIT_TIME) 175*cfb92d14SAndroid Build Coastguard Worker 176*cfb92d14SAndroid Build Coastguard Worker# Now wait for them to get to zero timeout. 177*cfb92d14SAndroid Build Coastguard Worker 178*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 179*cfb92d14SAndroid Build Coastguard Worker 180*cfb92d14SAndroid Build Coastguard Worker 181*cfb92d14SAndroid Build Coastguard Workerdef check_cache_entry_in_retry_state_to_get_to_zero_timeout(): 182*cfb92d14SAndroid Build Coastguard Worker cache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 183*cfb92d14SAndroid Build Coastguard Worker for index in range(NUM_QUERY_ADDRS): 184*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_RETRY_QUERY) 185*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].timeout == 0) 186*cfb92d14SAndroid Build Coastguard Worker 187*cfb92d14SAndroid Build Coastguard Worker 188*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_cache_entry_in_retry_state_to_get_to_zero_timeout, WAIT_TIME) 189*cfb92d14SAndroid Build Coastguard Worker 190*cfb92d14SAndroid Build Coastguard Worker# Now try again using the same addresses. 191*cfb92d14SAndroid Build Coastguard Worker 192*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_QUERY_ADDRS): 193*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "800:" + str(num), PORT), "hi again nobody!", 1) 194*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 195*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 196*cfb92d14SAndroid Build Coastguard Worker 197*cfb92d14SAndroid Build Coastguard Worker# We expect now after the delay to see retries for same addresses. 198*cfb92d14SAndroid Build Coastguard Worker 199*cfb92d14SAndroid Build Coastguard Worker 200*cfb92d14SAndroid Build Coastguard Workerdef check_cache_entry_switch_to_query_state(): 201*cfb92d14SAndroid Build Coastguard Worker cache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 202*cfb92d14SAndroid Build Coastguard Worker for index in range(NUM_QUERY_ADDRS): 203*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_QUERY) 204*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].can_evict() == True) 205*cfb92d14SAndroid Build Coastguard Worker 206*cfb92d14SAndroid Build Coastguard Worker 207*cfb92d14SAndroid Build Coastguard Workerwpan.verify_within(check_cache_entry_switch_to_query_state, WAIT_TIME) 208*cfb92d14SAndroid Build Coastguard Worker 209*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 210*cfb92d14SAndroid Build Coastguard Worker 211*cfb92d14SAndroid Build Coastguard Worker# Verify snoop optimization. 212*cfb92d14SAndroid Build Coastguard Worker 213*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 214*cfb92d14SAndroid Build Coastguard Worker sender = r2.prepare_tx((PREFIX + "2:" + str(num), PORT), (r1_address, PORT), "hi r1 from r2 (snoop me)", 1) 215*cfb92d14SAndroid Build Coastguard Worker recver = r1.prepare_rx(sender) 216*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 217*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 218*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 219*cfb92d14SAndroid Build Coastguard Worker 220*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 221*cfb92d14SAndroid Build Coastguard Worker 222*cfb92d14SAndroid Build Coastguard Worker# We expect to see new "snooped" entries at the top of list. 223*cfb92d14SAndroid Build Coastguard Worker# Also verify that only MAX_SNOOPED_NON_EVICTABLE of snooped entries are non-evictable. 224*cfb92d14SAndroid Build Coastguard Worker 225*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) >= NUM_ADDRESSES) 226*cfb92d14SAndroid Build Coastguard Worker 227*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 228*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "2:" + str(NUM_ADDRESSES - index - 1)) 229*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == r2_rloc) 230*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_SNOOPED) 231*cfb92d14SAndroid Build Coastguard Worker if index < NUM_ADDRESSES - MAX_SNOOPED_NON_EVICTABLE: 232*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].can_evict() == True) 233*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].timeout == 0) 234*cfb92d14SAndroid Build Coastguard Worker else: 235*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].can_evict() == False) 236*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].timeout > 0) 237*cfb92d14SAndroid Build Coastguard Worker 238*cfb92d14SAndroid Build Coastguard Worker# From r1 send to r2 using the addresses from snooped entries: 239*cfb92d14SAndroid Build Coastguard Worker 240*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 241*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "2:" + str(num), PORT), "hi back r2 from r1", 1) 242*cfb92d14SAndroid Build Coastguard Worker recver = r2.prepare_rx(sender) 243*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 244*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 245*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 246*cfb92d14SAndroid Build Coastguard Worker 247*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 248*cfb92d14SAndroid Build Coastguard Worker 249*cfb92d14SAndroid Build Coastguard Worker# We expect to see the entries to be in "cached" state now. 250*cfb92d14SAndroid Build Coastguard Worker 251*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) >= NUM_ADDRESSES) 252*cfb92d14SAndroid Build Coastguard Worker 253*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 254*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "2:" + str(NUM_ADDRESSES - index - 1)) 255*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == r2_rloc) 256*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_CACHED) 257*cfb92d14SAndroid Build Coastguard Worker 258*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 259*cfb92d14SAndroid Build Coastguard Worker 260*cfb92d14SAndroid Build Coastguard Worker# Check query requests, last transaction time 261*cfb92d14SAndroid Build Coastguard Worker 262*cfb92d14SAndroid Build Coastguard Worker# Send from r1 to all addresses on r3. 263*cfb92d14SAndroid Build Coastguard Worker 264*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 265*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "3:" + str(num), PORT), "hi r3 from r1", 1) 266*cfb92d14SAndroid Build Coastguard Worker recver = r3.prepare_rx(sender) 267*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 268*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 269*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 270*cfb92d14SAndroid Build Coastguard Worker cache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 271*cfb92d14SAndroid Build Coastguard Worker 272*cfb92d14SAndroid Build Coastguard Worker# We expect to see the cache entries for the addresses pointing to r3. 273*cfb92d14SAndroid Build Coastguard Worker 274*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 275*cfb92d14SAndroid Build Coastguard Worker 276*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 277*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "3:" + str(NUM_ADDRESSES - index - 1)) 278*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == r3_rloc) 279*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_CACHED) 280*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].last_trans == 0) 281*cfb92d14SAndroid Build Coastguard Worker 282*cfb92d14SAndroid Build Coastguard Worker# Send from r1 to all addresses on c3 (sleepy child of r3) 283*cfb92d14SAndroid Build Coastguard Worker 284*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 285*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "c3:" + str(num), PORT), "hi c3 from r1", 1) 286*cfb92d14SAndroid Build Coastguard Worker recver = c3.prepare_rx(sender) 287*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 288*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 289*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 290*cfb92d14SAndroid Build Coastguard Worker 291*cfb92d14SAndroid Build Coastguard Worker# We expect to see the cache entries for c3 addresses pointing to r3. 292*cfb92d14SAndroid Build Coastguard Worker 293*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 294*cfb92d14SAndroid Build Coastguard Worker 295*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 296*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "c3:" + str(NUM_ADDRESSES - index - 1)) 297*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == r3_rloc) 298*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_CACHED) 299*cfb92d14SAndroid Build Coastguard Worker # SED's keep-alive period (`POLL_PERIOD`) is 200ms, `last_trans` should always be 0 as it is 300*cfb92d14SAndroid Build Coastguard Worker # the number of seconds since a keep-alive was last received from the child. 301*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].last_trans == 0) 302*cfb92d14SAndroid Build Coastguard Worker 303*cfb92d14SAndroid Build Coastguard Worker# Send again to r2. This should cause the related cache entries to be moved to top of the list: 304*cfb92d14SAndroid Build Coastguard Worker 305*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 306*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "2:" + str(num), PORT), "hi again r2 from r1", 1) 307*cfb92d14SAndroid Build Coastguard Worker recver = r2.prepare_rx(sender) 308*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 309*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 310*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 311*cfb92d14SAndroid Build Coastguard Worker 312*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 313*cfb92d14SAndroid Build Coastguard Worker 314*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 315*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "2:" + str(NUM_ADDRESSES - index - 1)) 316*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == r2_rloc) 317*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_CACHED) 318*cfb92d14SAndroid Build Coastguard Worker 319*cfb92d14SAndroid Build Coastguard Worker# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - 320*cfb92d14SAndroid Build Coastguard Worker 321*cfb92d14SAndroid Build Coastguard Worker# Check behavior when cache table is full. 322*cfb92d14SAndroid Build Coastguard Worker 323*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) == MAX_CACHE_ENTRIES) 324*cfb92d14SAndroid Build Coastguard Worker 325*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_QUERY_ADDRS): 326*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "900:" + str(num), PORT), "hi nobody!", 1) 327*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 328*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 329*cfb92d14SAndroid Build Coastguard Worker 330*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 331*cfb92d14SAndroid Build Coastguard Worker 332*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) == MAX_CACHE_ENTRIES) 333*cfb92d14SAndroid Build Coastguard Worker 334*cfb92d14SAndroid Build Coastguard Worker# Send from c2 to r1 and verify that snoop optimization uses at most 335*cfb92d14SAndroid Build Coastguard Worker# `MAX_SNOOPED_NON_EVICTABLE` entries. 336*cfb92d14SAndroid Build Coastguard Worker 337*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 338*cfb92d14SAndroid Build Coastguard Worker sender = c2.prepare_tx((PREFIX + "c2:" + str(num), PORT), (r1_address, PORT), "hi r1 from c2 (snoop me)", 1) 339*cfb92d14SAndroid Build Coastguard Worker recver = r1.prepare_rx(sender) 340*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 341*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 342*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 343*cfb92d14SAndroid Build Coastguard Worker 344*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 345*cfb92d14SAndroid Build Coastguard Worker 346*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) == MAX_CACHE_ENTRIES) 347*cfb92d14SAndroid Build Coastguard Worker 348*cfb92d14SAndroid Build Coastguard Workersnooped_entries = [entry for entry in cache_table if entry.state == wpan.ADDRESS_CACHE_ENTRY_STATE_SNOOPED] 349*cfb92d14SAndroid Build Coastguard Workerverify(len(snooped_entries) == MAX_SNOOPED_NON_EVICTABLE) 350*cfb92d14SAndroid Build Coastguard Worker 351*cfb92d14SAndroid Build Coastguard Workerfor entry in snooped_entries: 352*cfb92d14SAndroid Build Coastguard Worker verify(entry.rloc16 == c2_rloc) 353*cfb92d14SAndroid Build Coastguard Worker verify(entry.state == wpan.ADDRESS_CACHE_ENTRY_STATE_SNOOPED) 354*cfb92d14SAndroid Build Coastguard Worker verify(entry.can_evict() == False) 355*cfb92d14SAndroid Build Coastguard Worker verify(entry.timeout > 0) 356*cfb92d14SAndroid Build Coastguard Worker 357*cfb92d14SAndroid Build Coastguard Worker# Now send from r1 to c2, some of the snooped entries would be used, 358*cfb92d14SAndroid Build Coastguard Worker# others would go through full address query. 359*cfb92d14SAndroid Build Coastguard Worker 360*cfb92d14SAndroid Build Coastguard Workerfor num in range(NUM_ADDRESSES): 361*cfb92d14SAndroid Build Coastguard Worker sender = r1.prepare_tx((r1_address, PORT), (PREFIX + "c2:" + str(num), PORT), "hi c2 from r1", 1) 362*cfb92d14SAndroid Build Coastguard Worker recver = c2.prepare_rx(sender) 363*cfb92d14SAndroid Build Coastguard Worker wpan.Node.perform_async_tx_rx() 364*cfb92d14SAndroid Build Coastguard Worker verify(sender.was_successful) 365*cfb92d14SAndroid Build Coastguard Worker verify(recver.was_successful) 366*cfb92d14SAndroid Build Coastguard Worker 367*cfb92d14SAndroid Build Coastguard Workercache_table = wpan.parse_address_cache_table_result(r1.get(wpan.WPAN_THREAD_ADDRESS_CACHE_TABLE)) 368*cfb92d14SAndroid Build Coastguard Worker 369*cfb92d14SAndroid Build Coastguard Workerverify(len(cache_table) == MAX_CACHE_ENTRIES) 370*cfb92d14SAndroid Build Coastguard Worker 371*cfb92d14SAndroid Build Coastguard Worker# Verify that c2 entries are now at the top of cache list. 372*cfb92d14SAndroid Build Coastguard Worker 373*cfb92d14SAndroid Build Coastguard Workerfor index in range(NUM_ADDRESSES): 374*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].address == PREFIX + "c2:" + str(NUM_ADDRESSES - index - 1)) 375*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].rloc16 == c2_rloc) 376*cfb92d14SAndroid Build Coastguard Worker verify(cache_table[index].state == wpan.ADDRESS_CACHE_ENTRY_STATE_CACHED) 377*cfb92d14SAndroid Build Coastguard Worker 378*cfb92d14SAndroid Build Coastguard Worker# ----------------------------------------------------------------------------------------------------------------------- 379*cfb92d14SAndroid Build Coastguard Worker# Test finished 380*cfb92d14SAndroid Build Coastguard Worker 381*cfb92d14SAndroid Build Coastguard Workerwpan.Node.finalize_all_nodes() 382*cfb92d14SAndroid Build Coastguard Worker 383*cfb92d14SAndroid Build Coastguard Workerprint('\'{}\' passed.'.format(test_name)) 384