1 /*
2  * Copyright (C) 2018 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.net.dhcp;
18 
19 import static android.net.dhcp.DhcpLease.EXPIRATION_NEVER;
20 import static android.net.dhcp.DhcpLease.inet4AddrToString;
21 
22 import static com.android.net.module.util.Inet4AddressUtils.inet4AddressToIntHTH;
23 import static com.android.net.module.util.Inet4AddressUtils.intToInet4AddressHTH;
24 import static com.android.net.module.util.Inet4AddressUtils.prefixLengthToV4NetmaskIntHTH;
25 import static com.android.net.module.util.NetworkStackConstants.IPV4_ADDR_ANY;
26 import static com.android.net.module.util.NetworkStackConstants.IPV4_ADDR_BITS;
27 
28 import static java.lang.Math.min;
29 
30 import android.net.IpPrefix;
31 import android.net.MacAddress;
32 import android.net.dhcp.DhcpServer.Clock;
33 import android.net.util.SharedLog;
34 import android.os.RemoteCallbackList;
35 import android.os.RemoteException;
36 import android.util.ArrayMap;
37 
38 import androidx.annotation.NonNull;
39 import androidx.annotation.Nullable;
40 import androidx.annotation.VisibleForTesting;
41 
42 import java.net.Inet4Address;
43 import java.util.ArrayList;
44 import java.util.Collections;
45 import java.util.HashSet;
46 import java.util.Iterator;
47 import java.util.LinkedHashMap;
48 import java.util.List;
49 import java.util.Map;
50 import java.util.Map.Entry;
51 import java.util.Objects;
52 import java.util.Set;
53 import java.util.function.Function;
54 
55 /**
56  * A repository managing IPv4 address assignments through DHCPv4.
57  *
58  * <p>This class is not thread-safe. All public methods should be called on a common thread or
59  * use some synchronization mechanism.
60  *
61  * <p>Methods are optimized for a small number of allocated leases, assuming that most of the time
62  * only 2~10 addresses will be allocated, which is the common case. Managing a large number of
63  * addresses is supported but will be slower: some operations have complexity in O(num_leases).
64  * @hide
65  */
66 class DhcpLeaseRepository {
67     public static final byte[] CLIENTID_UNSPEC = null;
68     public static final Inet4Address INETADDR_UNSPEC = null;
69 
70     @NonNull
71     private final SharedLog mLog;
72     @NonNull
73     private final Clock mClock;
74 
75     @NonNull
76     private IpPrefix mPrefix;
77     @NonNull
78     private Set<Inet4Address> mReservedAddrs;
79     private int mSubnetAddr;
80     private int mPrefixLength;
81     private int mSubnetMask;
82     private int mNumAddresses;
83     private long mLeaseTimeMs;
84     @Nullable
85     private Inet4Address mClientAddr;
86 
87     /**
88      * Next timestamp when committed or declined leases should be checked for expired ones. This
89      * will always be lower than or equal to the time for the first lease to expire: it's OK not to
90      * update this when removing entries, but it must always be updated when adding/updating.
91      */
92     private long mNextExpirationCheck = EXPIRATION_NEVER;
93 
94     @NonNull
95     private RemoteCallbackList<IDhcpEventCallbacks> mEventCallbacks = new RemoteCallbackList<>();
96 
97     static class DhcpLeaseException extends Exception {
98         DhcpLeaseException(String message) {
99             super(message);
100         }
101     }
102 
103     static class OutOfAddressesException extends DhcpLeaseException {
104         OutOfAddressesException(String message) {
105             super(message);
106         }
107     }
108 
109     static class InvalidAddressException extends DhcpLeaseException {
110         InvalidAddressException(String message) {
111             super(message);
112         }
113     }
114 
115     static class InvalidSubnetException extends DhcpLeaseException {
116         InvalidSubnetException(String message) {
117             super(message);
118         }
119     }
120 
121     /**
122      * Leases by IP address
123      */
124     private final ArrayMap<Inet4Address, DhcpLease> mCommittedLeases = new ArrayMap<>();
125 
126     /**
127      * Map address -> expiration timestamp in ms. Addresses are guaranteed to be valid as defined
128      * by {@link #isValidAddress(Inet4Address)}, but are not necessarily otherwise available for
129      * assignment.
130      */
131     private final LinkedHashMap<Inet4Address, Long> mDeclinedAddrs = new LinkedHashMap<>();
132 
133     DhcpLeaseRepository(@NonNull IpPrefix prefix, @NonNull Set<Inet4Address> reservedAddrs,
134             long leaseTimeMs, @Nullable Inet4Address clientAddr, @NonNull SharedLog log,
135             @NonNull Clock clock) {
136         mLog = log;
137         mClock = clock;
138         mClientAddr = clientAddr;
139         updateParams(prefix, reservedAddrs, leaseTimeMs, clientAddr);
140     }
141 
142     public void updateParams(@NonNull IpPrefix prefix, @NonNull Set<Inet4Address> reservedAddrs,
143             long leaseTimeMs, @Nullable Inet4Address clientAddr) {
144         mPrefix = prefix;
145         mReservedAddrs = Collections.unmodifiableSet(new HashSet<>(reservedAddrs));
146         mPrefixLength = prefix.getPrefixLength();
147         mSubnetMask = prefixLengthToV4NetmaskIntHTH(mPrefixLength);
148         mSubnetAddr = inet4AddressToIntHTH((Inet4Address) prefix.getAddress()) & mSubnetMask;
149         mNumAddresses = clientAddr != null ? 1 : 1 << (IPV4_ADDR_BITS - prefix.getPrefixLength());
150         mLeaseTimeMs = leaseTimeMs;
151         mClientAddr = clientAddr;
152 
153         cleanMap(mDeclinedAddrs);
154         if (cleanMap(mCommittedLeases)) {
155             notifyLeasesChanged();
156         }
157     }
158 
159     /**
160      * From a map keyed by {@link Inet4Address}, remove entries where the key is invalid (as
161      * specified by {@link #isValidAddress(Inet4Address)}), or is a reserved address.
162      * @return true if and only if at least one entry was removed.
163      */
164     private <T> boolean cleanMap(Map<Inet4Address, T> map) {
165         final Iterator<Entry<Inet4Address, T>> it = map.entrySet().iterator();
166         boolean removed = false;
167         while (it.hasNext()) {
168             final Inet4Address addr = it.next().getKey();
169             if (!isValidAddress(addr) || mReservedAddrs.contains(addr)) {
170                 it.remove();
171                 removed = true;
172             }
173         }
174         return removed;
175     }
176 
177     /**
178      * Get a DHCP offer, to reply to a DHCPDISCOVER. Follows RFC2131 #4.3.1.
179      *
180      * @param clientId Client identifier option if specified, or {@link #CLIENTID_UNSPEC}
181      * @param relayAddr Internet address of the relay (giaddr), can be {@link Inet4Address#ANY}
182      * @param reqAddr Requested address by the client (option 50), or {@link #INETADDR_UNSPEC}
183      * @param hostname Client-provided hostname, or {@link DhcpLease#HOSTNAME_NONE}
184      * @throws OutOfAddressesException The server does not have any available address
185      * @throws InvalidSubnetException The lease was requested from an unsupported subnet
186      */
187     @NonNull
188     public DhcpLease getOffer(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
189             @NonNull Inet4Address relayAddr, @Nullable Inet4Address reqAddr,
190             @Nullable String hostname) throws OutOfAddressesException, InvalidSubnetException {
191         final long currentTime = mClock.elapsedRealtime();
192         final long expTime = currentTime + mLeaseTimeMs;
193 
194         removeExpiredLeases(currentTime);
195         checkValidRelayAddr(relayAddr);
196 
197         final DhcpLease currentLease = findByClient(clientId, hwAddr);
198         final DhcpLease newLease;
199         if (currentLease != null) {
200             newLease = currentLease.renewedLease(expTime, hostname);
201             mLog.log("Offering extended lease " + newLease);
202             // Do not update lease time in the map: the offer is not committed yet.
203         } else if (reqAddr != null && isValidAddress(reqAddr) && isAvailable(reqAddr)) {
204             newLease = new DhcpLease(clientId, hwAddr, reqAddr, mPrefixLength, expTime, hostname);
205             mLog.log("Offering requested lease " + newLease);
206         } else {
207             newLease = makeNewOffer(clientId, hwAddr, expTime, hostname);
208             mLog.log("Offering new generated lease " + newLease);
209         }
210         return newLease;
211     }
212 
213     /**
214      * Get a rapid committed DHCP Lease, to reply to a DHCPDISCOVER w/ Rapid Commit option.
215      *
216      * @param clientId Client identifier option if specified, or {@link #CLIENTID_UNSPEC}
217      * @param relayAddr Internet address of the relay (giaddr), can be {@link Inet4Address#ANY}
218      * @param hostname Client-provided hostname, or {@link DhcpLease#HOSTNAME_NONE}
219      * @throws OutOfAddressesException The server does not have any available address
220      * @throws InvalidSubnetException The lease was requested from an unsupported subnet
221      */
222     @NonNull
223     public DhcpLease getCommittedLease(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
224             @NonNull Inet4Address relayAddr, @Nullable String hostname)
225             throws OutOfAddressesException, InvalidSubnetException {
226         final DhcpLease newLease = getOffer(clientId, hwAddr, relayAddr, null /* reqAddr */,
227                 hostname);
228         commitLease(newLease);
229         return newLease;
230     }
231 
232     private void checkValidRelayAddr(@Nullable Inet4Address relayAddr)
233             throws InvalidSubnetException {
234         // As per #4.3.1, addresses are assigned based on the relay address if present. This
235         // implementation only assigns addresses if the relayAddr is inside our configured subnet.
236         // This also applies when the client requested a specific address for consistency between
237         // requests, and with older behavior.
238         if (isIpAddrOutsidePrefix(mPrefix, relayAddr)) {
239             throw new InvalidSubnetException("Lease requested by relay from outside of subnet");
240         }
241     }
242 
243     private static boolean isIpAddrOutsidePrefix(@NonNull IpPrefix prefix,
244             @Nullable Inet4Address addr) {
245         return addr != null && !addr.equals(IPV4_ADDR_ANY) && !prefix.contains(addr);
246     }
247 
248     @Nullable
249     private DhcpLease findByClient(@Nullable byte[] clientId, @NonNull MacAddress hwAddr) {
250         for (DhcpLease lease : mCommittedLeases.values()) {
251             if (lease.matchesClient(clientId, hwAddr)) {
252                 return lease;
253             }
254         }
255 
256         // Note this differs from dnsmasq behavior, which would match by hwAddr if clientId was
257         // given but no lease keyed on clientId matched. This would prevent one interface from
258         // obtaining multiple leases with different clientId.
259         return null;
260     }
261 
262     /**
263      * Make a lease conformant to a client DHCPREQUEST or renew the client's existing lease,
264      * commit it to the repository and return it.
265      *
266      * <p>This method always succeeds and commits the lease if it does not throw, and has no side
267      * effects if it throws.
268      *
269      * @param clientId Client identifier option if specified, or {@link #CLIENTID_UNSPEC}
270      * @param reqAddr Requested address by the client (option 50), or {@link #INETADDR_UNSPEC}
271      * @param sidSet Whether the server identifier was set in the request
272      * @return The newly created or renewed lease
273      * @throws InvalidAddressException The client provided an address that conflicts with its
274      *                                 current configuration, or other committed/reserved leases.
275      */
276     @NonNull
277     public DhcpLease requestLease(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
278             @NonNull Inet4Address clientAddr, @NonNull Inet4Address relayAddr,
279             @Nullable Inet4Address reqAddr, boolean sidSet, @Nullable String hostname)
280             throws InvalidAddressException, InvalidSubnetException {
281         final long currentTime = mClock.elapsedRealtime();
282         removeExpiredLeases(currentTime);
283         checkValidRelayAddr(relayAddr);
284         final DhcpLease assignedLease = findByClient(clientId, hwAddr);
285 
286         final Inet4Address leaseAddr = reqAddr != null ? reqAddr : clientAddr;
287         if (assignedLease != null) {
288             if (sidSet && reqAddr != null) {
289                 // Client in SELECTING state; remove any current lease before creating a new one.
290                 // Do not notify of change as it will be done when the new lease is committed.
291                 removeLease(assignedLease.getNetAddr(), false /* notifyChange */);
292             } else if (!assignedLease.getNetAddr().equals(leaseAddr)) {
293                 // reqAddr null (RENEWING/REBINDING): client renewing its own lease for clientAddr.
294                 // reqAddr set with sid not set (INIT-REBOOT): client verifying configuration.
295                 // In both cases, throw if clientAddr or reqAddr does not match the known lease.
296                 throw new InvalidAddressException("Incorrect address for client in "
297                         + (reqAddr != null ? "INIT-REBOOT" : "RENEWING/REBINDING"));
298             }
299         }
300 
301         // In the init-reboot case, RFC2131 #4.3.2 says that the server must not reply if
302         // assignedLease == null, but dnsmasq will let the client use the requested address if
303         // available, when configured with --dhcp-authoritative. This is preferable to avoid issues
304         // if the server lost the lease DB: the client would not get a reply because the server
305         // does not know their lease.
306         // Similarly in RENEWING/REBINDING state, create a lease when possible if the
307         // client-provided lease is unknown.
308         final DhcpLease lease =
309                 checkClientAndMakeLease(clientId, hwAddr, leaseAddr, hostname, currentTime);
310         mLog.logf("DHCPREQUEST assignedLease %s, reqAddr=%s, sidSet=%s: created/renewed lease %s",
311                 assignedLease, inet4AddrToString(reqAddr), sidSet, lease);
312         return lease;
313     }
314 
315     /**
316      * Check that the client can request the specified address, make or renew the lease if yes, and
317      * commit it.
318      *
319      * <p>This method always succeeds and returns the lease if it does not throw, and has no
320      * side-effect if it throws.
321      *
322      * @return The newly created or renewed, committed lease
323      * @throws InvalidAddressException The client provided an address that conflicts with its
324      *                                 current configuration, or other committed/reserved leases.
325      */
326     private DhcpLease checkClientAndMakeLease(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
327             @NonNull Inet4Address addr, @Nullable String hostname, long currentTime)
328             throws InvalidAddressException {
329         final long expTime = currentTime + mLeaseTimeMs;
330         final DhcpLease currentLease = mCommittedLeases.getOrDefault(addr, null);
331         if (currentLease != null && !currentLease.matchesClient(clientId, hwAddr)) {
332             throw new InvalidAddressException("Address in use");
333         }
334 
335         final DhcpLease lease;
336         if (currentLease == null) {
337             if (isValidAddress(addr) && !mReservedAddrs.contains(addr)) {
338                 lease = new DhcpLease(clientId, hwAddr, addr, mPrefixLength, expTime, hostname);
339             } else {
340                 throw new InvalidAddressException("Lease not found and address unavailable");
341             }
342         } else {
343             lease = currentLease.renewedLease(expTime, hostname);
344         }
345         commitLease(lease);
346         return lease;
347     }
348 
349     private void commitLease(@NonNull DhcpLease lease) {
350         mCommittedLeases.put(lease.getNetAddr(), lease);
351         maybeUpdateEarliestExpiration(lease.getExpTime());
352         notifyLeasesChanged();
353     }
354 
355     private void removeLease(@NonNull Inet4Address address, boolean notifyChange) {
356         // Earliest expiration remains <= the first expiry time on remove, so no need to update it.
357         mCommittedLeases.remove(address);
358         if (notifyChange) notifyLeasesChanged();
359     }
360 
361     /**
362      * Delete a committed lease from the repository.
363      *
364      * @return true if a lease matching parameters was found.
365      */
366     public boolean releaseLease(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
367             @NonNull Inet4Address addr) {
368         final DhcpLease currentLease = mCommittedLeases.getOrDefault(addr, null);
369         if (currentLease == null) {
370             mLog.w("Could not release unknown lease for " + inet4AddrToString(addr));
371             return false;
372         }
373         if (currentLease.matchesClient(clientId, hwAddr)) {
374             mLog.log("Released lease " + currentLease);
375             removeLease(addr, true /* notifyChange */);
376             return true;
377         }
378         mLog.w(String.format("Not releasing lease %s: does not match client (cid %s, hwAddr %s)",
379                 currentLease, DhcpLease.clientIdToString(clientId), hwAddr));
380         return false;
381     }
382 
383     private void notifyLeasesChanged() {
384         final List<DhcpLeaseParcelable> leaseParcelables =
385                 new ArrayList<>(mCommittedLeases.size());
386         for (DhcpLease committedLease : mCommittedLeases.values()) {
387             leaseParcelables.add(committedLease.toParcelable());
388         }
389 
390         final int cbCount = mEventCallbacks.beginBroadcast();
391         for (int i = 0; i < cbCount; i++) {
392             try {
393                 mEventCallbacks.getBroadcastItem(i).onLeasesChanged(leaseParcelables);
394             } catch (RemoteException e) {
395                 mLog.e("Could not send lease callback", e);
396             }
397         }
398         mEventCallbacks.finishBroadcast();
399     }
400 
401     @VisibleForTesting
402     void markLeaseDeclined(@NonNull Inet4Address addr) {
403         if (mDeclinedAddrs.containsKey(addr) || !isValidAddress(addr)) {
404             mLog.logf("Not marking %s as declined: already declined or not assignable",
405                     inet4AddrToString(addr));
406             return;
407         }
408         final long expTime = mClock.elapsedRealtime() + mLeaseTimeMs;
409         mDeclinedAddrs.put(addr, expTime);
410         mLog.logf("Marked %s as declined expiring %d", inet4AddrToString(addr), expTime);
411         maybeUpdateEarliestExpiration(expTime);
412     }
413 
414     /**
415      * Mark a committed lease matching the passed in clientId and hardware address parameters to be
416      * declined, and delete it from the repository.
417      *
418      * @param clientId Client identifier option if specified, or {@link #CLIENTID_UNSPEC}
419      * @param hwAddr client's mac address
420      * @param Addr IPv4 address to be declined
421      * @return true if a lease matching parameters was removed from committed repository.
422      */
423     public boolean markAndReleaseDeclinedLease(@Nullable byte[] clientId,
424             @NonNull MacAddress hwAddr, @NonNull Inet4Address addr) {
425         if (!releaseLease(clientId, hwAddr, addr)) return false;
426         markLeaseDeclined(addr);
427         return true;
428     }
429 
430     /**
431      * Get the list of currently valid committed leases in the repository.
432      */
433     @NonNull
434     public List<DhcpLease> getCommittedLeases() {
435         removeExpiredLeases(mClock.elapsedRealtime());
436         return new ArrayList<>(mCommittedLeases.values());
437     }
438 
439     /**
440      * Get the set of addresses that have been marked as declined in the repository.
441      */
442     @NonNull
443     public Set<Inet4Address> getDeclinedAddresses() {
444         removeExpiredLeases(mClock.elapsedRealtime());
445         return new HashSet<>(mDeclinedAddrs.keySet());
446     }
447 
448     /**
449      * Add callbacks that will be called on leases update.
450      */
451     public void addLeaseCallbacks(@NonNull IDhcpEventCallbacks cb) {
452         Objects.requireNonNull(cb, "Callbacks must be non-null");
453         mEventCallbacks.register(cb);
454     }
455 
456     /**
457      * Given the expiration time of a new committed lease or declined address, update
458      * {@link #mNextExpirationCheck} so it stays lower than or equal to the time for the first lease
459      * to expire.
460      */
461     private void maybeUpdateEarliestExpiration(long expTime) {
462         if (expTime < mNextExpirationCheck) {
463             mNextExpirationCheck = expTime;
464         }
465     }
466 
467     /**
468      * Remove expired entries from a map keyed by {@link Inet4Address}.
469      *
470      * @param tag Type of lease in the map, for logging
471      * @param getExpTime Functor returning the expiration time for an object in the map.
472      *                   Must not return null.
473      * @return The lowest expiration time among entries remaining in the map
474      */
475     private <T> long removeExpired(long currentTime, @NonNull Map<Inet4Address, T> map,
476             @NonNull String tag, @NonNull Function<T, Long> getExpTime) {
477         final Iterator<Entry<Inet4Address, T>> it = map.entrySet().iterator();
478         long firstExpiration = EXPIRATION_NEVER;
479         while (it.hasNext()) {
480             final Entry<Inet4Address, T> lease = it.next();
481             final long expTime = getExpTime.apply(lease.getValue());
482             if (expTime <= currentTime) {
483                 mLog.logf("Removing expired %s lease for %s (expTime=%s, currentTime=%s)",
484                         tag, lease.getKey(), expTime, currentTime);
485                 it.remove();
486             } else {
487                 firstExpiration = min(firstExpiration, expTime);
488             }
489         }
490         return firstExpiration;
491     }
492 
493     /**
494      * Go through committed and declined leases and remove the expired ones.
495      */
496     private void removeExpiredLeases(long currentTime) {
497         if (currentTime < mNextExpirationCheck) {
498             return;
499         }
500 
501         final long commExp = removeExpired(
502                 currentTime, mCommittedLeases, "committed", DhcpLease::getExpTime);
503         final long declExp = removeExpired(
504                 currentTime, mDeclinedAddrs, "declined", Function.identity());
505 
506         mNextExpirationCheck = min(commExp, declExp);
507     }
508 
509     private boolean isAvailable(@NonNull Inet4Address addr) {
510         return !mReservedAddrs.contains(addr) && !mCommittedLeases.containsKey(addr);
511     }
512 
513     /**
514      * Get the 0-based index of an address in the subnet.
515      *
516      * <p>Given ordering of addresses 5.6.7.8 < 5.6.7.9 < 5.6.8.0, the index on a subnet is defined
517      * so that the first address is 0, the second 1, etc. For example on a /16, 192.168.0.0 -> 0,
518      * 192.168.0.1 -> 1, 192.168.1.0 -> 256
519      *
520      */
521     private int getAddrIndex(int addr) {
522         return addr & ~mSubnetMask;
523     }
524 
525     private int getAddrByIndex(int index) {
526         return mSubnetAddr | index;
527     }
528 
529     /**
530      * Get a valid address starting from the supplied one.
531      *
532      * <p>This only checks that the address is numerically valid for assignment, not whether it is
533      * already in use. The return value is always inside the configured prefix, even if the supplied
534      * address is not.
535      *
536      * <p>If the provided address is valid, it is returned as-is. Otherwise, the next valid
537      * address (with the ordering in {@link #getAddrIndex(int)}) is returned.
538      */
539     private int getValidAddress(int addr) {
540         // Only mClientAddr is valid if static client address is enforced.
541         if (mClientAddr != null) return inet4AddressToIntHTH(mClientAddr);
542 
543         final int lastByteMask = 0xff;
544         int addrIndex = getAddrIndex(addr); // 0-based index of the address in the subnet
545 
546         // Some OSes do not handle addresses in .255 or .0 correctly: avoid those.
547         final int lastByte = getAddrByIndex(addrIndex) & lastByteMask;
548         if (lastByte == lastByteMask) {
549             // Avoid .255 address, and .0 address that follows
550             addrIndex = (addrIndex + 2) % mNumAddresses;
551         } else if (lastByte == 0) {
552             // Avoid .0 address
553             addrIndex = (addrIndex + 1) % mNumAddresses;
554         }
555 
556         // Do not use first or last address of range
557         if (addrIndex == 0 || addrIndex == mNumAddresses - 1) {
558             // Always valid and not end of range since prefixLength is at most 30 in serving params
559             addrIndex = 1;
560         }
561         return getAddrByIndex(addrIndex);
562     }
563 
564     /**
565      * Returns whether the address is in the configured subnet and part of the assignable range.
566      */
567     private boolean isValidAddress(Inet4Address addr) {
568         final int intAddr = inet4AddressToIntHTH(addr);
569         return getValidAddress(intAddr) == intAddr;
570     }
571 
572     private int getNextAddress(int addr) {
573         final int addrIndex = getAddrIndex(addr);
574         final int nextAddress = getAddrByIndex((addrIndex + 1) % mNumAddresses);
575         return getValidAddress(nextAddress);
576     }
577 
578     /**
579      * Calculate a first candidate address for a client by hashing the hardware address.
580      *
581      * <p>This will be a valid address as checked by {@link #getValidAddress(int)}, but may be
582      * in use.
583      *
584      * @return An IPv4 address encoded as 32-bit int
585      */
586     private int getFirstClientAddress(MacAddress hwAddr) {
587         // This follows dnsmasq behavior. Advantages are: clients will often get the same
588         // offers for different DISCOVER even if the lease was not yet accepted or has expired,
589         // and address generation will generally not need to loop through many allocated addresses
590         // until it finds a free one.
591         int hash = 0;
592         for (byte b : hwAddr.toByteArray()) {
593             hash += b + (b << 8) + (b << 16);
594         }
595         // This implementation will not always result in the same IPs as dnsmasq would give out in
596         // Android <= P, because it includes invalid and reserved addresses in mNumAddresses while
597         // the configured ranges for dnsmasq did not.
598         final int addrIndex = hash % mNumAddresses;
599         return getValidAddress(getAddrByIndex(addrIndex));
600     }
601 
602     /**
603      * Create a lease that can be offered to respond to a client DISCOVER.
604      *
605      * <p>This method always succeeds and returns the lease if it does not throw. If no non-declined
606      * address is available, it will try to offer the oldest declined address if valid.
607      *
608      * @throws OutOfAddressesException The server has no address left to offer
609      */
610     private DhcpLease makeNewOffer(@Nullable byte[] clientId, @NonNull MacAddress hwAddr,
611             long expTime, @Nullable String hostname) throws OutOfAddressesException {
612         int intAddr = getFirstClientAddress(hwAddr);
613         // Loop until a free address is found, or there are no more addresses.
614         // There is slightly less than this many usable addresses, but some extra looping is OK
615         for (int i = 0; i < mNumAddresses; i++) {
616             final Inet4Address addr = intToInet4AddressHTH(intAddr);
617             if (isAvailable(addr) && !mDeclinedAddrs.containsKey(addr)) {
618                 return new DhcpLease(clientId, hwAddr, addr, mPrefixLength, expTime, hostname);
619             }
620             intAddr = getNextAddress(intAddr);
621         }
622 
623         // Try freeing DECLINEd addresses if out of addresses.
624         final Iterator<Inet4Address> it = mDeclinedAddrs.keySet().iterator();
625         while (it.hasNext()) {
626             final Inet4Address addr = it.next();
627             it.remove();
628             mLog.logf("Out of addresses in address pool: dropped declined addr %s",
629                     inet4AddrToString(addr));
630             // isValidAddress() is always verified for entries in mDeclinedAddrs.
631             // However declined addresses may have been requested (typically by the machine that was
632             // already using the address) after being declined.
633             if (isAvailable(addr)) {
634                 return new DhcpLease(clientId, hwAddr, addr, mPrefixLength, expTime, hostname);
635             }
636         }
637 
638         throw new OutOfAddressesException("No address available for offer");
639     }
640 }
641