/*
* Copyright (C) 2019 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.net.ipsec.ike;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP4_ADDRESS;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP4_DHCP;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP4_DNS;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP4_NETMASK;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP4_SUBNET;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP6_ADDRESS;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP6_DNS;
import static com.android.internal.net.ipsec.ike.message.IkeConfigPayload.CONFIG_ATTR_INTERNAL_IP6_SUBNET;
import android.annotation.NonNull;
import android.annotation.SystemApi;
import android.net.IpPrefix;
import android.net.LinkAddress;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttribute;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv4Address;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv4Dhcp;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv4Dns;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv4Netmask;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv4Subnet;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv6Address;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv6Dns;
import com.android.internal.net.ipsec.ike.message.IkeConfigPayload.ConfigAttributeIpv6Subnet;
import java.net.InetAddress;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;
/**
* ChildSessionConfiguration represents the negotiated configuration for a Child Session.
*
*
Configurations include traffic selectors and internal network information.
*
* @hide
*/
@SystemApi
public final class ChildSessionConfiguration {
private static final int IPv4_DEFAULT_PREFIX_LEN = 32;
private final List mInboundTs;
private final List mOutboundTs;
private final List mInternalAddressList;
private final List mInternalDnsAddressList;
private final List mSubnetAddressList;
private final List mInternalDhcpAddressList;
/**
* Construct an instance of {@link ChildSessionConfiguration}.
*
* It is only supported to build a {@link ChildSessionConfiguration} with a Configure(Reply)
* Payload.
*
* @hide
*/
public ChildSessionConfiguration(
List inTs,
List outTs,
IkeConfigPayload configPayload) {
this(inTs, outTs);
if (configPayload.configType != IkeConfigPayload.CONFIG_TYPE_REPLY) {
throw new IllegalArgumentException(
"Cannot build ChildSessionConfiguration with configuration type: "
+ configPayload.configType);
}
// It is validated in IkeConfigPayload that a config reply only has at most one non-empty
// netmask and netmask exists only when IPv4 internal address exists.
ConfigAttributeIpv4Netmask netmaskAttr = null;
for (ConfigAttribute att : configPayload.recognizedAttributeList) {
if (att.attributeType == CONFIG_ATTR_INTERNAL_IP4_NETMASK && !att.isEmptyValue()) {
netmaskAttr = (ConfigAttributeIpv4Netmask) att;
}
}
for (ConfigAttribute att : configPayload.recognizedAttributeList) {
if (att.isEmptyValue()) continue;
switch (att.attributeType) {
case CONFIG_ATTR_INTERNAL_IP4_ADDRESS:
ConfigAttributeIpv4Address addressAttr = (ConfigAttributeIpv4Address) att;
if (netmaskAttr != null) {
mInternalAddressList.add(
new LinkAddress(addressAttr.address, netmaskAttr.getPrefixLen()));
} else {
mInternalAddressList.add(
new LinkAddress(addressAttr.address, IPv4_DEFAULT_PREFIX_LEN));
}
break;
case CONFIG_ATTR_INTERNAL_IP4_NETMASK:
// No action.
break;
case CONFIG_ATTR_INTERNAL_IP6_ADDRESS:
mInternalAddressList.add(((ConfigAttributeIpv6Address) att).linkAddress);
break;
case CONFIG_ATTR_INTERNAL_IP4_DNS:
mInternalDnsAddressList.add(((ConfigAttributeIpv4Dns) att).address);
break;
case CONFIG_ATTR_INTERNAL_IP6_DNS:
mInternalDnsAddressList.add(((ConfigAttributeIpv6Dns) att).address);
break;
case CONFIG_ATTR_INTERNAL_IP4_SUBNET:
ConfigAttributeIpv4Subnet ipv4SubnetAttr = (ConfigAttributeIpv4Subnet) att;
mSubnetAddressList.add(
new IpPrefix(
ipv4SubnetAttr.linkAddress.getAddress(),
ipv4SubnetAttr.linkAddress.getPrefixLength()));
break;
case CONFIG_ATTR_INTERNAL_IP6_SUBNET:
ConfigAttributeIpv6Subnet ipV6SubnetAttr = (ConfigAttributeIpv6Subnet) att;
mSubnetAddressList.add(
new IpPrefix(
ipV6SubnetAttr.linkAddress.getAddress(),
ipV6SubnetAttr.linkAddress.getPrefixLength()));
break;
case CONFIG_ATTR_INTERNAL_IP4_DHCP:
mInternalDhcpAddressList.add(((ConfigAttributeIpv4Dhcp) att).address);
break;
default:
// Not relevant to child session
}
}
}
/**
* Construct an instance of {@link ChildSessionConfiguration}.
*
* @hide
*/
public ChildSessionConfiguration(
List inTs, List outTs) {
mInboundTs = Collections.unmodifiableList(inTs);
mOutboundTs = Collections.unmodifiableList(outTs);
mInternalAddressList = new LinkedList<>();
mInternalDnsAddressList = new LinkedList<>();
mSubnetAddressList = new LinkedList<>();
mInternalDhcpAddressList = new LinkedList<>();
}
/**
* Returns the negotiated inbound traffic selectors.
*
* Only inbound traffic within the range is acceptable to the Child Session.
*
*
The Android platform does not support port-based routing. Port ranges of traffic selectors
* are only informational.
*
* @return the inbound traffic selectors.
*/
@NonNull
public List getInboundTrafficSelectors() {
return mInboundTs;
}
/**
* Returns the negotiated outbound traffic selectors.
*
* Only outbound traffic within the range is acceptable to the Child Session.
*
*
The Android platform does not support port-based routing. Port ranges of traffic selectors
* are only informational.
*
* @return the outbound traffic selectors.
*/
@NonNull
public List getOutboundTrafficSelectors() {
return mOutboundTs;
}
/**
* Returns the assigned internal addresses.
*
* @return the assigned internal addresses, or an empty list when no addresses are assigned by
* the remote IKE server (e.g. for a non-tunnel mode Child Session).
*/
@NonNull
public List getInternalAddresses() {
return Collections.unmodifiableList(mInternalAddressList);
}
/**
* Returns the internal subnets protected by the IKE server.
*
* @return the internal subnets, or an empty list when no information of protected subnets is
* provided by the IKE server (e.g. for a non-tunnel mode Child Session).
*/
@NonNull
public List getInternalSubnets() {
return Collections.unmodifiableList(mSubnetAddressList);
}
/**
* Returns the internal DNS server addresses.
*
* @return the internal DNS server addresses, or an empty list when no DNS server is provided by
* the IKE server (e.g. for a non-tunnel mode Child Session).
*/
@NonNull
public List getInternalDnsServers() {
return Collections.unmodifiableList(mInternalDnsAddressList);
}
/**
* Returns the internal DHCP server addresses.
*
* @return the internal DHCP server addresses, or an empty list when no DHCP server is provided
* by the IKE server (e.g. for a non-tunnel mode Child Session).
*/
@NonNull
public List getInternalDhcpServers() {
return Collections.unmodifiableList(mInternalDhcpAddressList);
}
}