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.internal.net.ipsec.ike.message; 18 19 import android.annotation.IntDef; 20 import android.net.LinkAddress; 21 import android.net.ipsec.ike.IkeManager; 22 import android.net.ipsec.ike.IkeSessionParams; 23 import android.net.ipsec.ike.IkeSessionParams.ConfigRequestIpv4PcscfServer; 24 import android.net.ipsec.ike.IkeSessionParams.ConfigRequestIpv6PcscfServer; 25 import android.net.ipsec.ike.IkeSessionParams.IkeConfigRequest; 26 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv4Address; 27 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv4DhcpServer; 28 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv4DnsServer; 29 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv4Netmask; 30 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv6Address; 31 import android.net.ipsec.ike.TunnelModeChildSessionParams.ConfigRequestIpv6DnsServer; 32 import android.net.ipsec.ike.TunnelModeChildSessionParams.TunnelModeChildConfigRequest; 33 import android.net.ipsec.ike.exceptions.InvalidSyntaxException; 34 import android.os.PersistableBundle; 35 36 import com.android.internal.annotations.VisibleForTesting; 37 import com.android.server.vcn.util.PersistableBundleUtils; 38 39 import java.lang.annotation.Retention; 40 import java.lang.annotation.RetentionPolicy; 41 import java.net.Inet4Address; 42 import java.net.Inet6Address; 43 import java.net.InetAddress; 44 import java.net.UnknownHostException; 45 import java.nio.BufferUnderflowException; 46 import java.nio.ByteBuffer; 47 import java.nio.charset.Charset; 48 import java.nio.charset.StandardCharsets; 49 import java.util.LinkedList; 50 import java.util.List; 51 import java.util.Objects; 52 53 /** 54 * This class represents Configuration payload. 55 * 56 * <p>Configuration payload is used to exchange configuration information between IKE peers. 57 * 58 * <p>Configuration type should be consistent with the IKE message direction (e.g. a request Config 59 * Payload should be in a request IKE message). IKE library will ignore Config Payload with 60 * inconsistent type or with unrecognized type. 61 * 62 * @see <a href="https://tools.ietf.org/html/rfc7296#section-3.6">RFC 7296, Internet Key Exchange 63 * Protocol Version 2 (IKEv2)</a> 64 */ 65 public final class IkeConfigPayload extends IkePayload { 66 private static final int CONFIG_HEADER_RESERVED_LEN = 3; 67 private static final int CONFIG_HEADER_LEN = 4; 68 69 @Retention(RetentionPolicy.SOURCE) 70 @IntDef({ 71 CONFIG_ATTR_INTERNAL_IP4_ADDRESS, 72 CONFIG_ATTR_INTERNAL_IP4_NETMASK, 73 CONFIG_ATTR_INTERNAL_IP4_DNS, 74 CONFIG_ATTR_INTERNAL_IP4_DHCP, 75 CONFIG_ATTR_APPLICATION_VERSION, 76 CONFIG_ATTR_INTERNAL_IP6_ADDRESS, 77 CONFIG_ATTR_INTERNAL_IP6_DNS, 78 CONFIG_ATTR_INTERNAL_IP4_SUBNET, 79 CONFIG_ATTR_SUPPORTED_ATTRIBUTES, 80 CONFIG_ATTR_INTERNAL_IP6_SUBNET, 81 CONFIG_ATTR_IP4_PCSCF, 82 CONFIG_ATTR_IP6_PCSCF 83 }) 84 public @interface ConfigAttr {} 85 86 public static final int CONFIG_ATTR_INTERNAL_IP4_ADDRESS = 1; 87 public static final int CONFIG_ATTR_INTERNAL_IP4_NETMASK = 2; 88 public static final int CONFIG_ATTR_INTERNAL_IP4_DNS = 3; 89 public static final int CONFIG_ATTR_INTERNAL_IP4_DHCP = 6; 90 public static final int CONFIG_ATTR_APPLICATION_VERSION = 7; 91 public static final int CONFIG_ATTR_INTERNAL_IP6_ADDRESS = 8; 92 public static final int CONFIG_ATTR_INTERNAL_IP6_DNS = 10; 93 public static final int CONFIG_ATTR_INTERNAL_IP4_SUBNET = 13; 94 public static final int CONFIG_ATTR_SUPPORTED_ATTRIBUTES = 14; 95 public static final int CONFIG_ATTR_INTERNAL_IP6_SUBNET = 15; 96 public static final int CONFIG_ATTR_IP4_PCSCF = 20; 97 public static final int CONFIG_ATTR_IP6_PCSCF = 21; 98 99 @Retention(RetentionPolicy.SOURCE) 100 @IntDef({CONFIG_TYPE_REQUEST, CONFIG_TYPE_REPLY}) 101 public @interface ConfigType {} 102 103 // We don't support CONFIG_TYPE_SET and CONFIG_TYPE_ACK 104 public static final int CONFIG_TYPE_REQUEST = 1; 105 public static final int CONFIG_TYPE_REPLY = 2; 106 107 @ConfigType public final int configType; 108 public final List<ConfigAttribute> recognizedAttributeList; 109 110 /** Build an IkeConfigPayload from a decoded inbound IKE packet. */ IkeConfigPayload(boolean critical, byte[] payloadBody)111 IkeConfigPayload(boolean critical, byte[] payloadBody) throws InvalidSyntaxException { 112 super(PAYLOAD_TYPE_CP, critical); 113 114 ByteBuffer inputBuffer = ByteBuffer.wrap(payloadBody); 115 configType = Byte.toUnsignedInt(inputBuffer.get()); 116 inputBuffer.get(new byte[CONFIG_HEADER_RESERVED_LEN]); 117 118 recognizedAttributeList = ConfigAttribute.decodeAttributesFrom(inputBuffer); 119 120 // For an inbound Config Payload, IKE library is only able to handle a Config Reply or IKE 121 // Session attribute requests in a Config Request. For interoperability, netmask validation 122 // will be skipped for Config(Request) and config payloads with unsupported config types. 123 if (configType == CONFIG_TYPE_REPLY) { 124 validateNetmaskInReply(); 125 } 126 } 127 128 /** Build an IkeConfigPayload instance for an outbound IKE packet. */ IkeConfigPayload(boolean isReply, List<ConfigAttribute> attributeList)129 public IkeConfigPayload(boolean isReply, List<ConfigAttribute> attributeList) { 130 super(PAYLOAD_TYPE_CP, false); 131 this.configType = isReply ? CONFIG_TYPE_REPLY : CONFIG_TYPE_REQUEST; 132 this.recognizedAttributeList = attributeList; 133 } 134 validateNetmaskInReply()135 private void validateNetmaskInReply() throws InvalidSyntaxException { 136 boolean hasIpv4Address = false; 137 int numNetmask = 0; 138 139 for (ConfigAttribute attr : recognizedAttributeList) { 140 if (attr.isEmptyValue()) { 141 IkeManager.getIkeLog() 142 .d( 143 "IkeConfigPayload", 144 "Found empty attribute in a Config Payload reply " 145 + attr.attributeType); 146 } 147 switch (attr.attributeType) { 148 case CONFIG_ATTR_INTERNAL_IP4_ADDRESS: 149 if (!attr.isEmptyValue()) hasIpv4Address = true; 150 break; 151 case CONFIG_ATTR_INTERNAL_IP4_NETMASK: 152 if (!attr.isEmptyValue()) numNetmask++; 153 break; 154 default: 155 continue; 156 } 157 } 158 159 if (!hasIpv4Address && numNetmask > 0) { 160 throw new InvalidSyntaxException( 161 "Found INTERNAL_IP4_NETMASK attribute but no INTERNAL_IP4_ADDRESS attribute"); 162 } 163 164 if (numNetmask > 1) { 165 throw new InvalidSyntaxException("Found more than one INTERNAL_IP4_NETMASK"); 166 } 167 } 168 169 // TODO: Create ConfigAttribute subclasses for each attribute. 170 171 /** This class represents common information of all Configuration Attributes. */ 172 public abstract static class ConfigAttribute { 173 private static final String ENCODED_ATTRIBUTE_BYTES_KEY = "encodedAttribute"; 174 175 private static final int ATTRIBUTE_TYPE_MASK = 0x7fff; 176 177 private static final int ATTRIBUTE_HEADER_LEN = 4; 178 private static final int IPV4_PREFIX_LEN_MAX = 32; 179 180 protected static final int VALUE_LEN_NOT_INCLUDED = 0; 181 182 protected static final int IPV4_ADDRESS_LEN = 4; 183 protected static final int IPV6_ADDRESS_LEN = 16; 184 protected static final int PREFIX_LEN_LEN = 1; 185 186 public final int attributeType; 187 ConfigAttribute(int attributeType)188 protected ConfigAttribute(int attributeType) { 189 this.attributeType = attributeType; 190 } 191 ConfigAttribute(int attributeType, int len)192 protected ConfigAttribute(int attributeType, int len) throws InvalidSyntaxException { 193 this(attributeType); 194 195 if (!isLengthValid(len)) { 196 throw new InvalidSyntaxException("Invalid configuration length"); 197 } 198 } 199 200 /** 201 * Constructs this object by deserializing a PersistableBundle. 202 * 203 * <p>Constructed ConfigAttributes are guaranteed to be valid, as checked by 204 * #decodeAttributesFrom(ByteBuffer) 205 */ fromPersistableBundle(PersistableBundle in)206 public static ConfigAttribute fromPersistableBundle(PersistableBundle in) { 207 Objects.requireNonNull(in, "PersistableBundle is null"); 208 209 PersistableBundle byteArrayBundle = 210 in.getPersistableBundle(ENCODED_ATTRIBUTE_BYTES_KEY); 211 ByteBuffer buffer = 212 ByteBuffer.wrap(PersistableBundleUtils.toByteArray(byteArrayBundle)); 213 214 ConfigAttribute attribute; 215 try { 216 attribute = decodeSingleAttributeFrom(buffer); 217 } catch (NegativeArraySizeException 218 | BufferUnderflowException 219 | InvalidSyntaxException e) { 220 throw new IllegalArgumentException( 221 "PersistableBundle contains invalid Config request"); 222 } 223 224 if (buffer.hasRemaining()) { 225 throw new IllegalArgumentException( 226 "Unexpected trailing bytes in Config request PersistableBundle"); 227 } 228 229 return attribute; 230 } 231 232 /** Serializes this object to a PersistableBundle */ toPersistableBundle()233 public PersistableBundle toPersistableBundle() { 234 final PersistableBundle result = new PersistableBundle(); 235 236 ByteBuffer buffer = ByteBuffer.allocate(getAttributeLen()); 237 encodeAttributeToByteBuffer(buffer); 238 239 result.putPersistableBundle( 240 ENCODED_ATTRIBUTE_BYTES_KEY, 241 PersistableBundleUtils.fromByteArray(buffer.array())); 242 return result; 243 } 244 245 /** 246 * Package private method to decode ConfigAttribute list from an inbound packet 247 * 248 * <p>NegativeArraySizeException and BufferUnderflowException will be caught in {@link 249 * IkeMessage} 250 */ decodeAttributesFrom(ByteBuffer inputBuffer)251 static List<ConfigAttribute> decodeAttributesFrom(ByteBuffer inputBuffer) 252 throws InvalidSyntaxException { 253 List<ConfigAttribute> configList = new LinkedList(); 254 255 while (inputBuffer.hasRemaining()) { 256 ConfigAttribute attribute = decodeSingleAttributeFrom(inputBuffer); 257 if (attribute != null) { 258 configList.add(attribute); 259 } 260 } 261 262 return configList; 263 } 264 265 /** 266 * Method to decode a single ConfigAttribute from a ByteBuffer. 267 * 268 * <p>Caller should be responsible for handling NegativeArraySizeException and 269 * BufferUnderflowException. 270 */ decodeSingleAttributeFrom(ByteBuffer inputBuffer)271 private static ConfigAttribute decodeSingleAttributeFrom(ByteBuffer inputBuffer) 272 throws InvalidSyntaxException { 273 int attributeType = Short.toUnsignedInt(inputBuffer.getShort()); 274 int length = Short.toUnsignedInt(inputBuffer.getShort()); 275 byte[] value = new byte[length]; 276 inputBuffer.get(value); 277 278 switch (attributeType) { 279 case CONFIG_ATTR_INTERNAL_IP4_ADDRESS: 280 return new ConfigAttributeIpv4Address(value); 281 case CONFIG_ATTR_INTERNAL_IP4_NETMASK: 282 return new ConfigAttributeIpv4Netmask(value); 283 case CONFIG_ATTR_INTERNAL_IP4_DNS: 284 return new ConfigAttributeIpv4Dns(value); 285 case CONFIG_ATTR_INTERNAL_IP4_DHCP: 286 return new ConfigAttributeIpv4Dhcp(value); 287 case CONFIG_ATTR_APPLICATION_VERSION: 288 return new ConfigAttributeAppVersion(value); 289 case CONFIG_ATTR_INTERNAL_IP6_ADDRESS: 290 return new ConfigAttributeIpv6Address(value); 291 case CONFIG_ATTR_INTERNAL_IP6_DNS: 292 return new ConfigAttributeIpv6Dns(value); 293 case CONFIG_ATTR_INTERNAL_IP4_SUBNET: 294 return new ConfigAttributeIpv4Subnet(value); 295 case CONFIG_ATTR_INTERNAL_IP6_SUBNET: 296 return new ConfigAttributeIpv6Subnet(value); 297 case CONFIG_ATTR_IP4_PCSCF: 298 return new ConfigAttributeIpv4Pcscf(value); 299 case CONFIG_ATTR_IP6_PCSCF: 300 return new ConfigAttributeIpv6Pcscf(value); 301 default: 302 IkeManager.getIkeLog() 303 .i("IkeConfigPayload", "Unrecognized attribute type: " + attributeType); 304 return null; 305 } 306 } 307 308 /** Encode attribute to ByteBuffer. */ encodeAttributeToByteBuffer(ByteBuffer buffer)309 public void encodeAttributeToByteBuffer(ByteBuffer buffer) { 310 buffer.putShort((short) (attributeType & ATTRIBUTE_TYPE_MASK)) 311 .putShort((short) getValueLength()); 312 encodeValueToByteBuffer(buffer); 313 } 314 315 /** Get attribute length. */ getAttributeLen()316 public int getAttributeLen() { 317 return ATTRIBUTE_HEADER_LEN + getValueLength(); 318 } 319 320 /** Returns if this attribute value is empty. */ isEmptyValue()321 public boolean isEmptyValue() { 322 return getValueLength() == VALUE_LEN_NOT_INCLUDED; 323 } 324 325 @Override hashCode()326 public int hashCode() { 327 return Objects.hash(attributeType); 328 } 329 330 @Override equals(Object o)331 public boolean equals(Object o) { 332 if (!(o instanceof ConfigAttribute)) { 333 return false; 334 } 335 336 return attributeType == ((ConfigAttribute) o).attributeType; 337 } 338 netmaskToPrefixLen(Inet4Address address)339 protected static int netmaskToPrefixLen(Inet4Address address) { 340 byte[] bytes = address.getAddress(); 341 342 int netmaskInt = ByteBuffer.wrap(bytes).getInt(); 343 int leftmostBitMask = 0x80000000; 344 345 int prefixLen = 0; 346 while ((netmaskInt & leftmostBitMask) == leftmostBitMask) { 347 prefixLen++; 348 netmaskInt <<= 1; 349 } 350 351 if (netmaskInt != 0) { 352 throw new IllegalArgumentException("Invalid netmask address"); 353 } 354 355 return prefixLen; 356 } 357 prefixToNetmaskBytes(int prefixLen)358 protected static byte[] prefixToNetmaskBytes(int prefixLen) { 359 if (prefixLen > IPV4_PREFIX_LEN_MAX || prefixLen < 0) { 360 throw new IllegalArgumentException("Invalid IPv4 prefix length."); 361 } 362 363 int netmaskInt = (int) (((long) 0xffffffff) << (IPV4_PREFIX_LEN_MAX - prefixLen)); 364 byte[] netmask = new byte[IPV4_ADDRESS_LEN]; 365 366 ByteBuffer buffer = ByteBuffer.allocate(IPV4_ADDRESS_LEN); 367 buffer.putInt(netmaskInt); 368 return buffer.array(); 369 } 370 encodeValueToByteBuffer(ByteBuffer buffer)371 protected abstract void encodeValueToByteBuffer(ByteBuffer buffer); 372 getValueLength()373 protected abstract int getValueLength(); 374 isLengthValid(int length)375 protected abstract boolean isLengthValid(int length); 376 } 377 378 /** This class supports strong typing for IkeConfigRequest(s) */ 379 public abstract static class IkeConfigAttribute extends ConfigAttribute 380 implements IkeConfigRequest { IkeConfigAttribute(int attributeType)381 protected IkeConfigAttribute(int attributeType) { 382 super(attributeType); 383 } 384 IkeConfigAttribute(int attributeType, int len)385 protected IkeConfigAttribute(int attributeType, int len) throws InvalidSyntaxException { 386 super(attributeType, len); 387 } 388 } 389 390 /** This class supports strong typing for TunnelModeChildConfigRequest(s) */ 391 public abstract static class TunnelModeChildConfigAttribute extends ConfigAttribute 392 implements TunnelModeChildConfigRequest { TunnelModeChildConfigAttribute(int attributeType)393 protected TunnelModeChildConfigAttribute(int attributeType) { 394 super(attributeType); 395 } 396 TunnelModeChildConfigAttribute(int attributeType, int len)397 protected TunnelModeChildConfigAttribute(int attributeType, int len) 398 throws InvalidSyntaxException { 399 super(attributeType, len); 400 } 401 } 402 403 /** 404 * This class represents common information of all Tunnel Mode Child Session Configuration 405 * Attributes for which the value is one IPv4 address or empty. 406 */ 407 abstract static class TunnelModeChildConfigAttrIpv4AddressBase 408 extends TunnelModeChildConfigAttribute implements TunnelModeChildConfigRequest { 409 public final Inet4Address address; 410 TunnelModeChildConfigAttrIpv4AddressBase( int attributeType, Inet4Address address)411 protected TunnelModeChildConfigAttrIpv4AddressBase( 412 int attributeType, Inet4Address address) { 413 super(attributeType); 414 this.address = address; 415 } 416 TunnelModeChildConfigAttrIpv4AddressBase(int attributeType)417 protected TunnelModeChildConfigAttrIpv4AddressBase(int attributeType) { 418 super(attributeType); 419 this.address = null; 420 } 421 TunnelModeChildConfigAttrIpv4AddressBase(int attributeType, byte[] value)422 protected TunnelModeChildConfigAttrIpv4AddressBase(int attributeType, byte[] value) 423 throws InvalidSyntaxException { 424 super(attributeType, value.length); 425 426 if (value.length == VALUE_LEN_NOT_INCLUDED) { 427 address = null; 428 return; 429 } 430 431 try { 432 InetAddress netAddress = InetAddress.getByAddress(value); 433 434 if (!(netAddress instanceof Inet4Address)) { 435 throw new InvalidSyntaxException("Invalid IPv4 address."); 436 } 437 address = (Inet4Address) netAddress; 438 } catch (UnknownHostException e) { 439 throw new InvalidSyntaxException("Invalid attribute value", e); 440 } 441 } 442 443 @Override encodeValueToByteBuffer(ByteBuffer buffer)444 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 445 if (address == null) return; // No encoding necessary 446 447 buffer.put(address.getAddress()); 448 } 449 450 @Override getValueLength()451 protected int getValueLength() { 452 return address == null ? 0 : IPV4_ADDRESS_LEN; 453 } 454 455 @Override isLengthValid(int length)456 protected boolean isLengthValid(int length) { 457 return length == IPV4_ADDRESS_LEN || length == VALUE_LEN_NOT_INCLUDED; 458 } 459 460 @Override hashCode()461 public int hashCode() { 462 return Objects.hash(super.hashCode(), address); 463 } 464 465 @Override equals(Object o)466 public boolean equals(Object o) { 467 if (!super.equals(o) || !(o instanceof TunnelModeChildConfigAttrIpv4AddressBase)) { 468 return false; 469 } 470 471 TunnelModeChildConfigAttrIpv4AddressBase other = 472 (TunnelModeChildConfigAttrIpv4AddressBase) o; 473 474 return Objects.equals(address, other.address); 475 } 476 } 477 478 /** 479 * This class represents common information of all IKE Session Configuration Attributes for 480 * which the value is one IPv4 address or empty. 481 */ 482 abstract static class IkeConfigAttrIpv4AddressBase extends IkeConfigAttribute 483 implements IkeSessionParams.IkeConfigRequest { 484 public final Inet4Address address; 485 IkeConfigAttrIpv4AddressBase(int attributeType, Inet4Address address)486 protected IkeConfigAttrIpv4AddressBase(int attributeType, Inet4Address address) { 487 super(attributeType); 488 this.address = address; 489 } 490 IkeConfigAttrIpv4AddressBase(int attributeType)491 protected IkeConfigAttrIpv4AddressBase(int attributeType) { 492 super(attributeType); 493 this.address = null; 494 } 495 IkeConfigAttrIpv4AddressBase(int attributeType, byte[] value)496 protected IkeConfigAttrIpv4AddressBase(int attributeType, byte[] value) 497 throws InvalidSyntaxException { 498 super(attributeType, value.length); 499 500 if (value.length == VALUE_LEN_NOT_INCLUDED) { 501 address = null; 502 return; 503 } 504 505 try { 506 InetAddress netAddress = InetAddress.getByAddress(value); 507 508 if (!(netAddress instanceof Inet4Address)) { 509 throw new InvalidSyntaxException("Invalid IPv4 address."); 510 } 511 address = (Inet4Address) netAddress; 512 } catch (UnknownHostException e) { 513 throw new InvalidSyntaxException("Invalid attribute value", e); 514 } 515 } 516 517 @Override encodeValueToByteBuffer(ByteBuffer buffer)518 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 519 if (address == null) return; // No encoding necessary 520 521 buffer.put(address.getAddress()); 522 } 523 524 @Override getValueLength()525 protected int getValueLength() { 526 return address == null ? 0 : IPV4_ADDRESS_LEN; 527 } 528 529 @Override isLengthValid(int length)530 protected boolean isLengthValid(int length) { 531 return length == IPV4_ADDRESS_LEN || length == VALUE_LEN_NOT_INCLUDED; 532 } 533 534 @Override hashCode()535 public int hashCode() { 536 return Objects.hash(super.hashCode(), address); 537 } 538 539 @Override equals(Object o)540 public boolean equals(Object o) { 541 if (!super.equals(o) || !(o instanceof IkeConfigAttrIpv4AddressBase)) { 542 return false; 543 } 544 545 IkeConfigAttrIpv4AddressBase other = (IkeConfigAttrIpv4AddressBase) o; 546 547 return Objects.equals(address, other.address); 548 } 549 } 550 551 /** This class represents Configuration Attribute for IPv4 internal address. */ 552 public static class ConfigAttributeIpv4Address extends TunnelModeChildConfigAttrIpv4AddressBase 553 implements ConfigRequestIpv4Address { 554 /** Construct an instance with specified address for an outbound packet. */ ConfigAttributeIpv4Address(Inet4Address ipv4Address)555 public ConfigAttributeIpv4Address(Inet4Address ipv4Address) { 556 super(CONFIG_ATTR_INTERNAL_IP4_ADDRESS, ipv4Address); 557 } 558 559 /** 560 * Construct an instance without a specified address for an outbound packet. 561 * 562 * <p>It must be only used in a configuration request. 563 */ ConfigAttributeIpv4Address()564 public ConfigAttributeIpv4Address() { 565 super(CONFIG_ATTR_INTERNAL_IP4_ADDRESS); 566 } 567 568 /** Construct an instance with a decoded inbound packet. */ 569 @VisibleForTesting ConfigAttributeIpv4Address(byte[] value)570 ConfigAttributeIpv4Address(byte[] value) throws InvalidSyntaxException { 571 super(CONFIG_ATTR_INTERNAL_IP4_ADDRESS, value); 572 } 573 574 @Override getAddress()575 public Inet4Address getAddress() { 576 return address; 577 } 578 } 579 580 /** 581 * This class represents Configuration Attribute for IPv4 netmask. 582 * 583 * <p>Non-empty values for this attribute in a CFG_REQUEST do not make sense and thus MUST NOT 584 * be included 585 */ 586 public static class ConfigAttributeIpv4Netmask extends TunnelModeChildConfigAttrIpv4AddressBase 587 implements ConfigRequestIpv4Netmask { 588 /** 589 * Construct an instance without a specified netmask for an outbound packet. 590 * 591 * <p>It must be only used in a configuration request. 592 */ ConfigAttributeIpv4Netmask()593 public ConfigAttributeIpv4Netmask() { 594 super(CONFIG_ATTR_INTERNAL_IP4_NETMASK); 595 } 596 597 /** Construct an instance with a decoded inbound packet. */ 598 @VisibleForTesting ConfigAttributeIpv4Netmask(byte[] value)599 public ConfigAttributeIpv4Netmask(byte[] value) throws InvalidSyntaxException { 600 super(CONFIG_ATTR_INTERNAL_IP4_NETMASK, value); 601 602 if (address == null) return; 603 try { 604 netmaskToPrefixLen(address); 605 } catch (IllegalArgumentException e) { 606 throw new InvalidSyntaxException("Invalid attribute value", e); 607 } 608 } 609 610 /** Convert netmask to prefix length. */ getPrefixLen()611 public int getPrefixLen() { 612 return netmaskToPrefixLen(address); 613 } 614 } 615 616 /** This class represents Configuration Attribute for IPv4 DHCP server. */ 617 public static class ConfigAttributeIpv4Dhcp extends TunnelModeChildConfigAttrIpv4AddressBase 618 implements ConfigRequestIpv4DhcpServer { 619 /** Construct an instance with specified DHCP server address for an outbound packet. */ ConfigAttributeIpv4Dhcp(Inet4Address ipv4Address)620 public ConfigAttributeIpv4Dhcp(Inet4Address ipv4Address) { 621 super(CONFIG_ATTR_INTERNAL_IP4_DHCP, ipv4Address); 622 } 623 624 /** 625 * Construct an instance without a specified DHCP server address for an outbound packet. 626 * 627 * <p>It must be only used in a configuration request. 628 */ ConfigAttributeIpv4Dhcp()629 public ConfigAttributeIpv4Dhcp() { 630 super(CONFIG_ATTR_INTERNAL_IP4_DHCP); 631 } 632 633 /** Construct an instance with a decoded inbound packet. */ 634 @VisibleForTesting ConfigAttributeIpv4Dhcp(byte[] value)635 ConfigAttributeIpv4Dhcp(byte[] value) throws InvalidSyntaxException { 636 super(CONFIG_ATTR_INTERNAL_IP4_DHCP, value); 637 } 638 getAddress()639 public Inet4Address getAddress() { 640 return address; 641 } 642 } 643 644 /** 645 * This class represents Configuration Attribute for IPv4 DNS. 646 * 647 * <p>There is no use case to create a DNS request for a specfic DNS server address. As an IKE 648 * client, we will only support building an empty DNS attribute for an outbound IKE packet. 649 */ 650 public static class ConfigAttributeIpv4Dns extends TunnelModeChildConfigAttrIpv4AddressBase 651 implements ConfigRequestIpv4DnsServer { 652 /** Construct an instance with specified DNS server address for an outbound packet. */ ConfigAttributeIpv4Dns(Inet4Address ipv4Address)653 public ConfigAttributeIpv4Dns(Inet4Address ipv4Address) { 654 super(CONFIG_ATTR_INTERNAL_IP4_DNS, ipv4Address); 655 } 656 657 /** 658 * Construct an instance without a specified DNS server address for an outbound packet. 659 * 660 * <p>It must be only used in a configuration request. 661 */ ConfigAttributeIpv4Dns()662 public ConfigAttributeIpv4Dns() { 663 super(CONFIG_ATTR_INTERNAL_IP4_DNS); 664 } 665 666 /** Construct an instance with a decoded inbound packet. */ 667 @VisibleForTesting ConfigAttributeIpv4Dns(byte[] value)668 ConfigAttributeIpv4Dns(byte[] value) throws InvalidSyntaxException { 669 super(CONFIG_ATTR_INTERNAL_IP4_DNS, value); 670 } 671 getAddress()672 public Inet4Address getAddress() { 673 return address; 674 } 675 } 676 677 // TODO: b/145454043 Remove constructors for building outbound 678 // INTERNAL_IP4_SUBNET/INTERNAL_IP6_SUBNET because they should never be in a config request and 679 // IKE library, as an IKE client, will never send them in a config reply either. 680 681 /** 682 * This class represents Configuration Attribute for IPv4 subnets. 683 * 684 * <p>According to RFC 7296, INTERNAL_IP4_SUBNET in configuration requests cannot be used 685 * reliably because the meaning is unclear. 686 */ 687 public static class ConfigAttributeIpv4Subnet extends TunnelModeChildConfigAttribute { 688 private static final int VALUE_LEN = 2 * IPV4_ADDRESS_LEN; 689 690 public final LinkAddress linkAddress; 691 692 /** Construct an instance with specified subnet for an outbound packet. */ ConfigAttributeIpv4Subnet(LinkAddress ipv4LinkAddress)693 public ConfigAttributeIpv4Subnet(LinkAddress ipv4LinkAddress) { 694 super(CONFIG_ATTR_INTERNAL_IP4_SUBNET); 695 696 if (!ipv4LinkAddress.isIpv4()) { 697 throw new IllegalArgumentException("Input LinkAddress is not IPv4"); 698 } 699 700 this.linkAddress = ipv4LinkAddress; 701 } 702 703 /** 704 * Construct an instance without a specified subnet for an outbound packet. 705 * 706 * <p>It must be only used in a configuration request. 707 */ ConfigAttributeIpv4Subnet()708 public ConfigAttributeIpv4Subnet() { 709 super(CONFIG_ATTR_INTERNAL_IP4_SUBNET); 710 this.linkAddress = null; 711 } 712 713 /** Construct an instance with a decoded inbound packet. */ 714 @VisibleForTesting ConfigAttributeIpv4Subnet(byte[] value)715 ConfigAttributeIpv4Subnet(byte[] value) throws InvalidSyntaxException { 716 super(CONFIG_ATTR_INTERNAL_IP4_SUBNET, value.length); 717 718 if (value.length == VALUE_LEN_NOT_INCLUDED) { 719 linkAddress = null; 720 return; 721 } 722 723 try { 724 ByteBuffer inputBuffer = ByteBuffer.wrap(value); 725 byte[] ipBytes = new byte[IPV4_ADDRESS_LEN]; 726 inputBuffer.get(ipBytes); 727 byte[] netmaskBytes = new byte[IPV4_ADDRESS_LEN]; 728 inputBuffer.get(netmaskBytes); 729 730 InetAddress address = InetAddress.getByAddress(ipBytes); 731 InetAddress netmask = InetAddress.getByAddress(netmaskBytes); 732 validateInet4AddressTypeOrThrow(address); 733 validateInet4AddressTypeOrThrow(netmask); 734 735 linkAddress = new LinkAddress(address, netmaskToPrefixLen((Inet4Address) netmask)); 736 } catch (UnknownHostException | IllegalArgumentException e) { 737 throw new InvalidSyntaxException("Invalid attribute value", e); 738 } 739 } 740 validateInet4AddressTypeOrThrow(InetAddress address)741 private void validateInet4AddressTypeOrThrow(InetAddress address) { 742 if (!(address instanceof Inet4Address)) { 743 throw new IllegalArgumentException("Input InetAddress is not IPv4"); 744 } 745 } 746 747 @Override encodeValueToByteBuffer(ByteBuffer buffer)748 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 749 if (linkAddress == null) { 750 buffer.put(new byte[VALUE_LEN_NOT_INCLUDED]); 751 return; 752 } 753 byte[] netmaskBytes = prefixToNetmaskBytes(linkAddress.getPrefixLength()); 754 buffer.put(linkAddress.getAddress().getAddress()).put(netmaskBytes); 755 } 756 757 @Override getValueLength()758 protected int getValueLength() { 759 return linkAddress == null ? 0 : VALUE_LEN; 760 } 761 762 @Override isLengthValid(int length)763 protected boolean isLengthValid(int length) { 764 return length == VALUE_LEN || length == VALUE_LEN_NOT_INCLUDED; 765 } 766 767 @Override hashCode()768 public int hashCode() { 769 return Objects.hash(super.hashCode(), linkAddress); 770 } 771 772 @Override equals(Object o)773 public boolean equals(Object o) { 774 if (!super.equals(o) || !(o instanceof ConfigAttributeIpv4Subnet)) { 775 return false; 776 } 777 778 ConfigAttributeIpv4Subnet other = (ConfigAttributeIpv4Subnet) o; 779 780 return Objects.equals(linkAddress, other.linkAddress); 781 } 782 } 783 784 /** This class represents an IPv4 P_CSCF address attribute */ 785 public static class ConfigAttributeIpv4Pcscf extends IkeConfigAttrIpv4AddressBase 786 implements ConfigRequestIpv4PcscfServer { 787 /** Construct an instance with a specified P_CSCF server address for an outbound packet. */ ConfigAttributeIpv4Pcscf(Inet4Address ipv4Address)788 public ConfigAttributeIpv4Pcscf(Inet4Address ipv4Address) { 789 super(CONFIG_ATTR_IP4_PCSCF, ipv4Address); 790 } 791 792 /** 793 * Construct an instance without a specified P_CSCF server address for an outbound packet. 794 * 795 * <p>It must be only used in a configuration request. 796 */ ConfigAttributeIpv4Pcscf()797 public ConfigAttributeIpv4Pcscf() { 798 super(CONFIG_ATTR_IP4_PCSCF); 799 } 800 801 /** Construct an instance with a decoded inbound packet. */ 802 @VisibleForTesting ConfigAttributeIpv4Pcscf(byte[] value)803 ConfigAttributeIpv4Pcscf(byte[] value) throws InvalidSyntaxException { 804 super(CONFIG_ATTR_IP4_PCSCF, value); 805 } 806 807 @Override getAddress()808 public Inet4Address getAddress() { 809 return address; 810 } 811 } 812 813 /** 814 * This class represents common information of all Tunnel Mode Child Session Configuration 815 * Attributes for which the value is one IPv6 address or empty. 816 */ 817 abstract static class TunnelModeChildConfigAttrIpv6AddressBase 818 extends TunnelModeChildConfigAttribute implements TunnelModeChildConfigRequest { 819 public final Inet6Address address; 820 TunnelModeChildConfigAttrIpv6AddressBase( int attributeType, Inet6Address address)821 protected TunnelModeChildConfigAttrIpv6AddressBase( 822 int attributeType, Inet6Address address) { 823 super(attributeType); 824 this.address = address; 825 } 826 TunnelModeChildConfigAttrIpv6AddressBase(int attributeType)827 protected TunnelModeChildConfigAttrIpv6AddressBase(int attributeType) { 828 super(attributeType); 829 this.address = null; 830 } 831 TunnelModeChildConfigAttrIpv6AddressBase(int attributeType, byte[] value)832 protected TunnelModeChildConfigAttrIpv6AddressBase(int attributeType, byte[] value) 833 throws InvalidSyntaxException { 834 super(attributeType, value.length); 835 836 if (value.length == VALUE_LEN_NOT_INCLUDED) { 837 address = null; 838 return; 839 } 840 841 try { 842 InetAddress netAddress = InetAddress.getByAddress(value); 843 844 if (!(netAddress instanceof Inet6Address)) { 845 throw new InvalidSyntaxException("Invalid IPv6 address."); 846 } 847 address = (Inet6Address) netAddress; 848 } catch (UnknownHostException e) { 849 throw new InvalidSyntaxException("Invalid attribute value", e); 850 } 851 } 852 853 @Override encodeValueToByteBuffer(ByteBuffer buffer)854 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 855 if (address == null) return; // No encoding necessary 856 857 buffer.put(address.getAddress()); 858 } 859 860 @Override getValueLength()861 protected int getValueLength() { 862 return address == null ? 0 : IPV6_ADDRESS_LEN; 863 } 864 865 @Override isLengthValid(int length)866 protected boolean isLengthValid(int length) { 867 return length == IPV6_ADDRESS_LEN || length == VALUE_LEN_NOT_INCLUDED; 868 } 869 870 @Override hashCode()871 public int hashCode() { 872 return Objects.hash(super.hashCode(), address); 873 } 874 875 @Override equals(Object o)876 public boolean equals(Object o) { 877 if (!super.equals(o) || !(o instanceof TunnelModeChildConfigAttrIpv6AddressBase)) { 878 return false; 879 } 880 881 TunnelModeChildConfigAttrIpv6AddressBase other = 882 (TunnelModeChildConfigAttrIpv6AddressBase) o; 883 884 return Objects.equals(address, other.address); 885 } 886 } 887 888 /** 889 * This class represents common information of all IKE Session Configuration Attributes for 890 * which the value is one IPv6 address or empty. 891 */ 892 abstract static class IkeConfigAttrIpv6AddressBase extends IkeConfigAttribute 893 implements IkeConfigRequest { 894 public final Inet6Address address; 895 IkeConfigAttrIpv6AddressBase(int attributeType, Inet6Address address)896 protected IkeConfigAttrIpv6AddressBase(int attributeType, Inet6Address address) { 897 super(attributeType); 898 this.address = address; 899 } 900 IkeConfigAttrIpv6AddressBase(int attributeType)901 protected IkeConfigAttrIpv6AddressBase(int attributeType) { 902 super(attributeType); 903 this.address = null; 904 } 905 IkeConfigAttrIpv6AddressBase(int attributeType, byte[] value)906 protected IkeConfigAttrIpv6AddressBase(int attributeType, byte[] value) 907 throws InvalidSyntaxException { 908 super(attributeType, value.length); 909 910 if (value.length == VALUE_LEN_NOT_INCLUDED) { 911 address = null; 912 return; 913 } 914 915 try { 916 InetAddress netAddress = InetAddress.getByAddress(value); 917 918 if (!(netAddress instanceof Inet6Address)) { 919 throw new InvalidSyntaxException("Invalid IPv6 address."); 920 } 921 address = (Inet6Address) netAddress; 922 } catch (UnknownHostException e) { 923 throw new InvalidSyntaxException("Invalid attribute value", e); 924 } 925 } 926 927 @Override encodeValueToByteBuffer(ByteBuffer buffer)928 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 929 if (address == null) return; // No encoding necessary 930 931 buffer.put(address.getAddress()); 932 } 933 934 @Override getValueLength()935 protected int getValueLength() { 936 return address == null ? 0 : IPV6_ADDRESS_LEN; 937 } 938 939 @Override isLengthValid(int length)940 protected boolean isLengthValid(int length) { 941 return length == IPV6_ADDRESS_LEN || length == VALUE_LEN_NOT_INCLUDED; 942 } 943 944 @Override hashCode()945 public int hashCode() { 946 return Objects.hash(super.hashCode(), address); 947 } 948 949 @Override equals(Object o)950 public boolean equals(Object o) { 951 if (!super.equals(o) || !(o instanceof IkeConfigAttrIpv6AddressBase)) { 952 return false; 953 } 954 955 IkeConfigAttrIpv6AddressBase other = (IkeConfigAttrIpv6AddressBase) o; 956 957 return Objects.equals(address, other.address); 958 } 959 } 960 961 /** 962 * This class represents common information of all Configuration Attributes for which the value 963 * is an IPv6 address range. 964 * 965 * <p>These attributes contains an IPv6 address and a prefix length. 966 */ 967 abstract static class TunnelModeChildConfigAttrIpv6AddrRangeBase 968 extends TunnelModeChildConfigAttribute { 969 private static final int VALUE_LEN = IPV6_ADDRESS_LEN + PREFIX_LEN_LEN; 970 971 public final LinkAddress linkAddress; 972 TunnelModeChildConfigAttrIpv6AddrRangeBase( int attributeType, LinkAddress ipv6LinkAddress)973 protected TunnelModeChildConfigAttrIpv6AddrRangeBase( 974 int attributeType, LinkAddress ipv6LinkAddress) { 975 super(attributeType); 976 977 validateIpv6LinkAddressTypeOrThrow(ipv6LinkAddress); 978 linkAddress = ipv6LinkAddress; 979 } 980 TunnelModeChildConfigAttrIpv6AddrRangeBase(int attributeType)981 protected TunnelModeChildConfigAttrIpv6AddrRangeBase(int attributeType) { 982 super(attributeType); 983 linkAddress = null; 984 } 985 TunnelModeChildConfigAttrIpv6AddrRangeBase(int attributeType, byte[] value)986 protected TunnelModeChildConfigAttrIpv6AddrRangeBase(int attributeType, byte[] value) 987 throws InvalidSyntaxException { 988 super(attributeType, value.length); 989 990 if (value.length == VALUE_LEN_NOT_INCLUDED) { 991 linkAddress = null; 992 return; 993 } 994 995 try { 996 ByteBuffer inputBuffer = ByteBuffer.wrap(value); 997 byte[] ip6AddrBytes = new byte[IPV6_ADDRESS_LEN]; 998 inputBuffer.get(ip6AddrBytes); 999 InetAddress address = InetAddress.getByAddress(ip6AddrBytes); 1000 1001 int prefixLen = Byte.toUnsignedInt(inputBuffer.get()); 1002 1003 linkAddress = new LinkAddress(address, prefixLen); 1004 validateIpv6LinkAddressTypeOrThrow(linkAddress); 1005 } catch (UnknownHostException | IllegalArgumentException e) { 1006 throw new InvalidSyntaxException("Invalid attribute value", e); 1007 } 1008 } 1009 validateIpv6LinkAddressTypeOrThrow(LinkAddress address)1010 private void validateIpv6LinkAddressTypeOrThrow(LinkAddress address) { 1011 if (!address.isIpv6()) { 1012 throw new IllegalArgumentException("Input LinkAddress is not IPv6"); 1013 } 1014 } 1015 1016 @Override encodeValueToByteBuffer(ByteBuffer buffer)1017 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 1018 if (linkAddress == null) { 1019 buffer.put(new byte[VALUE_LEN_NOT_INCLUDED]); 1020 return; 1021 } 1022 1023 buffer.put(linkAddress.getAddress().getAddress()) 1024 .put((byte) linkAddress.getPrefixLength()); 1025 } 1026 1027 @Override getValueLength()1028 protected int getValueLength() { 1029 return linkAddress == null ? VALUE_LEN_NOT_INCLUDED : VALUE_LEN; 1030 } 1031 1032 @Override isLengthValid(int length)1033 protected boolean isLengthValid(int length) { 1034 return length == VALUE_LEN || length == VALUE_LEN_NOT_INCLUDED; 1035 } 1036 1037 @Override hashCode()1038 public int hashCode() { 1039 return Objects.hash(super.hashCode(), linkAddress); 1040 } 1041 1042 @Override equals(Object o)1043 public boolean equals(Object o) { 1044 if (!super.equals(o) || !(o instanceof TunnelModeChildConfigAttrIpv6AddrRangeBase)) { 1045 return false; 1046 } 1047 1048 TunnelModeChildConfigAttrIpv6AddrRangeBase other = 1049 (TunnelModeChildConfigAttrIpv6AddrRangeBase) o; 1050 1051 return Objects.equals(linkAddress, other.linkAddress); 1052 } 1053 } 1054 1055 /** This class represents Configuration Attribute for IPv6 internal addresses. */ 1056 public static class ConfigAttributeIpv6Address 1057 extends TunnelModeChildConfigAttrIpv6AddrRangeBase implements ConfigRequestIpv6Address { 1058 /** Construct an instance with specified address for an outbound packet. */ ConfigAttributeIpv6Address(LinkAddress ipv6LinkAddress)1059 public ConfigAttributeIpv6Address(LinkAddress ipv6LinkAddress) { 1060 super(CONFIG_ATTR_INTERNAL_IP6_ADDRESS, ipv6LinkAddress); 1061 } 1062 1063 /** 1064 * Construct an instance without a specified address for an outbound packet. 1065 * 1066 * <p>It must be only used in a configuration request. 1067 */ ConfigAttributeIpv6Address()1068 public ConfigAttributeIpv6Address() { 1069 super(CONFIG_ATTR_INTERNAL_IP6_ADDRESS); 1070 } 1071 1072 /** Construct an instance with a decoded inbound packet. */ 1073 @VisibleForTesting ConfigAttributeIpv6Address(byte[] value)1074 ConfigAttributeIpv6Address(byte[] value) throws InvalidSyntaxException { 1075 super(CONFIG_ATTR_INTERNAL_IP6_ADDRESS, value); 1076 } 1077 1078 @Override getAddress()1079 public Inet6Address getAddress() { 1080 return linkAddress == null ? null : (Inet6Address) linkAddress.getAddress(); 1081 } 1082 1083 @Override getPrefixLength()1084 public int getPrefixLength() { 1085 return linkAddress == null ? 0 : linkAddress.getPrefixLength(); 1086 } 1087 } 1088 1089 /** 1090 * This class represents Configuration Attribute for IPv6 subnets. 1091 * 1092 * <p>According to RFC 7296, INTERNAL_IP6_SUBNET in configuration requests cannot be used 1093 * reliably because the meaning is unclear. 1094 */ 1095 public static class ConfigAttributeIpv6Subnet 1096 extends TunnelModeChildConfigAttrIpv6AddrRangeBase { 1097 /** Construct an instance with specified subnet for an outbound packet. */ ConfigAttributeIpv6Subnet(LinkAddress ipv6LinkAddress)1098 public ConfigAttributeIpv6Subnet(LinkAddress ipv6LinkAddress) { 1099 super(CONFIG_ATTR_INTERNAL_IP6_SUBNET, ipv6LinkAddress); 1100 } 1101 1102 /** 1103 * Construct an instance without a specified subnet for an outbound packet. 1104 * 1105 * <p>It must be only used in a configuration request. 1106 */ ConfigAttributeIpv6Subnet()1107 public ConfigAttributeIpv6Subnet() { 1108 super(CONFIG_ATTR_INTERNAL_IP6_SUBNET); 1109 } 1110 1111 /** Construct an instance with a decoded inbound packet. */ 1112 @VisibleForTesting ConfigAttributeIpv6Subnet(byte[] value)1113 ConfigAttributeIpv6Subnet(byte[] value) throws InvalidSyntaxException { 1114 super(CONFIG_ATTR_INTERNAL_IP6_SUBNET, value); 1115 } 1116 } 1117 1118 /** 1119 * This class represents Configuration Attribute for IPv6 DNS. 1120 * 1121 * <p>There is no use case to create a DNS request for a specfic DNS server address. As an IKE 1122 * client, we will only support building an empty DNS attribute for an outbound IKE packet. 1123 */ 1124 public static class ConfigAttributeIpv6Dns extends TunnelModeChildConfigAttrIpv6AddressBase 1125 implements ConfigRequestIpv6DnsServer { 1126 /** Construct an instance with specified DNS server address for an outbound packet. */ ConfigAttributeIpv6Dns(Inet6Address ipv6Address)1127 public ConfigAttributeIpv6Dns(Inet6Address ipv6Address) { 1128 super(CONFIG_ATTR_INTERNAL_IP6_DNS, ipv6Address); 1129 } 1130 1131 /** 1132 * Construct an instance without a specified DNS server address for an outbound packet. 1133 * 1134 * <p>It must be only used in a configuration request. 1135 */ ConfigAttributeIpv6Dns()1136 public ConfigAttributeIpv6Dns() { 1137 super(CONFIG_ATTR_INTERNAL_IP6_DNS); 1138 } 1139 ConfigAttributeIpv6Dns(byte[] value)1140 protected ConfigAttributeIpv6Dns(byte[] value) throws InvalidSyntaxException { 1141 super(CONFIG_ATTR_INTERNAL_IP6_DNS, value); 1142 } 1143 getAddress()1144 public Inet6Address getAddress() { 1145 return address; 1146 } 1147 } 1148 1149 /** This class represents an IPv6 P_CSCF address attribute */ 1150 public static class ConfigAttributeIpv6Pcscf extends IkeConfigAttrIpv6AddressBase 1151 implements ConfigRequestIpv6PcscfServer { 1152 /** Construct an instance with a specified P_CSCF server address for an outbound packet. */ ConfigAttributeIpv6Pcscf(Inet6Address ipv6Address)1153 public ConfigAttributeIpv6Pcscf(Inet6Address ipv6Address) { 1154 super(CONFIG_ATTR_IP6_PCSCF, ipv6Address); 1155 } 1156 1157 /** 1158 * Construct an instance without a specified P_CSCF server address for an outbound packet. 1159 * 1160 * <p>It must be only used in a configuration request. 1161 */ ConfigAttributeIpv6Pcscf()1162 public ConfigAttributeIpv6Pcscf() { 1163 super(CONFIG_ATTR_IP6_PCSCF); 1164 } 1165 ConfigAttributeIpv6Pcscf(byte[] value)1166 protected ConfigAttributeIpv6Pcscf(byte[] value) throws InvalidSyntaxException { 1167 super(CONFIG_ATTR_IP6_PCSCF, value); 1168 } 1169 1170 @Override getAddress()1171 public Inet6Address getAddress() { 1172 return address; 1173 } 1174 } 1175 1176 /** This class represents an application version attribute */ 1177 public static class ConfigAttributeAppVersion extends ConfigAttribute { 1178 private static final Charset ASCII = StandardCharsets.US_ASCII; 1179 private static final String APP_VERSION_NONE = ""; 1180 1181 public final String applicationVersion; 1182 1183 /** 1184 * Construct an instance for an outbound packet for requesting remote application version. 1185 */ ConfigAttributeAppVersion()1186 public ConfigAttributeAppVersion() { 1187 this(APP_VERSION_NONE); 1188 } 1189 1190 /** Construct an instance for an outbound packet with local application version. */ ConfigAttributeAppVersion(String localAppVersion)1191 public ConfigAttributeAppVersion(String localAppVersion) { 1192 super(CONFIG_ATTR_APPLICATION_VERSION); 1193 applicationVersion = localAppVersion; 1194 } 1195 1196 /** Construct an instance from a decoded inbound packet. */ ConfigAttributeAppVersion(byte[] value)1197 protected ConfigAttributeAppVersion(byte[] value) throws InvalidSyntaxException { 1198 super(CONFIG_ATTR_APPLICATION_VERSION); 1199 applicationVersion = new String(value, ASCII); 1200 } 1201 encodeValueToByteBuffer(ByteBuffer buffer)1202 protected void encodeValueToByteBuffer(ByteBuffer buffer) { 1203 buffer.put(applicationVersion.getBytes(ASCII)); 1204 } 1205 getValueLength()1206 protected int getValueLength() { 1207 return applicationVersion.getBytes(ASCII).length; 1208 } 1209 1210 @Override isLengthValid(int length)1211 protected boolean isLengthValid(int length) { 1212 return length >= 0; 1213 } 1214 1215 @Override hashCode()1216 public int hashCode() { 1217 return Objects.hash(super.hashCode(), applicationVersion); 1218 } 1219 1220 @Override equals(Object o)1221 public boolean equals(Object o) { 1222 if (!super.equals(o) || !(o instanceof ConfigAttributeAppVersion)) { 1223 return false; 1224 } 1225 1226 ConfigAttributeAppVersion other = (ConfigAttributeAppVersion) o; 1227 1228 return Objects.equals(applicationVersion, other.applicationVersion); 1229 } 1230 } 1231 1232 /** 1233 * Encode Configuration payload to ByteBUffer. 1234 * 1235 * @param nextPayload type of payload that follows this payload. 1236 * @param byteBuffer destination ByteBuffer that stores encoded payload. 1237 */ 1238 @Override encodeToByteBuffer(@ayloadType int nextPayload, ByteBuffer byteBuffer)1239 protected void encodeToByteBuffer(@PayloadType int nextPayload, ByteBuffer byteBuffer) { 1240 encodePayloadHeaderToByteBuffer(nextPayload, getPayloadLength(), byteBuffer); 1241 byteBuffer.put((byte) configType).put(new byte[CONFIG_HEADER_RESERVED_LEN]); 1242 1243 for (ConfigAttribute attr : recognizedAttributeList) { 1244 attr.encodeAttributeToByteBuffer(byteBuffer); 1245 } 1246 } 1247 1248 /** 1249 * Get entire payload length. 1250 * 1251 * @return entire payload length. 1252 */ 1253 @Override getPayloadLength()1254 protected int getPayloadLength() { 1255 int len = GENERIC_HEADER_LENGTH + CONFIG_HEADER_LEN; 1256 1257 for (ConfigAttribute attr : recognizedAttributeList) { 1258 len += attr.getAttributeLen(); 1259 } 1260 1261 return len; 1262 } 1263 1264 /** 1265 * Return the payload type as a String. 1266 * 1267 * @return the payload type as a String. 1268 */ 1269 @Override getTypeString()1270 public String getTypeString() { 1271 switch (configType) { 1272 case CONFIG_TYPE_REQUEST: 1273 return "CP(Req)"; 1274 case CONFIG_TYPE_REPLY: 1275 return "CP(Reply)"; 1276 default: 1277 return "CP(" + configType + ")"; 1278 } 1279 } 1280 } 1281