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 implements of MLE types and constants.
32  */
33 
34 #include "mle_types.hpp"
35 
36 #include "common/array.hpp"
37 #include "common/code_utils.hpp"
38 #include "common/message.hpp"
39 #include "common/random.hpp"
40 
41 namespace ot {
42 namespace Mle {
43 
44 //---------------------------------------------------------------------------------------------------------------------
45 // DeviceMode
46 
Get(ModeConfig & aModeConfig) const47 void DeviceMode::Get(ModeConfig &aModeConfig) const
48 {
49     aModeConfig.mRxOnWhenIdle = IsRxOnWhenIdle();
50     aModeConfig.mDeviceType   = IsFullThreadDevice();
51     aModeConfig.mNetworkData  = (GetNetworkDataType() == NetworkData::kFullSet);
52 }
53 
Set(const ModeConfig & aModeConfig)54 void DeviceMode::Set(const ModeConfig &aModeConfig)
55 {
56     mMode = kModeReserved;
57     mMode |= aModeConfig.mRxOnWhenIdle ? kModeRxOnWhenIdle : 0;
58     mMode |= aModeConfig.mDeviceType ? kModeFullThreadDevice : 0;
59     mMode |= aModeConfig.mNetworkData ? kModeFullNetworkData : 0;
60 }
61 
ToString(void) const62 DeviceMode::InfoString DeviceMode::ToString(void) const
63 {
64     InfoString string;
65 
66     string.Append("rx-on:%s ftd:%s full-net:%s", ToYesNo(IsRxOnWhenIdle()), ToYesNo(IsFullThreadDevice()),
67                   ToYesNo(GetNetworkDataType() == NetworkData::kFullSet));
68 
69     return string;
70 }
71 
72 //---------------------------------------------------------------------------------------------------------------------
73 // DeviceProperties
74 
75 #if OPENTHREAD_FTD && OPENTHREAD_CONFIG_MLE_DEVICE_PROPERTY_LEADER_WEIGHT_ENABLE
76 
DeviceProperties(void)77 DeviceProperties::DeviceProperties(void)
78 {
79     Clear();
80 
81     mPowerSupply            = OPENTHREAD_CONFIG_DEVICE_POWER_SUPPLY;
82     mLeaderWeightAdjustment = kDefaultAdjustment;
83 #if OPENTHREAD_CONFIG_BORDER_ROUTING_ENABLE
84     mIsBorderRouter = true;
85 #endif
86 }
87 
ClampWeightAdjustment(void)88 void DeviceProperties::ClampWeightAdjustment(void)
89 {
90     mLeaderWeightAdjustment = Clamp(mLeaderWeightAdjustment, kMinAdjustment, kMaxAdjustment);
91 }
92 
CalculateLeaderWeight(void) const93 uint8_t DeviceProperties::CalculateLeaderWeight(void) const
94 {
95     static const int8_t kPowerSupplyIncs[] = {
96         kPowerBatteryInc,          // (0) kPowerSupplyBattery
97         kPowerExternalInc,         // (1) kPowerSupplyExternal
98         kPowerExternalStableInc,   // (2) kPowerSupplyExternalStable
99         kPowerExternalUnstableInc, // (3) kPowerSupplyExternalUnstable
100     };
101 
102     static_assert(0 == kPowerSupplyBattery, "kPowerSupplyBattery value is incorrect");
103     static_assert(1 == kPowerSupplyExternal, "kPowerSupplyExternal value is incorrect");
104     static_assert(2 == kPowerSupplyExternalStable, "kPowerSupplyExternalStable value is incorrect");
105     static_assert(3 == kPowerSupplyExternalUnstable, "kPowerSupplyExternalUnstable value is incorrect");
106 
107     uint8_t     weight      = kBaseWeight;
108     PowerSupply powerSupply = MapEnum(mPowerSupply);
109 
110     if (mIsBorderRouter)
111     {
112         weight += (mSupportsCcm ? kCcmBorderRouterInc : kBorderRouterInc);
113     }
114 
115     if (powerSupply < GetArrayLength(kPowerSupplyIncs))
116     {
117         weight += kPowerSupplyIncs[powerSupply];
118     }
119 
120     if (mIsUnstable)
121     {
122         switch (powerSupply)
123         {
124         case kPowerSupplyBattery:
125         case kPowerSupplyExternalUnstable:
126             break;
127 
128         default:
129             weight += kIsUnstableInc;
130         }
131     }
132 
133     weight += mLeaderWeightAdjustment;
134 
135     return weight;
136 }
137 
138 #endif // #if OPENTHREAD_FTD && OPENTHREAD_CONFIG_MLE_DEVICE_PROPERTY_LEADER_WEIGHT_ENABLE
139 
140 //---------------------------------------------------------------------------------------------------------------------
141 // RouterIdSet
142 
GetNumberOfAllocatedIds(void) const143 uint8_t RouterIdSet::GetNumberOfAllocatedIds(void) const
144 {
145     uint8_t count = 0;
146 
147     for (uint8_t byte : mRouterIdSet)
148     {
149         count += CountBitsInMask(byte);
150     }
151 
152     return count;
153 }
154 
155 //---------------------------------------------------------------------------------------------------------------------
156 // TxChallenge
157 
GenerateRandom(void)158 void TxChallenge::GenerateRandom(void) { IgnoreError(Random::Crypto::Fill(*this)); }
159 
160 //---------------------------------------------------------------------------------------------------------------------
161 // RxChallenge
162 
ReadFrom(const Message & aMessage,const OffsetRange & aOffsetRange)163 Error RxChallenge::ReadFrom(const Message &aMessage, const OffsetRange &aOffsetRange)
164 {
165     Error       error       = kErrorNone;
166     OffsetRange offsetRange = aOffsetRange;
167 
168     Clear();
169 
170     offsetRange.ShrinkLength(kMaxSize);
171 
172     VerifyOrExit(offsetRange.Contains(kMinSize), error = kErrorParse);
173 
174     SuccessOrExit(error = aMessage.Read(offsetRange, mArray.GetArrayBuffer(), offsetRange.GetLength()));
175     mArray.SetLength(static_cast<uint8_t>(offsetRange.GetLength()));
176 
177 exit:
178     return error;
179 }
180 
operator ==(const TxChallenge & aTxChallenge) const181 bool RxChallenge::operator==(const TxChallenge &aTxChallenge) const
182 {
183     return (mArray.GetLength() == kMaxSize) && (memcmp(mArray.GetArrayBuffer(), aTxChallenge.m8, kMaxSize) == 0);
184 }
185 
186 //---------------------------------------------------------------------------------------------------------------------
187 
RoleToString(DeviceRole aRole)188 const char *RoleToString(DeviceRole aRole)
189 {
190     static const char *const kRoleStrings[] = {
191         "disabled", // (0) kRoleDisabled
192         "detached", // (1) kRoleDetached
193         "child",    // (2) kRoleChild
194         "router",   // (3) kRoleRouter
195         "leader",   // (4) kRoleLeader
196     };
197 
198     static_assert(kRoleDisabled == 0, "kRoleDisabled value is incorrect");
199     static_assert(kRoleDetached == 1, "kRoleDetached value is incorrect");
200     static_assert(kRoleChild == 2, "kRoleChild value is incorrect");
201     static_assert(kRoleRouter == 3, "kRoleRouter value is incorrect");
202     static_assert(kRoleLeader == 4, "kRoleLeader value is incorrect");
203 
204     return (aRole < GetArrayLength(kRoleStrings)) ? kRoleStrings[aRole] : "invalid";
205 }
206 
207 } // namespace Mle
208 } // namespace ot
209