summaryrefslogtreecommitdiffstats
path: root/services
diff options
context:
space:
mode:
authorLorenzo Colitti <lorenzo@google.com>2015-03-11 00:01:25 +0900
committerLorenzo Colitti <lorenzo@google.com>2015-03-17 13:21:21 +0900
commit04b8d3aab6acdf382c74d12a89b5fc50fd8d0158 (patch)
treeac8da6040fd3bf9c151c2017a7974db854cffb73 /services
parent29cb944bef7b56ef4b657cd7f2d0c2be96d670d6 (diff)
downloadframeworks_base-04b8d3aab6acdf382c74d12a89b5fc50fd8d0158.zip
frameworks_base-04b8d3aab6acdf382c74d12a89b5fc50fd8d0158.tar.gz
frameworks_base-04b8d3aab6acdf382c74d12a89b5fc50fd8d0158.tar.bz2
DHCP: protocol changes.
1. Define and add parsing code for MTU, max message size, T1, T2. 2. Add common TLVs (message size, hostname, vendor ID) to all packets sent by the client. 3. Don't include requested IP and server ID in renew messages, since the RFC says MUST NOT. 4. Don't hardcode the broadcast flag to true in DISCOVER packets, use what the caller passed in. 5. Make some methods static. Bug: 19704592 Change-Id: I42a0997e468b12e19cad9b403b98fe266e6cea73
Diffstat (limited to 'services')
-rw-r--r--services/net/java/android/net/dhcp/DhcpDiscoverPacket.java3
-rw-r--r--services/net/java/android/net/dhcp/DhcpPacket.java95
-rw-r--r--services/net/java/android/net/dhcp/DhcpRequestPacket.java11
3 files changed, 91 insertions, 18 deletions
diff --git a/services/net/java/android/net/dhcp/DhcpDiscoverPacket.java b/services/net/java/android/net/dhcp/DhcpDiscoverPacket.java
index fba43e6..a031080 100644
--- a/services/net/java/android/net/dhcp/DhcpDiscoverPacket.java
+++ b/services/net/java/android/net/dhcp/DhcpDiscoverPacket.java
@@ -42,7 +42,7 @@ class DhcpDiscoverPacket extends DhcpPacket {
public ByteBuffer buildPacket(int encap, short destUdp, short srcUdp) {
ByteBuffer result = ByteBuffer.allocate(MAX_LENGTH);
fillInPacket(encap, INADDR_BROADCAST, INADDR_ANY, destUdp,
- srcUdp, result, DHCP_BOOTREQUEST, true);
+ srcUdp, result, DHCP_BOOTREQUEST, mBroadcast);
result.flip();
return result;
}
@@ -52,6 +52,7 @@ class DhcpDiscoverPacket extends DhcpPacket {
*/
void finishPacket(ByteBuffer buffer) {
addTlv(buffer, DHCP_MESSAGE_TYPE, DHCP_MESSAGE_TYPE_DISCOVER);
+ addCommonClientTlvs(buffer);
addTlv(buffer, DHCP_PARAMETER_LIST, mRequestedParams);
addTlvEnd(buffer);
}
diff --git a/services/net/java/android/net/dhcp/DhcpPacket.java b/services/net/java/android/net/dhcp/DhcpPacket.java
index b825a39..fd18513 100644
--- a/services/net/java/android/net/dhcp/DhcpPacket.java
+++ b/services/net/java/android/net/dhcp/DhcpPacket.java
@@ -9,6 +9,7 @@ import android.system.OsConstants;
import java.net.Inet4Address;
import java.net.UnknownHostException;
+import java.nio.BufferUnderflowException;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.charset.StandardCharsets;
@@ -32,8 +33,6 @@ abstract class DhcpPacket {
(byte) 0xff, (byte) 0xff, (byte) 0xff,
(byte) 0xff, (byte) 0xff, (byte) 0xff,
};
- // Minimum length of a DHCP packet w/o options: Ethernet header, IP header, fixed bootp format.
- public static final int MIN_PACKET_LENGTH_L2 = 14 + 20 + 300;
/**
* Packet encapsulations.
@@ -129,6 +128,12 @@ abstract class DhcpPacket {
protected String mDomainName;
/**
+ * DHCP Optional Type: DHCP Interface MTU
+ */
+ protected static final byte DHCP_MTU = 26;
+ protected Short mMtu;
+
+ /**
* DHCP Optional Type: DHCP BROADCAST ADDRESS
*/
protected static final byte DHCP_BROADCAST_ADDRESS = 28;
@@ -178,9 +183,22 @@ abstract class DhcpPacket {
protected String mMessage;
/**
+ * DHCP Optional Type: Maximum DHCP Message Size
+ */
+ protected static final byte DHCP_MAX_MESSAGE_SIZE = 57;
+ protected Short mMaxMessageSize;
+
+ /**
* DHCP Optional Type: DHCP Renewal Time Value
*/
protected static final byte DHCP_RENEWAL_TIME = 58;
+ protected Integer mT1;
+
+ /**
+ * DHCP Optional Type: Rebinding Time Value
+ */
+ protected static final byte DHCP_REBINDING_TIME = 59;
+ protected Integer mT2;
/**
* DHCP Optional Type: Vendor Class Identifier
@@ -437,7 +455,7 @@ abstract class DhcpPacket {
/**
* Adds an optional parameter containing a single byte value.
*/
- protected void addTlv(ByteBuffer buf, byte type, byte value) {
+ protected static void addTlv(ByteBuffer buf, byte type, byte value) {
buf.put(type);
buf.put((byte) 1);
buf.put(value);
@@ -446,7 +464,7 @@ abstract class DhcpPacket {
/**
* Adds an optional parameter containing an array of bytes.
*/
- protected void addTlv(ByteBuffer buf, byte type, byte[] payload) {
+ protected static void addTlv(ByteBuffer buf, byte type, byte[] payload) {
if (payload != null) {
buf.put(type);
buf.put((byte) payload.length);
@@ -457,7 +475,7 @@ abstract class DhcpPacket {
/**
* Adds an optional parameter containing an IP address.
*/
- protected void addTlv(ByteBuffer buf, byte type, Inet4Address addr) {
+ protected static void addTlv(ByteBuffer buf, byte type, Inet4Address addr) {
if (addr != null) {
addTlv(buf, type, addr.getAddress());
}
@@ -466,7 +484,7 @@ abstract class DhcpPacket {
/**
* Adds an optional parameter containing a list of IP addresses.
*/
- protected void addTlv(ByteBuffer buf, byte type, List<Inet4Address> addrs) {
+ protected static void addTlv(ByteBuffer buf, byte type, List<Inet4Address> addrs) {
if (addrs != null && addrs.size() > 0) {
buf.put(type);
buf.put((byte)(4 * addrs.size()));
@@ -478,9 +496,20 @@ abstract class DhcpPacket {
}
/**
+ * Adds an optional parameter containing a short integer
+ */
+ protected static void addTlv(ByteBuffer buf, byte type, Short value) {
+ if (value != null) {
+ buf.put(type);
+ buf.put((byte) 2);
+ buf.putShort(value.shortValue());
+ }
+ }
+
+ /**
* Adds an optional parameter containing a simple integer
*/
- protected void addTlv(ByteBuffer buf, byte type, Integer value) {
+ protected static void addTlv(ByteBuffer buf, byte type, Integer value) {
if (value != null) {
buf.put(type);
buf.put((byte) 4);
@@ -491,7 +520,7 @@ abstract class DhcpPacket {
/**
* Adds an optional parameter containing and ASCII string.
*/
- protected void addTlv(ByteBuffer buf, byte type, String str) {
+ protected static void addTlv(ByteBuffer buf, byte type, String str) {
if (str != null) {
buf.put(type);
buf.put((byte) str.length());
@@ -505,11 +534,23 @@ abstract class DhcpPacket {
/**
* Adds the special end-of-optional-parameters indicator.
*/
- protected void addTlvEnd(ByteBuffer buf) {
+ protected static void addTlvEnd(ByteBuffer buf) {
buf.put((byte) 0xFF);
}
/**
+ * Adds common client TLVs.
+ *
+ * TODO: Does this belong here? The alternative would be to modify all the buildXyzPacket
+ * methods to take them.
+ */
+ protected void addCommonClientTlvs(ByteBuffer buf) {
+ addTlv(buf, DHCP_MAX_MESSAGE_SIZE, (short) MAX_LENGTH);
+ addTlv(buf, DHCP_VENDOR_CLASS_ID, "android-dhcp-" + Build.VERSION.RELEASE);
+ addTlv(buf, DHCP_HOST_NAME, SystemProperties.get("net.hostname"));
+ }
+
+ /**
* Converts a MAC from an array of octets to an ASCII string.
*/
public static String macToString(byte[] mac) {
@@ -585,7 +626,6 @@ abstract class DhcpPacket {
byte[] clientMac;
List<Inet4Address> dnsServers = new ArrayList<Inet4Address>();
Inet4Address gateway = null; // aka router
- Integer leaseTime = null;
Inet4Address serverIdentifier = null;
Inet4Address netMask = null;
String message = null;
@@ -598,6 +638,16 @@ abstract class DhcpPacket {
Inet4Address bcAddr = null;
Inet4Address requestedIp = null;
+ // The following are all unsigned integers. Internally we store them as signed integers of
+ // the same length because that way we're guaranteed that they can't be out of the range of
+ // the unsigned field in the packet. Callers wanting to pass in an unsigned value will need
+ // to cast it.
+ Short mtu = null;
+ Short maxMessageSize = null;
+ Integer leaseTime = null;
+ Integer T1 = null;
+ Integer T2 = null;
+
// dhcp options
byte dhcpType = (byte) 0xFF;
@@ -605,7 +655,6 @@ abstract class DhcpPacket {
// check to see if we need to parse L2, IP, and UDP encaps
if (pktType == ENCAP_L2) {
- // System.out.println("buffer len " + packet.limit());
byte[] l2dst = new byte[6];
byte[] l2src = new byte[6];
@@ -618,8 +667,7 @@ abstract class DhcpPacket {
return null;
}
- if ((pktType == ENCAP_L2) || (pktType == ENCAP_L3)) {
- // assume l2type is 0x0800, i.e. IP
+ if (pktType <= ENCAP_L3) {
byte ipTypeAndLength = packet.get();
int ipVersion = (ipTypeAndLength & 0xf0) >> 4;
if (ipVersion != 4) {
@@ -658,7 +706,6 @@ abstract class DhcpPacket {
return null;
}
- // assume bootp
byte type = packet.get();
byte hwType = packet.get();
byte addrLen = packet.get();
@@ -728,6 +775,10 @@ abstract class DhcpPacket {
expectedLen = optionLen;
hostName = readAsciiString(packet, optionLen);
break;
+ case DHCP_MTU:
+ expectedLen = 2;
+ mtu = Short.valueOf(packet.getShort());
+ break;
case DHCP_DOMAIN_NAME:
expectedLen = optionLen;
domainName = readAsciiString(packet, optionLen);
@@ -761,6 +812,18 @@ abstract class DhcpPacket {
expectedLen = optionLen;
message = readAsciiString(packet, optionLen);
break;
+ case DHCP_MAX_MESSAGE_SIZE:
+ expectedLen = 2;
+ maxMessageSize = Short.valueOf(packet.getShort());
+ break;
+ case DHCP_RENEWAL_TIME:
+ expectedLen = 4;
+ T1 = Integer.valueOf(packet.getInt());
+ break;
+ case DHCP_REBINDING_TIME:
+ expectedLen = 4;
+ T2 = Integer.valueOf(packet.getInt());
+ break;
case DHCP_VENDOR_CLASS_ID:
expectedLen = optionLen;
vendorId = readAsciiString(packet, optionLen);
@@ -831,10 +894,14 @@ abstract class DhcpPacket {
newPacket.mHostName = hostName;
newPacket.mLeaseTime = leaseTime;
newPacket.mMessage = message;
+ newPacket.mMtu = mtu;
newPacket.mRequestedIp = requestedIp;
newPacket.mRequestedParams = expectedParams;
newPacket.mServerIdentifier = serverIdentifier;
newPacket.mSubnetMask = netMask;
+ newPacket.mMaxMessageSize = maxMessageSize;
+ newPacket.mT1 = T1;
+ newPacket.mT2 = T2;
newPacket.mVendorId = vendorId;
return newPacket;
}
diff --git a/services/net/java/android/net/dhcp/DhcpRequestPacket.java b/services/net/java/android/net/dhcp/DhcpRequestPacket.java
index 337fdf0..42b7b0c 100644
--- a/services/net/java/android/net/dhcp/DhcpRequestPacket.java
+++ b/services/net/java/android/net/dhcp/DhcpRequestPacket.java
@@ -63,10 +63,15 @@ class DhcpRequestPacket extends DhcpPacket {
System.arraycopy(mClientMac, 0, clientId, 1, 6);
addTlv(buffer, DHCP_MESSAGE_TYPE, DHCP_MESSAGE_TYPE_REQUEST);
- addTlv(buffer, DHCP_PARAMETER_LIST, mRequestedParams);
- addTlv(buffer, DHCP_REQUESTED_IP, mRequestedIp);
- addTlv(buffer, DHCP_SERVER_IDENTIFIER, mServerIdentifier);
+ if (!INADDR_ANY.equals(mRequestedIp)) {
+ addTlv(buffer, DHCP_REQUESTED_IP, mRequestedIp);
+ }
+ if (!INADDR_ANY.equals(mServerIdentifier)) {
+ addTlv(buffer, DHCP_SERVER_IDENTIFIER, mServerIdentifier);
+ }
addTlv(buffer, DHCP_CLIENT_IDENTIFIER, clientId);
+ addCommonClientTlvs(buffer);
+ addTlv(buffer, DHCP_PARAMETER_LIST, mRequestedParams);
addTlvEnd(buffer);
}
}