1 /* 2 * Copyright (c) 2016, 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 definitions for the Commissioner role. 32 */ 33 34 #ifndef COMMISSIONER_HPP_ 35 #define COMMISSIONER_HPP_ 36 37 #include "openthread-core-config.h" 38 39 #if OPENTHREAD_FTD && OPENTHREAD_CONFIG_COMMISSIONER_ENABLE 40 41 #include <openthread/commissioner.h> 42 43 #include "coap/coap_secure.hpp" 44 #include "common/as_core_type.hpp" 45 #include "common/callback.hpp" 46 #include "common/clearable.hpp" 47 #include "common/locator.hpp" 48 #include "common/log.hpp" 49 #include "common/non_copyable.hpp" 50 #include "common/timer.hpp" 51 #include "mac/mac_types.hpp" 52 #include "meshcop/announce_begin_client.hpp" 53 #include "meshcop/energy_scan_client.hpp" 54 #include "meshcop/panid_query_client.hpp" 55 #include "meshcop/secure_transport.hpp" 56 #include "net/ip6_address.hpp" 57 #include "net/udp6.hpp" 58 #include "thread/key_manager.hpp" 59 #include "thread/mle.hpp" 60 #include "thread/tmf.hpp" 61 62 namespace ot { 63 64 namespace MeshCoP { 65 66 class Commissioner : public InstanceLocator, private NonCopyable 67 { 68 friend class Tmf::Agent; 69 friend class Tmf::SecureAgent; 70 71 public: 72 /** 73 * Type represents the Commissioner State. 74 * 75 */ 76 enum State : uint8_t 77 { 78 kStateDisabled = OT_COMMISSIONER_STATE_DISABLED, ///< Disabled. 79 kStatePetition = OT_COMMISSIONER_STATE_PETITION, ///< Petitioning to become a Commissioner. 80 kStateActive = OT_COMMISSIONER_STATE_ACTIVE, ///< Active Commissioner. 81 }; 82 83 /** 84 * Type represents Joiner Event. 85 * 86 */ 87 enum JoinerEvent : uint8_t 88 { 89 kJoinerEventStart = OT_COMMISSIONER_JOINER_START, 90 kJoinerEventConnected = OT_COMMISSIONER_JOINER_CONNECTED, 91 kJoinerEventFinalize = OT_COMMISSIONER_JOINER_FINALIZE, 92 kJoinerEventEnd = OT_COMMISSIONER_JOINER_END, 93 kJoinerEventRemoved = OT_COMMISSIONER_JOINER_REMOVED, 94 }; 95 96 typedef otCommissionerStateCallback StateCallback; ///< State change callback function pointer type. 97 typedef otCommissionerJoinerCallback JoinerCallback; ///< Joiner state change callback function pointer type. 98 99 /** 100 * Initializes the Commissioner object. 101 * 102 * @param[in] aInstance A reference to the OpenThread instance. 103 * 104 */ 105 explicit Commissioner(Instance &aInstance); 106 107 /** 108 * Starts the Commissioner service. 109 * 110 * @param[in] aStateCallback A pointer to a function that is called when the commissioner state changes. 111 * @param[in] aJoinerCallback A pointer to a function that is called when a joiner event occurs. 112 * @param[in] aCallbackContext A pointer to application-specific context. 113 * 114 * @retval kErrorNone Successfully started the Commissioner service. 115 * @retval kErrorAlready Commissioner is already started. 116 * @retval kErrorInvalidState Device is not currently attached to a network. 117 * 118 */ 119 Error Start(StateCallback aStateCallback, JoinerCallback aJoinerCallback, void *aCallbackContext); 120 121 /** 122 * Stops the Commissioner service. 123 * 124 * @retval kErrorNone Successfully stopped the Commissioner service. 125 * @retval kErrorAlready Commissioner is already stopped. 126 * 127 */ Stop(void)128 Error Stop(void) { return Stop(kSendKeepAliveToResign); } 129 130 /** 131 * Returns the Commissioner Id. 132 * 133 * @returns The Commissioner Id. 134 * 135 */ GetId(void) const136 const char *GetId(void) const { return mCommissionerId; } 137 138 /** 139 * Sets the Commissioner Id. 140 * 141 * @param[in] aId A pointer to a string character array. Must be null terminated. 142 * 143 * @retval kErrorNone Successfully set the Commissioner Id. 144 * @retval kErrorInvalidArgs Given name is too long. 145 * @retval kErrorInvalidState The commissioner is active and id cannot be changed. 146 * 147 */ 148 Error SetId(const char *aId); 149 150 /** 151 * Clears all Joiner entries. 152 * 153 */ 154 void ClearJoiners(void); 155 156 /** 157 * Adds a Joiner entry accepting any Joiner. 158 * 159 * @param[in] aPskd A pointer to the PSKd. 160 * @param[in] aTimeout A time after which a Joiner is automatically removed, in seconds. 161 * 162 * @retval kErrorNone Successfully added the Joiner. 163 * @retval kErrorNoBufs No buffers available to add the Joiner. 164 * @retval kErrorInvalidState Commissioner service is not started. 165 * 166 */ AddJoinerAny(const char * aPskd,uint32_t aTimeout)167 Error AddJoinerAny(const char *aPskd, uint32_t aTimeout) { return AddJoiner(nullptr, nullptr, aPskd, aTimeout); } 168 169 /** 170 * Adds a Joiner entry. 171 * 172 * @param[in] aEui64 The Joiner's IEEE EUI-64. 173 * @param[in] aPskd A pointer to the PSKd. 174 * @param[in] aTimeout A time after which a Joiner is automatically removed, in seconds. 175 * 176 * @retval kErrorNone Successfully added the Joiner. 177 * @retval kErrorNoBufs No buffers available to add the Joiner. 178 * @retval kErrorInvalidState Commissioner service is not started. 179 * 180 */ AddJoiner(const Mac::ExtAddress & aEui64,const char * aPskd,uint32_t aTimeout)181 Error AddJoiner(const Mac::ExtAddress &aEui64, const char *aPskd, uint32_t aTimeout) 182 { 183 return AddJoiner(&aEui64, nullptr, aPskd, aTimeout); 184 } 185 186 /** 187 * Adds a Joiner entry with a Joiner Discerner. 188 * 189 * @param[in] aDiscerner A Joiner Discerner. 190 * @param[in] aPskd A pointer to the PSKd. 191 * @param[in] aTimeout A time after which a Joiner is automatically removed, in seconds. 192 * 193 * @retval kErrorNone Successfully added the Joiner. 194 * @retval kErrorNoBufs No buffers available to add the Joiner. 195 * @retval kErrorInvalidState Commissioner service is not started. 196 * 197 */ AddJoiner(const JoinerDiscerner & aDiscerner,const char * aPskd,uint32_t aTimeout)198 Error AddJoiner(const JoinerDiscerner &aDiscerner, const char *aPskd, uint32_t aTimeout) 199 { 200 return AddJoiner(nullptr, &aDiscerner, aPskd, aTimeout); 201 } 202 203 /** 204 * Get joiner info at aIterator position. 205 * 206 * @param[in,out] aIterator A iterator to the index of the joiner. 207 * @param[out] aJoiner A reference to Joiner info. 208 * 209 * @retval kErrorNone Successfully get the Joiner info. 210 * @retval kErrorNotFound Not found next Joiner. 211 * 212 */ 213 Error GetNextJoinerInfo(uint16_t &aIterator, otJoinerInfo &aJoiner) const; 214 215 /** 216 * Removes a Joiner entry accepting any Joiner. 217 * 218 * @param[in] aDelay The delay to remove Joiner (in seconds). 219 * 220 * @retval kErrorNone Successfully added the Joiner. 221 * @retval kErrorNotFound The Joiner entry accepting any Joiner was not found. 222 * @retval kErrorInvalidState Commissioner service is not started. 223 * 224 */ RemoveJoinerAny(uint32_t aDelay)225 Error RemoveJoinerAny(uint32_t aDelay) { return RemoveJoiner(nullptr, nullptr, aDelay); } 226 227 /** 228 * Removes a Joiner entry. 229 * 230 * @param[in] aEui64 The Joiner's IEEE EUI-64. 231 * @param[in] aDelay The delay to remove Joiner (in seconds). 232 * 233 * @retval kErrorNone Successfully added the Joiner. 234 * @retval kErrorNotFound The Joiner specified by @p aEui64 was not found. 235 * @retval kErrorInvalidState Commissioner service is not started. 236 * 237 */ RemoveJoiner(const Mac::ExtAddress & aEui64,uint32_t aDelay)238 Error RemoveJoiner(const Mac::ExtAddress &aEui64, uint32_t aDelay) 239 { 240 return RemoveJoiner(&aEui64, nullptr, aDelay); 241 } 242 243 /** 244 * Removes a Joiner entry. 245 * 246 * @param[in] aDiscerner A Joiner Discerner. 247 * @param[in] aDelay The delay to remove Joiner (in seconds). 248 * 249 * @retval kErrorNone Successfully added the Joiner. 250 * @retval kErrorNotFound The Joiner specified by @p aEui64 was not found. 251 * @retval kErrorInvalidState Commissioner service is not started. 252 * 253 */ RemoveJoiner(const JoinerDiscerner & aDiscerner,uint32_t aDelay)254 Error RemoveJoiner(const JoinerDiscerner &aDiscerner, uint32_t aDelay) 255 { 256 return RemoveJoiner(nullptr, &aDiscerner, aDelay); 257 } 258 259 /** 260 * Gets the Provisioning URL. 261 * 262 * @returns A pointer to char buffer containing the URL string. 263 * 264 */ GetProvisioningUrl(void) const265 const char *GetProvisioningUrl(void) const { return mProvisioningUrl; } 266 267 /** 268 * Sets the Provisioning URL. 269 * 270 * @param[in] aProvisioningUrl A pointer to the Provisioning URL (may be `nullptr` to set URL to empty string). 271 * 272 * @retval kErrorNone Successfully set the Provisioning URL. 273 * @retval kErrorInvalidArgs @p aProvisioningUrl is invalid (too long). 274 * 275 */ 276 Error SetProvisioningUrl(const char *aProvisioningUrl); 277 278 /** 279 * Returns the Commissioner Session ID. 280 * 281 * @returns The Commissioner Session ID. 282 * 283 */ GetSessionId(void) const284 uint16_t GetSessionId(void) const { return mSessionId; } 285 286 /** 287 * Indicates whether or not the Commissioner role is active. 288 * 289 * @returns TRUE if the Commissioner role is active, FALSE otherwise. 290 * 291 */ IsActive(void) const292 bool IsActive(void) const { return mState == kStateActive; } 293 294 /** 295 * Indicates whether or not the Commissioner role is disabled. 296 * 297 * @returns TRUE if the Commissioner role is disabled, FALSE otherwise. 298 * 299 */ IsDisabled(void) const300 bool IsDisabled(void) const { return mState == kStateDisabled; } 301 302 /** 303 * Gets the Commissioner State. 304 * 305 * @returns The Commissioner State. 306 * 307 */ GetState(void) const308 State GetState(void) const { return mState; } 309 310 /** 311 * Sends MGMT_COMMISSIONER_GET. 312 * 313 * @param[in] aTlvs A pointer to Commissioning Data TLVs. 314 * @param[in] aLength The length of requested TLVs in bytes. 315 * 316 * @retval kErrorNone Send MGMT_COMMISSIONER_GET successfully. 317 * @retval kErrorNoBufs Insufficient buffer space to send. 318 * @retval kErrorInvalidState Commissioner service is not started. 319 * 320 */ 321 Error SendMgmtCommissionerGetRequest(const uint8_t *aTlvs, uint8_t aLength); 322 323 /** 324 * Sends MGMT_COMMISSIONER_SET. 325 * 326 * @param[in] aDataset A reference to Commissioning Data. 327 * @param[in] aTlvs A pointer to user specific Commissioning Data TLVs. 328 * @param[in] aLength The length of user specific TLVs in bytes. 329 * 330 * @retval kErrorNone Send MGMT_COMMISSIONER_SET successfully. 331 * @retval kErrorNoBufs Insufficient buffer space to send. 332 * @retval kErrorInvalidState Commissioner service is not started. 333 * 334 */ 335 Error SendMgmtCommissionerSetRequest(const CommissioningDataset &aDataset, const uint8_t *aTlvs, uint8_t aLength); 336 337 /** 338 * Returns a reference to the AnnounceBeginClient instance. 339 * 340 * @returns A reference to the AnnounceBeginClient instance. 341 * 342 */ GetAnnounceBeginClient(void)343 AnnounceBeginClient &GetAnnounceBeginClient(void) { return mAnnounceBegin; } 344 345 /** 346 * Returns a reference to the EnergyScanClient instance. 347 * 348 * @returns A reference to the EnergyScanClient instance. 349 * 350 */ GetEnergyScanClient(void)351 EnergyScanClient &GetEnergyScanClient(void) { return mEnergyScan; } 352 353 /** 354 * Returns a reference to the PanIdQueryClient instance. 355 * 356 * @returns A reference to the PanIdQueryClient instance. 357 * 358 */ GetPanIdQueryClient(void)359 PanIdQueryClient &GetPanIdQueryClient(void) { return mPanIdQuery; } 360 361 private: 362 static constexpr uint32_t kPetitionAttemptDelay = 5; // COMM_PET_ATTEMPT_DELAY (seconds) 363 static constexpr uint8_t kPetitionRetryCount = 2; // COMM_PET_RETRY_COUNT 364 static constexpr uint32_t kPetitionRetryDelay = 1; // COMM_PET_RETRY_DELAY (seconds) 365 static constexpr uint32_t kKeepAliveTimeout = 50; // TIMEOUT_COMM_PET (seconds) 366 static constexpr uint32_t kRemoveJoinerDelay = 20; // Delay to remove successfully joined joiner 367 368 static constexpr uint32_t kJoinerSessionTimeoutMillis = 369 1000 * OPENTHREAD_CONFIG_COMMISSIONER_JOINER_SESSION_TIMEOUT; // Expiration time for active Joiner session 370 371 enum ResignMode : uint8_t 372 { 373 kSendKeepAliveToResign, 374 kDoNotSendKeepAlive, 375 }; 376 377 struct Joiner 378 { 379 enum Type : uint8_t 380 { 381 kTypeUnused = 0, // Need to be 0 to ensure `memset()` clears all `Joiners` 382 kTypeAny, 383 kTypeEui64, 384 kTypeDiscerner, 385 }; 386 387 TimeMilli mExpirationTime; 388 389 union 390 { 391 Mac::ExtAddress mEui64; 392 JoinerDiscerner mDiscerner; 393 } mSharedId; 394 395 JoinerPskd mPskd; 396 Type mType; 397 398 void CopyToJoinerInfo(otJoinerInfo &aJoiner) const; 399 }; 400 401 Error Stop(ResignMode aResignMode); 402 Joiner *GetUnusedJoinerEntry(void); 403 Joiner *FindJoinerEntry(const Mac::ExtAddress *aEui64); 404 Joiner *FindJoinerEntry(const JoinerDiscerner &aDiscerner); 405 Joiner *FindBestMatchingJoinerEntry(const Mac::ExtAddress &aReceivedJoinerId); 406 void RemoveJoinerEntry(Joiner &aJoiner); 407 408 Error AddJoiner(const Mac::ExtAddress *aEui64, 409 const JoinerDiscerner *aDiscerner, 410 const char *aPskd, 411 uint32_t aTimeout); 412 Error RemoveJoiner(const Mac::ExtAddress *aEui64, const JoinerDiscerner *aDiscerner, uint32_t aDelay); 413 void RemoveJoiner(Joiner &aJoiner, uint32_t aDelay); 414 415 void HandleTimer(void); 416 void HandleJoinerExpirationTimer(void); 417 418 static void HandleMgmtCommissionerSetResponse(void *aContext, 419 otMessage *aMessage, 420 const otMessageInfo *aMessageInfo, 421 Error aResult); 422 void HandleMgmtCommissionerSetResponse(Coap::Message *aMessage, 423 const Ip6::MessageInfo *aMessageInfo, 424 Error aResult); 425 static void HandleMgmtCommissionerGetResponse(void *aContext, 426 otMessage *aMessage, 427 const otMessageInfo *aMessageInfo, 428 Error aResult); 429 void HandleMgmtCommissionerGetResponse(Coap::Message *aMessage, 430 const Ip6::MessageInfo *aMessageInfo, 431 Error aResult); 432 static void HandleLeaderPetitionResponse(void *aContext, 433 otMessage *aMessage, 434 const otMessageInfo *aMessageInfo, 435 Error aResult); 436 void HandleLeaderPetitionResponse(Coap::Message *aMessage, const Ip6::MessageInfo *aMessageInfo, Error aResult); 437 static void HandleLeaderKeepAliveResponse(void *aContext, 438 otMessage *aMessage, 439 const otMessageInfo *aMessageInfo, 440 Error aResult); 441 void HandleLeaderKeepAliveResponse(Coap::Message *aMessage, const Ip6::MessageInfo *aMessageInfo, Error aResult); 442 443 static void HandleSecureAgentConnectEvent(SecureTransport::ConnectEvent aEvent, void *aContext); 444 void HandleSecureAgentConnectEvent(SecureTransport::ConnectEvent aEvent); 445 446 template <Uri kUri> void HandleTmf(Coap::Message &aMessage, const Ip6::MessageInfo &aMessageInfo); 447 448 void HandleRelayReceive(Coap::Message &aMessage, const Ip6::MessageInfo &aMessageInfo); 449 450 void HandleJoinerSessionTimer(void); 451 452 void SendJoinFinalizeResponse(const Coap::Message &aRequest, StateTlv::State aState); 453 454 static Error SendRelayTransmit(void *aContext, Message &aMessage, const Ip6::MessageInfo &aMessageInfo); 455 Error SendRelayTransmit(Message &aMessage, const Ip6::MessageInfo &aMessageInfo); 456 457 void ComputeBloomFilter(SteeringData &aSteeringData) const; 458 void SendCommissionerSet(void); 459 Error SendPetition(void); 460 void SendKeepAlive(void); 461 void SendKeepAlive(uint16_t aSessionId); 462 463 void SetState(State aState); 464 void SignalJoinerEvent(JoinerEvent aEvent, const Joiner *aJoiner) const; 465 void LogJoinerEntry(const char *aAction, const Joiner &aJoiner) const; 466 467 static const char *StateToString(State aState); 468 469 using JoinerExpirationTimer = TimerMilliIn<Commissioner, &Commissioner::HandleJoinerExpirationTimer>; 470 using CommissionerTimer = TimerMilliIn<Commissioner, &Commissioner::HandleTimer>; 471 using JoinerSessionTimer = TimerMilliIn<Commissioner, &Commissioner::HandleJoinerSessionTimer>; 472 473 Joiner mJoiners[OPENTHREAD_CONFIG_COMMISSIONER_MAX_JOINER_ENTRIES]; 474 475 Joiner *mActiveJoiner; 476 Ip6::InterfaceIdentifier mJoinerIid; 477 uint16_t mJoinerPort; 478 uint16_t mJoinerRloc; 479 uint16_t mSessionId; 480 uint8_t mTransmitAttempts; 481 JoinerExpirationTimer mJoinerExpirationTimer; 482 CommissionerTimer mTimer; 483 JoinerSessionTimer mJoinerSessionTimer; 484 485 AnnounceBeginClient mAnnounceBegin; 486 EnergyScanClient mEnergyScan; 487 PanIdQueryClient mPanIdQuery; 488 489 Ip6::Netif::UnicastAddress mCommissionerAloc; 490 491 ProvisioningUrlTlv::StringType mProvisioningUrl; 492 CommissionerIdTlv::StringType mCommissionerId; 493 494 State mState; 495 496 Callback<StateCallback> mStateCallback; 497 Callback<JoinerCallback> mJoinerCallback; 498 }; 499 500 DeclareTmfHandler(Commissioner, kUriDatasetChanged); 501 DeclareTmfHandler(Commissioner, kUriRelayRx); 502 DeclareTmfHandler(Commissioner, kUriJoinerFinalize); 503 504 } // namespace MeshCoP 505 506 DefineMapEnum(otCommissionerState, MeshCoP::Commissioner::State); 507 DefineMapEnum(otCommissionerJoinerEvent, MeshCoP::Commissioner::JoinerEvent); 508 509 } // namespace ot 510 511 #endif // OPENTHREAD_FTD && OPENTHREAD_CONFIG_COMMISSIONER_ENABLE 512 513 #endif // COMMISSIONER_HPP_ 514