blob: a6549253dcab3baa1d8f750444cf84244020b067 [file] [log] [blame]
/*
* Copyright (C) 2010 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;
import android.content.Context;
import android.location.Country;
import android.location.CountryListener;
import android.location.ICountryDetector;
import android.location.ICountryListener;
import android.os.Handler;
import android.os.IBinder;
import android.os.RemoteException;
import android.text.TextUtils;
import android.util.PrintWriterPrinter;
import android.util.Printer;
import android.util.Slog;
import com.android.internal.R;
import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.os.BackgroundThread;
import com.android.internal.util.DumpUtils;
import com.android.server.location.countrydetector.ComprehensiveCountryDetector;
import com.android.server.location.countrydetector.CountryDetectorBase;
import java.io.FileDescriptor;
import java.io.PrintWriter;
import java.lang.reflect.InvocationTargetException;
import java.util.HashMap;
/**
* This class detects the country that the user is in. The default country detection is made through
* {@link ComprehensiveCountryDetector}. It is possible to overlay the detection algorithm by
* overlaying the attribute R.string.config_customCountryDetector with the custom class name to use
* instead. The custom class must extend {@link CountryDetectorBase}
*
* @hide
*/
public class CountryDetectorService extends ICountryDetector.Stub {
/**
* The class represents the remote listener, it will also removes itself from listener list when
* the remote process was died.
*/
private final class Receiver implements IBinder.DeathRecipient {
private final ICountryListener mListener;
private final IBinder mKey;
public Receiver(ICountryListener listener) {
mListener = listener;
mKey = listener.asBinder();
}
public void binderDied() {
removeListener(mKey);
}
@Override
public boolean equals(Object otherObj) {
if (otherObj instanceof Receiver) {
return mKey.equals(((Receiver) otherObj).mKey);
}
return false;
}
@Override
public int hashCode() {
return mKey.hashCode();
}
public ICountryListener getListener() {
return mListener;
}
}
private static final String TAG = "CountryDetector";
/**
* Whether to dump the state of the country detector service to bugreports
*/
private static final boolean DEBUG = false;
private final HashMap<IBinder, Receiver> mReceivers;
private final Context mContext;
private CountryDetectorBase mCountryDetector;
private boolean mSystemReady;
private Handler mHandler;
private CountryListener mLocationBasedDetectorListener;
public CountryDetectorService(Context context) {
this(context, BackgroundThread.getHandler());
}
@VisibleForTesting
CountryDetectorService(Context context, Handler handler) {
super();
mReceivers = new HashMap<>();
mContext = context;
mHandler = handler;
}
@Override
public Country detectCountry() {
if (!mSystemReady) {
return null; // server not yet active
} else {
return mCountryDetector.detectCountry();
}
}
/**
* Add the ICountryListener into the listener list.
*/
@Override
public void addCountryListener(ICountryListener listener) throws RemoteException {
if (!mSystemReady) {
throw new RemoteException();
}
addListener(listener);
}
/**
* Remove the ICountryListener from the listener list.
*/
@Override
public void removeCountryListener(ICountryListener listener) throws RemoteException {
if (!mSystemReady) {
throw new RemoteException();
}
removeListener(listener.asBinder());
}
private void addListener(ICountryListener listener) {
synchronized (mReceivers) {
Receiver r = new Receiver(listener);
try {
listener.asBinder().linkToDeath(r, 0);
final Country country = detectCountry();
if (country != null) {
listener.onCountryDetected(country);
}
mReceivers.put(listener.asBinder(), r);
if (mReceivers.size() == 1) {
Slog.d(TAG, "The first listener is added");
setCountryListener(mLocationBasedDetectorListener);
}
} catch (RemoteException e) {
Slog.e(TAG, "linkToDeath failed:", e);
}
}
}
private void removeListener(IBinder key) {
synchronized (mReceivers) {
mReceivers.remove(key);
if (mReceivers.isEmpty()) {
setCountryListener(null);
Slog.d(TAG, "No listener is left");
}
}
}
protected void notifyReceivers(Country country) {
synchronized (mReceivers) {
for (Receiver receiver : mReceivers.values()) {
try {
receiver.getListener().onCountryDetected(country);
} catch (RemoteException e) {
// TODO: Shall we remove the receiver?
Slog.e(TAG, "notifyReceivers failed:", e);
}
}
}
}
void systemRunning() {
// Shall we wait for the initialization finish.
mHandler.post(
() -> {
initialize();
mSystemReady = true;
});
}
@VisibleForTesting
void initialize() {
final String customCountryClass = mContext.getString(R.string.config_customCountryDetector);
if (!TextUtils.isEmpty(customCountryClass)) {
mCountryDetector = loadCustomCountryDetectorIfAvailable(customCountryClass);
}
if (mCountryDetector == null) {
Slog.d(TAG, "Using default country detector");
mCountryDetector = new ComprehensiveCountryDetector(mContext);
}
mLocationBasedDetectorListener = country -> mHandler.post(() -> notifyReceivers(country));
}
protected void setCountryListener(final CountryListener listener) {
mHandler.post(() -> mCountryDetector.setCountryListener(listener));
}
@VisibleForTesting
CountryDetectorBase getCountryDetector() {
return mCountryDetector;
}
@VisibleForTesting
boolean isSystemReady() {
return mSystemReady;
}
private CountryDetectorBase loadCustomCountryDetectorIfAvailable(
final String customCountryClass) {
CountryDetectorBase customCountryDetector = null;
Slog.d(TAG, "Using custom country detector class: " + customCountryClass);
try {
customCountryDetector = Class.forName(customCountryClass).asSubclass(
CountryDetectorBase.class).getConstructor(Context.class).newInstance(
mContext);
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException
| NoSuchMethodException | InvocationTargetException e) {
Slog.e(TAG, "Could not instantiate the custom country detector class");
}
return customCountryDetector;
}
@SuppressWarnings("unused")
@Override
protected void dump(FileDescriptor fd, PrintWriter fout, String[] args) {
if (!DumpUtils.checkDumpPermission(mContext, TAG, fout)) return;
if (!DEBUG) return;
try {
final Printer p = new PrintWriterPrinter(fout);
p.println("CountryDetectorService state:");
p.println("Country detector class=" + mCountryDetector.getClass().getName());
p.println(" Number of listeners=" + mReceivers.keySet().size());
if (mCountryDetector == null) {
p.println(" CountryDetector not initialized");
} else {
p.println(" " + mCountryDetector.toString());
}
} catch (Exception e) {
Slog.e(TAG, "Failed to dump CountryDetectorService: ", e);
}
}
}