1 /*
2 * Copyright (c) 2019, The OpenThread Authors.
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 * 3. Neither the name of the copyright holder nor the
13 * names of its contributors may be used to endorse or promote products
14 * derived from this software without specific prior written permission.
15 *
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26 * POSSIBILITY OF SUCH DAMAGE.
27 */
28
29 /**
30 * @file
31 * This file includes implementation of radio selector (for multi radio links).
32 */
33
34 #include "radio_selector.hpp"
35
36 #if OPENTHREAD_CONFIG_MULTI_RADIO
37
38 #include "common/code_utils.hpp"
39 #include "common/locator_getters.hpp"
40 #include "common/log.hpp"
41 #include "common/random.hpp"
42 #include "instance/instance.hpp"
43
44 namespace ot {
45
46 RegisterLogModule("RadioSelector");
47
48 // This array defines the order in which different radio link types are
49 // selected for message tx (direct message).
50 const Mac::RadioType RadioSelector::sRadioSelectionOrder[Mac::kNumRadioTypes] = {
51 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
52 Mac::kRadioTypeTrel,
53 #endif
54 #if OPENTHREAD_CONFIG_RADIO_LINK_IEEE_802_15_4_ENABLE
55 Mac::kRadioTypeIeee802154,
56 #endif
57 };
58
RadioSelector(Instance & aInstance)59 RadioSelector::RadioSelector(Instance &aInstance)
60 : InstanceLocator(aInstance)
61 {
62 }
63
PopulateMultiRadioInfo(MultiRadioInfo & aInfo)64 void RadioSelector::NeighborInfo::PopulateMultiRadioInfo(MultiRadioInfo &aInfo)
65 {
66 ClearAllBytes(aInfo);
67
68 #if OPENTHREAD_CONFIG_RADIO_LINK_IEEE_802_15_4_ENABLE
69 if (GetSupportedRadioTypes().Contains(Mac::kRadioTypeIeee802154))
70 {
71 aInfo.mSupportsIeee802154 = true;
72 aInfo.mIeee802154Info.mPreference = GetRadioPreference(Mac::kRadioTypeIeee802154);
73 }
74 #endif
75
76 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
77 if (GetSupportedRadioTypes().Contains(Mac::kRadioTypeTrel))
78 {
79 aInfo.mSupportsTrelUdp6 = true;
80 aInfo.mTrelUdp6Info.mPreference = GetRadioPreference(Mac::kRadioTypeTrel);
81 }
82 #endif
83 }
84
UpdatePreference(Neighbor & aNeighbor,Mac::RadioType aRadioType,int16_t aDifference)85 LogLevel RadioSelector::UpdatePreference(Neighbor &aNeighbor, Mac::RadioType aRadioType, int16_t aDifference)
86 {
87 uint8_t old = aNeighbor.GetRadioPreference(aRadioType);
88 int16_t preference = static_cast<int16_t>(old);
89
90 preference += aDifference;
91
92 if (preference > kMaxPreference)
93 {
94 preference = kMaxPreference;
95 }
96
97 if (preference < kMinPreference)
98 {
99 preference = kMinPreference;
100 }
101
102 aNeighbor.SetRadioPreference(aRadioType, static_cast<uint8_t>(preference));
103
104 // We check whether the update to the preference value caused it
105 // to cross the threshold `kHighPreference`. Based on this we
106 // return a suggested log level. If there is cross, suggest info
107 // log level, otherwise debug log level.
108
109 return ((old >= kHighPreference) != (preference >= kHighPreference)) ? kLogLevelInfo : kLogLevelDebg;
110 }
111
UpdateOnReceive(Neighbor & aNeighbor,Mac::RadioType aRadioType,bool aIsDuplicate)112 void RadioSelector::UpdateOnReceive(Neighbor &aNeighbor, Mac::RadioType aRadioType, bool aIsDuplicate)
113 {
114 LogLevel logLevel = kLogLevelInfo;
115
116 if (aNeighbor.GetSupportedRadioTypes().Contains(aRadioType))
117 {
118 logLevel = UpdatePreference(aNeighbor, aRadioType,
119 aIsDuplicate ? kPreferenceChangeOnRxDuplicate : kPreferenceChangeOnRx);
120
121 Log(logLevel, aIsDuplicate ? "UpdateOnDupRx" : "UpdateOnRx", aRadioType, aNeighbor);
122 }
123 else
124 {
125 aNeighbor.AddSupportedRadioType(aRadioType);
126 aNeighbor.SetRadioPreference(aRadioType, kInitPreference);
127
128 Log(logLevel, "NewRadio(OnRx)", aRadioType, aNeighbor);
129 }
130 }
131
UpdateOnSendDone(Mac::TxFrame & aFrame,Error aTxError)132 void RadioSelector::UpdateOnSendDone(Mac::TxFrame &aFrame, Error aTxError)
133 {
134 LogLevel logLevel = kLogLevelInfo;
135 Mac::RadioType radioType = aFrame.GetRadioType();
136 Mac::Address macDest;
137 Neighbor *neighbor;
138
139 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
140 if (radioType == Mac::kRadioTypeTrel)
141 {
142 // TREL radio link uses deferred ack model. We ignore
143 // `SendDone` event from `Mac` layer with success status and
144 // wait for deferred ack callback.
145 VerifyOrExit(aTxError != kErrorNone);
146 }
147 #endif
148
149 VerifyOrExit(aFrame.GetAckRequest());
150
151 IgnoreError(aFrame.GetDstAddr(macDest));
152 neighbor = Get<NeighborTable>().FindNeighbor(macDest, Neighbor::kInStateAnyExceptInvalid);
153 VerifyOrExit(neighbor != nullptr);
154
155 if (neighbor->GetSupportedRadioTypes().Contains(radioType))
156 {
157 logLevel = UpdatePreference(
158 *neighbor, radioType, (aTxError == kErrorNone) ? kPreferenceChangeOnTxSuccess : kPreferenceChangeOnTxError);
159
160 Log(logLevel, (aTxError == kErrorNone) ? "UpdateOnTxSucc" : "UpdateOnTxErr", radioType, *neighbor);
161 }
162 else
163 {
164 VerifyOrExit(aTxError == kErrorNone);
165 neighbor->AddSupportedRadioType(radioType);
166 neighbor->SetRadioPreference(radioType, kInitPreference);
167
168 Log(logLevel, "NewRadio(OnTx)", radioType, *neighbor);
169 }
170
171 exit:
172 return;
173 }
174
175 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
UpdateOnDeferredAck(Neighbor & aNeighbor,Error aTxError,bool & aAllowNeighborRemove)176 void RadioSelector::UpdateOnDeferredAck(Neighbor &aNeighbor, Error aTxError, bool &aAllowNeighborRemove)
177 {
178 LogLevel logLevel = kLogLevelInfo;
179
180 aAllowNeighborRemove = true;
181
182 if (aNeighbor.GetSupportedRadioTypes().Contains(Mac::kRadioTypeTrel))
183 {
184 logLevel = UpdatePreference(aNeighbor, Mac::kRadioTypeTrel,
185 (aTxError == kErrorNone) ? kPreferenceChangeOnDeferredAckSuccess
186 : kPreferenceChangeOnDeferredAckTimeout);
187
188 Log(logLevel, (aTxError == kErrorNone) ? "UpdateOnDefAckSucc" : "UpdateOnDefAckFail", Mac::kRadioTypeTrel,
189 aNeighbor);
190
191 // In case of deferred ack timeout, we check if the neighbor
192 // has any other radio link (with high preference) for future
193 // tx. If it it does, we set `aAllowNeighborRemove` to `false`
194 // to ensure neighbor is not removed yet.
195
196 VerifyOrExit(aTxError != kErrorNone);
197
198 for (Mac::RadioType radio : sRadioSelectionOrder)
199 {
200 if ((radio != Mac::kRadioTypeTrel) && aNeighbor.GetSupportedRadioTypes().Contains(radio) &&
201 aNeighbor.GetRadioPreference(radio) >= kHighPreference)
202 {
203 aAllowNeighborRemove = false;
204 ExitNow();
205 }
206 }
207 }
208 else
209 {
210 VerifyOrExit(aTxError == kErrorNone);
211 aNeighbor.AddSupportedRadioType(Mac::kRadioTypeTrel);
212 aNeighbor.SetRadioPreference(Mac::kRadioTypeTrel, kInitPreference);
213
214 Log(logLevel, "NewRadio(OnDefAckSucc)", Mac::kRadioTypeTrel, aNeighbor);
215 }
216
217 exit:
218 return;
219 }
220 #endif // OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
221
Select(Mac::RadioTypes aRadioOptions,const Neighbor & aNeighbor)222 Mac::RadioType RadioSelector::Select(Mac::RadioTypes aRadioOptions, const Neighbor &aNeighbor)
223 {
224 Mac::RadioType selectedRadio = sRadioSelectionOrder[0];
225 uint8_t selectedPreference = 0;
226 bool found = false;
227
228 // Select the first radio links with preference higher than
229 // threshold `kHighPreference`. The radio links are checked in the
230 // order defined by the `sRadioSelectionOrder` array. If no radio
231 // link has preference higher then threshold, select the one with
232 // highest preference.
233
234 for (Mac::RadioType radio : sRadioSelectionOrder)
235 {
236 if (aRadioOptions.Contains(radio))
237 {
238 uint8_t preference = aNeighbor.GetRadioPreference(radio);
239
240 if (preference >= kHighPreference)
241 {
242 selectedRadio = radio;
243 break;
244 }
245
246 if (!found || (selectedPreference < preference))
247 {
248 found = true;
249 selectedRadio = radio;
250 selectedPreference = preference;
251 }
252 }
253 }
254
255 return selectedRadio;
256 }
257
SelectRadio(Message & aMessage,const Mac::Address & aMacDest,Mac::TxFrames & aTxFrames)258 Mac::TxFrame &RadioSelector::SelectRadio(Message &aMessage, const Mac::Address &aMacDest, Mac::TxFrames &aTxFrames)
259 {
260 Neighbor *neighbor;
261 Mac::RadioType selectedRadio;
262 Mac::RadioTypes selections;
263
264 if (aMacDest.IsBroadcast() || aMacDest.IsNone())
265 {
266 aMessage.ClearRadioType();
267 ExitNow(selections.AddAll());
268 }
269
270 // If the radio type is already set when message was created we
271 // use the selected radio type. (e.g., MLE Discovery Response
272 // selects the radio link from which MLE Discovery Request is
273 // received.
274
275 if (aMessage.IsRadioTypeSet())
276 {
277 ExitNow(selections.Add(aMessage.GetRadioType()));
278 }
279
280 neighbor = Get<NeighborTable>().FindNeighbor(aMacDest, Neighbor::kInStateAnyExceptInvalid);
281
282 if ((neighbor == nullptr) || neighbor->GetSupportedRadioTypes().IsEmpty())
283 {
284 // If we do not have a corresponding neighbor or do not yet
285 // know the supported radio types, we try sending on all radio
286 // links in parallel. As an example, such a situation can
287 // happen when recovering a non-sleepy child (sending MLE
288 // Child Update Request to it) after device itself was reset.
289
290 aMessage.ClearRadioType();
291 ExitNow(selections.AddAll());
292 }
293
294 selectedRadio = Select(neighbor->GetSupportedRadioTypes(), *neighbor);
295 selections.Add(selectedRadio);
296
297 Log(kLogLevelDebg, "SelectRadio", selectedRadio, *neighbor);
298
299 aMessage.SetRadioType(selectedRadio);
300
301 // We (probabilistically) decide whether to probe on another radio
302 // link for the current frame tx. When probing we allow the same
303 // frame to be sent in parallel over multiple radio links but only
304 // care about the tx outcome (ack status) on the main selected
305 // radio link. This is done by setting the "required radio types"
306 // (`SetRequiredRadioTypes()`) to match the main selection on
307 // `aTxFrames`. We allow probe on TREL radio link if it is not
308 // currently usable (thus not selected) but is/was supported by
309 // the neighbor (i.e., we did rx/tx on TREL link from/to this
310 // neighbor in the past). The probe process helps detect whether
311 // the TREL link is usable again allowing us to switch over
312 // faster.
313
314 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
315 if (!selections.Contains(Mac::kRadioTypeTrel) && neighbor->GetSupportedRadioTypes().Contains(Mac::kRadioTypeTrel) &&
316 (Random::NonCrypto::GetUint8InRange(0, 100) < kTrelProbeProbability))
317 {
318 aTxFrames.SetRequiredRadioTypes(selections);
319 selections.Add(Mac::kRadioTypeTrel);
320
321 Log(kLogLevelDebg, "Probe", Mac::kRadioTypeTrel, *neighbor);
322 }
323 #endif
324
325 exit:
326 return aTxFrames.GetTxFrame(selections);
327 }
328
SelectPollFrameRadio(const Neighbor & aParent)329 Mac::RadioType RadioSelector::SelectPollFrameRadio(const Neighbor &aParent)
330 {
331 // This array defines the order in which different radio link types
332 // are selected for data poll frame tx.
333 static const Mac::RadioType selectionOrder[Mac::kNumRadioTypes] = {
334 #if OPENTHREAD_CONFIG_RADIO_LINK_IEEE_802_15_4_ENABLE
335 Mac::kRadioTypeIeee802154,
336 #endif
337 #if OPENTHREAD_CONFIG_RADIO_LINK_TREL_ENABLE
338 Mac::kRadioTypeTrel,
339 #endif
340 };
341
342 Mac::RadioType selection = selectionOrder[0];
343
344 for (Mac::RadioType radio : selectionOrder)
345 {
346 if (aParent.GetSupportedRadioTypes().Contains(radio))
347 {
348 selection = radio;
349 break;
350 }
351 }
352
353 return selection;
354 }
355
356 // LCOV_EXCL_START
357
358 #if OT_SHOULD_LOG_AT(OT_LOG_LEVEL_INFO)
359
Log(LogLevel aLogLevel,const char * aActionText,Mac::RadioType aRadioType,const Neighbor & aNeighbor)360 void RadioSelector::Log(LogLevel aLogLevel,
361 const char *aActionText,
362 Mac::RadioType aRadioType,
363 const Neighbor &aNeighbor)
364 {
365 String<kRadioPreferenceStringSize> preferenceString;
366 bool isFirstEntry = true;
367
368 VerifyOrExit(Instance::GetLogLevel() >= aLogLevel);
369
370 for (Mac::RadioType radio : sRadioSelectionOrder)
371 {
372 if (aNeighbor.GetSupportedRadioTypes().Contains(radio))
373 {
374 preferenceString.Append("%s%s:%d", isFirstEntry ? "" : " ", RadioTypeToString(radio),
375 aNeighbor.GetRadioPreference(radio));
376 isFirstEntry = false;
377 }
378 }
379
380 LogAt(aLogLevel, "RadioSelector: %s %s - neighbor:[%s rloc16:0x%04x radio-pref:{%s} state:%s]", aActionText,
381 RadioTypeToString(aRadioType), aNeighbor.GetExtAddress().ToString().AsCString(), aNeighbor.GetRloc16(),
382 preferenceString.AsCString(), Neighbor::StateToString(aNeighbor.GetState()));
383
384 exit:
385 return;
386 }
387
388 #else // #if OT_SHOULD_LOG_AT(OT_LOG_LEVEL_INFO)
389
Log(LogLevel,const char *,Mac::RadioType,const Neighbor &)390 void RadioSelector::Log(LogLevel, const char *, Mac::RadioType, const Neighbor &) {}
391
392 #endif // #if OT_SHOULD_LOG_AT(OT_LOG_LEVEL_INFO)
393
394 // LCOV_EXCL_STOP
395
396 } // namespace ot
397
398 #endif // #if OPENTHREAD_CONFIG_MULTI_RADIO
399