1 /*
2  *  Copyright (c) 2019-2022, 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 #include "common/array.hpp"
30 #include "common/encoding.hpp"
31 #include "common/string.hpp"
32 #include "net/ip4_types.hpp"
33 #include "net/ip6_address.hpp"
34 
35 #include "test_util.h"
36 
37 namespace ot {
38 
39 template <typename AddressType> struct TestVector
40 {
41     const char   *mString;
42     const uint8_t mAddr[sizeof(AddressType)];
43     Error         mError;
44 };
45 
checkAddressFromString(TestVector<AddressType> * aTestVector)46 template <typename AddressType> static void checkAddressFromString(TestVector<AddressType> *aTestVector)
47 {
48     Error       error;
49     AddressType address;
50 
51     address.Clear();
52 
53     error = address.FromString(aTestVector->mString);
54 
55     printf("%-42s -> %-42s\n", aTestVector->mString,
56            (error == kErrorNone) ? address.ToString().AsCString() : "(parse error)");
57 
58     VerifyOrQuit(error == aTestVector->mError, "Address::FromString returned unexpected error code");
59 
60     if (error == kErrorNone)
61     {
62         VerifyOrQuit(0 == memcmp(address.GetBytes(), aTestVector->mAddr, sizeof(AddressType)),
63                      "Address::FromString parsing failed");
64     }
65 }
66 
TestIp6AddressFromString(void)67 void TestIp6AddressFromString(void)
68 {
69     typedef TestVector<Ip6::Address> Ip6AddressTestVector;
70 
71     Ip6AddressTestVector testVectors[] = {
72         // Valid full IPv6 address.
73         {"0102:0304:0506:0708:090a:0b0c:0d0e:0f00",
74          {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x00},
75          kErrorNone},
76 
77         // Valid full address using capital letters.
78         {"0102:0304:0506:0708:090A:0B0C:0D0E:0F00",
79          {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x00},
80          kErrorNone},
81 
82         // Valid full IPv6 address with mixed capital and small letters.
83         {"0102:0304:0506:0708:090a:0B0C:0d0E:0F00",
84          {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f, 0x00},
85          kErrorNone},
86 
87         // Short prefix and full IID.
88         {"fd11::abcd:e0e0:d10e:0001",
89          {0xfd, 0x11, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xab, 0xcd, 0xe0, 0xe0, 0xd1, 0x0e, 0x00, 0x01},
90          kErrorNone},
91 
92         // Valid IPv6 address with unnecessary :: symbol.
93         {"fd11:1234:5678:abcd::abcd:e0e0:d10e:1000",
94          {0xfd, 0x11, 0x12, 0x34, 0x56, 0x78, 0xab, 0xcd, 0xab, 0xcd, 0xe0, 0xe0, 0xd1, 0x0e, 0x10, 0x00},
95          kErrorNone},
96 
97         // Short multicast address.
98         {"ff03::0b",
99          {0xff, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x0b},
100          kErrorNone},
101 
102         // Unspecified address.
103         {"::", {0}, kErrorNone},
104 
105         // Starts with ::
106         {"::1:2:3:4",
107          {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x02, 0x00, 0x03, 0x00, 0x04},
108          kErrorNone},
109 
110         // Ends with ::
111         {"1001:2002:3003:4004::",
112          {0x10, 0x01, 0x20, 0x02, 0x30, 0x03, 0x40, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
113          kErrorNone},
114 
115         // Valid embedded IPv4 address.
116         {"64:ff9b::100.200.15.4",
117          {0x00, 0x64, 0xff, 0x9b, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x64, 0xc8, 0x0f, 0x04},
118          kErrorNone},
119 
120         // Valid embedded IPv4 address.
121         {"2001:db8::abc:def1:127.0.0.1",
122          {0x20, 0x01, 0x0d, 0xb8, 0x00, 0x00, 0x00, 0x00, 0x0a, 0xbc, 0xde, 0xf1, 0x7f, 0x00, 0x00, 0x01},
123          kErrorNone},
124 
125         // Valid embedded IPv4 address.
126         {"1:2:3:4:5:6:127.1.2.3",
127          {0x00, 0x01, 0x00, 0x02, 0x00, 0x03, 0x00, 0x04, 0x00, 0x05, 0x00, 0x06, 0x7f, 0x01, 0x02, 0x03},
128          kErrorNone},
129 
130         // Two :: should cause a parse error.
131         {"2001:db8::a::b", {0}, kErrorParse},
132 
133         // The "g" and "h" are not the hex characters.
134         {"2001:db8::abcd:efgh", {0}, kErrorParse},
135 
136         // Too many colons.
137         {"1:2:3:4:5:6:7:8:9", {0}, kErrorParse},
138 
139         // Too many characters in a single part.
140         {"2001:db8::abc:def12:1:2", {0}, kErrorParse},
141 
142         // Invalid embedded IPv4 address.
143         {"64:ff9b::123.231.0.257", {0}, kErrorParse},
144 
145         // Invalid embedded IPv4 address.
146         {"64:ff9b::1.22.33", {0}, kErrorParse},
147 
148         // Invalid embedded IPv4 address.
149         {"64:ff9b::1.22.33.44.5", {0}, kErrorParse},
150 
151         // Too long with embedded IPv4 address.
152         {"1:2:3:4:5:6:7:127.1.2.3", {0}, kErrorParse},
153 
154         // Invalid embedded IPv4 address.
155         {".", {0}, kErrorParse},
156 
157         // Invalid embedded IPv4 address.
158         {":.", {0}, kErrorParse},
159 
160         // Invalid embedded IPv4 address.
161         {"::.", {0}, kErrorParse},
162 
163         // Invalid embedded IPv4 address.
164         {":f:0:0:c:0:f:f:.", {0}, kErrorParse},
165     };
166 
167     for (Ip6AddressTestVector &testVector : testVectors)
168     {
169         checkAddressFromString(&testVector);
170     }
171 
172     // Validate parsing all test vectors now as an IPv6 prefix.
173 
174     for (Ip6AddressTestVector &testVector : testVectors)
175     {
176         constexpr uint16_t kMaxString = 80;
177 
178         Ip6::Prefix prefix;
179         char        string[kMaxString];
180         uint16_t    length;
181 
182         length = StringLength(testVector.mString, kMaxString);
183         memcpy(string, testVector.mString, length);
184         VerifyOrQuit(length + sizeof("/128") <= kMaxString);
185         strcpy(&string[length], "/128");
186 
187         printf("%s\n", string);
188 
189         VerifyOrQuit(prefix.FromString(string) == testVector.mError);
190 
191         if (testVector.mError == kErrorNone)
192         {
193             VerifyOrQuit(memcmp(prefix.GetBytes(), testVector.mAddr, sizeof(Ip6::Address)) == 0);
194             VerifyOrQuit(prefix.GetLength() == 128);
195         }
196     }
197 }
198 
TestIp6PrefixFromString(void)199 void TestIp6PrefixFromString(void)
200 {
201     Ip6::Prefix prefix;
202 
203     SuccessOrQuit(prefix.FromString("::/128"));
204     VerifyOrQuit(prefix.GetLength() == 128);
205 
206     SuccessOrQuit(prefix.FromString("::/0128"));
207     VerifyOrQuit(prefix.GetLength() == 128);
208 
209     SuccessOrQuit(prefix.FromString("::/5"));
210     VerifyOrQuit(prefix.GetLength() == 5);
211 
212     SuccessOrQuit(prefix.FromString("::/0"));
213     VerifyOrQuit(prefix.GetLength() == 0);
214 
215     VerifyOrQuit(prefix.FromString("::") == kErrorParse);
216     VerifyOrQuit(prefix.FromString("::/") == kErrorParse);
217     VerifyOrQuit(prefix.FromString("::/129") == kErrorParse);
218     VerifyOrQuit(prefix.FromString(":: /12") == kErrorParse);
219     VerifyOrQuit(prefix.FromString("::/a1") == kErrorParse);
220     VerifyOrQuit(prefix.FromString("::/12 ") == kErrorParse);
221 }
222 
TestIp4AddressFromString(void)223 void TestIp4AddressFromString(void)
224 {
225     typedef TestVector<Ip4::Address> Ip4AddressTestVector;
226 
227     Ip4AddressTestVector testVectors[] = {
228         {"0.0.0.0", {0, 0, 0, 0}, kErrorNone},
229         {"255.255.255.255", {255, 255, 255, 255}, kErrorNone},
230         {"127.0.0.1", {127, 0, 0, 1}, kErrorNone},
231         {"1.2.3.4", {1, 2, 3, 4}, kErrorNone},
232         {"001.002.003.004", {1, 2, 3, 4}, kErrorNone},
233         {"00000127.000.000.000001", {127, 0, 0, 1}, kErrorNone},
234         {"123.231.0.256", {0}, kErrorParse},    // Invalid byte value.
235         {"100123.231.0.256", {0}, kErrorParse}, // Invalid byte value.
236         {"1.22.33", {0}, kErrorParse},          // Too few bytes.
237         {"1.22.33.44.5", {0}, kErrorParse},     // Too many bytes.
238         {"a.b.c.d", {0}, kErrorParse},          // Wrong digit char.
239         {"123.23.45 .12", {0}, kErrorParse},    // Extra space.
240         {".", {0}, kErrorParse},                // Invalid.
241     };
242 
243     for (Ip4AddressTestVector &testVector : testVectors)
244     {
245         checkAddressFromString(&testVector);
246     }
247 }
248 
249 struct CidrTestVector
250 {
251     const char   *mString;
252     const uint8_t mAddr[sizeof(otIp4Address)];
253     const uint8_t mLength;
254     Error         mError;
255 };
256 
checkCidrFromString(CidrTestVector * aTestVector)257 static void checkCidrFromString(CidrTestVector *aTestVector)
258 {
259     Error     error;
260     Ip4::Cidr cidr;
261 
262     cidr.Clear();
263 
264     error = cidr.FromString(aTestVector->mString);
265 
266     printf("%-42s -> %-42s\n", aTestVector->mString,
267            (error == kErrorNone) ? cidr.ToString().AsCString() : "(parse error)");
268 
269     VerifyOrQuit(error == aTestVector->mError, "Address::FromString returned unexpected error code");
270 
271     if (error == kErrorNone)
272     {
273         VerifyOrQuit(0 == memcmp(cidr.GetBytes(), aTestVector->mAddr, sizeof(aTestVector->mAddr)),
274                      "Cidr::FromString parsing failed");
275         VerifyOrQuit(cidr.mLength == aTestVector->mLength, "Cidr::FromString parsing failed");
276     }
277 }
278 
TestIp4CidrFromString(void)279 void TestIp4CidrFromString(void)
280 {
281     CidrTestVector testVectors[] = {
282         {"0.0.0.0/0", {0, 0, 0, 0}, 0, kErrorNone},
283         {"255.255.255.255/32", {255, 255, 255, 255}, 32, kErrorNone},
284         {"127.0.0.1/8", {127, 0, 0, 1}, 8, kErrorNone},
285         {"1.2.3.4/24", {1, 2, 3, 4}, 24, kErrorNone},
286         {"001.002.003.004/20", {1, 2, 3, 4}, 20, kErrorNone},
287         {"00000127.000.000.000001/8", {127, 0, 0, 1}, 8, kErrorNone},
288         // Valid suffix, invalid address
289         {"123.231.0.256/4", {0}, 0, kErrorParse},    // Invalid byte value.
290         {"100123.231.0.256/4", {0}, 0, kErrorParse}, // Invalid byte value.
291         {"1.22.33/4", {0}, 0, kErrorParse},          // Too few bytes.
292         {"1.22.33.44.5/4", {0}, 0, kErrorParse},     // Too many bytes.
293         {"a.b.c.d/4", {0}, 0, kErrorParse},          // Wrong digit char.
294         {"123.23.45 .12/4", {0}, 0, kErrorParse},    // Extra space.
295         {"./4", {0}, 0, kErrorParse},                // Invalid.
296         // valid address, invalid suffix
297         {"1.2.3.4/33", {0}, 0, kErrorParse},       // Prefix length too large
298         {"1.2.3.4/12345678", {0}, 0, kErrorParse}, // Prefix length too large?
299         {"1.2.3.4/12a", {0}, 0, kErrorParse},      // Extra char after prefix length.
300         {"1.2.3.4/-1", {0}, 0, kErrorParse},       // Not even a non-negative integer.
301         {"1.2.3.4/3.14", {0}, 0, kErrorParse},     // Not even a integer.
302         {"1.2.3.4/abcd", {0}, 0, kErrorParse},     // Not even a number.
303         {"1.2.3.4/", {0}, 0, kErrorParse},         // Where is the suffix?
304         {"1.2.3.4", {0}, 0, kErrorParse},          // Where is the suffix?
305         // invalid address and invalid suffix
306         {"123.231.0.256/41", {0}, 0, kErrorParse},     // Invalid byte value.
307         {"100123.231.0.256/abc", {0}, 0, kErrorParse}, // Invalid byte value.
308         {"1.22.33", {0}, 0, kErrorParse},              // Too few bytes.
309         {"1.22.33.44.5/36", {0}, 0, kErrorParse},      // Too many bytes.
310         {"a.b.c.d/99", {0}, 0, kErrorParse},           // Wrong digit char.
311         {"123.23.45 .12", {0}, 0, kErrorParse},        // Extra space.
312         {".", {0}, 0, kErrorParse},                    // Invalid.
313     };
314 
315     for (CidrTestVector &testVector : testVectors)
316     {
317         checkCidrFromString(&testVector);
318     }
319 }
320 
CheckPrefix(const Ip6::Address & aAddress,const uint8_t * aPrefix,uint8_t aPrefixLength)321 bool CheckPrefix(const Ip6::Address &aAddress, const uint8_t *aPrefix, uint8_t aPrefixLength)
322 {
323     // Check the first aPrefixLength bits of aAddress to match the given aPrefix.
324 
325     bool matches = true;
326 
327     for (uint8_t bit = 0; bit < aPrefixLength; bit++)
328     {
329         uint8_t index = bit / kBitsPerByte;
330         uint8_t mask  = (0x80 >> (bit % kBitsPerByte));
331 
332         if ((aAddress.mFields.m8[index] & mask) != (aPrefix[index] & mask))
333         {
334             matches = false;
335             break;
336         }
337     }
338 
339     return matches;
340 }
341 
CheckPrefixInIid(const Ip6::InterfaceIdentifier & aIid,const uint8_t * aPrefix,uint8_t aPrefixLength)342 bool CheckPrefixInIid(const Ip6::InterfaceIdentifier &aIid, const uint8_t *aPrefix, uint8_t aPrefixLength)
343 {
344     // Check the IID to contain the prefix bits (applicable when prefix length is longer than 64).
345 
346     bool matches = true;
347 
348     for (uint8_t bit = 64; bit < aPrefixLength; bit++)
349     {
350         uint8_t index = bit / kBitsPerByte;
351         uint8_t mask  = (0x80 >> (bit % kBitsPerByte));
352 
353         if ((aIid.mFields.m8[index - 8] & mask) != (aPrefix[index] & mask))
354         {
355             matches = false;
356             break;
357         }
358     }
359 
360     return matches;
361 }
362 
CheckInterfaceId(const Ip6::Address & aAddress1,const Ip6::Address & aAddress2,uint8_t aPrefixLength)363 bool CheckInterfaceId(const Ip6::Address &aAddress1, const Ip6::Address &aAddress2, uint8_t aPrefixLength)
364 {
365     // Check whether all the bits after aPrefixLength of the two given IPv6 Address match or not.
366 
367     bool matches = true;
368 
369     for (size_t bit = aPrefixLength; bit < sizeof(Ip6::Address) * kBitsPerByte; bit++)
370     {
371         uint8_t index = bit / kBitsPerByte;
372         uint8_t mask  = (0x80 >> (bit % kBitsPerByte));
373 
374         if ((aAddress1.mFields.m8[index] & mask) != (aAddress2.mFields.m8[index] & mask))
375         {
376             matches = false;
377             break;
378         }
379     }
380 
381     return matches;
382 }
383 
TestIp6AddressSetPrefix(void)384 void TestIp6AddressSetPrefix(void)
385 {
386     const uint8_t kPrefixes[][OT_IP6_ADDRESS_SIZE] = {
387         {0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef, 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef},
388         {0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55},
389         {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00},
390         {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
391     };
392 
393     Ip6::Address address;
394     Ip6::Address allZeroAddress;
395     Ip6::Address allOneAddress;
396     Ip6::Prefix  ip6Prefix;
397 
398     allZeroAddress.Clear();
399     memset(&allOneAddress, 0xff, sizeof(allOneAddress));
400 
401     for (auto prefix : kPrefixes)
402     {
403         memcpy(address.mFields.m8, prefix, sizeof(address));
404         printf("Prefix is %s\n", address.ToString().AsCString());
405 
406         for (size_t prefixLength = 0; prefixLength <= sizeof(Ip6::Address) * kBitsPerByte; prefixLength++)
407         {
408             ip6Prefix.Clear();
409             ip6Prefix.Set(prefix, prefixLength);
410 
411             address = allZeroAddress;
412             address.SetPrefix(ip6Prefix);
413             printf("   prefix-len:%-3zu --> %s\n", prefixLength, address.ToString().AsCString());
414             VerifyOrQuit(CheckPrefix(address, prefix, prefixLength), "Prefix does not match after SetPrefix()");
415             VerifyOrQuit(CheckInterfaceId(address, allZeroAddress, prefixLength),
416                          "SetPrefix changed bits beyond the prefix length");
417 
418             address = allOneAddress;
419             address.SetPrefix(ip6Prefix);
420             VerifyOrQuit(CheckPrefix(address, prefix, prefixLength), "Prefix does not match after SetPrefix()");
421             VerifyOrQuit(CheckInterfaceId(address, allOneAddress, prefixLength),
422                          "SetPrefix changed bits beyond the prefix length");
423 
424             address = allZeroAddress;
425             address.GetIid().ApplyPrefix(ip6Prefix);
426             VerifyOrQuit(CheckPrefixInIid(address.GetIid(), prefix, prefixLength), "IID is not correct");
427             VerifyOrQuit(CheckInterfaceId(address, allZeroAddress, prefixLength),
428                          "Iid:ApplyPrefrix() changed bits beyond the prefix length");
429 
430             address = allOneAddress;
431             address.GetIid().ApplyPrefix(ip6Prefix);
432             VerifyOrQuit(CheckPrefixInIid(address.GetIid(), prefix, prefixLength), "IID is not correct");
433             VerifyOrQuit(CheckInterfaceId(address, allOneAddress, prefixLength),
434                          "Iid:ApplyPrefrix() changed bits beyond the prefix length");
435         }
436     }
437 }
438 
PrefixFrom(const char * aAddressString,uint8_t aPrefixLength)439 Ip6::Prefix PrefixFrom(const char *aAddressString, uint8_t aPrefixLength)
440 {
441     Ip6::Prefix  prefix;
442     Ip6::Address address;
443 
444     SuccessOrQuit(address.FromString(aAddressString));
445     prefix.Set(address.GetBytes(), aPrefixLength);
446 
447     return prefix;
448 }
449 
TestIp6Prefix(void)450 void TestIp6Prefix(void)
451 {
452     const uint8_t kPrefixes[][OT_IP6_ADDRESS_SIZE] = {
453         {0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef, 0x01, 0x23, 0x45, 0x67, 0x89, 0xab, 0xcd, 0xef},
454         {0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55, 0xaa, 0x55},
455         {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff},
456     };
457 
458     Ip6::Prefix  prefix;
459     Ip6::Address address1, address2;
460 
461     for (auto prefixBytes : kPrefixes)
462     {
463         memcpy(address1.mFields.m8, prefixBytes, sizeof(address1));
464         address2 = address1;
465         address2.mFields.m8[0] ^= 0x80; // Change first bit.
466 
467         for (uint8_t prefixLength = 1; prefixLength <= Ip6::Prefix::kMaxLength; prefixLength++)
468         {
469             prefix.Set(prefixBytes, prefixLength);
470 
471             printf("Prefix %s\n", prefix.ToString().AsCString());
472 
473             VerifyOrQuit(prefix.GetLength() == prefixLength);
474             VerifyOrQuit(prefix.IsValid());
475             VerifyOrQuit(prefix.IsEqual(prefixBytes, prefixLength));
476 
477             VerifyOrQuit(address1.MatchesPrefix(prefix));
478             VerifyOrQuit(!address2.MatchesPrefix(prefix));
479 
480             VerifyOrQuit(prefix == prefix);
481             VerifyOrQuit(!(prefix < prefix));
482 
483             for (uint8_t subPrefixLength = 1; subPrefixLength <= prefixLength; subPrefixLength++)
484             {
485                 Ip6::Prefix subPrefix;
486 
487                 subPrefix.Set(prefixBytes, subPrefixLength);
488 
489                 VerifyOrQuit(prefix.ContainsPrefix(subPrefix));
490 
491                 if (prefixLength == subPrefixLength)
492                 {
493                     VerifyOrQuit(prefix == subPrefix);
494                     VerifyOrQuit(prefix.IsEqual(subPrefix.GetBytes(), subPrefix.GetLength()));
495                     VerifyOrQuit(!(subPrefix < prefix));
496                 }
497                 else
498                 {
499                     VerifyOrQuit(prefix != subPrefix);
500                     VerifyOrQuit(!prefix.IsEqual(subPrefix.GetBytes(), subPrefix.GetLength()));
501                     VerifyOrQuit(subPrefix < prefix);
502                 }
503             }
504 
505             for (uint8_t bitNumber = 0; bitNumber < prefixLength; bitNumber++)
506             {
507                 Ip6::Prefix prefix2;
508                 uint8_t     mask  = static_cast<uint8_t>(1U << (7 - (bitNumber & 7)));
509                 uint8_t     index = (bitNumber / 8);
510                 bool        isPrefixSmaller;
511 
512                 prefix2 = prefix;
513                 VerifyOrQuit(prefix == prefix2);
514 
515                 // Flip the `bitNumber` bit between `prefix` and `prefix2`
516 
517                 prefix2.mPrefix.mFields.m8[index] ^= mask;
518                 VerifyOrQuit(prefix != prefix2);
519 
520                 isPrefixSmaller = ((prefix.GetBytes()[index] & mask) == 0);
521 
522                 VerifyOrQuit((prefix < prefix2) == isPrefixSmaller);
523                 VerifyOrQuit((prefix2 < prefix) == !isPrefixSmaller);
524             }
525         }
526     }
527 
528     {
529         struct TestCase
530         {
531             Ip6::Prefix mPrefixA;
532             Ip6::Prefix mPrefixB;
533         };
534 
535         TestCase kTestCases[] = {
536             {PrefixFrom("fd00::", 16), PrefixFrom("fd01::", 16)},
537             {PrefixFrom("fc00::", 16), PrefixFrom("fd00::", 16)},
538             {PrefixFrom("fd00::", 15), PrefixFrom("fd00::", 16)},
539             {PrefixFrom("fd00::", 16), PrefixFrom("fd00:0::", 32)},
540             {PrefixFrom("2001:0:0:0::", 64), PrefixFrom("fd00::", 8)},
541             {PrefixFrom("2001:dba::", 32), PrefixFrom("fd12:3456:1234:abcd::", 64)},
542             {PrefixFrom("910b:1000:0::", 48), PrefixFrom("910b:2000::", 32)},
543             {PrefixFrom("::", 0), PrefixFrom("fd00::", 8)},
544             {PrefixFrom("::", 0), PrefixFrom("::", 16)},
545             {PrefixFrom("fd00:2:2::", 33), PrefixFrom("fd00:2:2::", 35)},
546             {PrefixFrom("1:2:3:ffff::", 62), PrefixFrom("1:2:3:ffff::", 63)},
547         };
548 
549         printf("\nCompare Prefixes:\n");
550 
551         for (const TestCase &testCase : kTestCases)
552         {
553             printf(" %26s  <  %s\n", testCase.mPrefixA.ToString().AsCString(),
554                    testCase.mPrefixB.ToString().AsCString());
555             VerifyOrQuit(testCase.mPrefixA < testCase.mPrefixB);
556             VerifyOrQuit(!(testCase.mPrefixB < testCase.mPrefixA));
557         }
558     }
559 
560     // `IsLinkLocal()` - should contain `fe80::/10`.
561     VerifyOrQuit(PrefixFrom("fe80::", 10).IsLinkLocal());
562     VerifyOrQuit(PrefixFrom("fe80::", 11).IsLinkLocal());
563     VerifyOrQuit(PrefixFrom("fea0::", 16).IsLinkLocal());
564     VerifyOrQuit(!PrefixFrom("fe80::", 9).IsLinkLocal());
565     VerifyOrQuit(!PrefixFrom("ff80::", 10).IsLinkLocal());
566     VerifyOrQuit(!PrefixFrom("fe00::", 10).IsLinkLocal());
567     VerifyOrQuit(!PrefixFrom("fec0::", 10).IsLinkLocal());
568 
569     // `IsMulticast()` - should contain `ff00::/8`.
570     VerifyOrQuit(PrefixFrom("ff00::", 8).IsMulticast());
571     VerifyOrQuit(PrefixFrom("ff80::", 9).IsMulticast());
572     VerifyOrQuit(PrefixFrom("ffff::", 16).IsMulticast());
573     VerifyOrQuit(!PrefixFrom("ff00::", 7).IsMulticast());
574     VerifyOrQuit(!PrefixFrom("fe00::", 8).IsMulticast());
575 
576     // `IsUniqueLocal()` - should contain `fc00::/7`.
577     VerifyOrQuit(PrefixFrom("fc00::", 7).IsUniqueLocal());
578     VerifyOrQuit(PrefixFrom("fd00::", 8).IsUniqueLocal());
579     VerifyOrQuit(PrefixFrom("fc10::", 16).IsUniqueLocal());
580     VerifyOrQuit(!PrefixFrom("fc00::", 6).IsUniqueLocal());
581     VerifyOrQuit(!PrefixFrom("f800::", 7).IsUniqueLocal());
582     VerifyOrQuit(!PrefixFrom("fe00::", 7).IsUniqueLocal());
583 }
584 
TestIp6PrefixTidy(void)585 void TestIp6PrefixTidy(void)
586 {
587     struct TestVector
588     {
589         uint8_t     originalPrefix[OT_IP6_ADDRESS_SIZE];
590         const char *prefixStringAfterTidy[129];
591     };
592     const TestVector kPrefixes[] = {
593         {
594             .originalPrefix = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
595                                0xff},
596             .prefixStringAfterTidy =
597                 {
598                     "::/0",
599                     "8000::/1",
600                     "c000::/2",
601                     "e000::/3",
602                     "f000::/4",
603                     "f800::/5",
604                     "fc00::/6",
605                     "fe00::/7",
606                     "ff00::/8",
607                     "ff80::/9",
608                     "ffc0::/10",
609                     "ffe0::/11",
610                     "fff0::/12",
611                     "fff8::/13",
612                     "fffc::/14",
613                     "fffe::/15",
614                     "ffff::/16",
615                     "ffff:8000::/17",
616                     "ffff:c000::/18",
617                     "ffff:e000::/19",
618                     "ffff:f000::/20",
619                     "ffff:f800::/21",
620                     "ffff:fc00::/22",
621                     "ffff:fe00::/23",
622                     "ffff:ff00::/24",
623                     "ffff:ff80::/25",
624                     "ffff:ffc0::/26",
625                     "ffff:ffe0::/27",
626                     "ffff:fff0::/28",
627                     "ffff:fff8::/29",
628                     "ffff:fffc::/30",
629                     "ffff:fffe::/31",
630                     "ffff:ffff::/32",
631                     "ffff:ffff:8000::/33",
632                     "ffff:ffff:c000::/34",
633                     "ffff:ffff:e000::/35",
634                     "ffff:ffff:f000::/36",
635                     "ffff:ffff:f800::/37",
636                     "ffff:ffff:fc00::/38",
637                     "ffff:ffff:fe00::/39",
638                     "ffff:ffff:ff00::/40",
639                     "ffff:ffff:ff80::/41",
640                     "ffff:ffff:ffc0::/42",
641                     "ffff:ffff:ffe0::/43",
642                     "ffff:ffff:fff0::/44",
643                     "ffff:ffff:fff8::/45",
644                     "ffff:ffff:fffc::/46",
645                     "ffff:ffff:fffe::/47",
646                     "ffff:ffff:ffff::/48",
647                     "ffff:ffff:ffff:8000::/49",
648                     "ffff:ffff:ffff:c000::/50",
649                     "ffff:ffff:ffff:e000::/51",
650                     "ffff:ffff:ffff:f000::/52",
651                     "ffff:ffff:ffff:f800::/53",
652                     "ffff:ffff:ffff:fc00::/54",
653                     "ffff:ffff:ffff:fe00::/55",
654                     "ffff:ffff:ffff:ff00::/56",
655                     "ffff:ffff:ffff:ff80::/57",
656                     "ffff:ffff:ffff:ffc0::/58",
657                     "ffff:ffff:ffff:ffe0::/59",
658                     "ffff:ffff:ffff:fff0::/60",
659                     "ffff:ffff:ffff:fff8::/61",
660                     "ffff:ffff:ffff:fffc::/62",
661                     "ffff:ffff:ffff:fffe::/63",
662                     "ffff:ffff:ffff:ffff::/64",
663                     "ffff:ffff:ffff:ffff:8000::/65",
664                     "ffff:ffff:ffff:ffff:c000::/66",
665                     "ffff:ffff:ffff:ffff:e000::/67",
666                     "ffff:ffff:ffff:ffff:f000::/68",
667                     "ffff:ffff:ffff:ffff:f800::/69",
668                     "ffff:ffff:ffff:ffff:fc00::/70",
669                     "ffff:ffff:ffff:ffff:fe00::/71",
670                     "ffff:ffff:ffff:ffff:ff00::/72",
671                     "ffff:ffff:ffff:ffff:ff80::/73",
672                     "ffff:ffff:ffff:ffff:ffc0::/74",
673                     "ffff:ffff:ffff:ffff:ffe0::/75",
674                     "ffff:ffff:ffff:ffff:fff0::/76",
675                     "ffff:ffff:ffff:ffff:fff8::/77",
676                     "ffff:ffff:ffff:ffff:fffc::/78",
677                     "ffff:ffff:ffff:ffff:fffe::/79",
678                     "ffff:ffff:ffff:ffff:ffff::/80",
679                     "ffff:ffff:ffff:ffff:ffff:8000::/81",
680                     "ffff:ffff:ffff:ffff:ffff:c000::/82",
681                     "ffff:ffff:ffff:ffff:ffff:e000::/83",
682                     "ffff:ffff:ffff:ffff:ffff:f000::/84",
683                     "ffff:ffff:ffff:ffff:ffff:f800::/85",
684                     "ffff:ffff:ffff:ffff:ffff:fc00::/86",
685                     "ffff:ffff:ffff:ffff:ffff:fe00::/87",
686                     "ffff:ffff:ffff:ffff:ffff:ff00::/88",
687                     "ffff:ffff:ffff:ffff:ffff:ff80::/89",
688                     "ffff:ffff:ffff:ffff:ffff:ffc0::/90",
689                     "ffff:ffff:ffff:ffff:ffff:ffe0::/91",
690                     "ffff:ffff:ffff:ffff:ffff:fff0::/92",
691                     "ffff:ffff:ffff:ffff:ffff:fff8::/93",
692                     "ffff:ffff:ffff:ffff:ffff:fffc::/94",
693                     "ffff:ffff:ffff:ffff:ffff:fffe::/95",
694                     "ffff:ffff:ffff:ffff:ffff:ffff::/96",
695                     // Note: The result of /97 to /112 does not meet RFC requirements:
696                     // 4.2.2.  Handling One 16-Bit 0 Field
697                     // The symbol "::" MUST NOT be used to shorten just one 16-bit 0 field.
698                     "ffff:ffff:ffff:ffff:ffff:ffff:8000::/97",
699                     "ffff:ffff:ffff:ffff:ffff:ffff:c000::/98",
700                     "ffff:ffff:ffff:ffff:ffff:ffff:e000::/99",
701                     "ffff:ffff:ffff:ffff:ffff:ffff:f000::/100",
702                     "ffff:ffff:ffff:ffff:ffff:ffff:f800::/101",
703                     "ffff:ffff:ffff:ffff:ffff:ffff:fc00::/102",
704                     "ffff:ffff:ffff:ffff:ffff:ffff:fe00::/103",
705                     "ffff:ffff:ffff:ffff:ffff:ffff:ff00::/104",
706                     "ffff:ffff:ffff:ffff:ffff:ffff:ff80::/105",
707                     "ffff:ffff:ffff:ffff:ffff:ffff:ffc0::/106",
708                     "ffff:ffff:ffff:ffff:ffff:ffff:ffe0::/107",
709                     "ffff:ffff:ffff:ffff:ffff:ffff:fff0::/108",
710                     "ffff:ffff:ffff:ffff:ffff:ffff:fff8::/109",
711                     "ffff:ffff:ffff:ffff:ffff:ffff:fffc::/110",
712                     "ffff:ffff:ffff:ffff:ffff:ffff:fffe::/111",
713                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff::/112",
714                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:8000/113",
715                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:c000/114",
716                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:e000/115",
717                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:f000/116",
718                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:f800/117",
719                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fc00/118",
720                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fe00/119",
721                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ff00/120",
722                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ff80/121",
723                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffc0/122",
724                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffe0/123",
725                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fff0/124",
726                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fff8/125",
727                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fffc/126",
728                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:fffe/127",
729                     "ffff:ffff:ffff:ffff:ffff:ffff:ffff:ffff/128",
730                 },
731         },
732     };
733 
734     printf("Tidy Prefixes:\n");
735 
736     for (auto test : kPrefixes)
737     {
738         for (uint16_t i = 0; i < GetArrayLength(test.prefixStringAfterTidy); i++)
739         {
740             Ip6::Prefix prefix, answer;
741 
742             SuccessOrQuit(answer.FromString(test.prefixStringAfterTidy[i]));
743             prefix.Set(test.originalPrefix, i);
744             prefix.Tidy();
745 
746             {
747                 Ip6::Prefix::InfoString prefixString = prefix.ToString();
748 
749                 printf("Prefix: %-36s  TidyResult: %-36s\n", test.prefixStringAfterTidy[i],
750                        prefix.ToString().AsCString());
751 
752                 VerifyOrQuit(memcmp(answer.mPrefix.mFields.m8, prefix.mPrefix.mFields.m8,
753                                     sizeof(answer.mPrefix.mFields.m8)) == 0);
754                 VerifyOrQuit(prefix.mLength == answer.mLength);
755                 VerifyOrQuit(strcmp(test.prefixStringAfterTidy[i], prefixString.AsCString()) == 0);
756             }
757         }
758     }
759 }
760 
TestIp4MappedIp6Address(void)761 void TestIp4MappedIp6Address(void)
762 {
763     const uint8_t kIp4Address[] = {192, 0, 2, 33};
764 
765     const char *kInvalidIp4MappedFormats[] = {
766         "8000::ffff:192.0.2.23",     "0:400::ffff:192.0.2.23", "0:0:1::ffff:192.0.2.23", "0:0:0:4:0:ffff:192.0.2.23",
767         "0:0:0:0:1:ffff:192.0.2.23", "::fffe:192.0.2.23",      "::efff:192.0.2.23",
768     };
769 
770     Ip4::Address expectedIp4Address;
771     Ip4::Address ip4Address;
772     Ip6::Address expectedIp6Address;
773     Ip6::Address ip6Address;
774 
775     printf("\nTestIp4MappedIp6Address()\n");
776 
777     expectedIp4Address.SetBytes(kIp4Address);
778 
779     SuccessOrQuit(expectedIp6Address.FromString("::ffff:192.0.2.33"));
780     ip6Address.SetToIp4Mapped(expectedIp4Address);
781 
782     printf("IPv4-mapped IPv6 address: %s\n", ip6Address.ToString().AsCString());
783 
784     VerifyOrQuit(ip6Address.IsIp4Mapped());
785     VerifyOrQuit(ip6Address == expectedIp6Address);
786 
787     SuccessOrQuit(ip4Address.ExtractFromIp4MappedIp6Address(ip6Address));
788     VerifyOrQuit(ip4Address == expectedIp4Address);
789 
790     for (const char *invalidIp4MappedAddr : kInvalidIp4MappedFormats)
791     {
792         SuccessOrQuit(ip6Address.FromString(invalidIp4MappedAddr));
793         printf("Invalid IPv4-mapped IPv6 address: %s -> %s\n", invalidIp4MappedAddr, ip6Address.ToString().AsCString());
794         VerifyOrQuit(!ip6Address.IsIp4Mapped());
795         VerifyOrQuit(ip4Address.ExtractFromIp4MappedIp6Address(ip6Address) != kErrorNone);
796     }
797 }
798 
TestIp4Ip6Translation(void)799 void TestIp4Ip6Translation(void)
800 {
801     struct TestCase
802     {
803         const char *mPrefix;     // NAT64 prefix
804         uint8_t     mLength;     // Prefix length in bits
805         const char *mIp6Address; // Expected IPv6 address (with embedded IPv4 "192.0.2.33").
806     };
807 
808     // The test cases are from RFC 6052 - section 2.4
809 
810     const TestCase kTestCases[] = {
811         {"2001:db8::", 32, "2001:db8:c000:221::"},
812         {"2001:db8:100::", 40, "2001:db8:1c0:2:21::"},
813         {"2001:db8:122::", 48, "2001:db8:122:c000:2:2100::"},
814         {"2001:db8:122:300::", 56, "2001:db8:122:3c0:0:221::"},
815         {"2001:db8:122:344::", 64, "2001:db8:122:344:c0:2:2100::"},
816         {"2001:db8:122:344::", 96, "2001:db8:122:344::192.0.2.33"},
817         {"64:ff9b::", 96, "64:ff9b::192.0.2.33"},
818     };
819 
820     const uint8_t kIp4Address[] = {192, 0, 2, 33};
821 
822     Ip4::Address ip4Address;
823 
824     printf("\nTestIp4Ip6Translation()\n");
825 
826     ip4Address.SetBytes(kIp4Address);
827 
828     for (const TestCase &testCase : kTestCases)
829     {
830         Ip6::Prefix  prefix;
831         Ip6::Address address;
832         Ip6::Address expectedAddress;
833 
834         SuccessOrQuit(address.FromString(testCase.mPrefix));
835         prefix.Set(address.GetBytes(), testCase.mLength);
836 
837         SuccessOrQuit(expectedAddress.FromString(testCase.mIp6Address));
838 
839         address.SynthesizeFromIp4Address(prefix, ip4Address);
840 
841         printf("Prefix: %-26s IPv4Addr: %-12s Ipv6Address: %-36s Expected: %s (%s)\n", prefix.ToString().AsCString(),
842                ip4Address.ToString().AsCString(), address.ToString().AsCString(), testCase.mIp6Address,
843                expectedAddress.ToString().AsCString());
844 
845         VerifyOrQuit(address == expectedAddress, "Ip6::SynthesizeFromIp4Address() failed");
846     }
847 
848     for (const TestCase &testCase : kTestCases)
849     {
850         const Ip4::Address expectedAddress = ip4Address;
851         Ip4::Address       address;
852         Ip6::Address       ip6Address;
853 
854         SuccessOrQuit(ip6Address.FromString(testCase.mIp6Address));
855 
856         address.ExtractFromIp6Address(testCase.mLength, ip6Address);
857 
858         printf("Ipv6Address: %-36s IPv4Addr: %-12s Expected: %s\n", testCase.mIp6Address,
859                address.ToString().AsCString(), expectedAddress.ToString().AsCString());
860 
861         VerifyOrQuit(address == expectedAddress, "Ip4::ExtractFromIp6Address() failed");
862     }
863 }
864 
TestIp4Cidr(void)865 void TestIp4Cidr(void)
866 {
867     struct TestCase
868     {
869         const char    *mNetwork;
870         const uint8_t  mLength;
871         const uint32_t mHost;
872         const char    *mOutcome;
873     };
874 
875     const TestCase kTestCases[] = {
876         {"172.16.12.34", 32, 0x12345678, "172.16.12.34"},  {"172.16.12.34", 31, 0x12345678, "172.16.12.34"},
877         {"172.16.12.34", 30, 0x12345678, "172.16.12.32"},  {"172.16.12.34", 29, 0x12345678, "172.16.12.32"},
878         {"172.16.12.34", 28, 0x12345678, "172.16.12.40"},  {"172.16.12.34", 27, 0x12345678, "172.16.12.56"},
879         {"172.16.12.34", 26, 0x12345678, "172.16.12.56"},  {"172.16.12.34", 25, 0x12345678, "172.16.12.120"},
880         {"172.16.12.34", 24, 0x12345678, "172.16.12.120"}, {"172.16.12.34", 23, 0x12345678, "172.16.12.120"},
881         {"172.16.12.34", 22, 0x12345678, "172.16.14.120"}, {"172.16.12.34", 21, 0x12345678, "172.16.14.120"},
882         {"172.16.12.34", 20, 0x12345678, "172.16.6.120"},  {"172.16.12.34", 19, 0x12345678, "172.16.22.120"},
883         {"172.16.12.34", 18, 0x12345678, "172.16.22.120"}, {"172.16.12.34", 17, 0x12345678, "172.16.86.120"},
884         {"172.16.12.34", 16, 0x12345678, "172.16.86.120"}, {"172.16.12.34", 15, 0x12345678, "172.16.86.120"},
885         {"172.16.12.34", 14, 0x12345678, "172.16.86.120"}, {"172.16.12.34", 13, 0x12345678, "172.20.86.120"},
886         {"172.16.12.34", 12, 0x12345678, "172.20.86.120"}, {"172.16.12.34", 11, 0x12345678, "172.20.86.120"},
887         {"172.16.12.34", 10, 0x12345678, "172.52.86.120"}, {"172.16.12.34", 9, 0x12345678, "172.52.86.120"},
888         {"172.16.12.34", 8, 0x12345678, "172.52.86.120"},  {"172.16.12.34", 7, 0x12345678, "172.52.86.120"},
889         {"172.16.12.34", 6, 0x12345678, "174.52.86.120"},  {"172.16.12.34", 5, 0x12345678, "170.52.86.120"},
890         {"172.16.12.34", 4, 0x12345678, "162.52.86.120"},  {"172.16.12.34", 3, 0x12345678, "178.52.86.120"},
891         {"172.16.12.34", 2, 0x12345678, "146.52.86.120"},  {"172.16.12.34", 1, 0x12345678, "146.52.86.120"},
892         {"172.16.12.34", 0, 0x12345678, "18.52.86.120"},
893     };
894 
895     for (const TestCase &testCase : kTestCases)
896     {
897         Ip4::Address network;
898         Ip4::Cidr    cidr;
899         Ip4::Address generated;
900 
901         SuccessOrQuit(network.FromString(testCase.mNetwork));
902         cidr.mAddress = network;
903         cidr.mLength  = testCase.mLength;
904 
905         generated.SynthesizeFromCidrAndHost(cidr, testCase.mHost);
906 
907         printf("CIDR: %-18s HostID: %-8x Host: %-14s Expected: %s\n", cidr.ToString().AsCString(), testCase.mHost,
908                generated.ToString().AsCString(), testCase.mOutcome);
909 
910         VerifyOrQuit(strcmp(generated.ToString().AsCString(), testCase.mOutcome) == 0,
911                      "Ip4::Address::SynthesizeFromCidrAndHost() failed");
912     }
913 }
914 
915 } // namespace ot
916 
main(void)917 int main(void)
918 {
919     ot::TestIp6AddressSetPrefix();
920     ot::TestIp4AddressFromString();
921     ot::TestIp6AddressFromString();
922     ot::TestIp6PrefixFromString();
923     ot::TestIp6Prefix();
924     ot::TestIp6PrefixTidy();
925     ot::TestIp4MappedIp6Address();
926     ot::TestIp4Ip6Translation();
927     ot::TestIp4Cidr();
928     ot::TestIp4CidrFromString();
929 
930     printf("All tests passed\n");
931     return 0;
932 }
933