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 IPv6 addresses.
32  */
33 
34 #ifndef IP6_ADDRESS_HPP_
35 #define IP6_ADDRESS_HPP_
36 
37 #include "openthread-core-config.h"
38 
39 #include <openthread/ip6.h>
40 
41 #include "common/as_core_type.hpp"
42 #include "common/clearable.hpp"
43 #include "common/encoding.hpp"
44 #include "common/equatable.hpp"
45 #include "common/num_utils.hpp"
46 #include "common/numeric_limits.hpp"
47 #include "common/string.hpp"
48 #include "mac/mac_types.hpp"
49 
50 namespace ot {
51 
52 namespace Ip4 {
53 // Forward declaration for SynthesizeFromIp4Address
54 class Address;
55 } // namespace Ip4
56 
57 namespace Ip6 {
58 
59 /**
60  * @addtogroup core-ip6-ip6
61  *
62  * @{
63  *
64  */
65 
66 /**
67  * Represents the Network Prefix of an IPv6 address (most significant 64 bits of the address).
68  *
69  */
70 OT_TOOL_PACKED_BEGIN
71 class NetworkPrefix : public otIp6NetworkPrefix, public Equatable<NetworkPrefix>, public Clearable<NetworkPrefix>
72 {
73 public:
74     static constexpr uint8_t kSize   = OT_IP6_PREFIX_SIZE;   ///< Size in bytes.
75     static constexpr uint8_t kLength = kSize * kBitsPerByte; ///< Length of Network Prefix in bits.
76 
77     /**
78      * Generates and sets the Network Prefix to a crypto-secure random Unique Local Address (ULA) based
79      * on the pattern `fdxx:xxxx:xxxx:` (RFC 4193).
80      *
81      * @retval kErrorNone     Successfully generated a random ULA Network Prefix
82      * @retval kErrorFailed   Failed to generate random ULA Network Prefix.
83      *
84      */
85     Error GenerateRandomUla(void);
86 
87 } OT_TOOL_PACKED_END;
88 
89 /**
90  * Represents an IPv6 Prefix.
91  *
92  */
93 OT_TOOL_PACKED_BEGIN
94 class Prefix : public otIp6Prefix, public Clearable<Prefix>, public Unequatable<Prefix>
95 {
96 public:
97     static constexpr uint8_t kMaxSize   = OT_IP6_ADDRESS_SIZE;     ///< Max (byte) size of a prefix.
98     static constexpr uint8_t kMaxLength = kMaxSize * kBitsPerByte; ///< Max length of a prefix in bits.
99 
100     static constexpr uint16_t kInfoStringSize = OT_IP6_PREFIX_STRING_SIZE; ///< Info string size (`ToString()`).
101 
102     /**
103      * Defines the fixed-length `String` object returned from `ToString()`.
104      *
105      */
106     typedef String<kInfoStringSize> InfoString;
107 
108     /**
109      * Gets the prefix as a pointer to a byte array.
110      *
111      * @returns A pointer to a byte array containing the Prefix.
112      *
113      */
GetBytes(void) const114     const uint8_t *GetBytes(void) const { return mPrefix.mFields.m8; }
115 
116     /**
117      * Gets the subnet ID of the prefix.
118      *
119      * @returns The 16-bit subnet ID.
120      *
121      */
GetSubnetId(void) const122     uint16_t GetSubnetId(void) const { return BigEndian::HostSwap16(mPrefix.mFields.m16[3]); }
123 
124     /**
125      * Gets the prefix length (in bits).
126      *
127      * @returns The prefix length (in bits).
128      *
129      */
GetLength(void) const130     uint8_t GetLength(void) const { return mLength; }
131 
132     /**
133      * Returns the size (in bytes) of the prefix.
134      *
135      * @returns The size (in bytes) of the prefix array.
136      *
137      */
GetBytesSize(void) const138     uint8_t GetBytesSize(void) const { return SizeForLength(mLength); }
139 
140     /**
141      * Sets the prefix.
142      *
143      * @param[in] aPrefix  A pointer to buffer containing the prefix bytes.
144      * @param[in] aLength  The length or prefix in bits.
145      *
146      */
147     void Set(const uint8_t *aPrefix, uint8_t aLength);
148 
149     /**
150      * Sets the prefix from a given Network Prefix.
151      *
152      * @param[in] aNetworkPrefix    A Network Prefix.
153      *
154      */
Set(const NetworkPrefix & aNetworkPrefix)155     void Set(const NetworkPrefix &aNetworkPrefix) { Set(aNetworkPrefix.m8, NetworkPrefix::kLength); }
156 
157     /**
158      * Sets the subnet ID of the prefix.
159      *
160      * @param[in] aSubnetId  A 16-bit subnet ID.
161      *
162      */
SetSubnetId(uint16_t aSubnetId)163     void SetSubnetId(uint16_t aSubnetId) { mPrefix.mFields.m16[3] = BigEndian::HostSwap16(aSubnetId); }
164 
165     /**
166      * Set the prefix length.
167      *
168      * @param[in] aLength  The new prefix length (in bits).
169      *
170      */
SetLength(uint8_t aLength)171     void SetLength(uint8_t aLength) { mLength = aLength; }
172 
173     /**
174      * Sets the bits after the prefix length to 0.
175      *
176      */
177     void Tidy(void);
178 
179     /**
180      * Indicates whether prefix length is valid (smaller or equal to max length).
181      *
182      * @retval TRUE   The prefix length is valid.
183      * @retval FALSE  The prefix length is not valid.
184      *
185      */
IsValid(void) const186     bool IsValid(void) const { return (mLength <= kMaxLength); }
187 
188     /**
189      * Indicates whether the prefix is a Link-Local prefix.
190      *
191      * @retval TRUE   The prefix is a Link-Local prefix.
192      * @retval FALSE  The prefix is not a Link-Local prefix.
193      *
194      */
195     bool IsLinkLocal(void) const;
196 
197     /**
198      * Indicates whether the prefix is a Multicast prefix.
199      *
200      * @retval TRUE   The prefix is a Multicast prefix.
201      * @retval FALSE  The prefix is not a Multicast prefix.
202      *
203      */
204     bool IsMulticast(void) const;
205 
206     /**
207      * Indicates whether the prefix is a Unique-Local prefix.
208      *
209      * @retval TRUE   The prefix is a Unique-Local prefix.
210      * @retval FALSE  The prefix is not a Unique-Local prefix.
211      *
212      */
213     bool IsUniqueLocal(void) const;
214 
215     /**
216      * Indicates whether the prefix is equal to a given prefix.
217      *
218      * @param[in] aPrefixBytes   A pointer to buffer containing the prefix bytes to compare with.
219      * @param[in] aPrefixLength  The length of prefix (in bits) specified by @p aPrefixBytes.
220      *
221      * @retval TRUE   If the prefix is equal to the specified prefix by @p aPrefixBytes and @p aPrefixLength.
222      * @retval FALSE  If the prefix is not equal to the specified prefix by @p aPrefixBytes and @p aPrefixLength.
223      *
224      */
225     bool IsEqual(const uint8_t *aPrefixBytes, uint8_t aPrefixLength) const;
226 
227     /**
228      * Indicates whether the prefix contains a sub-prefix.
229      *
230      * @param[in] aSubPrefix  A sub-prefix.
231      *
232      * @retval TRUE   The prefix contains the @p aSubPrefix
233      * @retval FALSE  The prefix does not contains the @p aSubPrefix.
234      *
235      */
236     bool ContainsPrefix(const Prefix &aSubPrefix) const;
237 
238     /**
239      * Indicates whether the prefix contains a sub-prefix (given as a `NetworkPrefix`).
240      *
241      * @param[in] aSubPrefix  A sub-prefix (as a `NetworkPrefix`).
242      *
243      * @retval TRUE   The prefix contains the @p aSubPrefix
244      * @retval FALSE  The prefix does not contains the @p aSubPrefix.
245      *
246      */
247     bool ContainsPrefix(const NetworkPrefix &aSubPrefix) const;
248 
249     /**
250      * Overloads operator `==` to evaluate whether or not two prefixes are equal.
251      *
252      * @param[in]  aOther  The other prefix to compare with.
253      *
254      * @retval TRUE   If the two prefixes are equal.
255      * @retval FALSE  If the two prefixes are not equal.
256      *
257      */
258     bool operator==(const Prefix &aOther) const;
259 
260     /**
261      * Overloads operator `<` to compare two prefixes.
262      *
263      * If the two prefixes have the same length N, then the bytes are compared directly (as two big-endian N-bit
264      * numbers). If the two prefix have different lengths, the shorter prefix is padded by `0` bit up to the longer
265      * prefix length N before the bytes are compared (as big-endian N-bit numbers). If all bytes are equal, the prefix
266      * with shorter length is considered smaller.
267      *
268      * @param[in] aOther  The other prefix to compare against.
269      *
270      * @retval TRUE   If the prefix is smaller than @p aOther.
271      * @retval FALSE  If the prefix is not smaller than @p aOther.
272      *
273      */
274     bool operator<(const Prefix &aOther) const;
275 
276     /**
277      * Converts a prefix length (in bits) to size (number of bytes).
278      *
279      * @param[in] aLength   A prefix length (in bits).
280      *
281      * @returns The size (in bytes) of the prefix.
282      *
283      */
SizeForLength(uint8_t aLength)284     static uint8_t SizeForLength(uint8_t aLength) { return BytesForBitSize(aLength); }
285 
286     /**
287      * Returns the number of IPv6 prefix bits that match.
288      *
289      * @param[in]  aPrefixA     A pointer to a byte array containing a first prefix.
290      * @param[in]  aPrefixB     A pointer to a byte array containing a second prefix.
291      * @param[in]  aMaxSize     Number of bytes of the two prefixes.
292      *
293      * @returns The number of prefix bits that match.
294      *
295      */
296     static uint8_t MatchLength(const uint8_t *aPrefixA, const uint8_t *aPrefixB, uint8_t aMaxSize);
297 
298     /**
299      * Indicates whether or not a given prefix length is valid for use as a NAT64 prefix.
300      *
301      * A NAT64 prefix must have one of the following lengths: 32, 40, 48, 56, 64, or 96 (per RFC 6052).
302      *
303      * @param[in] aLength The length of the prefix.
304      *
305      * @retval TRUE   If the prefix has a valid length for use as a NAT64 prefix.
306      * @retval FALSE  If the prefix does not have a valid length for use as a NAT64 prefix.
307      *
308      */
309     static bool IsValidNat64PrefixLength(uint8_t aLength);
310 
311     /**
312      * Indicates whether or not the prefix has a valid length for use as a NAT64 prefix.
313      *
314      * A NAT64 prefix must have one of the following lengths: 32, 40, 48, 56, 64, or 96 (per RFC 6052).
315      *
316      * @retval TRUE   If the prefix has a valid length for use as a NAT64 prefix.
317      * @retval FALSE  If the prefix does not have a valid length for use as a NAT64 prefix.
318      *
319      */
IsValidNat64(void) const320     bool IsValidNat64(void) const { return IsValidNat64PrefixLength(mLength); }
321 
322     /**
323      * Parses a given IPv6 prefix string and sets the prefix.
324      *
325      * @param[in]  aString         A null-terminated string, with format "<prefix>/<plen>"
326      *
327      * @retval kErrorNone          Successfully parsed the IPv6 prefix from @p aString.
328      * @retval kErrorParse         Failed to parse the IPv6 prefix from @p aString.
329      *
330      */
331     Error FromString(const char *aString);
332 
333     /**
334      * Converts the prefix to a string.
335      *
336      * The IPv6 prefix string is formatted as "%x:%x:%x:...[::]/plen".
337      *
338      * @returns An `InfoString` containing the string representation of the Prefix.
339      *
340      */
341     InfoString ToString(void) const;
342 
343     /**
344      * Converts the prefix to a string.
345      *
346      * The IPv6 prefix string is formatted as "%x:%x:%x:...[::]/plen".
347      *
348      * If the resulting string does not fit in @p aBuffer (within its @p aSize characters), the string will be
349      * truncated but the outputted string is always null-terminated.
350      *
351      * @param[out] aBuffer   A pointer to a char array to output the string (MUST NOT be `nullptr`).
352      * @param[in]  aSize     The size of @p aBuffer (in bytes).
353      *
354      */
355     void ToString(char *aBuffer, uint16_t aSize) const;
356 
357 private:
358     uint8_t ByteAfterTidy(uint8_t aIndex);
359     void    ToString(StringWriter &aWriter) const;
360 } OT_TOOL_PACKED_END;
361 
362 /**
363  * Represents the Interface Identifier of an IPv6 address.
364  *
365  */
366 OT_TOOL_PACKED_BEGIN
367 class InterfaceIdentifier : public otIp6InterfaceIdentifier,
368                             public Equatable<InterfaceIdentifier>,
369                             public Clearable<InterfaceIdentifier>
370 {
371     friend class Address;
372 
373 public:
374     static constexpr uint8_t kSize = OT_IP6_IID_SIZE; ///< Size of an IPv6 Interface Identifier (in bytes).
375 
376     static constexpr uint16_t kInfoStringSize = 17; ///< Max chars for the info string (`ToString()`).
377 
378     /**
379      * Defines the fixed-length `String` object returned from `ToString()`.
380      *
381      */
382     typedef String<kInfoStringSize> InfoString;
383 
384     /**
385      * Indicates whether or not the Interface Identifier is unspecified.
386      *
387      * @retval true  If the Interface Identifier is unspecified.
388      * @retval false If the Interface Identifier is not unspecified.
389      *
390      */
391     bool IsUnspecified(void) const;
392 
393     /**
394      * Indicates whether or not the Interface Identifier is reserved (RFC 5453).
395      *
396      * @retval true  If the Interface Identifier is reserved.
397      * @retval false If the Interface Identifier is not reserved.
398      *
399      */
400     bool IsReserved(void) const;
401 
402     /**
403      * Indicates whether or not the Interface Identifier is Subnet-Router Anycast (RFC 4291).
404      *
405      * @retval TRUE   If the Interface Identifier is a Subnet-Router Anycast address.
406      * @retval FALSE  If the Interface Identifier is not a Subnet-Router Anycast address.
407      *
408      */
409     bool IsSubnetRouterAnycast(void) const;
410 
411     /**
412      * Indicates whether or not the Interface Identifier is Reserved Subnet Anycast (RFC 2526).
413      *
414      * @retval TRUE   If the Interface Identifier is a Reserved Subnet Anycast address.
415      * @retval FALSE  If the Interface Identifier is not a Reserved Subnet Anycast address.
416      *
417      */
418     bool IsReservedSubnetAnycast(void) const;
419 
420     /**
421      * Generates and sets the Interface Identifier to a crypto-secure random byte sequence.
422      *
423      */
424     void GenerateRandom(void);
425 
426     /**
427      * Gets the Interface Identifier as a pointer to a byte array.
428      *
429      * @returns A pointer to a byte array (of size `kSize`) containing the Interface Identifier.
430      *
431      */
GetBytes(void) const432     const uint8_t *GetBytes(void) const { return mFields.m8; }
433 
434     /**
435      * Sets the Interface Identifier from a given byte array.
436      *
437      * @param[in] aBuffer    Pointer to an array containing the Interface Identifier. `kSize` bytes from the buffer
438      *                       are copied to form the Interface Identifier.
439      *
440      */
441     void SetBytes(const uint8_t *aBuffer);
442 
443     /**
444      * Sets the Interface Identifier from a given IEEE 802.15.4 Extended Address.
445      *
446      * @param[in] aExtAddress  An Extended Address.
447      *
448      */
449     void SetFromExtAddress(const Mac::ExtAddress &aExtAddress);
450 
451     /**
452      * Converts the Interface Identifier to an IEEE 802.15.4 Extended Address.
453      *
454      * @param[out]  aExtAddress  A reference to an Extended Address where the converted address is placed.
455      *
456      */
457     void ConvertToExtAddress(Mac::ExtAddress &aExtAddress) const;
458 
459     /**
460      * Converts the Interface Identifier to an IEEE 802.15.4 MAC Address.
461      *
462      * @param[out]  aMacAddress  A reference to a MAC Address where the converted address is placed.
463      *
464      */
465     void ConvertToMacAddress(Mac::Address &aMacAddress) const;
466 
467     /**
468      * Sets the Interface Identifier to Routing/Anycast Locator pattern `0000:00ff:fe00:xxxx` with a given
469      * locator (RLOC16 or ALOC16) value.
470      *
471      * @param[in]  aLocator    RLOC16 or ALOC16.
472      *
473      */
474     void SetToLocator(uint16_t aLocator);
475 
476     /**
477      * Indicates whether or not the Interface Identifier matches the locator pattern `0000:00ff:fe00:xxxx`.
478      *
479      * @retval TRUE   If the IID matches the locator pattern.
480      * @retval FALSE  If the IID does not match the locator pattern.
481      *
482      */
483     bool IsLocator(void) const;
484 
485     /**
486      * Indicates whether or not the Interface Identifier (IID) matches a Routing Locator (RLOC).
487      *
488      * In addition to checking that the IID matches the locator pattern (`0000:00ff:fe00:xxxx`), this method also
489      * checks that the locator value is a valid RLOC16.
490      *
491      * @retval TRUE   If the IID matches a RLOC address.
492      * @retval FALSE  If the IID does not match a RLOC address.
493      *
494      */
495     bool IsRoutingLocator(void) const;
496 
497     /**
498      * Indicates whether or not the Interface Identifier (IID) matches an Anycast Locator (ALOC).
499      *
500      * In addition to checking that the IID matches the locator pattern (`0000:00ff:fe00:xxxx`), this method also
501      * checks that the locator value is any valid ALOC16 (0xfc00 - 0xfcff).
502      *
503      * @retval TRUE   If the IID matches a ALOC address.
504      * @retval FALSE  If the IID does not match a ALOC address.
505      *
506      */
507     bool IsAnycastLocator(void) const;
508 
509     /**
510      * Indicates whether or not the Interface Identifier (IID) matches a Service Anycast Locator (ALOC).
511      *
512      * In addition to checking that the IID matches the locator pattern (`0000:00ff:fe00:xxxx`), this method also
513      * checks that the locator value is a valid Service ALOC16 (0xfc10 – 0xfc2f).
514      *
515      * @retval TRUE   If the IID matches a ALOC address.
516      * @retval FALSE  If the IID does not match a ALOC address.
517      *
518      */
519     bool IsAnycastServiceLocator(void) const;
520 
521     /**
522      * Gets the Interface Identifier (IID) address locator fields.
523      *
524      * Assumes the IID to match the locator pattern `0000:00ff:fe00:xxxx` (does not explicitly check this)
525      * and returns the last `uint16` portion of the IID.
526      *
527      * @returns The RLOC16 or ALOC16.
528      *
529      */
GetLocator(void) const530     uint16_t GetLocator(void) const { return BigEndian::HostSwap16(mFields.m16[3]); }
531 
532     /**
533      * Sets the Interface Identifier (IID) address locator field.
534      *
535      * Unlike `SetToLocator()`, this method only changes the last 2 bytes of the IID and keeps the rest of the address
536      * as before.
537      *
538      * @param[in]  aLocator   RLOC16 or ALOC16.
539      *
540      */
SetLocator(uint16_t aLocator)541     void SetLocator(uint16_t aLocator) { mFields.m16[3] = BigEndian::HostSwap16(aLocator); }
542 
543     /**
544      * Applies a prefix to IID.
545      *
546      * If the prefix length is longer than 64 bits, the prefix bits after 64 are written into the IID. This method only
547      * changes the bits in IID up the prefix length and keeps the rest of the bits in IID as before.
548      *
549      * @param[in] aPrefix   An IPv6 prefix.
550      *
551      */
552     void ApplyPrefix(const Prefix &aPrefix);
553 
554     /**
555      * Converts an Interface Identifier to a string.
556      *
557      * @returns An `InfoString` containing the string representation of the Interface Identifier.
558      *
559      */
560     InfoString ToString(void) const;
561 
562 private:
563     static constexpr uint8_t kAloc16Mask            = 0xfc; // The mask for ALOC16.
564     static constexpr uint8_t kRloc16ReservedBitMask = 0x02; // The mask for the reserved bit of RLOC16.
565 
566 } OT_TOOL_PACKED_END;
567 
568 /**
569  * Implements an IPv6 address object.
570  *
571  */
572 OT_TOOL_PACKED_BEGIN
573 class Address : public otIp6Address, public Equatable<Address>, public Clearable<Address>
574 {
575     friend class Prefix;
576     friend class InterfaceIdentifier;
577 
578 public:
579     static constexpr uint8_t kAloc16Mask = InterfaceIdentifier::kAloc16Mask; ///< The mask for ALOC16.
580 
581     static constexpr uint8_t kSize = OT_IP6_ADDRESS_SIZE; ///< Size of an IPv6 Address (in bytes).
582 
583     static constexpr uint16_t kInfoStringSize = OT_IP6_ADDRESS_STRING_SIZE; ///< String Size for IPv6 address.
584 
585     // IPv6 Address Scopes
586     static constexpr uint8_t kNodeLocalScope      = 0;  ///< Node-Local scope
587     static constexpr uint8_t kInterfaceLocalScope = 1;  ///< Interface-Local scope
588     static constexpr uint8_t kLinkLocalScope      = 2;  ///< Link-Local scope
589     static constexpr uint8_t kRealmLocalScope     = 3;  ///< Realm-Local scope
590     static constexpr uint8_t kAdminLocalScope     = 4;  ///< Admin-Local scope
591     static constexpr uint8_t kSiteLocalScope      = 5;  ///< Site-Local scope
592     static constexpr uint8_t kOrgLocalScope       = 8;  ///< Organization-Local scope
593     static constexpr uint8_t kGlobalScope         = 14; ///< Global scope
594 
595     /**
596      * Defines IPv6 address type filter.
597      *
598      */
599     enum TypeFilter : uint8_t
600     {
601         kTypeAny,                           ///< Accept any IPv6 address (unicast or multicast).
602         kTypeUnicast,                       ///< Accept unicast IPv6 addresses only.
603         kTypeMulticast,                     ///< Accept multicast IPv6 addresses only.
604         kTypeMulticastLargerThanRealmLocal, ///< Accept multicast IPv6 addresses with scope larger than Realm Local.
605     };
606 
607     /**
608      * Defines the fixed-length `String` object returned from `ToString()`.
609      *
610      */
611     typedef String<kInfoStringSize> InfoString;
612 
613     /**
614      * Gets the IPv6 address as a pointer to a byte array.
615      *
616      * @returns A pointer to a byte array containing the IPv6 address.
617      *
618      */
GetBytes(void) const619     const uint8_t *GetBytes(void) const { return mFields.m8; }
620 
621     /**
622      * Sets the IPv6 address from a given byte array.
623      *
624      * @param[in] aBuffer    Pointer to an array containing the IPv6 address. `kSize` bytes from the buffer
625      *                       are copied to form the IPv6 address.
626      *
627      */
SetBytes(const uint8_t * aBuffer)628     void SetBytes(const uint8_t *aBuffer) { memcpy(mFields.m8, aBuffer, kSize); }
629 
630     /**
631      * Indicates whether or not the IPv6 address is the Unspecified Address.
632      *
633      * @retval TRUE   If the IPv6 address is the Unspecified Address.
634      * @retval FALSE  If the IPv6 address is not the Unspecified Address.
635      *
636      */
637     bool IsUnspecified(void) const;
638 
639     /**
640      * Indicates whether or not the IPv6 address is the Loopback Address.
641      *
642      * @retval TRUE   If the IPv6 address is the Loopback Address.
643      * @retval FALSE  If the IPv6 address is not the Loopback Address.
644      *
645      */
646     bool IsLoopback(void) const;
647 
648     /**
649      * Indicates whether or not the IPv6 address is a Link-Local unicast address.
650      *
651      * @retval TRUE   If the IPv6 address is a Link-Local unicast address.
652      * @retval FALSE  If the IPv6 address is not a Link-Local unicast address.
653      *
654      */
655     bool IsLinkLocalUnicast(void) const;
656 
657     /**
658      * Sets the IPv6 address to a Link-Local address with Interface Identifier generated from a given
659      * MAC Extended Address.
660      *
661      * @param[in]  aExtAddress  A MAC Extended Address (used to generate the IID).
662      *
663      */
664     void SetToLinkLocalAddress(const Mac::ExtAddress &aExtAddress);
665 
666     /**
667      * Sets the IPv6 address to a Link-Local address with a given Interface Identifier.
668      *
669      * @param[in]  aIid   An Interface Identifier.
670      *
671      */
672     void SetToLinkLocalAddress(const InterfaceIdentifier &aIid);
673 
674     /**
675      * Indicates whether or not the IPv6 address is multicast address.
676      *
677      * @retval TRUE   If the IPv6 address is a multicast address.
678      * @retval FALSE  If the IPv6 address scope is not a multicast address.
679      *
680      */
IsMulticast(void) const681     bool IsMulticast(void) const { return mFields.m8[0] == 0xff; }
682 
683     /**
684      * Indicates whether or not the IPv6 address is a link-local multicast address.
685      *
686      * @retval TRUE   If the IPv6 address is a link-local multicast address.
687      * @retval FALSE  If the IPv6 address scope is not a link-local multicast address.
688      *
689      */
690     bool IsLinkLocalMulticast(void) const;
691 
692     /**
693      * Indicates whether or not the IPv6 address is a link-local unicast or a link-local multicast address.
694      *
695      * @retval TRUE   If the IPv6 address is a link-local unicast or multicast address.
696      * @retval FALSE  If the IPv6 address is not a link-local unicast and not a link-local multicast address.
697      *
698      */
699     bool IsLinkLocalUnicastOrMulticast(void) const;
700 
701     /**
702      * Indicates whether or not the IPv6 address is a link-local all nodes multicast address (ff02::01).
703      *
704      * @retval TRUE   If the IPv6 address is a link-local all nodes multicast address.
705      * @retval FALSE  If the IPv6 address is not a link-local all nodes multicast address.
706      *
707      */
708     bool IsLinkLocalAllNodesMulticast(void) const;
709 
710     /**
711      * Sets the IPv6 address to the link-local all nodes multicast address (ff02::01).
712      *
713      */
714     void SetToLinkLocalAllNodesMulticast(void);
715 
716     /**
717      * Indicates whether or not the IPv6 address is a link-local all routers multicast address (ff02::02).
718      *
719      * @retval TRUE   If the IPv6 address is a link-local all routers multicast address.
720      * @retval FALSE  If the IPv6 address is not a link-local all routers multicast address.
721      *
722      */
723     bool IsLinkLocalAllRoutersMulticast(void) const;
724 
725     /**
726      * Sets the IPv6 address to the link-local all routers multicast address (ff02::02).
727      *
728      */
729     void SetToLinkLocalAllRoutersMulticast(void);
730 
731     /**
732      * Indicates whether or not the IPv6 address is a realm-local multicast address.
733      *
734      * @retval TRUE   If the IPv6 address is a realm-local multicast address.
735      * @retval FALSE  If the IPv6 address scope is not a realm-local multicast address.
736      *
737      */
738     bool IsRealmLocalMulticast(void) const;
739 
740     /**
741      * Indicates whether or not the IPv6 address is a realm-local all nodes multicast address (ff03::01).
742      *
743      * @retval TRUE   If the IPv6 address is a realm-local all nodes multicast address.
744      * @retval FALSE  If the IPv6 address is not a realm-local all nodes multicast address.
745      *
746      */
747     bool IsRealmLocalAllNodesMulticast(void) const;
748 
749     /**
750      * Sets the IPv6 address to the realm-local all nodes multicast address (ff03::01)
751      *
752      */
753     void SetToRealmLocalAllNodesMulticast(void);
754 
755     /**
756      * Indicates whether or not the IPv6 address is a realm-local all routers multicast address (ff03::02).
757      *
758      * @retval TRUE   If the IPv6 address is a realm-local all routers multicast address.
759      * @retval FALSE  If the IPv6 address is not a realm-local all routers multicast address.
760      *
761      */
762     bool IsRealmLocalAllRoutersMulticast(void) const;
763 
764     /**
765      * Sets the IPv6 address to the realm-local all routers multicast address (ff03::02).
766      *
767      */
768     void SetToRealmLocalAllRoutersMulticast(void);
769 
770     /**
771      * Indicates whether or not the IPv6 address is a realm-local all MPL forwarders address (ff03::fc).
772      *
773      * @retval TRUE   If the IPv6 address is a realm-local all MPL forwarders address.
774      * @retval FALSE  If the IPv6 address is not a realm-local all MPL forwarders address.
775      *
776      */
777     bool IsRealmLocalAllMplForwarders(void) const;
778 
779     /**
780      * Sets the the IPv6 address to the realm-local all MPL forwarders address (ff03::fc).
781      *
782      */
783     void SetToRealmLocalAllMplForwarders(void);
784 
785     /**
786      * Indicates whether or not the IPv6 address is multicast larger than realm local.
787      *
788      * @retval TRUE   If the IPv6 address is multicast larger than realm local.
789      * @retval FALSE  If the IPv6 address is not multicast or the scope is not larger than realm local.
790      *
791      */
792     bool IsMulticastLargerThanRealmLocal(void) const;
793 
794     /**
795      * Sets the IPv6 address to a Routing Locator (RLOC) IPv6 address with a given Network Prefix and
796      * RLOC16 value.
797      *
798      * @param[in]  aNetworkPrefix    A Network Prefix.
799      * @param[in]  aRloc16           A RLOC16 value.
800      *
801      */
SetToRoutingLocator(const NetworkPrefix & aNetworkPrefix,uint16_t aRloc16)802     void SetToRoutingLocator(const NetworkPrefix &aNetworkPrefix, uint16_t aRloc16)
803     {
804         SetToLocator(aNetworkPrefix, aRloc16);
805     }
806 
807     /**
808      * Sets the IPv6 address to a Anycast Locator (ALOC) IPv6 address with a given Network Prefix and
809      * ALOC16 value.
810      *
811      * @param[in]  aNetworkPrefix    A Network Prefix.
812      * @param[in]  aAloc16           A ALOC16 value.
813      *
814      */
SetToAnycastLocator(const NetworkPrefix & aNetworkPrefix,uint16_t aAloc16)815     void SetToAnycastLocator(const NetworkPrefix &aNetworkPrefix, uint16_t aAloc16)
816     {
817         SetToLocator(aNetworkPrefix, aAloc16);
818     }
819 
820     /**
821      * Indicates whether or not the IPv6 address follows the IPv4-mapped format.
822      *
823      * An IPv4-mapped IPv6 address consists of an 80-bit prefix of zeros, the next 16 bits set to  ones, and the
824      * remaining, least-significant 32 bits contain the IPv4 address, e.g., `::ffff:192.0.2.128` representing
825      * `192.0.2.128` IPv4 address.
826      *
827      * @retval TRUE   If the IPv6 address follows the IPv4-mapped format.
828      * @retval FALSE  If the IPv6 address does not follow the IPv4-mapped format.
829      *
830      */
831     bool IsIp4Mapped(void) const;
832 
833     /**
834      * Sets the IPv6 address to follow the IPv4-mapped IPv6 address for a given IPv4 address.
835      *
836      * @param[in] aIp4Address  An IPv4 address.
837      *
838      */
839     void SetToIp4Mapped(const Ip4::Address &aIp4Address);
840 
841     /**
842      * Returns the Network Prefix of the IPv6 address (most significant 64 bits of the address).
843      *
844      * @returns A reference to the Network Prefix.
845      *
846      */
GetPrefix(void) const847     const NetworkPrefix &GetPrefix(void) const
848     {
849         return static_cast<const NetworkPrefix &>(mFields.mComponents.mNetworkPrefix);
850     }
851 
852     /**
853      * Gets a prefix of the IPv6 address with a given length.
854      *
855      * @param[in]  aLength  The length of prefix in bits.
856      * @param[out] aPrefix  A reference to a prefix to output the fetched prefix.
857      *
858      */
GetPrefix(uint8_t aLength,Prefix & aPrefix) const859     void GetPrefix(uint8_t aLength, Prefix &aPrefix) const { aPrefix.Set(mFields.m8, aLength); }
860 
861     /**
862      * Indicates whether the IPv6 address matches a given prefix.
863      *
864      * @param[in] aPrefix  An IPv6 prefix to match with.
865      *
866      * @retval TRUE   The IPv6 address matches the @p aPrefix.
867      * @retval FALSE  The IPv6 address does not match the @p aPrefix.
868      *
869      */
870     bool MatchesPrefix(const Prefix &aPrefix) const;
871 
872     /**
873      * Indicates whether the IPv6 address matches a given prefix.
874      *
875      * @param[in]  aPrefix         A buffer containing the prefix.
876      * @param[in]  aPrefixLength   The prefix length (in bits).
877      *
878      * @retval TRUE   The IPv6 address matches the @p aPrefix.
879      * @retval FALSE  The IPv6 address does not match the @p aPrefix.
880      *
881      */
882     bool MatchesPrefix(const uint8_t *aPrefix, uint8_t aPrefixLength) const;
883 
884     /**
885      * Sets the IPv6 address prefix.
886      *
887      * Only changes the first @p aPrefixLength bits of the address and keeps the rest of the bits in the
888      * address as before.
889      *
890      * @param[in]  aPrefix         A buffer containing the prefix.
891      * @param[in]  aPrefixLength   The prefix length (in bits).
892      *
893      */
SetPrefix(const uint8_t * aPrefix,uint8_t aPrefixLength)894     void SetPrefix(const uint8_t *aPrefix, uint8_t aPrefixLength) { CopyBits(mFields.m8, aPrefix, aPrefixLength); }
895 
896     /**
897      * Sets the IPv6 address prefix to the given Network Prefix.
898      *
899      * @param[in]  aNetworkPrefix   A Network Prefix.
900      *
901      */
902     void SetPrefix(const NetworkPrefix &aNetworkPrefix);
903 
904     /**
905      * Sets the IPv6 address prefix.
906      *
907      * Only changes the initial prefix length bits of the IPv6 address and keeps the rest of the bits in
908      * the address as before.
909      *
910      * @param[in]  aPrefix         An IPv6 prefix.
911      *
912      */
913     void SetPrefix(const Prefix &aPrefix);
914 
915     /**
916      * Sets the prefix content of the Prefix-Based Multicast Address.
917      *
918      * @param[in]  aPrefix         A buffer containing the prefix.
919      * @param[in]  aPrefixLength   The prefix length (in bits).
920      *
921      */
922     void SetMulticastNetworkPrefix(const uint8_t *aPrefix, uint8_t aPrefixLength);
923 
924     /**
925      * Sets the prefix content of Prefix-Based Multicast Address.
926      *
927      * @param[in]  aNetworkPrefix   A reference to a Network Prefix.
928      *
929      */
SetMulticastNetworkPrefix(const NetworkPrefix & aNetworkPrefix)930     void SetMulticastNetworkPrefix(const NetworkPrefix &aNetworkPrefix)
931     {
932         SetMulticastNetworkPrefix(aNetworkPrefix.m8, NetworkPrefix::kLength);
933     }
934 
935     /**
936      * Sets the prefix content of Prefix-Based Multicast Address.
937      *
938      * @param[in]  aPrefix  An IPv6 Prefix.
939      *
940      */
SetMulticastNetworkPrefix(const Prefix & aPrefix)941     void SetMulticastNetworkPrefix(const Prefix &aPrefix)
942     {
943         SetMulticastNetworkPrefix(aPrefix.GetBytes(), aPrefix.GetLength());
944     }
945 
946     /**
947      * Returns the Interface Identifier of the IPv6 address.
948      *
949      * @returns A reference to the Interface Identifier.
950      *
951      */
GetIid(void) const952     const InterfaceIdentifier &GetIid(void) const
953     {
954         return static_cast<const InterfaceIdentifier &>(mFields.mComponents.mIid);
955     }
956 
957     /**
958      * Returns the Interface Identifier of the IPv6 address.
959      *
960      * @returns A reference to the Interface Identifier.
961      *
962      */
GetIid(void)963     InterfaceIdentifier &GetIid(void) { return static_cast<InterfaceIdentifier &>(mFields.mComponents.mIid); }
964 
965     /**
966      * Sets the Interface Identifier.
967      *
968      * @param[in]  aIid  An Interface Identifier.
969      *
970      */
SetIid(const InterfaceIdentifier & aIid)971     void SetIid(const InterfaceIdentifier &aIid) { GetIid() = aIid; }
972 
973     /**
974      * Returns the IPv6 address scope.
975      *
976      * @returns The IPv6 address scope.
977      *
978      */
979     uint8_t GetScope(void) const;
980 
981     /**
982      * Returns the number of IPv6 prefix bits that match.
983      *
984      * @param[in]  aOther  The IPv6 address to match against.
985      *
986      * @returns The number of IPv6 prefix bits that match.
987      *
988      */
989     uint8_t PrefixMatch(const Address &aOther) const;
990 
991     /**
992      * Indicates whether address matches a given type filter.
993      *
994      * @param[in] aFilter   An address type filter.
995      *
996      * @retval TRUE   The address matches @p aFilter.
997      * @retval FALSE  The address does not match @p aFilter.
998      *
999      */
1000     bool MatchesFilter(TypeFilter aFilter) const;
1001 
1002     /**
1003      * Sets the IPv6 address by performing NAT64 address translation from a given IPv4 address as specified
1004      * in RFC 6052.
1005      *
1006      * The NAT64 @p aPrefix MUST have one of the following lengths: 32, 40, 48, 56, 64, or 96, otherwise the behavior
1007      * of this method is undefined.
1008      *
1009      * @param[in] aPrefix      The prefix to use for IPv4/IPv6 translation.
1010      * @param[in] aIp4Address  The IPv4 address to translate to IPv6.
1011      *
1012      */
1013     void SynthesizeFromIp4Address(const Prefix &aPrefix, const Ip4::Address &aIp4Address);
1014 
1015     /**
1016      * Converts an IPv6 address string to binary.
1017      *
1018      * @param[in]  aString  A pointer to the null-terminated string.
1019      *
1020      * @retval kErrorNone          Successfully parsed the IPv6 address string.
1021      * @retval kErrorParse         Failed to parse the IPv6 address string.
1022      *
1023      */
1024     Error FromString(const char *aString);
1025 
1026     /**
1027      * Converts the IPv6 address to a string.
1028      *
1029      * The IPv6 address string is formatted as 16 hex values separated by ':' (i.e., "%x:%x:%x:...:%x").
1030      *
1031      * @returns An `InfoString` representing the IPv6 address.
1032      *
1033      */
1034     InfoString ToString(void) const;
1035 
1036     /**
1037      * Convert the IPv6 address to a C string.
1038      *
1039      * The IPv6 address string is formatted as 16 hex values separated by ':' (i.e., "%x:%x:%x:...:%x").
1040      *
1041      * If the resulting string does not fit in @p aBuffer (within its @p aSize characters), the string will be
1042      * truncated but the outputted string is always null-terminated.
1043      *
1044      * @param[out] aBuffer   A pointer to a char array to output the string (MUST NOT be `nullptr`).
1045      * @param[in]  aSize     The size of @p aBuffer (in bytes).
1046      *
1047      */
1048     void ToString(char *aBuffer, uint16_t aSize) const;
1049 
1050     /**
1051      * Overloads operator `<` to compare two IPv6 addresses.
1052      *
1053      * @param[in] aOther  The other IPv6 address to compare with.
1054      *
1055      * @retval true   The IPv6 address is smaller than @p aOther.
1056      * @retval false  The IPv6 address is larger than or equal to @p aOther.
1057      *
1058      */
operator <(const Address & aOther) const1059     bool operator<(const Address &aOther) const { return memcmp(mFields.m8, aOther.mFields.m8, sizeof(Address)) < 0; }
1060 
1061 private:
1062     static constexpr uint8_t kMulticastNetworkPrefixLengthOffset = 3; // Prefix-Based Multicast Address (RFC3306)
1063     static constexpr uint8_t kMulticastNetworkPrefixOffset       = 4; // Prefix-Based Multicast Address (RFC3306)
1064 
1065     void SetToLocator(const NetworkPrefix &aNetworkPrefix, uint16_t aLocator);
1066     void ToString(StringWriter &aWriter) const;
1067     void AppendHexWords(StringWriter &aWriter, uint8_t aLength) const;
1068 
1069     static const Address &GetLinkLocalAllNodesMulticast(void);
1070     static const Address &GetLinkLocalAllRoutersMulticast(void);
1071     static const Address &GetRealmLocalAllNodesMulticast(void);
1072     static const Address &GetRealmLocalAllRoutersMulticast(void);
1073     static const Address &GetRealmLocalAllMplForwarders(void);
1074 
1075     static void CopyBits(uint8_t *aDst, const uint8_t *aSrc, uint8_t aNumBits);
1076 
1077     Error ParseFrom(const char *aString, char aTerminatorChar);
1078 
1079 } OT_TOOL_PACKED_END;
1080 
1081 /**
1082  * @}
1083  *
1084  */
1085 
1086 } // namespace Ip6
1087 
1088 DefineCoreType(otIp6NetworkPrefix, Ip6::NetworkPrefix);
1089 DefineCoreType(otIp6Prefix, Ip6::Prefix);
1090 DefineCoreType(otIp6InterfaceIdentifier, Ip6::InterfaceIdentifier);
1091 DefineCoreType(otIp6Address, Ip6::Address);
1092 
1093 } // namespace ot
1094 
1095 #endif // IP6_ADDRESS_HPP_
1096