|  | // Copyright 2023 The Chromium Authors | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "ip_util.h" | 
|  |  | 
|  | BSSL_NAMESPACE_BEGIN | 
|  |  | 
|  | bool IsValidNetmask(der::Input mask) { | 
|  | if (mask.size() != kIPv4AddressSize && mask.size() != kIPv6AddressSize) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | for (size_t i = 0; i < mask.size(); i++) { | 
|  | uint8_t b = mask[i]; | 
|  | if (b != 0xff) { | 
|  | // b must be all ones followed by all zeros, so ~b must be all zeros | 
|  | // followed by all ones. | 
|  | uint8_t inv = ~b; | 
|  | if ((inv & (inv + 1)) != 0) { | 
|  | return false; | 
|  | } | 
|  | // The remaining bytes must be all zeros. | 
|  | for (size_t j = i + 1; j < mask.size(); j++) { | 
|  | if (mask[j] != 0) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool IPAddressMatchesWithNetmask(der::Input addr1, der::Input addr2, | 
|  | der::Input mask) { | 
|  | if (addr1.size() != addr2.size() || addr1.size() != mask.size()) { | 
|  | return false; | 
|  | } | 
|  | for (size_t i = 0; i < addr1.size(); i++) { | 
|  | if ((addr1[i] & mask[i]) != (addr2[i] & mask[i])) { | 
|  | return false; | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | BSSL_NAMESPACE_END |