1 /*
2  * Copyright (C) 2019 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 com.android.net.module.util.ip;
18 
19 import static android.net.INetd.IF_STATE_DOWN;
20 import static android.net.INetd.IF_STATE_UP;
21 
22 import android.net.INetd;
23 import android.net.InterfaceConfigurationParcel;
24 import android.net.LinkAddress;
25 import android.os.RemoteException;
26 import android.os.ServiceSpecificException;
27 import android.system.OsConstants;
28 
29 import com.android.net.module.util.SharedLog;
30 
31 import java.net.Inet4Address;
32 import java.net.InetAddress;
33 
34 /**
35  * Encapsulates the multiple IP configuration operations performed on an interface.
36  *
37  * TODO: refactor/eliminate the redundant ways to set and clear addresses.
38  *
39  * @hide
40  */
41 public class InterfaceController {
42     private static final boolean DBG = false;
43 
44     private final String mIfName;
45     private final INetd mNetd;
46     private final SharedLog mLog;
47 
InterfaceController(String ifname, INetd netd, SharedLog log)48     public InterfaceController(String ifname, INetd netd, SharedLog log) {
49         mIfName = ifname;
50         mNetd = netd;
51         mLog = log;
52     }
53 
54     /**
55      * Set the IPv4 address and also optionally bring the interface up or down.
56      */
setInterfaceConfiguration(final LinkAddress ipv4Addr, final Boolean setIfaceUp)57     public boolean setInterfaceConfiguration(final LinkAddress ipv4Addr,
58             final Boolean setIfaceUp) {
59         if (!(ipv4Addr.getAddress() instanceof Inet4Address)) {
60             throw new IllegalArgumentException("Invalid or mismatched Inet4Address");
61         }
62         // Note: currently netd only support INetd#IF_STATE_UP and #IF_STATE_DOWN.
63         // Other flags would be ignored.
64 
65         final InterfaceConfigurationParcel ifConfig = new InterfaceConfigurationParcel();
66         ifConfig.ifName = mIfName;
67         ifConfig.ipv4Addr = ipv4Addr.getAddress().getHostAddress();
68         ifConfig.prefixLength = ipv4Addr.getPrefixLength();
69         // Netd ignores hwaddr in interfaceSetCfg.
70         ifConfig.hwAddr = "";
71         if (setIfaceUp == null) {
72             // Empty array means no change.
73             ifConfig.flags = new String[0];
74         } else {
75             // Netd ignores any flag that's not IF_STATE_UP or IF_STATE_DOWN in interfaceSetCfg.
76             ifConfig.flags = setIfaceUp.booleanValue()
77                     ? new String[] {IF_STATE_UP} : new String[] {IF_STATE_DOWN};
78         }
79         try {
80             mNetd.interfaceSetCfg(ifConfig);
81         } catch (RemoteException | ServiceSpecificException e) {
82             logError("Setting IPv4 address to %s/%d failed: %s",
83                     ifConfig.ipv4Addr, ifConfig.prefixLength, e);
84             return false;
85         }
86         return true;
87     }
88 
89     /**
90      * Set the IPv4 address of the interface.
91      */
setIPv4Address(final LinkAddress address)92     public boolean setIPv4Address(final LinkAddress address) {
93         return setInterfaceConfiguration(address, null);
94     }
95 
96     /**
97      * Clear the IPv4Address of the interface.
98      */
clearIPv4Address()99     public boolean clearIPv4Address() {
100         return setIPv4Address(new LinkAddress("0.0.0.0/0"));
101     }
102 
setEnableIPv6(boolean enabled)103     private boolean setEnableIPv6(boolean enabled) {
104         try {
105             mNetd.interfaceSetEnableIPv6(mIfName, enabled);
106         } catch (RemoteException | ServiceSpecificException e) {
107             logError("%s IPv6 failed: %s", (enabled ? "enabling" : "disabling"), e);
108             return false;
109         }
110         return true;
111     }
112 
113     /**
114      * Enable IPv6 on the interface.
115      */
enableIPv6()116     public boolean enableIPv6() {
117         return setEnableIPv6(true);
118     }
119 
120     /**
121      * Disable IPv6 on the interface.
122      */
disableIPv6()123     public boolean disableIPv6() {
124         return setEnableIPv6(false);
125     }
126 
127     /**
128      * Enable or disable IPv6 privacy extensions on the interface.
129      * @param enabled Whether the extensions should be enabled.
130      */
setIPv6PrivacyExtensions(boolean enabled)131     public boolean setIPv6PrivacyExtensions(boolean enabled) {
132         try {
133             mNetd.interfaceSetIPv6PrivacyExtensions(mIfName, enabled);
134         } catch (RemoteException | ServiceSpecificException e) {
135             logError("error %s IPv6 privacy extensions: %s",
136                     (enabled ? "enabling" : "disabling"), e);
137             return false;
138         }
139         return true;
140     }
141 
142     /**
143      * Set IPv6 address generation mode on the interface.
144      *
145      * <p>IPv6 should be disabled before changing the mode.
146      */
setIPv6AddrGenModeIfSupported(int mode)147     public boolean setIPv6AddrGenModeIfSupported(int mode) {
148         try {
149             mNetd.setIPv6AddrGenMode(mIfName, mode);
150         } catch (RemoteException e) {
151             logError("Unable to set IPv6 addrgen mode: %s", e);
152             return false;
153         } catch (ServiceSpecificException e) {
154             if (e.errorCode != OsConstants.EOPNOTSUPP) {
155                 logError("Unable to set IPv6 addrgen mode: %s", e);
156                 return false;
157             }
158         }
159         return true;
160     }
161 
162     /**
163      * Add an address to the interface.
164      */
addAddress(LinkAddress addr)165     public boolean addAddress(LinkAddress addr) {
166         return addAddress(addr.getAddress(), addr.getPrefixLength());
167     }
168 
169     /**
170      * Add an address to the interface.
171      */
addAddress(InetAddress ip, int prefixLen)172     public boolean addAddress(InetAddress ip, int prefixLen) {
173         try {
174             mNetd.interfaceAddAddress(mIfName, ip.getHostAddress(), prefixLen);
175         } catch (ServiceSpecificException | RemoteException e) {
176             logError("failed to add %s/%d: %s", ip, prefixLen, e);
177             return false;
178         }
179         return true;
180     }
181 
182     /**
183      * Remove an address from the interface.
184      */
removeAddress(InetAddress ip, int prefixLen)185     public boolean removeAddress(InetAddress ip, int prefixLen) {
186         try {
187             mNetd.interfaceDelAddress(mIfName, ip.getHostAddress(), prefixLen);
188         } catch (ServiceSpecificException | RemoteException e) {
189             logError("failed to remove %s/%d: %s", ip, prefixLen, e);
190             return false;
191         }
192         return true;
193     }
194 
195     /**
196      * Remove all addresses from the interface.
197      */
clearAllAddresses()198     public boolean clearAllAddresses() {
199         try {
200             mNetd.interfaceClearAddrs(mIfName);
201         } catch (Exception e) {
202             logError("Failed to clear addresses: %s", e);
203             return false;
204         }
205         return true;
206     }
207 
logError(String fmt, Object... args)208     private void logError(String fmt, Object... args) {
209         mLog.e(String.format(fmt, args));
210     }
211 }
212