/* * 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 com.android.systemui.car; import android.car.Car; import android.content.Context; import androidx.annotation.AnyThread; import androidx.annotation.VisibleForTesting; import com.android.internal.annotations.GuardedBy; import com.android.systemui.car.dagger.CarSysUIDumpable; import java.util.ArrayList; import java.util.List; import javax.inject.Inject; import javax.inject.Singleton; /** Provides a common connection to the car service that can be shared. */ // It needs to be @Singleton scoped because it is used by a number of components in @WMSingleton & // @SysUISingleton scopes. @Singleton public class CarServiceProvider { private final Context mContext; /** * mListeners is guarded by itself - when requiring locks on both mListeners and mCar, always * obtain the car lock before the listeners. */ @GuardedBy("mListeners") private final List mListeners = new ArrayList<>(); private final Object mCarLock = new Object(); /** * mCar is guarded by mCarLock - when requiring locks on both mListeners and mCar, always * obtain the car lock before the listeners. */ @GuardedBy("mCarLock") private Car mCar; @GuardedBy("mCarLock") private boolean mIsCarReady; @Inject public CarServiceProvider(@CarSysUIDumpable Context context) { mContext = context; mCar = Car.createCar(mContext, /* handler= */ null, Car.CAR_WAIT_TIMEOUT_DO_NOT_WAIT, (car, ready) -> { synchronized (mCarLock) { synchronized (mListeners) { mIsCarReady = ready; mCar = car; if (ready) { for (CarServiceOnConnectedListener listener : mListeners) { listener.onConnected(mCar); } } } } }); } @VisibleForTesting public CarServiceProvider(Context context, Car car) { mContext = context; mCar = car; } /** * Let's other components hook into the connection to the car service. If we're already * connected to the car service, the callback is immediately triggered. */ @AnyThread public void addListener(CarServiceOnConnectedListener listener) { synchronized (mCarLock) { if (mIsCarReady) { listener.onConnected(mCar); } synchronized (mListeners) { mListeners.add(listener); } } } /** * Remove a car service connection listener. */ @AnyThread public void removeListener(CarServiceOnConnectedListener listener) { synchronized (mListeners) { mListeners.remove(listener); } } /** * Listener which is triggered when Car Service is connected. */ public interface CarServiceOnConnectedListener { /** This will be called when the car service has successfully been connected. */ void onConnected(Car car); } }