am 892f371c: Merge "Encapsulate locks in UEventObservers." into jb-mr1-dev
* commit '892f371c6b8c33268dceaeb89ac09c73ad72ad79': Encapsulate locks in UEventObservers.
This commit is contained in:
@@ -37,14 +37,79 @@ import java.util.HashMap;
|
|||||||
* @hide
|
* @hide
|
||||||
*/
|
*/
|
||||||
public abstract class UEventObserver {
|
public abstract class UEventObserver {
|
||||||
private static final String TAG = UEventObserver.class.getSimpleName();
|
private static UEventThread sThread;
|
||||||
|
|
||||||
|
private static native void native_setup();
|
||||||
|
private static native int next_event(byte[] buffer);
|
||||||
|
|
||||||
|
public UEventObserver() {
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void finalize() throws Throwable {
|
||||||
|
try {
|
||||||
|
stopObserving();
|
||||||
|
} finally {
|
||||||
|
super.finalize();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static UEventThread getThread() {
|
||||||
|
synchronized (UEventObserver.class) {
|
||||||
|
if (sThread == null) {
|
||||||
|
sThread = new UEventThread();
|
||||||
|
sThread.start();
|
||||||
|
}
|
||||||
|
return sThread;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static UEventThread peekThread() {
|
||||||
|
synchronized (UEventObserver.class) {
|
||||||
|
return sThread;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Begin observation of UEvent's.<p>
|
||||||
|
* This method will cause the UEvent thread to start if this is the first
|
||||||
|
* invocation of startObserving in this process.<p>
|
||||||
|
* Once called, the UEvent thread will call onUEvent() when an incoming
|
||||||
|
* UEvent matches the specified string.<p>
|
||||||
|
* This method can be called multiple times to register multiple matches.
|
||||||
|
* Only one call to stopObserving is required even with multiple registered
|
||||||
|
* matches.
|
||||||
|
* @param match A substring of the UEvent to match. Use "" to match all
|
||||||
|
* UEvent's
|
||||||
|
*/
|
||||||
|
public final void startObserving(String match) {
|
||||||
|
final UEventThread t = getThread();
|
||||||
|
t.addObserver(match, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* End observation of UEvent's.<p>
|
||||||
|
* This process's UEvent thread will never call onUEvent() on this
|
||||||
|
* UEventObserver after this call. Repeated calls have no effect.
|
||||||
|
*/
|
||||||
|
public final void stopObserving() {
|
||||||
|
final UEventThread t = getThread();
|
||||||
|
if (t != null) {
|
||||||
|
t.removeObserver(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subclasses of UEventObserver should override this method to handle
|
||||||
|
* UEvents.
|
||||||
|
*/
|
||||||
|
public abstract void onUEvent(UEvent event);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Representation of a UEvent.
|
* Representation of a UEvent.
|
||||||
*/
|
*/
|
||||||
static public class UEvent {
|
public static final class UEvent {
|
||||||
// collection of key=value pairs parsed from the uevent message
|
// collection of key=value pairs parsed from the uevent message
|
||||||
public HashMap<String,String> mMap = new HashMap<String,String>();
|
private final HashMap<String,String> mMap = new HashMap<String,String>();
|
||||||
|
|
||||||
public UEvent(String message) {
|
public UEvent(String message) {
|
||||||
int offset = 0;
|
int offset = 0;
|
||||||
@@ -79,17 +144,17 @@ public abstract class UEventObserver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static UEventThread sThread;
|
private static final class UEventThread extends Thread {
|
||||||
private static boolean sThreadStarted = false;
|
|
||||||
|
|
||||||
private static class UEventThread extends Thread {
|
|
||||||
/** Many to many mapping of string match to observer.
|
/** Many to many mapping of string match to observer.
|
||||||
* Multimap would be better, but not available in android, so use
|
* Multimap would be better, but not available in android, so use
|
||||||
* an ArrayList where even elements are the String match and odd
|
* an ArrayList where even elements are the String match and odd
|
||||||
* elements the corresponding UEventObserver observer */
|
* elements the corresponding UEventObserver observer */
|
||||||
private ArrayList<Object> mObservers = new ArrayList<Object>();
|
private final ArrayList<Object> mKeysAndObservers = new ArrayList<Object>();
|
||||||
|
|
||||||
UEventThread() {
|
private final ArrayList<UEventObserver> mTempObserversToSignal =
|
||||||
|
new ArrayList<UEventObserver>();
|
||||||
|
|
||||||
|
public UEventThread() {
|
||||||
super("UEventObserver");
|
super("UEventObserver");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -101,91 +166,54 @@ public abstract class UEventObserver {
|
|||||||
while (true) {
|
while (true) {
|
||||||
len = next_event(buffer);
|
len = next_event(buffer);
|
||||||
if (len > 0) {
|
if (len > 0) {
|
||||||
String bufferStr = new String(buffer, 0, len); // easier to search a String
|
sendEvent(new String(buffer, 0, len));
|
||||||
synchronized (mObservers) {
|
|
||||||
for (int i = 0; i < mObservers.size(); i += 2) {
|
|
||||||
if (bufferStr.indexOf((String)mObservers.get(i)) != -1) {
|
|
||||||
((UEventObserver)mObservers.get(i+1))
|
|
||||||
.onUEvent(new UEvent(bufferStr));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
public void addObserver(String match, UEventObserver observer) {
|
|
||||||
synchronized(mObservers) {
|
private void sendEvent(String message) {
|
||||||
mObservers.add(match);
|
synchronized (mKeysAndObservers) {
|
||||||
mObservers.add(observer);
|
final int N = mKeysAndObservers.size();
|
||||||
|
for (int i = 0; i < N; i += 2) {
|
||||||
|
final String key = (String)mKeysAndObservers.get(i);
|
||||||
|
if (message.indexOf(key) != -1) {
|
||||||
|
final UEventObserver observer =
|
||||||
|
(UEventObserver)mKeysAndObservers.get(i + 1);
|
||||||
|
mTempObserversToSignal.add(observer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!mTempObserversToSignal.isEmpty()) {
|
||||||
|
final UEvent event = new UEvent(message);
|
||||||
|
final int N = mTempObserversToSignal.size();
|
||||||
|
for (int i = 0; i < N; i++) {
|
||||||
|
final UEventObserver observer = mTempObserversToSignal.get(i);
|
||||||
|
observer.onUEvent(event);
|
||||||
|
}
|
||||||
|
mTempObserversToSignal.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void addObserver(String match, UEventObserver observer) {
|
||||||
|
synchronized (mKeysAndObservers) {
|
||||||
|
mKeysAndObservers.add(match);
|
||||||
|
mKeysAndObservers.add(observer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** Removes every key/value pair where value=observer from mObservers */
|
/** Removes every key/value pair where value=observer from mObservers */
|
||||||
public void removeObserver(UEventObserver observer) {
|
public void removeObserver(UEventObserver observer) {
|
||||||
synchronized(mObservers) {
|
synchronized (mKeysAndObservers) {
|
||||||
boolean found = true;
|
for (int i = 0; i < mKeysAndObservers.size(); ) {
|
||||||
while (found) {
|
if (mKeysAndObservers.get(i + 1) == observer) {
|
||||||
found = false;
|
mKeysAndObservers.remove(i + 1);
|
||||||
for (int i = 0; i < mObservers.size(); i += 2) {
|
mKeysAndObservers.remove(i);
|
||||||
if (mObservers.get(i+1) == observer) {
|
} else {
|
||||||
mObservers.remove(i+1);
|
i += 2;
|
||||||
mObservers.remove(i);
|
|
||||||
found = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static native void native_setup();
|
|
||||||
private static native int next_event(byte[] buffer);
|
|
||||||
|
|
||||||
private static final synchronized void ensureThreadStarted() {
|
|
||||||
if (sThreadStarted == false) {
|
|
||||||
sThread = new UEventThread();
|
|
||||||
sThread.start();
|
|
||||||
sThreadStarted = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Begin observation of UEvent's.<p>
|
|
||||||
* This method will cause the UEvent thread to start if this is the first
|
|
||||||
* invocation of startObserving in this process.<p>
|
|
||||||
* Once called, the UEvent thread will call onUEvent() when an incoming
|
|
||||||
* UEvent matches the specified string.<p>
|
|
||||||
* This method can be called multiple times to register multiple matches.
|
|
||||||
* Only one call to stopObserving is required even with multiple registered
|
|
||||||
* matches.
|
|
||||||
* @param match A substring of the UEvent to match. Use "" to match all
|
|
||||||
* UEvent's
|
|
||||||
*/
|
|
||||||
public final synchronized void startObserving(String match) {
|
|
||||||
ensureThreadStarted();
|
|
||||||
sThread.addObserver(match, this);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* End observation of UEvent's.<p>
|
|
||||||
* This process's UEvent thread will never call onUEvent() on this
|
|
||||||
* UEventObserver after this call. Repeated calls have no effect.
|
|
||||||
*/
|
|
||||||
public final synchronized void stopObserving() {
|
|
||||||
sThread.removeObserver(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Subclasses of UEventObserver should override this method to handle
|
|
||||||
* UEvents.
|
|
||||||
*/
|
|
||||||
public abstract void onUEvent(UEvent event);
|
|
||||||
|
|
||||||
protected void finalize() throws Throwable {
|
|
||||||
try {
|
|
||||||
stopObserving();
|
|
||||||
} finally {
|
|
||||||
super.finalize();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,10 +18,6 @@ package com.android.server;
|
|||||||
|
|
||||||
import static android.provider.Settings.Secure.SCREENSAVER_ACTIVATE_ON_DOCK;
|
import static android.provider.Settings.Secure.SCREENSAVER_ACTIVATE_ON_DOCK;
|
||||||
|
|
||||||
import com.android.server.power.PowerManagerService;
|
|
||||||
|
|
||||||
import android.bluetooth.BluetoothAdapter;
|
|
||||||
import android.bluetooth.BluetoothDevice;
|
|
||||||
import android.content.ContentResolver;
|
import android.content.ContentResolver;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
@@ -30,6 +26,7 @@ import android.media.Ringtone;
|
|||||||
import android.media.RingtoneManager;
|
import android.media.RingtoneManager;
|
||||||
import android.net.Uri;
|
import android.net.Uri;
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
|
import android.os.Looper;
|
||||||
import android.os.Message;
|
import android.os.Message;
|
||||||
import android.os.RemoteException;
|
import android.os.RemoteException;
|
||||||
import android.os.ServiceManager;
|
import android.os.ServiceManager;
|
||||||
@@ -47,7 +44,7 @@ import java.io.FileReader;
|
|||||||
/**
|
/**
|
||||||
* <p>DockObserver monitors for a docking station.
|
* <p>DockObserver monitors for a docking station.
|
||||||
*/
|
*/
|
||||||
class DockObserver extends UEventObserver {
|
final class DockObserver extends UEventObserver {
|
||||||
private static final String TAG = DockObserver.class.getSimpleName();
|
private static final String TAG = DockObserver.class.getSimpleName();
|
||||||
private static final boolean LOG = false;
|
private static final boolean LOG = false;
|
||||||
|
|
||||||
@@ -56,7 +53,9 @@ class DockObserver extends UEventObserver {
|
|||||||
|
|
||||||
private static final int DEFAULT_DOCK = 1;
|
private static final int DEFAULT_DOCK = 1;
|
||||||
|
|
||||||
private static final int MSG_DOCK_STATE = 0;
|
private static final int MSG_DOCK_STATE_CHANGED = 0;
|
||||||
|
|
||||||
|
private final Object mLock = new Object();
|
||||||
|
|
||||||
private int mDockState = Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
private int mDockState = Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
||||||
private int mPreviousDockState = Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
private int mPreviousDockState = Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
||||||
@@ -78,7 +77,7 @@ class DockObserver extends UEventObserver {
|
|||||||
Slog.v(TAG, "Dock UEVENT: " + event.toString());
|
Slog.v(TAG, "Dock UEVENT: " + event.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
synchronized (this) {
|
synchronized (mLock) {
|
||||||
try {
|
try {
|
||||||
int newState = Integer.parseInt(event.get("SWITCH_STATE"));
|
int newState = Integer.parseInt(event.get("SWITCH_STATE"));
|
||||||
if (newState != mDockState) {
|
if (newState != mDockState) {
|
||||||
@@ -96,7 +95,7 @@ class DockObserver extends UEventObserver {
|
|||||||
(PowerManager)mContext.getSystemService(Context.POWER_SERVICE);
|
(PowerManager)mContext.getSystemService(Context.POWER_SERVICE);
|
||||||
pm.wakeUp(SystemClock.uptimeMillis());
|
pm.wakeUp(SystemClock.uptimeMillis());
|
||||||
}
|
}
|
||||||
update();
|
updateLocked();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (NumberFormatException e) {
|
} catch (NumberFormatException e) {
|
||||||
@@ -105,132 +104,142 @@ class DockObserver extends UEventObserver {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private final void init() {
|
private void init() {
|
||||||
char[] buffer = new char[1024];
|
synchronized (mLock) {
|
||||||
|
try {
|
||||||
try {
|
char[] buffer = new char[1024];
|
||||||
FileReader file = new FileReader(DOCK_STATE_PATH);
|
FileReader file = new FileReader(DOCK_STATE_PATH);
|
||||||
int len = file.read(buffer, 0, 1024);
|
try {
|
||||||
file.close();
|
int len = file.read(buffer, 0, 1024);
|
||||||
mPreviousDockState = mDockState = Integer.valueOf((new String(buffer, 0, len)).trim());
|
mDockState = Integer.valueOf((new String(buffer, 0, len)).trim());
|
||||||
} catch (FileNotFoundException e) {
|
mPreviousDockState = mDockState;
|
||||||
Slog.w(TAG, "This kernel does not have dock station support");
|
} finally {
|
||||||
} catch (Exception e) {
|
file.close();
|
||||||
Slog.e(TAG, "" , e);
|
}
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
Slog.w(TAG, "This kernel does not have dock station support");
|
||||||
|
} catch (Exception e) {
|
||||||
|
Slog.e(TAG, "" , e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void systemReady() {
|
void systemReady() {
|
||||||
synchronized (this) {
|
synchronized (mLock) {
|
||||||
// don't bother broadcasting undocked here
|
// don't bother broadcasting undocked here
|
||||||
if (mDockState != Intent.EXTRA_DOCK_STATE_UNDOCKED) {
|
if (mDockState != Intent.EXTRA_DOCK_STATE_UNDOCKED) {
|
||||||
update();
|
updateLocked();
|
||||||
}
|
}
|
||||||
mSystemReady = true;
|
mSystemReady = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private final void update() {
|
private void updateLocked() {
|
||||||
mHandler.sendEmptyMessage(MSG_DOCK_STATE);
|
mHandler.sendEmptyMessage(MSG_DOCK_STATE_CHANGED);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void handleDockStateChange() {
|
||||||
|
synchronized (mLock) {
|
||||||
|
Slog.i(TAG, "Dock state changed: " + mDockState);
|
||||||
|
|
||||||
|
final ContentResolver cr = mContext.getContentResolver();
|
||||||
|
|
||||||
|
if (Settings.Secure.getInt(cr,
|
||||||
|
Settings.Secure.DEVICE_PROVISIONED, 0) == 0) {
|
||||||
|
Slog.i(TAG, "Device not provisioned, skipping dock broadcast");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pack up the values and broadcast them to everyone
|
||||||
|
Intent intent = new Intent(Intent.ACTION_DOCK_EVENT);
|
||||||
|
intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
|
||||||
|
intent.putExtra(Intent.EXTRA_DOCK_STATE, mDockState);
|
||||||
|
|
||||||
|
// Check if this is Bluetooth Dock
|
||||||
|
// TODO(BT): Get Dock address.
|
||||||
|
// String address = null;
|
||||||
|
// if (address != null) {
|
||||||
|
// intent.putExtra(BluetoothDevice.EXTRA_DEVICE,
|
||||||
|
// BluetoothAdapter.getDefaultAdapter().getRemoteDevice(address));
|
||||||
|
// }
|
||||||
|
|
||||||
|
// User feedback to confirm dock connection. Particularly
|
||||||
|
// useful for flaky contact pins...
|
||||||
|
if (Settings.System.getInt(cr,
|
||||||
|
Settings.System.DOCK_SOUNDS_ENABLED, 1) == 1) {
|
||||||
|
String whichSound = null;
|
||||||
|
if (mDockState == Intent.EXTRA_DOCK_STATE_UNDOCKED) {
|
||||||
|
if ((mPreviousDockState == Intent.EXTRA_DOCK_STATE_DESK) ||
|
||||||
|
(mPreviousDockState == Intent.EXTRA_DOCK_STATE_LE_DESK) ||
|
||||||
|
(mPreviousDockState == Intent.EXTRA_DOCK_STATE_HE_DESK)) {
|
||||||
|
whichSound = Settings.System.DESK_UNDOCK_SOUND;
|
||||||
|
} else if (mPreviousDockState == Intent.EXTRA_DOCK_STATE_CAR) {
|
||||||
|
whichSound = Settings.System.CAR_UNDOCK_SOUND;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if ((mDockState == Intent.EXTRA_DOCK_STATE_DESK) ||
|
||||||
|
(mDockState == Intent.EXTRA_DOCK_STATE_LE_DESK) ||
|
||||||
|
(mDockState == Intent.EXTRA_DOCK_STATE_HE_DESK)) {
|
||||||
|
whichSound = Settings.System.DESK_DOCK_SOUND;
|
||||||
|
} else if (mDockState == Intent.EXTRA_DOCK_STATE_CAR) {
|
||||||
|
whichSound = Settings.System.CAR_DOCK_SOUND;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (whichSound != null) {
|
||||||
|
final String soundPath = Settings.System.getString(cr, whichSound);
|
||||||
|
if (soundPath != null) {
|
||||||
|
final Uri soundUri = Uri.parse("file://" + soundPath);
|
||||||
|
if (soundUri != null) {
|
||||||
|
final Ringtone sfx = RingtoneManager.getRingtone(mContext, soundUri);
|
||||||
|
if (sfx != null) {
|
||||||
|
sfx.setStreamType(AudioManager.STREAM_SYSTEM);
|
||||||
|
sfx.play();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
IDreamManager mgr = IDreamManager.Stub.asInterface(ServiceManager.getService("dreams"));
|
||||||
|
if (mgr != null) {
|
||||||
|
// dreams feature enabled
|
||||||
|
boolean undocked = mDockState == Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
||||||
|
if (undocked) {
|
||||||
|
try {
|
||||||
|
if (mgr.isDreaming()) {
|
||||||
|
mgr.awaken();
|
||||||
|
}
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
Slog.w(TAG, "Unable to awaken!", e);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (isScreenSaverActivatedOnDock(mContext)) {
|
||||||
|
try {
|
||||||
|
mgr.dream();
|
||||||
|
} catch (RemoteException e) {
|
||||||
|
Slog.w(TAG, "Unable to dream!", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// dreams feature not enabled, send legacy intent
|
||||||
|
mContext.sendStickyBroadcast(intent);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static boolean isScreenSaverActivatedOnDock(Context context) {
|
private static boolean isScreenSaverActivatedOnDock(Context context) {
|
||||||
return 0 != Settings.Secure.getInt(
|
return Settings.Secure.getInt(context.getContentResolver(),
|
||||||
context.getContentResolver(), SCREENSAVER_ACTIVATE_ON_DOCK, DEFAULT_DOCK);
|
SCREENSAVER_ACTIVATE_ON_DOCK, DEFAULT_DOCK) != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private final Handler mHandler = new Handler() {
|
private final Handler mHandler = new Handler(Looper.myLooper(), null, true) {
|
||||||
@Override
|
@Override
|
||||||
public void handleMessage(Message msg) {
|
public void handleMessage(Message msg) {
|
||||||
switch (msg.what) {
|
switch (msg.what) {
|
||||||
case MSG_DOCK_STATE:
|
case MSG_DOCK_STATE_CHANGED:
|
||||||
synchronized (this) {
|
handleDockStateChange();
|
||||||
Slog.i(TAG, "Dock state changed: " + mDockState);
|
|
||||||
|
|
||||||
final ContentResolver cr = mContext.getContentResolver();
|
|
||||||
|
|
||||||
if (Settings.Secure.getInt(cr,
|
|
||||||
Settings.Secure.DEVICE_PROVISIONED, 0) == 0) {
|
|
||||||
Slog.i(TAG, "Device not provisioned, skipping dock broadcast");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Pack up the values and broadcast them to everyone
|
|
||||||
Intent intent = new Intent(Intent.ACTION_DOCK_EVENT);
|
|
||||||
intent.addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING);
|
|
||||||
intent.putExtra(Intent.EXTRA_DOCK_STATE, mDockState);
|
|
||||||
|
|
||||||
// Check if this is Bluetooth Dock
|
|
||||||
// TODO(BT): Get Dock address.
|
|
||||||
String address = null;
|
|
||||||
if (address != null)
|
|
||||||
intent.putExtra(BluetoothDevice.EXTRA_DEVICE,
|
|
||||||
BluetoothAdapter.getDefaultAdapter().getRemoteDevice(address));
|
|
||||||
|
|
||||||
// User feedback to confirm dock connection. Particularly
|
|
||||||
// useful for flaky contact pins...
|
|
||||||
if (Settings.System.getInt(cr,
|
|
||||||
Settings.System.DOCK_SOUNDS_ENABLED, 1) == 1)
|
|
||||||
{
|
|
||||||
String whichSound = null;
|
|
||||||
if (mDockState == Intent.EXTRA_DOCK_STATE_UNDOCKED) {
|
|
||||||
if ((mPreviousDockState == Intent.EXTRA_DOCK_STATE_DESK) ||
|
|
||||||
(mPreviousDockState == Intent.EXTRA_DOCK_STATE_LE_DESK) ||
|
|
||||||
(mPreviousDockState == Intent.EXTRA_DOCK_STATE_HE_DESK)) {
|
|
||||||
whichSound = Settings.System.DESK_UNDOCK_SOUND;
|
|
||||||
} else if (mPreviousDockState == Intent.EXTRA_DOCK_STATE_CAR) {
|
|
||||||
whichSound = Settings.System.CAR_UNDOCK_SOUND;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if ((mDockState == Intent.EXTRA_DOCK_STATE_DESK) ||
|
|
||||||
(mDockState == Intent.EXTRA_DOCK_STATE_LE_DESK) ||
|
|
||||||
(mDockState == Intent.EXTRA_DOCK_STATE_HE_DESK)) {
|
|
||||||
whichSound = Settings.System.DESK_DOCK_SOUND;
|
|
||||||
} else if (mDockState == Intent.EXTRA_DOCK_STATE_CAR) {
|
|
||||||
whichSound = Settings.System.CAR_DOCK_SOUND;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (whichSound != null) {
|
|
||||||
final String soundPath = Settings.System.getString(cr, whichSound);
|
|
||||||
if (soundPath != null) {
|
|
||||||
final Uri soundUri = Uri.parse("file://" + soundPath);
|
|
||||||
if (soundUri != null) {
|
|
||||||
final Ringtone sfx = RingtoneManager.getRingtone(mContext, soundUri);
|
|
||||||
if (sfx != null) {
|
|
||||||
sfx.setStreamType(AudioManager.STREAM_SYSTEM);
|
|
||||||
sfx.play();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
IDreamManager mgr = IDreamManager.Stub.asInterface(ServiceManager.getService("dreams"));
|
|
||||||
if (mgr != null) {
|
|
||||||
// dreams feature enabled
|
|
||||||
boolean undocked = mDockState == Intent.EXTRA_DOCK_STATE_UNDOCKED;
|
|
||||||
if (undocked) {
|
|
||||||
try {
|
|
||||||
if (mgr.isDreaming()) {
|
|
||||||
mgr.awaken();
|
|
||||||
}
|
|
||||||
} catch (RemoteException e) {
|
|
||||||
Slog.w(TAG, "Unable to awaken!", e);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isScreenSaverActivatedOnDock(mContext)) {
|
|
||||||
try {
|
|
||||||
mgr.dream();
|
|
||||||
} catch (RemoteException e) {
|
|
||||||
Slog.w(TAG, "Unable to dream!", e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// dreams feature not enabled, send legacy intent
|
|
||||||
mContext.sendStickyBroadcast(intent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,12 +16,12 @@
|
|||||||
|
|
||||||
package com.android.server;
|
package com.android.server;
|
||||||
|
|
||||||
import android.app.ActivityManagerNative;
|
|
||||||
import android.content.BroadcastReceiver;
|
import android.content.BroadcastReceiver;
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.content.Intent;
|
import android.content.Intent;
|
||||||
import android.content.IntentFilter;
|
import android.content.IntentFilter;
|
||||||
import android.os.Handler;
|
import android.os.Handler;
|
||||||
|
import android.os.Looper;
|
||||||
import android.os.Message;
|
import android.os.Message;
|
||||||
import android.os.PowerManager;
|
import android.os.PowerManager;
|
||||||
import android.os.PowerManager.WakeLock;
|
import android.os.PowerManager.WakeLock;
|
||||||
@@ -39,7 +39,7 @@ import java.util.List;
|
|||||||
/**
|
/**
|
||||||
* <p>WiredAccessoryObserver monitors for a wired headset on the main board or dock.
|
* <p>WiredAccessoryObserver monitors for a wired headset on the main board or dock.
|
||||||
*/
|
*/
|
||||||
class WiredAccessoryObserver extends UEventObserver {
|
final class WiredAccessoryObserver extends UEventObserver {
|
||||||
private static final String TAG = WiredAccessoryObserver.class.getSimpleName();
|
private static final String TAG = WiredAccessoryObserver.class.getSimpleName();
|
||||||
private static final boolean LOG = true;
|
private static final boolean LOG = true;
|
||||||
private static final int BIT_HEADSET = (1 << 0);
|
private static final int BIT_HEADSET = (1 << 0);
|
||||||
@@ -50,40 +50,177 @@ class WiredAccessoryObserver extends UEventObserver {
|
|||||||
private static final int SUPPORTED_HEADSETS = (BIT_HEADSET|BIT_HEADSET_NO_MIC|
|
private static final int SUPPORTED_HEADSETS = (BIT_HEADSET|BIT_HEADSET_NO_MIC|
|
||||||
BIT_USB_HEADSET_ANLG|BIT_USB_HEADSET_DGTL|
|
BIT_USB_HEADSET_ANLG|BIT_USB_HEADSET_DGTL|
|
||||||
BIT_HDMI_AUDIO);
|
BIT_HDMI_AUDIO);
|
||||||
private static final int HEADSETS_WITH_MIC = BIT_HEADSET;
|
|
||||||
|
|
||||||
private static class UEventInfo {
|
private final Object mLock = new Object();
|
||||||
private final String mDevName;
|
|
||||||
private final int mState1Bits;
|
|
||||||
private final int mState2Bits;
|
|
||||||
|
|
||||||
public UEventInfo(String devName, int state1Bits, int state2Bits) {
|
private final Context mContext;
|
||||||
mDevName = devName;
|
private final WakeLock mWakeLock; // held while there is a pending route change
|
||||||
mState1Bits = state1Bits;
|
private final AudioManager mAudioManager;
|
||||||
mState2Bits = state2Bits;
|
private final List<UEventInfo> mUEventInfo;
|
||||||
|
|
||||||
|
private int mHeadsetState;
|
||||||
|
private int mPrevHeadsetState;
|
||||||
|
private String mHeadsetName;
|
||||||
|
|
||||||
|
public WiredAccessoryObserver(Context context) {
|
||||||
|
mContext = context;
|
||||||
|
|
||||||
|
PowerManager pm = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
|
||||||
|
mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "WiredAccessoryObserver");
|
||||||
|
mWakeLock.setReferenceCounted(false);
|
||||||
|
mAudioManager = (AudioManager)context.getSystemService(Context.AUDIO_SERVICE);
|
||||||
|
|
||||||
|
mUEventInfo = makeObservedUEventList();
|
||||||
|
|
||||||
|
context.registerReceiver(new BootCompletedReceiver(),
|
||||||
|
new IntentFilter(Intent.ACTION_BOOT_COMPLETED), null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onUEvent(UEventObserver.UEvent event) {
|
||||||
|
if (LOG) Slog.v(TAG, "Headset UEVENT: " + event.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
String devPath = event.get("DEVPATH");
|
||||||
|
String name = event.get("SWITCH_NAME");
|
||||||
|
int state = Integer.parseInt(event.get("SWITCH_STATE"));
|
||||||
|
synchronized (mLock) {
|
||||||
|
updateStateLocked(devPath, name, state);
|
||||||
|
}
|
||||||
|
} catch (NumberFormatException e) {
|
||||||
|
Slog.e(TAG, "Could not parse switch state from event " + event);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void bootCompleted() {
|
||||||
|
synchronized (mLock) {
|
||||||
|
char[] buffer = new char[1024];
|
||||||
|
mPrevHeadsetState = mHeadsetState;
|
||||||
|
|
||||||
|
if (LOG) Slog.v(TAG, "init()");
|
||||||
|
|
||||||
|
for (int i = 0; i < mUEventInfo.size(); ++i) {
|
||||||
|
UEventInfo uei = mUEventInfo.get(i);
|
||||||
|
try {
|
||||||
|
int curState;
|
||||||
|
FileReader file = new FileReader(uei.getSwitchStatePath());
|
||||||
|
int len = file.read(buffer, 0, 1024);
|
||||||
|
file.close();
|
||||||
|
curState = Integer.valueOf((new String(buffer, 0, len)).trim());
|
||||||
|
|
||||||
|
if (curState > 0) {
|
||||||
|
updateStateLocked(uei.getDevPath(), uei.getDevName(), curState);
|
||||||
|
}
|
||||||
|
} catch (FileNotFoundException e) {
|
||||||
|
Slog.w(TAG, uei.getSwitchStatePath() +
|
||||||
|
" not found while attempting to determine initial switch state");
|
||||||
|
} catch (Exception e) {
|
||||||
|
Slog.e(TAG, "" , e);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getDevName() { return mDevName; }
|
// At any given time accessories could be inserted
|
||||||
|
// one on the board, one on the dock and one on HDMI:
|
||||||
|
// observe three UEVENTs
|
||||||
|
for (int i = 0; i < mUEventInfo.size(); ++i) {
|
||||||
|
UEventInfo uei = mUEventInfo.get(i);
|
||||||
|
startObserving("DEVPATH="+uei.getDevPath());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public String getDevPath() {
|
private void updateStateLocked(String devPath, String name, int state) {
|
||||||
return String.format("/devices/virtual/switch/%s", mDevName);
|
for (int i = 0; i < mUEventInfo.size(); ++i) {
|
||||||
|
UEventInfo uei = mUEventInfo.get(i);
|
||||||
|
if (devPath.equals(uei.getDevPath())) {
|
||||||
|
updateLocked(name, uei.computeNewHeadsetState(mHeadsetState, state));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateLocked(String newName, int newState) {
|
||||||
|
// Retain only relevant bits
|
||||||
|
int headsetState = newState & SUPPORTED_HEADSETS;
|
||||||
|
int usb_headset_anlg = headsetState & BIT_USB_HEADSET_ANLG;
|
||||||
|
int usb_headset_dgtl = headsetState & BIT_USB_HEADSET_DGTL;
|
||||||
|
int h2w_headset = headsetState & (BIT_HEADSET | BIT_HEADSET_NO_MIC);
|
||||||
|
boolean h2wStateChange = true;
|
||||||
|
boolean usbStateChange = true;
|
||||||
|
// reject all suspect transitions: only accept state changes from:
|
||||||
|
// - a: 0 heaset to 1 headset
|
||||||
|
// - b: 1 headset to 0 headset
|
||||||
|
if (LOG) Slog.v(TAG, "newState = "+newState+", headsetState = "+headsetState+","
|
||||||
|
+ "mHeadsetState = "+mHeadsetState);
|
||||||
|
if (mHeadsetState == headsetState || ((h2w_headset & (h2w_headset - 1)) != 0)) {
|
||||||
|
Log.e(TAG, "unsetting h2w flag");
|
||||||
|
h2wStateChange = false;
|
||||||
|
}
|
||||||
|
// - c: 0 usb headset to 1 usb headset
|
||||||
|
// - d: 1 usb headset to 0 usb headset
|
||||||
|
if ((usb_headset_anlg >> 2) == 1 && (usb_headset_dgtl >> 3) == 1) {
|
||||||
|
Log.e(TAG, "unsetting usb flag");
|
||||||
|
usbStateChange = false;
|
||||||
|
}
|
||||||
|
if (!h2wStateChange && !usbStateChange) {
|
||||||
|
Log.e(TAG, "invalid transition, returning ...");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
public String getSwitchStatePath() {
|
mHeadsetName = newName;
|
||||||
return String.format("/sys/class/switch/%s/state", mDevName);
|
mPrevHeadsetState = mHeadsetState;
|
||||||
|
mHeadsetState = headsetState;
|
||||||
|
|
||||||
|
mWakeLock.acquire();
|
||||||
|
|
||||||
|
Message msg = mHandler.obtainMessage(0, mHeadsetState, mPrevHeadsetState, mHeadsetName);
|
||||||
|
mHandler.sendMessage(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setDevicesState(
|
||||||
|
int headsetState, int prevHeadsetState, String headsetName) {
|
||||||
|
synchronized (mLock) {
|
||||||
|
int allHeadsets = SUPPORTED_HEADSETS;
|
||||||
|
for (int curHeadset = 1; allHeadsets != 0; curHeadset <<= 1) {
|
||||||
|
if ((curHeadset & allHeadsets) != 0) {
|
||||||
|
setDeviceStateLocked(curHeadset, headsetState, prevHeadsetState, headsetName);
|
||||||
|
allHeadsets &= ~curHeadset;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public boolean checkSwitchExists() {
|
private void setDeviceStateLocked(int headset,
|
||||||
File f = new File(getSwitchStatePath());
|
int headsetState, int prevHeadsetState, String headsetName) {
|
||||||
return ((null != f) && f.exists());
|
if ((headsetState & headset) != (prevHeadsetState & headset)) {
|
||||||
}
|
int device;
|
||||||
|
int state;
|
||||||
|
|
||||||
public int computeNewHeadsetState(int headsetState, int switchState) {
|
if ((headsetState & headset) != 0) {
|
||||||
int preserveMask = ~(mState1Bits | mState2Bits);
|
state = 1;
|
||||||
int setBits = ((switchState == 1) ? mState1Bits :
|
} else {
|
||||||
((switchState == 2) ? mState2Bits : 0));
|
state = 0;
|
||||||
|
}
|
||||||
|
|
||||||
return ((headsetState & preserveMask) | setBits);
|
if (headset == BIT_HEADSET) {
|
||||||
|
device = AudioManager.DEVICE_OUT_WIRED_HEADSET;
|
||||||
|
} else if (headset == BIT_HEADSET_NO_MIC){
|
||||||
|
device = AudioManager.DEVICE_OUT_WIRED_HEADPHONE;
|
||||||
|
} else if (headset == BIT_USB_HEADSET_ANLG) {
|
||||||
|
device = AudioManager.DEVICE_OUT_ANLG_DOCK_HEADSET;
|
||||||
|
} else if (headset == BIT_USB_HEADSET_DGTL) {
|
||||||
|
device = AudioManager.DEVICE_OUT_DGTL_DOCK_HEADSET;
|
||||||
|
} else if (headset == BIT_HDMI_AUDIO) {
|
||||||
|
device = AudioManager.DEVICE_OUT_AUX_DIGITAL;
|
||||||
|
} else {
|
||||||
|
Slog.e(TAG, "setDeviceState() invalid headset type: "+headset);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (LOG)
|
||||||
|
Slog.v(TAG, "device "+headsetName+((state == 1) ? " connected" : " disconnected"));
|
||||||
|
|
||||||
|
mAudioManager.setWiredDeviceConnectionState(device, state, headsetName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -130,189 +267,53 @@ class WiredAccessoryObserver extends UEventObserver {
|
|||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static List<UEventInfo> uEventInfo = makeObservedUEventList();
|
private final Handler mHandler = new Handler(Looper.myLooper(), null, true) {
|
||||||
|
|
||||||
private int mHeadsetState;
|
|
||||||
private int mPrevHeadsetState;
|
|
||||||
private String mHeadsetName;
|
|
||||||
|
|
||||||
private final Context mContext;
|
|
||||||
private final WakeLock mWakeLock; // held while there is a pending route change
|
|
||||||
|
|
||||||
private final AudioManager mAudioManager;
|
|
||||||
|
|
||||||
public WiredAccessoryObserver(Context context) {
|
|
||||||
mContext = context;
|
|
||||||
PowerManager pm = (PowerManager)context.getSystemService(Context.POWER_SERVICE);
|
|
||||||
mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "WiredAccessoryObserver");
|
|
||||||
mWakeLock.setReferenceCounted(false);
|
|
||||||
mAudioManager = (AudioManager)context.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
|
|
||||||
context.registerReceiver(new BootCompletedReceiver(),
|
|
||||||
new IntentFilter(Intent.ACTION_BOOT_COMPLETED), null, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
private final class BootCompletedReceiver extends BroadcastReceiver {
|
|
||||||
@Override
|
|
||||||
public void onReceive(Context context, Intent intent) {
|
|
||||||
// At any given time accessories could be inserted
|
|
||||||
// one on the board, one on the dock and one on HDMI:
|
|
||||||
// observe three UEVENTs
|
|
||||||
init(); // set initial status
|
|
||||||
for (int i = 0; i < uEventInfo.size(); ++i) {
|
|
||||||
UEventInfo uei = uEventInfo.get(i);
|
|
||||||
startObserving("DEVPATH="+uei.getDevPath());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onUEvent(UEventObserver.UEvent event) {
|
|
||||||
if (LOG) Slog.v(TAG, "Headset UEVENT: " + event.toString());
|
|
||||||
|
|
||||||
try {
|
|
||||||
String devPath = event.get("DEVPATH");
|
|
||||||
String name = event.get("SWITCH_NAME");
|
|
||||||
int state = Integer.parseInt(event.get("SWITCH_STATE"));
|
|
||||||
updateState(devPath, name, state);
|
|
||||||
} catch (NumberFormatException e) {
|
|
||||||
Slog.e(TAG, "Could not parse switch state from event " + event);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private synchronized final void updateState(String devPath, String name, int state)
|
|
||||||
{
|
|
||||||
for (int i = 0; i < uEventInfo.size(); ++i) {
|
|
||||||
UEventInfo uei = uEventInfo.get(i);
|
|
||||||
if (devPath.equals(uei.getDevPath())) {
|
|
||||||
update(name, uei.computeNewHeadsetState(mHeadsetState, state));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private synchronized final void init() {
|
|
||||||
char[] buffer = new char[1024];
|
|
||||||
mPrevHeadsetState = mHeadsetState;
|
|
||||||
|
|
||||||
if (LOG) Slog.v(TAG, "init()");
|
|
||||||
|
|
||||||
for (int i = 0; i < uEventInfo.size(); ++i) {
|
|
||||||
UEventInfo uei = uEventInfo.get(i);
|
|
||||||
try {
|
|
||||||
int curState;
|
|
||||||
FileReader file = new FileReader(uei.getSwitchStatePath());
|
|
||||||
int len = file.read(buffer, 0, 1024);
|
|
||||||
file.close();
|
|
||||||
curState = Integer.valueOf((new String(buffer, 0, len)).trim());
|
|
||||||
|
|
||||||
if (curState > 0) {
|
|
||||||
updateState(uei.getDevPath(), uei.getDevName(), curState);
|
|
||||||
}
|
|
||||||
|
|
||||||
} catch (FileNotFoundException e) {
|
|
||||||
Slog.w(TAG, uei.getSwitchStatePath() +
|
|
||||||
" not found while attempting to determine initial switch state");
|
|
||||||
} catch (Exception e) {
|
|
||||||
Slog.e(TAG, "" , e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private synchronized final void update(String newName, int newState) {
|
|
||||||
// Retain only relevant bits
|
|
||||||
int headsetState = newState & SUPPORTED_HEADSETS;
|
|
||||||
int newOrOld = headsetState | mHeadsetState;
|
|
||||||
int delay = 0;
|
|
||||||
int usb_headset_anlg = headsetState & BIT_USB_HEADSET_ANLG;
|
|
||||||
int usb_headset_dgtl = headsetState & BIT_USB_HEADSET_DGTL;
|
|
||||||
int h2w_headset = headsetState & (BIT_HEADSET | BIT_HEADSET_NO_MIC);
|
|
||||||
boolean h2wStateChange = true;
|
|
||||||
boolean usbStateChange = true;
|
|
||||||
// reject all suspect transitions: only accept state changes from:
|
|
||||||
// - a: 0 heaset to 1 headset
|
|
||||||
// - b: 1 headset to 0 headset
|
|
||||||
if (LOG) Slog.v(TAG, "newState = "+newState+", headsetState = "+headsetState+","
|
|
||||||
+ "mHeadsetState = "+mHeadsetState);
|
|
||||||
if (mHeadsetState == headsetState || ((h2w_headset & (h2w_headset - 1)) != 0)) {
|
|
||||||
Log.e(TAG, "unsetting h2w flag");
|
|
||||||
h2wStateChange = false;
|
|
||||||
}
|
|
||||||
// - c: 0 usb headset to 1 usb headset
|
|
||||||
// - d: 1 usb headset to 0 usb headset
|
|
||||||
if ((usb_headset_anlg >> 2) == 1 && (usb_headset_dgtl >> 3) == 1) {
|
|
||||||
Log.e(TAG, "unsetting usb flag");
|
|
||||||
usbStateChange = false;
|
|
||||||
}
|
|
||||||
if (!h2wStateChange && !usbStateChange) {
|
|
||||||
Log.e(TAG, "invalid transition, returning ...");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mHeadsetName = newName;
|
|
||||||
mPrevHeadsetState = mHeadsetState;
|
|
||||||
mHeadsetState = headsetState;
|
|
||||||
|
|
||||||
mWakeLock.acquire();
|
|
||||||
mHandler.sendMessage(mHandler.obtainMessage(0,
|
|
||||||
mHeadsetState,
|
|
||||||
mPrevHeadsetState,
|
|
||||||
mHeadsetName));
|
|
||||||
}
|
|
||||||
|
|
||||||
private synchronized final void setDevicesState(int headsetState,
|
|
||||||
int prevHeadsetState,
|
|
||||||
String headsetName) {
|
|
||||||
int allHeadsets = SUPPORTED_HEADSETS;
|
|
||||||
for (int curHeadset = 1; allHeadsets != 0; curHeadset <<= 1) {
|
|
||||||
if ((curHeadset & allHeadsets) != 0) {
|
|
||||||
setDeviceState(curHeadset, headsetState, prevHeadsetState, headsetName);
|
|
||||||
allHeadsets &= ~curHeadset;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private final void setDeviceState(int headset,
|
|
||||||
int headsetState,
|
|
||||||
int prevHeadsetState,
|
|
||||||
String headsetName) {
|
|
||||||
if ((headsetState & headset) != (prevHeadsetState & headset)) {
|
|
||||||
int device;
|
|
||||||
int state;
|
|
||||||
|
|
||||||
if ((headsetState & headset) != 0) {
|
|
||||||
state = 1;
|
|
||||||
} else {
|
|
||||||
state = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (headset == BIT_HEADSET) {
|
|
||||||
device = AudioManager.DEVICE_OUT_WIRED_HEADSET;
|
|
||||||
} else if (headset == BIT_HEADSET_NO_MIC){
|
|
||||||
device = AudioManager.DEVICE_OUT_WIRED_HEADPHONE;
|
|
||||||
} else if (headset == BIT_USB_HEADSET_ANLG) {
|
|
||||||
device = AudioManager.DEVICE_OUT_ANLG_DOCK_HEADSET;
|
|
||||||
} else if (headset == BIT_USB_HEADSET_DGTL) {
|
|
||||||
device = AudioManager.DEVICE_OUT_DGTL_DOCK_HEADSET;
|
|
||||||
} else if (headset == BIT_HDMI_AUDIO) {
|
|
||||||
device = AudioManager.DEVICE_OUT_AUX_DIGITAL;
|
|
||||||
} else {
|
|
||||||
Slog.e(TAG, "setDeviceState() invalid headset type: "+headset);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (LOG)
|
|
||||||
Slog.v(TAG, "device "+headsetName+((state == 1) ? " connected" : " disconnected"));
|
|
||||||
|
|
||||||
mAudioManager.setWiredDeviceConnectionState(device, state, headsetName);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private final Handler mHandler = new Handler() {
|
|
||||||
@Override
|
@Override
|
||||||
public void handleMessage(Message msg) {
|
public void handleMessage(Message msg) {
|
||||||
setDevicesState(msg.arg1, msg.arg2, (String)msg.obj);
|
setDevicesState(msg.arg1, msg.arg2, (String)msg.obj);
|
||||||
mWakeLock.release();
|
mWakeLock.release();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private static final class UEventInfo {
|
||||||
|
private final String mDevName;
|
||||||
|
private final int mState1Bits;
|
||||||
|
private final int mState2Bits;
|
||||||
|
|
||||||
|
public UEventInfo(String devName, int state1Bits, int state2Bits) {
|
||||||
|
mDevName = devName;
|
||||||
|
mState1Bits = state1Bits;
|
||||||
|
mState2Bits = state2Bits;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getDevName() { return mDevName; }
|
||||||
|
|
||||||
|
public String getDevPath() {
|
||||||
|
return String.format("/devices/virtual/switch/%s", mDevName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getSwitchStatePath() {
|
||||||
|
return String.format("/sys/class/switch/%s/state", mDevName);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean checkSwitchExists() {
|
||||||
|
File f = new File(getSwitchStatePath());
|
||||||
|
return ((null != f) && f.exists());
|
||||||
|
}
|
||||||
|
|
||||||
|
public int computeNewHeadsetState(int headsetState, int switchState) {
|
||||||
|
int preserveMask = ~(mState1Bits | mState2Bits);
|
||||||
|
int setBits = ((switchState == 1) ? mState1Bits :
|
||||||
|
((switchState == 2) ? mState2Bits : 0));
|
||||||
|
|
||||||
|
return ((headsetState & preserveMask) | setBits);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private final class BootCompletedReceiver extends BroadcastReceiver {
|
||||||
|
@Override
|
||||||
|
public void onReceive(Context context, Intent intent) {
|
||||||
|
bootCompleted();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user