|  | // Copyright 2023 The Chromium Authors | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     https://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | #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 |