blob: 5433a80300a50c434259a78062eb89a4bfcaac7b [file] [log] [blame]
/*
* Copyright (C) 2020 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 com.android.internal.net.ipsec.ike.keepalive;
import static android.net.SocketKeepalive.ERROR_HARDWARE_ERROR;
import static android.net.SocketKeepalive.ERROR_INSUFFICIENT_RESOURCES;
import static android.net.SocketKeepalive.ERROR_INVALID_INTERVAL;
import static android.net.SocketKeepalive.ERROR_INVALID_IP_ADDRESS;
import static android.net.SocketKeepalive.ERROR_INVALID_LENGTH;
import static android.net.SocketKeepalive.ERROR_INVALID_NETWORK;
import static android.net.SocketKeepalive.ERROR_INVALID_PORT;
import static android.net.SocketKeepalive.ERROR_INVALID_SOCKET;
import static android.net.SocketKeepalive.ERROR_SOCKET_NOT_IDLE;
import static android.net.SocketKeepalive.ERROR_UNSUPPORTED;
import static android.net.ipsec.ike.IkeManager.getIkeLog;
import static android.net.ipsec.ike.IkeSessionParams.IKE_OPTION_AUTOMATIC_KEEPALIVE_ON_OFF;
import android.annotation.Nullable;
import android.content.Context;
import android.net.ConnectivityManager;
import android.net.IpSecManager.UdpEncapsulationSocket;
import android.net.Network;
import android.net.SocketKeepalive;
import android.net.ipsec.ike.IkeSessionParams;
import com.android.internal.net.ipsec.ike.shim.ShimUtils;
import java.net.Inet4Address;
import java.util.concurrent.Executors;
/** This class provides methods to manage hardware offload NAT-T keepalive. */
public class HardwareKeepaliveImpl implements IkeNattKeepalive.NattKeepalive {
private static final String TAG = "HardwareKeepaliveImpl";
private final int mKeepaliveDelaySeconds;
private final SocketKeepalive mSocketKeepalive;
private final HardwareKeepaliveCallback mHardwareKeepaliveCb;
/***
* The NATT keepalive start options.
*
* This must be a value from {@link SocketKeepalive#StartFlags}.
*/
private final int mKeepaliveOptions;
/** Network underpinned by the IKE session, which can be monitored for automatic keepalive */
@Nullable private final Network mUnderpinnedNetwork;
/** Construct an instance of HardwareKeepaliveImpl */
public HardwareKeepaliveImpl(
Context context,
ConnectivityManager connectMgr,
int keepaliveDelaySeconds,
IkeSessionParams ikeParams,
Inet4Address src,
Inet4Address dest,
UdpEncapsulationSocket socket,
Network network,
Network underpinnedNetwork,
HardwareKeepaliveCallback hardwareKeepaliveCb) {
// Setup for hardware offload keepalive. Fail to create mSocketKeepalive will cause
// MySocketKeepaliveCb#onError to be fired
mKeepaliveDelaySeconds = keepaliveDelaySeconds;
mHardwareKeepaliveCb = hardwareKeepaliveCb;
mKeepaliveOptions = getKeepaliveStartOptions(ikeParams);
mUnderpinnedNetwork = underpinnedNetwork;
mSocketKeepalive =
connectMgr.createSocketKeepalive(
network,
socket,
src,
dest,
Executors.newSingleThreadExecutor(),
new MySocketKeepaliveCb());
}
@Override
public void start() {
ShimUtils.getInstance().startKeepalive(
mSocketKeepalive, mKeepaliveDelaySeconds, mKeepaliveOptions, mUnderpinnedNetwork);
}
@Override
public void stop() {
mSocketKeepalive.stop();
}
@Override
public void onAlarmFired() {
// Do thing. Should never be called
}
private static int getKeepaliveStartOptions(IkeSessionParams ikeParams) {
int flags = 0;
if (ikeParams.hasIkeOption(IKE_OPTION_AUTOMATIC_KEEPALIVE_ON_OFF)) {
flags |= SocketKeepalive.FLAG_AUTOMATIC_ON_OFF;
}
return flags;
}
/** Callback interface to receive states change of hardware keepalive */
public interface HardwareKeepaliveCallback {
/** Called when there is a hardware error for keepalive. */
void onHardwareOffloadError();
/**
* Called when there is a network or configuration error which cause sending keepalive
* packet to fail
*/
void onNetworkError();
/** Called when the keepalive has stopped */
void onStopped(HardwareKeepaliveImpl hardwareKeepalive);
}
class MySocketKeepaliveCb extends SocketKeepalive.Callback {
@Override
public void onError(int error) {
getIkeLog().d(TAG, "Hardware offload failed on error: " + error);
switch (error) {
case ERROR_INVALID_NETWORK: // fallthrough
case ERROR_INVALID_PORT: // fallthrough
case ERROR_INVALID_LENGTH: // fallthrough
case ERROR_INVALID_INTERVAL: // fallthrough
case ERROR_INVALID_SOCKET: // fallthrough
case ERROR_SOCKET_NOT_IDLE: // fallthrough
mHardwareKeepaliveCb.onNetworkError();
return;
case ERROR_INVALID_IP_ADDRESS:
// Hardware keepalive is not supported on 464XLAT and this error will be thrown.
// So fallthrough to use software keepalive.
case ERROR_UNSUPPORTED: // fallthrough
case ERROR_HARDWARE_ERROR: // fallthrough
case ERROR_INSUFFICIENT_RESOURCES:
mHardwareKeepaliveCb.onHardwareOffloadError();
return;
default:
mHardwareKeepaliveCb.onNetworkError();
}
}
@Override
public void onStopped() {
mHardwareKeepaliveCb.onStopped(HardwareKeepaliveImpl.this);
}
}
}