Fixed b/8276827 Vendor might want to provide their own implementation of "network location", "fused location" and "geocoder" service. Location manager now allows those service to be replaced by packages that have the same signature as one of the packages in config_locationProviderPackageNames. Such behavior might not be desirable on some devices. This change make this behavior configurable by 3 boolean flags. Details: - Added three boolean flags in core/res/res/values/config.xml to enable or disable NLP/FLP/Geocoder overlay - Added 3 package name for the stock NLP/FLP/Geocoder. They are needed only when overlay is disabled because LocationManagerService need to know which package is preferred when searching for NLP/FLP/Geocoder service. - Made ServiceWatcher able to handle non-overlayable services. - Fixed an NPE isue in ServiceWatcher. mPm.queryIntentServicesAsUser might return null. - Fixed an bug: justCheckThisPackage in bindBestPackageLocked is always ignored. Change-Id: Id221961ac7c3aa8ad44b894f9523f04f770ae237
158 lines
5.8 KiB
Java
158 lines
5.8 KiB
Java
/*
|
|
* Copyright (C) 2013 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.location;
|
|
|
|
import android.content.ComponentName;
|
|
import android.content.Intent;
|
|
import android.content.ServiceConnection;
|
|
import android.hardware.location.GeofenceHardwareService;
|
|
import android.hardware.location.IGeofenceHardware;
|
|
import android.location.IGeofenceProvider;
|
|
import android.location.IGpsGeofenceHardware;
|
|
import android.content.Context;
|
|
import android.os.Handler;
|
|
import android.os.IBinder;
|
|
import android.os.Message;
|
|
import android.os.RemoteException;
|
|
import android.os.UserHandle;
|
|
import android.util.Log;
|
|
import com.android.server.ServiceWatcher;
|
|
|
|
import java.util.List;
|
|
|
|
/**
|
|
* @hide
|
|
*/
|
|
public final class GeofenceProxy {
|
|
private static final String TAG = "GeofenceProxy";
|
|
private static final String SERVICE_ACTION =
|
|
"com.android.location.service.GeofenceProvider";
|
|
private ServiceWatcher mServiceWatcher;
|
|
private Context mContext;
|
|
private IGeofenceHardware mGeofenceHardware;
|
|
private IGpsGeofenceHardware mGpsGeofenceHardware;
|
|
|
|
private static final int GEOFENCE_PROVIDER_CONNECTED = 1;
|
|
private static final int GEOFENCE_HARDWARE_CONNECTED = 2;
|
|
private static final int GEOFENCE_HARDWARE_DISCONNECTED = 3;
|
|
private static final int GEOFENCE_GPS_HARDWARE_CONNECTED = 4;
|
|
private static final int GEOFENCE_GPS_HARDWARE_DISCONNECTED = 5;
|
|
|
|
private Runnable mRunnable = new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
mHandler.sendEmptyMessage(GEOFENCE_PROVIDER_CONNECTED);
|
|
}
|
|
};
|
|
|
|
public static GeofenceProxy createAndBind(Context context,
|
|
int overlaySwitchResId, int defaultServicePackageNameResId,
|
|
int initialPackageNamesResId, Handler handler, IGpsGeofenceHardware gpsGeofence) {
|
|
GeofenceProxy proxy = new GeofenceProxy(context, overlaySwitchResId,
|
|
defaultServicePackageNameResId, initialPackageNamesResId, handler, gpsGeofence);
|
|
if (proxy.bindGeofenceProvider()) {
|
|
return proxy;
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private GeofenceProxy(Context context,
|
|
int overlaySwitchResId, int defaultServicePackageNameResId,
|
|
int initialPackageNamesResId, Handler handler, IGpsGeofenceHardware gpsGeofence) {
|
|
mContext = context;
|
|
mServiceWatcher = new ServiceWatcher(context, TAG, SERVICE_ACTION, overlaySwitchResId,
|
|
defaultServicePackageNameResId, initialPackageNamesResId, mRunnable, handler);
|
|
mGpsGeofenceHardware = gpsGeofence;
|
|
bindHardwareGeofence();
|
|
}
|
|
|
|
private boolean bindGeofenceProvider() {
|
|
return mServiceWatcher.start();
|
|
}
|
|
|
|
private IGeofenceProvider getGeofenceProviderService() {
|
|
return IGeofenceProvider.Stub.asInterface(mServiceWatcher.getBinder());
|
|
}
|
|
|
|
private void bindHardwareGeofence() {
|
|
mContext.bindServiceAsUser(new Intent(mContext, GeofenceHardwareService.class),
|
|
mServiceConnection, Context.BIND_AUTO_CREATE, UserHandle.OWNER);
|
|
}
|
|
|
|
private ServiceConnection mServiceConnection = new ServiceConnection() {
|
|
@Override
|
|
public void onServiceConnected(ComponentName name, IBinder service) {
|
|
mGeofenceHardware = IGeofenceHardware.Stub.asInterface(service);
|
|
mHandler.sendEmptyMessage(GEOFENCE_HARDWARE_CONNECTED);
|
|
}
|
|
|
|
@Override
|
|
public void onServiceDisconnected(ComponentName name) {
|
|
mGeofenceHardware = null;
|
|
mHandler.sendEmptyMessage(GEOFENCE_HARDWARE_DISCONNECTED);
|
|
}
|
|
};
|
|
|
|
private void setGeofenceHardwareInProvider() {
|
|
try {
|
|
getGeofenceProviderService().setGeofenceHardware(mGeofenceHardware);
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Remote Exception: setGeofenceHardwareInProvider: " + e);
|
|
}
|
|
}
|
|
|
|
private void setGpsGeofence() {
|
|
try {
|
|
mGeofenceHardware.setGpsGeofenceHardware(mGpsGeofenceHardware);
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "Error while connecting to GeofenceHardwareService");
|
|
}
|
|
}
|
|
|
|
|
|
// This needs to be reworked, when more services get added,
|
|
// Might need a state machine or add a framework utility class,
|
|
private Handler mHandler = new Handler() {
|
|
private boolean mGeofenceHardwareConnected = false;
|
|
private boolean mGeofenceProviderConnected = false;
|
|
|
|
|
|
@Override
|
|
public void handleMessage(Message msg) {
|
|
switch (msg.what) {
|
|
case GEOFENCE_PROVIDER_CONNECTED:
|
|
mGeofenceProviderConnected = true;
|
|
if (mGeofenceHardwareConnected) {
|
|
setGeofenceHardwareInProvider();
|
|
}
|
|
break;
|
|
case GEOFENCE_HARDWARE_CONNECTED:
|
|
setGpsGeofence();
|
|
mGeofenceHardwareConnected = true;
|
|
if (mGeofenceProviderConnected) {
|
|
setGeofenceHardwareInProvider();
|
|
}
|
|
break;
|
|
case GEOFENCE_HARDWARE_DISCONNECTED:
|
|
mGeofenceHardwareConnected = false;
|
|
setGeofenceHardwareInProvider();
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
}
|