| /* |
| * Copyright (C) 2018 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.car; |
| |
| import android.app.ActivityManager; |
| import android.car.ILocationManagerProxy; |
| import android.car.IPerUserCarService; |
| import android.car.drivingstate.CarDrivingStateEvent; |
| import android.car.drivingstate.ICarDrivingStateChangeListener; |
| import android.car.hardware.power.CarPowerManager; |
| import android.car.hardware.power.CarPowerManager.CarPowerStateListener; |
| import android.car.hardware.power.CarPowerManager.CarPowerStateListenerWithCompletion; |
| import android.content.BroadcastReceiver; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.location.Location; |
| import android.location.LocationManager; |
| import android.os.Handler; |
| import android.os.HandlerThread; |
| import android.os.RemoteException; |
| import android.os.SystemClock; |
| import android.os.UserHandle; |
| import android.os.UserManager; |
| import android.util.AtomicFile; |
| import android.util.IndentingPrintWriter; |
| import android.util.JsonReader; |
| import android.util.JsonWriter; |
| import android.util.Slog; |
| |
| import com.android.car.systeminterface.SystemInterface; |
| import com.android.internal.annotations.VisibleForTesting; |
| |
| import java.io.File; |
| import java.io.FileInputStream; |
| import java.io.FileNotFoundException; |
| import java.io.FileOutputStream; |
| import java.io.IOException; |
| import java.io.InputStreamReader; |
| import java.io.OutputStreamWriter; |
| import java.util.concurrent.CompletableFuture; |
| |
| /** |
| * This service stores the last known location from {@link LocationManager} when a car is parked |
| * and restores the location when the car is powered on. |
| */ |
| public class CarLocationService extends BroadcastReceiver implements CarServiceBase, |
| CarPowerStateListenerWithCompletion { |
| private static final String TAG = CarLog.tagFor(CarLocationService.class); |
| private static final String FILENAME = "location_cache.json"; |
| // The accuracy for the stored timestamp |
| private static final long GRANULARITY_ONE_DAY_MS = 24 * 60 * 60 * 1000L; |
| // The time-to-live for the cached location |
| private static final long TTL_THIRTY_DAYS_MS = 30 * GRANULARITY_ONE_DAY_MS; |
| // The maximum number of times to try injecting a location |
| private static final int MAX_LOCATION_INJECTION_ATTEMPTS = 10; |
| |
| // Constants for location serialization. |
| private static final String PROVIDER = "provider"; |
| private static final String LATITUDE = "latitude"; |
| private static final String LONGITUDE = "longitude"; |
| private static final String ALTITUDE = "altitude"; |
| private static final String SPEED = "speed"; |
| private static final String BEARING = "bearing"; |
| private static final String ACCURACY = "accuracy"; |
| private static final String VERTICAL_ACCURACY = "verticalAccuracy"; |
| private static final String SPEED_ACCURACY = "speedAccuracy"; |
| private static final String BEARING_ACCURACY = "bearingAccuracy"; |
| private static final String IS_FROM_MOCK_PROVIDER = "isFromMockProvider"; |
| private static final String CAPTURE_TIME = "captureTime"; |
| |
| // Used internally for mHandlerThread synchronization |
| private final Object mLock = new Object(); |
| |
| // Used internally for mILocationManagerProxy synchronization |
| private final Object mLocationManagerProxyLock = new Object(); |
| |
| private final Context mContext; |
| private final HandlerThread mHandlerThread = CarServiceUtils.getHandlerThread( |
| getClass().getSimpleName()); |
| private final Handler mHandler = new Handler(mHandlerThread.getLooper()); |
| |
| private CarPowerManager mCarPowerManager; |
| private CarDrivingStateService mCarDrivingStateService; |
| private PerUserCarServiceHelper mPerUserCarServiceHelper; |
| |
| // Allows us to interact with the {@link LocationManager} as the foreground user. |
| private ILocationManagerProxy mILocationManagerProxy; |
| |
| // Maintains mILocationManagerProxy for the current foreground user. |
| private final PerUserCarServiceHelper.ServiceCallback mUserServiceCallback = |
| new PerUserCarServiceHelper.ServiceCallback() { |
| @Override |
| public void onServiceConnected(IPerUserCarService perUserCarService) { |
| logd("Connected to PerUserCarService"); |
| if (perUserCarService == null) { |
| logd("IPerUserCarService is null. Cannot get location manager proxy"); |
| return; |
| } |
| synchronized (mLocationManagerProxyLock) { |
| try { |
| mILocationManagerProxy = perUserCarService.getLocationManagerProxy(); |
| } catch (RemoteException e) { |
| Slog.e(TAG, "RemoteException from IPerUserCarService", e); |
| return; |
| } |
| } |
| int currentUser = ActivityManager.getCurrentUser(); |
| logd("Current user: %s", currentUser); |
| if (UserManager.isHeadlessSystemUserMode() |
| && currentUser > UserHandle.USER_SYSTEM) { |
| asyncOperation(() -> loadLocation()); |
| } |
| } |
| |
| @Override |
| public void onPreUnbind() { |
| logd("Before Unbinding from PerUserCarService"); |
| synchronized (mLocationManagerProxyLock) { |
| mILocationManagerProxy = null; |
| } |
| } |
| |
| @Override |
| public void onServiceDisconnected() { |
| logd("Disconnected from PerUserCarService"); |
| synchronized (mLocationManagerProxyLock) { |
| mILocationManagerProxy = null; |
| } |
| } |
| }; |
| |
| private final ICarDrivingStateChangeListener mICarDrivingStateChangeEventListener = |
| new ICarDrivingStateChangeListener.Stub() { |
| @Override |
| public void onDrivingStateChanged(CarDrivingStateEvent event) { |
| logd("onDrivingStateChanged: %s", event); |
| if (event != null |
| && event.eventValue == CarDrivingStateEvent.DRIVING_STATE_MOVING) { |
| deleteCacheFile(); |
| if (mCarDrivingStateService != null) { |
| mCarDrivingStateService.unregisterDrivingStateChangeListener( |
| mICarDrivingStateChangeEventListener); |
| } |
| } |
| } |
| }; |
| |
| public CarLocationService(Context context) { |
| logd("constructed"); |
| mContext = context; |
| } |
| |
| @Override |
| public void init() { |
| logd("init"); |
| IntentFilter filter = new IntentFilter(); |
| filter.addAction(LocationManager.MODE_CHANGED_ACTION); |
| mContext.registerReceiver(this, filter); |
| mCarDrivingStateService = CarLocalServices.getService(CarDrivingStateService.class); |
| if (mCarDrivingStateService != null) { |
| CarDrivingStateEvent event = mCarDrivingStateService.getCurrentDrivingState(); |
| if (event != null && event.eventValue == CarDrivingStateEvent.DRIVING_STATE_MOVING) { |
| deleteCacheFile(); |
| } else { |
| mCarDrivingStateService.registerDrivingStateChangeListener( |
| mICarDrivingStateChangeEventListener); |
| } |
| } |
| mCarPowerManager = CarLocalServices.createCarPowerManager(mContext); |
| if (mCarPowerManager != null) { // null case happens for testing. |
| mCarPowerManager.setListenerWithCompletion(CarLocationService.this); |
| } |
| mPerUserCarServiceHelper = CarLocalServices.getService(PerUserCarServiceHelper.class); |
| if (mPerUserCarServiceHelper != null) { |
| mPerUserCarServiceHelper.registerServiceCallback(mUserServiceCallback); |
| } |
| } |
| |
| @Override |
| public void release() { |
| logd("release"); |
| if (mCarPowerManager != null) { |
| mCarPowerManager.clearListener(); |
| } |
| if (mCarDrivingStateService != null) { |
| mCarDrivingStateService.unregisterDrivingStateChangeListener( |
| mICarDrivingStateChangeEventListener); |
| } |
| if (mPerUserCarServiceHelper != null) { |
| mPerUserCarServiceHelper.unregisterServiceCallback(mUserServiceCallback); |
| } |
| mContext.unregisterReceiver(this); |
| } |
| |
| @Override |
| public void dump(IndentingPrintWriter writer) { |
| writer.println(TAG); |
| mPerUserCarServiceHelper.dump(writer); |
| writer.printf("Context: %s\n", mContext); |
| writer.printf("MAX_LOCATION_INJECTION_ATTEMPTS: %d\n", MAX_LOCATION_INJECTION_ATTEMPTS); |
| } |
| |
| @Override |
| public void onStateChanged(int state, CompletableFuture<Void> future) { |
| logd("onStateChanged: %s", state); |
| switch (state) { |
| case CarPowerStateListener.SHUTDOWN_PREPARE: |
| asyncOperation(() -> { |
| storeLocation(); |
| // Notify the CarPowerManager that it may proceed to shutdown or suspend. |
| if (future != null) { |
| future.complete(null); |
| } |
| }); |
| break; |
| case CarPowerStateListener.SUSPEND_EXIT: |
| if (mCarDrivingStateService != null) { |
| CarDrivingStateEvent event = mCarDrivingStateService.getCurrentDrivingState(); |
| if (event != null |
| && event.eventValue == CarDrivingStateEvent.DRIVING_STATE_MOVING) { |
| deleteCacheFile(); |
| } else { |
| logd("Registering to receive driving state."); |
| mCarDrivingStateService.registerDrivingStateChangeListener( |
| mICarDrivingStateChangeEventListener); |
| } |
| } |
| if (future != null) { |
| future.complete(null); |
| } |
| default: |
| // This service does not need to do any work for these events but should still |
| // notify the CarPowerManager that it may proceed. |
| if (future != null) { |
| future.complete(null); |
| } |
| break; |
| } |
| } |
| |
| @Override |
| public void onReceive(Context context, Intent intent) { |
| logd("onReceive %s", intent); |
| // If the system user is headless but the current user is still the system user, then we |
| // should not delete the location cache file due to missing location permissions. |
| if (isCurrentUserHeadlessSystemUser()) { |
| logd("Current user is headless system user."); |
| return; |
| } |
| synchronized (mLocationManagerProxyLock) { |
| if (mILocationManagerProxy == null) { |
| logd("Null location manager."); |
| return; |
| } |
| String action = intent.getAction(); |
| try { |
| if (action == LocationManager.MODE_CHANGED_ACTION) { |
| boolean locationEnabled = mILocationManagerProxy.isLocationEnabled(); |
| logd("isLocationEnabled(): %s", locationEnabled); |
| if (!locationEnabled) { |
| deleteCacheFile(); |
| } |
| } else { |
| logd("Unexpected intent."); |
| } |
| } catch (RemoteException e) { |
| Slog.e(TAG, "RemoteException from ILocationManagerProxy", e); |
| } |
| } |
| } |
| |
| /** Tells whether the current foreground user is the headless system user. */ |
| private boolean isCurrentUserHeadlessSystemUser() { |
| int currentUserId = ActivityManager.getCurrentUser(); |
| return UserManager.isHeadlessSystemUserMode() |
| && currentUserId == UserHandle.USER_SYSTEM; |
| } |
| |
| /** |
| * Gets the last known location from the location manager proxy and store it in a file. |
| */ |
| private void storeLocation() { |
| Location location = null; |
| synchronized (mLocationManagerProxyLock) { |
| if (mILocationManagerProxy == null) { |
| logd("Null location manager proxy."); |
| return; |
| } |
| try { |
| location = mILocationManagerProxy.getLastKnownLocation( |
| LocationManager.GPS_PROVIDER); |
| } catch (RemoteException e) { |
| Slog.e(TAG, "RemoteException from ILocationManagerProxy", e); |
| } |
| } |
| if (location == null) { |
| logd("Not storing null location"); |
| } else { |
| logd("Storing location"); |
| AtomicFile atomicFile = new AtomicFile(getLocationCacheFile()); |
| FileOutputStream fos = null; |
| try { |
| fos = atomicFile.startWrite(); |
| try (JsonWriter jsonWriter = new JsonWriter(new OutputStreamWriter(fos, "UTF-8"))) { |
| jsonWriter.beginObject(); |
| jsonWriter.name(PROVIDER).value(location.getProvider()); |
| jsonWriter.name(LATITUDE).value(location.getLatitude()); |
| jsonWriter.name(LONGITUDE).value(location.getLongitude()); |
| if (location.hasAltitude()) { |
| jsonWriter.name(ALTITUDE).value(location.getAltitude()); |
| } |
| if (location.hasSpeed()) { |
| jsonWriter.name(SPEED).value(location.getSpeed()); |
| } |
| if (location.hasBearing()) { |
| jsonWriter.name(BEARING).value(location.getBearing()); |
| } |
| if (location.hasAccuracy()) { |
| jsonWriter.name(ACCURACY).value(location.getAccuracy()); |
| } |
| if (location.hasVerticalAccuracy()) { |
| jsonWriter.name(VERTICAL_ACCURACY).value( |
| location.getVerticalAccuracyMeters()); |
| } |
| if (location.hasSpeedAccuracy()) { |
| jsonWriter.name(SPEED_ACCURACY).value( |
| location.getSpeedAccuracyMetersPerSecond()); |
| } |
| if (location.hasBearingAccuracy()) { |
| jsonWriter.name(BEARING_ACCURACY).value( |
| location.getBearingAccuracyDegrees()); |
| } |
| if (location.isFromMockProvider()) { |
| jsonWriter.name(IS_FROM_MOCK_PROVIDER).value(true); |
| } |
| long currentTime = location.getTime(); |
| // Round the time down to only be accurate within one day. |
| jsonWriter.name(CAPTURE_TIME).value( |
| currentTime - currentTime % GRANULARITY_ONE_DAY_MS); |
| jsonWriter.endObject(); |
| } |
| atomicFile.finishWrite(fos); |
| } catch (IOException e) { |
| Slog.e(TAG, "Unable to write to disk", e); |
| atomicFile.failWrite(fos); |
| } |
| } |
| } |
| |
| /** |
| * Reads a previously stored location and attempts to inject it into the location manager proxy. |
| */ |
| private void loadLocation() { |
| Location location = readLocationFromCacheFile(); |
| logd("Read location from timestamp %s", location.getTime()); |
| long currentTime = System.currentTimeMillis(); |
| if (location.getTime() + TTL_THIRTY_DAYS_MS < currentTime) { |
| logd("Location expired."); |
| deleteCacheFile(); |
| } else { |
| location.setTime(currentTime); |
| long elapsedTime = SystemClock.elapsedRealtimeNanos(); |
| location.setElapsedRealtimeNanos(elapsedTime); |
| if (location.isComplete()) { |
| injectLocation(location, 1); |
| } |
| } |
| } |
| |
| private Location readLocationFromCacheFile() { |
| Location location = new Location((String) null); |
| File file = getLocationCacheFile(); |
| AtomicFile atomicFile = new AtomicFile(file); |
| try (FileInputStream fis = atomicFile.openRead()) { |
| JsonReader reader = new JsonReader(new InputStreamReader(fis, "UTF-8")); |
| reader.beginObject(); |
| while (reader.hasNext()) { |
| String name = reader.nextName(); |
| switch (name) { |
| case PROVIDER: |
| location.setProvider(reader.nextString()); |
| break; |
| case LATITUDE: |
| location.setLatitude(reader.nextDouble()); |
| break; |
| case LONGITUDE: |
| location.setLongitude(reader.nextDouble()); |
| break; |
| case ALTITUDE: |
| location.setAltitude(reader.nextDouble()); |
| break; |
| case SPEED: |
| location.setSpeed((float) reader.nextDouble()); |
| break; |
| case BEARING: |
| location.setBearing((float) reader.nextDouble()); |
| break; |
| case ACCURACY: |
| location.setAccuracy((float) reader.nextDouble()); |
| break; |
| case VERTICAL_ACCURACY: |
| location.setVerticalAccuracyMeters((float) reader.nextDouble()); |
| break; |
| case SPEED_ACCURACY: |
| location.setSpeedAccuracyMetersPerSecond((float) reader.nextDouble()); |
| break; |
| case BEARING_ACCURACY: |
| location.setBearingAccuracyDegrees((float) reader.nextDouble()); |
| break; |
| case IS_FROM_MOCK_PROVIDER: |
| location.setIsFromMockProvider(reader.nextBoolean()); |
| break; |
| case CAPTURE_TIME: |
| location.setTime(reader.nextLong()); |
| break; |
| default: |
| Slog.w(TAG, "Unrecognized key: " + name); |
| reader.skipValue(); |
| } |
| } |
| reader.endObject(); |
| } catch (FileNotFoundException e) { |
| logd("Location cache file not found: %s", file); |
| } catch (IOException e) { |
| Slog.e(TAG, "Unable to read from disk", e); |
| } catch (NumberFormatException | IllegalStateException e) { |
| Slog.e(TAG, "Unexpected format", e); |
| } |
| return location; |
| } |
| |
| private void deleteCacheFile() { |
| File file = getLocationCacheFile(); |
| boolean deleted = file.delete(); |
| if (deleted) { |
| logd("Successfully deleted cache file at %s", file); |
| } else { |
| logd("Failed to delete cache file at %s", file); |
| } |
| } |
| |
| /** |
| * Attempts to inject the location multiple times in case the LocationManager was not fully |
| * initialized or has not updated its handle to the current user yet. |
| */ |
| private void injectLocation(Location location, int attemptCount) { |
| boolean success = false; |
| synchronized (mLocationManagerProxyLock) { |
| if (mILocationManagerProxy == null) { |
| logd("Null location manager proxy."); |
| } else { |
| try { |
| success = mILocationManagerProxy.injectLocation(location); |
| } catch (RemoteException e) { |
| Slog.e(TAG, "RemoteException from ILocationManagerProxy", e); |
| } |
| } |
| } |
| if (success) { |
| logd("Successfully injected stored location on attempt %s.", attemptCount); |
| return; |
| } else if (attemptCount <= MAX_LOCATION_INJECTION_ATTEMPTS) { |
| logd("Failed to inject stored location on attempt %s.", attemptCount); |
| asyncOperation(() -> { |
| injectLocation(location, attemptCount + 1); |
| }, 200 * attemptCount); |
| } else { |
| logd("No location injected."); |
| } |
| } |
| |
| private File getLocationCacheFile() { |
| SystemInterface systemInterface = CarLocalServices.getService(SystemInterface.class); |
| return new File(systemInterface.getSystemCarDir(), FILENAME); |
| } |
| |
| @VisibleForTesting |
| void asyncOperation(Runnable operation) { |
| asyncOperation(operation, 0); |
| } |
| |
| private void asyncOperation(Runnable operation, long delayMillis) { |
| mHandler.postDelayed(() -> operation.run(), delayMillis); |
| } |
| |
| private static void logd(String msg, Object... vals) { |
| // Disable logs here if they become too spammy. |
| Slog.d(TAG, String.format(msg, vals)); |
| } |
| } |