blob: 2c1aa4fcf97c98f0b2f6ba181bae8872ba3681b5 [file] [log] [blame]
/*
* Copyright (C) 2022 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.server.hdmi;
import android.hardware.tv.hdmi.earc.IEArc;
import android.hardware.tv.hdmi.earc.IEArcCallback;
import android.os.Handler;
import android.os.IBinder;
import android.os.Looper;
import android.os.RemoteException;
import android.os.ServiceManager;
import android.os.ServiceSpecificException;
import com.android.internal.annotations.VisibleForTesting;
final class HdmiEarcController {
private static final String TAG = "HdmiEarcController";
// Handler instance to process HAL calls.
private Handler mControlHandler;
private final HdmiControlService mService;
private EArcHalWrapper mEArcAidl;
private final class EArcHalWrapper implements IBinder.DeathRecipient {
private IEArc mEArc;
private IEArcCallback mEArcCallback;
@Override
public void binderDied() {
mEArc.asBinder().unlinkToDeath(this, 0);
connectToHal();
if (mEArcCallback != null) {
nativeSetCallback(mEArcCallback);
}
}
boolean connectToHal() {
mEArc =
IEArc.Stub.asInterface(
ServiceManager.getService(IEArc.DESCRIPTOR + "/default"));
if (mEArc == null) {
return false;
}
try {
mEArc.asBinder().linkToDeath(this, 0);
} catch (RemoteException e) {
HdmiLogger.error("Couldn't link callback object: ", e);
}
return true;
}
public boolean nativeInit() {
return connectToHal();
}
public void nativeSetEArcEnabled(boolean enabled) {
try {
mEArc.setEArcEnabled(enabled);
} catch (ServiceSpecificException sse) {
HdmiLogger.error(
"Could not set eARC enabled to " + enabled + ". Error: ", sse.errorCode);
} catch (RemoteException re) {
HdmiLogger.error("Could not set eARC enabled to " + enabled + ":. Exception: ", re);
}
}
public boolean nativeIsEArcEnabled() {
try {
return mEArc.isEArcEnabled();
} catch (RemoteException re) {
HdmiLogger.error("Could not read if eARC is enabled. Exception: ", re);
return false;
}
}
public void nativeSetCallback(IEArcCallback callback) {
mEArcCallback = callback;
try {
mEArc.setCallback(callback);
} catch (RemoteException re) {
HdmiLogger.error("Could not set callback. Exception: ", re);
}
}
public byte nativeGetState(int portId) {
try {
return mEArc.getState(portId);
} catch (RemoteException re) {
HdmiLogger.error("Could not get eARC state. Exception: ", re);
return -1;
}
}
public byte[] nativeGetLastReportedAudioCapabilities(int portId) {
try {
return mEArc.getLastReportedAudioCapabilities(portId);
} catch (RemoteException re) {
HdmiLogger.error(
"Could not read last reported audio capabilities. Exception: ", re);
return null;
}
}
}
// Private constructor. Use HdmiEarcController.create().
private HdmiEarcController(HdmiControlService service) {
mService = service;
}
/**
* A factory method to get {@link HdmiEarcController}. If it fails to initialize
* inner device or has no device it will return {@code null}.
*
* <p>Declared as package-private, accessed by {@link HdmiControlService} only.
* @param service {@link HdmiControlService} instance used to create internal handler
* and to pass callback for incoming message or event.
* @return {@link HdmiEarcController} if device is initialized successfully. Otherwise,
* returns {@code null}.
*/
static HdmiEarcController create(HdmiControlService service) {
// TODO add the native wrapper and return null if eARC HAL is not present.
HdmiEarcController controller = new HdmiEarcController(service);
if (!controller.init()) {
HdmiLogger.warning("Could not connect to eARC AIDL HAL.");
return null;
}
return controller;
}
private boolean init() {
mEArcAidl = new EArcHalWrapper();
if (mEArcAidl.nativeInit()) {
mControlHandler = new Handler(mService.getServiceLooper());
mEArcAidl.nativeSetCallback(new EarcAidlCallback());
return true;
}
return false;
}
private void assertRunOnServiceThread() {
if (Looper.myLooper() != mControlHandler.getLooper()) {
throw new IllegalStateException("Should run on service thread.");
}
}
@VisibleForTesting
void runOnServiceThread(Runnable runnable) {
mControlHandler.post(new WorkSourceUidPreservingRunnable(runnable));
}
/**
* Enable eARC in the HAL
* @param enabled
*/
@HdmiAnnotations.ServiceThreadOnly
void setEarcEnabled(boolean enabled) {
assertRunOnServiceThread();
mEArcAidl.nativeSetEArcEnabled(enabled);
}
/**
* Getter for the current eARC state.
* @param portId the ID of the port on which to get the connection state
* @return the current eARC state
*/
@HdmiAnnotations.ServiceThreadOnly
@Constants.EarcStatus
int getState(int portId) {
return mEArcAidl.nativeGetState(portId);
}
/**
* Ask the HAL to report the last eARC capabilities that the connected audio system reported.
*
* @return the raw eARC capabilities
*/
@HdmiAnnotations.ServiceThreadOnly
byte[] getLastReportedCaps(int portId) {
return mEArcAidl.nativeGetLastReportedAudioCapabilities(portId);
}
final class EarcAidlCallback extends IEArcCallback.Stub {
public void onStateChange(@Constants.EarcStatus byte status, int portId) {
runOnServiceThread(
() -> mService.handleEarcStateChange(status, portId));
}
public void onCapabilitiesReported(byte[] rawCapabilities, int portId) {
runOnServiceThread(
() -> mService.handleEarcCapabilitiesReported(rawCapabilities, portId));
}
@Override
public synchronized String getInterfaceHash() throws RemoteException {
return IEArcCallback.Stub.HASH;
}
@Override
public int getInterfaceVersion() throws RemoteException {
return IEArcCallback.Stub.VERSION;
}
}
}