blob: f9a2dbb1bdb4263a9c04f34f5da1ff6fdc9f521b [file] [log] [blame]
/**
* Copyright (C) 2017 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.hardware.radio;
import android.annotation.Nullable;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import com.android.internal.annotations.GuardedBy;
import java.util.List;
import java.util.Map;
import java.util.Objects;
/**
* Implements the ITunerCallback interface by forwarding calls to RadioTuner.Callback.
*/
final class TunerCallbackAdapter extends ITunerCallback.Stub {
private static final String TAG = "BroadcastRadio.TunerCallbackAdapter";
private final Object mLock = new Object();
private final RadioTuner.Callback mCallback;
private final Handler mHandler;
@GuardedBy("mLock")
@Nullable ProgramList mProgramList;
// cache for deprecated methods
@GuardedBy("mLock")
boolean mIsAntennaConnected = true;
@GuardedBy("mLock")
@Nullable List<RadioManager.ProgramInfo> mLastCompleteList;
@GuardedBy("mLock")
private boolean mDelayedCompleteCallback;
@GuardedBy("mLock")
@Nullable RadioManager.ProgramInfo mCurrentProgramInfo;
TunerCallbackAdapter(RadioTuner.Callback callback, @Nullable Handler handler) {
mCallback = Objects.requireNonNull(callback, "Callback cannot be null");
if (handler == null) {
mHandler = new Handler(Looper.getMainLooper());
} else {
mHandler = handler;
}
}
void close() {
synchronized (mLock) {
if (mProgramList != null) {
mProgramList.close();
}
}
}
void setProgramListObserver(@Nullable ProgramList programList,
ProgramList.OnCloseListener closeListener) {
Objects.requireNonNull(closeListener, "CloseListener cannot be null");
synchronized (mLock) {
if (mProgramList != null) {
Log.w(TAG, "Previous program list observer wasn't properly closed, closing it...");
mProgramList.close();
}
mProgramList = programList;
if (programList == null) {
return;
}
programList.setOnCloseListener(() -> {
synchronized (mLock) {
if (mProgramList != programList) {
return;
}
mProgramList = null;
mLastCompleteList = null;
}
closeListener.onClose();
});
programList.addOnCompleteListener(() -> {
synchronized (mLock) {
if (mProgramList != programList) {
return;
}
mLastCompleteList = programList.toList();
if (mDelayedCompleteCallback) {
Log.d(TAG, "Sending delayed onBackgroundScanComplete callback");
sendBackgroundScanCompleteLocked();
}
}
});
}
}
@Nullable List<RadioManager.ProgramInfo> getLastCompleteList() {
synchronized (mLock) {
return mLastCompleteList;
}
}
void clearLastCompleteList() {
synchronized (mLock) {
mLastCompleteList = null;
}
}
@Nullable RadioManager.ProgramInfo getCurrentProgramInformation() {
synchronized (mLock) {
return mCurrentProgramInfo;
}
}
boolean isAntennaConnected() {
boolean isConnected;
synchronized (mLock) {
isConnected = mIsAntennaConnected;
}
return isConnected;
}
@Override
public void onError(int status) {
mHandler.post(() -> mCallback.onError(status));
}
@Override
public void onTuneFailed(int status, @Nullable ProgramSelector selector) {
mHandler.post(() -> mCallback.onTuneFailed(status, selector));
int errorCode;
switch (status) {
case RadioTuner.TUNER_RESULT_CANCELED:
errorCode = RadioTuner.ERROR_CANCELLED;
break;
case RadioManager.STATUS_PERMISSION_DENIED:
case RadioManager.STATUS_DEAD_OBJECT:
errorCode = RadioTuner.ERROR_SERVER_DIED;
break;
case RadioManager.STATUS_ERROR:
case RadioManager.STATUS_NO_INIT:
case RadioManager.STATUS_BAD_VALUE:
case RadioManager.STATUS_INVALID_OPERATION:
case RadioTuner.TUNER_RESULT_INTERNAL_ERROR:
case RadioTuner.TUNER_RESULT_INVALID_ARGUMENTS:
case RadioTuner.TUNER_RESULT_INVALID_STATE:
case RadioTuner.TUNER_RESULT_NOT_SUPPORTED:
case RadioTuner.TUNER_RESULT_UNKNOWN_ERROR:
Log.i(TAG, "Got an error with no mapping to the legacy API (" + status
+ "), doing a best-effort conversion to ERROR_SCAN_TIMEOUT");
// fall through
case RadioManager.STATUS_TIMED_OUT:
case RadioTuner.TUNER_RESULT_TIMEOUT:
default:
errorCode = RadioTuner.ERROR_SCAN_TIMEOUT;
}
mHandler.post(() -> mCallback.onError(errorCode));
}
@Override
public void onConfigurationChanged(RadioManager.BandConfig config) {
mHandler.post(() -> mCallback.onConfigurationChanged(config));
}
@Override
public void onCurrentProgramInfoChanged(RadioManager.ProgramInfo info) {
if (info == null) {
Log.e(TAG, "ProgramInfo must not be null");
return;
}
synchronized (mLock) {
mCurrentProgramInfo = info;
}
mHandler.post(() -> {
mCallback.onProgramInfoChanged(info);
RadioMetadata metadata = info.getMetadata();
if (metadata != null) mCallback.onMetadataChanged(metadata);
});
}
@Override
public void onTrafficAnnouncement(boolean active) {
mHandler.post(() -> mCallback.onTrafficAnnouncement(active));
}
@Override
public void onEmergencyAnnouncement(boolean active) {
mHandler.post(() -> mCallback.onEmergencyAnnouncement(active));
}
@Override
public void onAntennaState(boolean connected) {
synchronized (mLock) {
mIsAntennaConnected = connected;
}
mHandler.post(() -> mCallback.onAntennaState(connected));
}
@Override
public void onBackgroundScanAvailabilityChange(boolean isAvailable) {
mHandler.post(() -> mCallback.onBackgroundScanAvailabilityChange(isAvailable));
}
@GuardedBy("mLock")
private void sendBackgroundScanCompleteLocked() {
mDelayedCompleteCallback = false;
mHandler.post(() -> mCallback.onBackgroundScanComplete());
}
@Override
public void onBackgroundScanComplete() {
synchronized (mLock) {
if (mLastCompleteList == null) {
Log.i(TAG, "Got onBackgroundScanComplete callback, but the "
+ "program list didn't get through yet. Delaying it...");
mDelayedCompleteCallback = true;
return;
}
sendBackgroundScanCompleteLocked();
}
}
@Override
public void onProgramListChanged() {
mHandler.post(() -> mCallback.onProgramListChanged());
}
@Override
public void onProgramListUpdated(ProgramList.Chunk chunk) {
mHandler.post(() -> {
synchronized (mLock) {
if (mProgramList == null) {
return;
}
mProgramList.apply(Objects.requireNonNull(chunk, "Chunk cannot be null"));
}
});
}
@Override
public void onConfigFlagUpdated(@RadioManager.ConfigFlag int flag, boolean value) {
mHandler.post(() -> mCallback.onConfigFlagUpdated(flag, value));
}
@Override
public void onParametersUpdated(Map<String, String> parameters) {
mHandler.post(() -> mCallback.onParametersUpdated(parameters));
}
}