/* * Copyright (C) 2008 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.app.AppOpsManager; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.pm.PackageManager; import android.database.ContentObserver; import android.hardware.input.InputManager; import android.os.BatteryStats; import android.os.Handler; import android.os.IVibratorService; import android.os.PowerManager; import android.os.PowerManagerInternal; import android.os.Process; import android.os.RemoteException; import android.os.IBinder; import android.os.Binder; import android.os.ServiceManager; import android.os.SystemClock; import android.os.UserHandle; import android.os.Vibrator; import android.os.WorkSource; import android.provider.Settings; import android.provider.Settings.SettingNotFoundException; import android.util.Slog; import android.view.InputDevice; import android.media.AudioAttributes; import com.android.internal.app.IAppOpsService; import com.android.internal.app.IBatteryStats; import java.util.ArrayList; import java.util.Iterator; import java.util.LinkedList; import java.util.ListIterator; public class VibratorService extends IVibratorService.Stub implements InputManager.InputDeviceListener { private static final String TAG = "VibratorService"; private static final boolean DEBUG = false; private final LinkedList mVibrations; private Vibration mCurrentVibration; private final WorkSource mTmpWorkSource = new WorkSource(); private final Handler mH = new Handler(); private final Context mContext; private final PowerManager.WakeLock mWakeLock; private final IAppOpsService mAppOpsService; private final IBatteryStats mBatteryStatsService; private PowerManagerInternal mPowerManagerInternal; private InputManager mIm; volatile VibrateThread mThread; // mInputDeviceVibrators lock should be acquired after mVibrations lock, if both are // to be acquired private final ArrayList mInputDeviceVibrators = new ArrayList(); private boolean mVibrateInputDevicesSetting; // guarded by mInputDeviceVibrators private boolean mInputDeviceListenerRegistered; // guarded by mInputDeviceVibrators private int mCurVibUid = -1; private boolean mLowPowerMode; private SettingsObserver mSettingObserver; native static boolean vibratorExists(); native static void vibratorOn(long milliseconds); native static void vibratorOff(); private class Vibration implements IBinder.DeathRecipient { private final IBinder mToken; private final long mTimeout; private final long mStartTime; private final long[] mPattern; private final int mRepeat; private final int mUsageHint; private final int mUid; private final String mOpPkg; Vibration(IBinder token, long millis, int usageHint, int uid, String opPkg) { this(token, millis, null, 0, usageHint, uid, opPkg); } Vibration(IBinder token, long[] pattern, int repeat, int usageHint, int uid, String opPkg) { this(token, 0, pattern, repeat, usageHint, uid, opPkg); } private Vibration(IBinder token, long millis, long[] pattern, int repeat, int usageHint, int uid, String opPkg) { mToken = token; mTimeout = millis; mStartTime = SystemClock.uptimeMillis(); mPattern = pattern; mRepeat = repeat; mUsageHint = usageHint; mUid = uid; mOpPkg = opPkg; } public void binderDied() { synchronized (mVibrations) { mVibrations.remove(this); if (this == mCurrentVibration) { doCancelVibrateLocked(); startNextVibrationLocked(); } } } public boolean hasLongerTimeout(long millis) { if (mTimeout == 0) { // This is a pattern, return false to play the simple // vibration. return false; } if ((mStartTime + mTimeout) < (SystemClock.uptimeMillis() + millis)) { // If this vibration will end before the time passed in, let // the new vibration play. return false; } return true; } public boolean isSystemHapticFeedback() { return (mUid == Process.SYSTEM_UID || mUid == 0) && mRepeat < 0; } } VibratorService(Context context) { // Reset the hardware to a default state, in case this is a runtime // restart instead of a fresh boot. vibratorOff(); mContext = context; PowerManager pm = (PowerManager)context.getSystemService( Context.POWER_SERVICE); mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "*vibrator*"); mWakeLock.setReferenceCounted(true); mAppOpsService = IAppOpsService.Stub.asInterface(ServiceManager.getService(Context.APP_OPS_SERVICE)); mBatteryStatsService = IBatteryStats.Stub.asInterface(ServiceManager.getService( BatteryStats.SERVICE_NAME)); mVibrations = new LinkedList(); IntentFilter filter = new IntentFilter(); filter.addAction(Intent.ACTION_SCREEN_OFF); context.registerReceiver(mIntentReceiver, filter); } public void systemReady() { mIm = (InputManager)mContext.getSystemService(Context.INPUT_SERVICE); mSettingObserver = new SettingsObserver(mH); mPowerManagerInternal = LocalServices.getService(PowerManagerInternal.class); mPowerManagerInternal.registerLowPowerModeObserver( new PowerManagerInternal.LowPowerModeListener() { @Override public void onLowPowerModeChanged(boolean enabled) { updateInputDeviceVibrators(); } }); mContext.getContentResolver().registerContentObserver( Settings.System.getUriFor(Settings.System.VIBRATE_INPUT_DEVICES), true, mSettingObserver, UserHandle.USER_ALL); mContext.registerReceiver(new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { updateInputDeviceVibrators(); } }, new IntentFilter(Intent.ACTION_USER_SWITCHED), null, mH); updateInputDeviceVibrators(); } private final class SettingsObserver extends ContentObserver { public SettingsObserver(Handler handler) { super(handler); } @Override public void onChange(boolean SelfChange) { updateInputDeviceVibrators(); } } @Override // Binder call public boolean hasVibrator() { return doVibratorExists(); } private void verifyIncomingUid(int uid) { if (uid == Binder.getCallingUid()) { return; } if (Binder.getCallingPid() == Process.myPid()) { return; } mContext.enforcePermission(android.Manifest.permission.UPDATE_APP_OPS_STATS, Binder.getCallingPid(), Binder.getCallingUid(), null); } @Override // Binder call public void vibrate(int uid, String opPkg, long milliseconds, int usageHint, IBinder token) { if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.VIBRATE) != PackageManager.PERMISSION_GRANTED) { throw new SecurityException("Requires VIBRATE permission"); } verifyIncomingUid(uid); // We're running in the system server so we cannot crash. Check for a // timeout of 0 or negative. This will ensure that a vibration has // either a timeout of > 0 or a non-null pattern. if (milliseconds <= 0 || (mCurrentVibration != null && mCurrentVibration.hasLongerTimeout(milliseconds))) { // Ignore this vibration since the current vibration will play for // longer than milliseconds. return; } if (DEBUG) { Slog.d(TAG, "Vibrating for " + milliseconds + " ms."); } Vibration vib = new Vibration(token, milliseconds, usageHint, uid, opPkg); final long ident = Binder.clearCallingIdentity(); try { synchronized (mVibrations) { removeVibrationLocked(token); doCancelVibrateLocked(); mCurrentVibration = vib; startVibrationLocked(vib); } } finally { Binder.restoreCallingIdentity(ident); } } private boolean isAll0(long[] pattern) { int N = pattern.length; for (int i = 0; i < N; i++) { if (pattern[i] != 0) { return false; } } return true; } @Override // Binder call public void vibratePattern(int uid, String packageName, long[] pattern, int repeat, int usageHint, IBinder token) { if (mContext.checkCallingOrSelfPermission(android.Manifest.permission.VIBRATE) != PackageManager.PERMISSION_GRANTED) { throw new SecurityException("Requires VIBRATE permission"); } verifyIncomingUid(uid); // so wakelock calls will succeed long identity = Binder.clearCallingIdentity(); try { if (DEBUG) { String s = ""; int N = pattern.length; for (int i=0; i= pattern.length || token == null) { return; } Vibration vib = new Vibration(token, pattern, repeat, usageHint, uid, packageName); try { token.linkToDeath(vib, 0); } catch (RemoteException e) { return; } synchronized (mVibrations) { removeVibrationLocked(token); doCancelVibrateLocked(); if (repeat >= 0) { mVibrations.addFirst(vib); startNextVibrationLocked(); } else { // A negative repeat means that this pattern is not meant // to repeat. Treat it like a simple vibration. mCurrentVibration = vib; startVibrationLocked(vib); } } } finally { Binder.restoreCallingIdentity(identity); } } @Override // Binder call public void cancelVibrate(IBinder token) { mContext.enforceCallingOrSelfPermission( android.Manifest.permission.VIBRATE, "cancelVibrate"); // so wakelock calls will succeed long identity = Binder.clearCallingIdentity(); try { synchronized (mVibrations) { final Vibration vib = removeVibrationLocked(token); if (vib == mCurrentVibration) { if (DEBUG) { Slog.d(TAG, "Canceling vibration."); } doCancelVibrateLocked(); startNextVibrationLocked(); } } } finally { Binder.restoreCallingIdentity(identity); } } private final Runnable mVibrationRunnable = new Runnable() { @Override public void run() { synchronized (mVibrations) { doCancelVibrateLocked(); startNextVibrationLocked(); } } }; // Lock held on mVibrations private void doCancelVibrateLocked() { if (mThread != null) { synchronized (mThread) { mThread.mDone = true; mThread.notify(); } mThread = null; } doVibratorOff(); mH.removeCallbacks(mVibrationRunnable); reportFinishVibrationLocked(); } // Lock held on mVibrations private void startNextVibrationLocked() { if (mVibrations.size() <= 0) { reportFinishVibrationLocked(); mCurrentVibration = null; return; } mCurrentVibration = mVibrations.getFirst(); startVibrationLocked(mCurrentVibration); } // Lock held on mVibrations private void startVibrationLocked(final Vibration vib) { try { if (mLowPowerMode && vib.mUsageHint != AudioAttributes.USAGE_NOTIFICATION_RINGTONE) { return; } int mode = mAppOpsService.checkAudioOperation(AppOpsManager.OP_VIBRATE, vib.mUsageHint, vib.mUid, vib.mOpPkg); if (mode == AppOpsManager.MODE_ALLOWED) { mode = mAppOpsService.startOperation(AppOpsManager.getToken(mAppOpsService), AppOpsManager.OP_VIBRATE, vib.mUid, vib.mOpPkg); } if (mode != AppOpsManager.MODE_ALLOWED) { if (mode == AppOpsManager.MODE_ERRORED) { Slog.w(TAG, "Would be an error: vibrate from uid " + vib.mUid); } mH.post(mVibrationRunnable); return; } } catch (RemoteException e) { } if (vib.mTimeout != 0) { doVibratorOn(vib.mTimeout, vib.mUid, vib.mUsageHint); mH.postDelayed(mVibrationRunnable, vib.mTimeout); } else { // mThread better be null here. doCancelVibrate should always be // called before startNextVibrationLocked or startVibrationLocked. mThread = new VibrateThread(vib); mThread.start(); } } private void reportFinishVibrationLocked() { if (mCurrentVibration != null) { try { mAppOpsService.finishOperation(AppOpsManager.getToken(mAppOpsService), AppOpsManager.OP_VIBRATE, mCurrentVibration.mUid, mCurrentVibration.mOpPkg); } catch (RemoteException e) { } mCurrentVibration = null; } } // Lock held on mVibrations private Vibration removeVibrationLocked(IBinder token) { ListIterator iter = mVibrations.listIterator(0); while (iter.hasNext()) { Vibration vib = iter.next(); if (vib.mToken == token) { iter.remove(); unlinkVibration(vib); return vib; } } // We might be looking for a simple vibration which is only stored in // mCurrentVibration. if (mCurrentVibration != null && mCurrentVibration.mToken == token) { unlinkVibration(mCurrentVibration); return mCurrentVibration; } return null; } private void unlinkVibration(Vibration vib) { if (vib.mPattern != null) { // If Vibration object has a pattern, // the Vibration object has also been linkedToDeath. vib.mToken.unlinkToDeath(vib, 0); } } private void updateInputDeviceVibrators() { synchronized (mVibrations) { doCancelVibrateLocked(); synchronized (mInputDeviceVibrators) { mVibrateInputDevicesSetting = false; try { mVibrateInputDevicesSetting = Settings.System.getIntForUser( mContext.getContentResolver(), Settings.System.VIBRATE_INPUT_DEVICES, UserHandle.USER_CURRENT) > 0; } catch (SettingNotFoundException snfe) { } mLowPowerMode = mPowerManagerInternal.getLowPowerModeEnabled(); if (mVibrateInputDevicesSetting) { if (!mInputDeviceListenerRegistered) { mInputDeviceListenerRegistered = true; mIm.registerInputDeviceListener(this, mH); } } else { if (mInputDeviceListenerRegistered) { mInputDeviceListenerRegistered = false; mIm.unregisterInputDeviceListener(this); } } mInputDeviceVibrators.clear(); if (mVibrateInputDevicesSetting) { int[] ids = mIm.getInputDeviceIds(); for (int i = 0; i < ids.length; i++) { InputDevice device = mIm.getInputDevice(ids[i]); Vibrator vibrator = device.getVibrator(); if (vibrator.hasVibrator()) { mInputDeviceVibrators.add(vibrator); } } } } startNextVibrationLocked(); } } @Override public void onInputDeviceAdded(int deviceId) { updateInputDeviceVibrators(); } @Override public void onInputDeviceChanged(int deviceId) { updateInputDeviceVibrators(); } @Override public void onInputDeviceRemoved(int deviceId) { updateInputDeviceVibrators(); } private boolean doVibratorExists() { // For now, we choose to ignore the presence of input devices that have vibrators // when reporting whether the device has a vibrator. Applications often use this // information to decide whether to enable certain features so they expect the // result of hasVibrator() to be constant. For now, just report whether // the device has a built-in vibrator. //synchronized (mInputDeviceVibrators) { // return !mInputDeviceVibrators.isEmpty() || vibratorExists(); //} return vibratorExists(); } private void doVibratorOn(long millis, int uid, int usageHint) { synchronized (mInputDeviceVibrators) { if (DEBUG) { Slog.d(TAG, "Turning vibrator on for " + millis + " ms."); } try { mBatteryStatsService.noteVibratorOn(uid, millis); mCurVibUid = uid; } catch (RemoteException e) { } final int vibratorCount = mInputDeviceVibrators.size(); if (vibratorCount != 0) { final AudioAttributes attributes = new AudioAttributes.Builder().setUsage(usageHint) .build(); for (int i = 0; i < vibratorCount; i++) { mInputDeviceVibrators.get(i).vibrate(millis, attributes); } } else { vibratorOn(millis); } } } private void doVibratorOff() { synchronized (mInputDeviceVibrators) { if (DEBUG) { Slog.d(TAG, "Turning vibrator off."); } if (mCurVibUid >= 0) { try { mBatteryStatsService.noteVibratorOff(mCurVibUid); } catch (RemoteException e) { } mCurVibUid = -1; } final int vibratorCount = mInputDeviceVibrators.size(); if (vibratorCount != 0) { for (int i = 0; i < vibratorCount; i++) { mInputDeviceVibrators.get(i).cancel(); } } else { vibratorOff(); } } } private class VibrateThread extends Thread { final Vibration mVibration; boolean mDone; VibrateThread(Vibration vib) { mVibration = vib; mTmpWorkSource.set(vib.mUid); mWakeLock.setWorkSource(mTmpWorkSource); mWakeLock.acquire(); } private void delay(long duration) { if (duration > 0) { long bedtime = duration + SystemClock.uptimeMillis(); do { try { this.wait(duration); } catch (InterruptedException e) { } if (mDone) { break; } duration = bedtime - SystemClock.uptimeMillis(); } while (duration > 0); } } public void run() { Process.setThreadPriority(Process.THREAD_PRIORITY_URGENT_DISPLAY); synchronized (this) { final long[] pattern = mVibration.mPattern; final int len = pattern.length; final int repeat = mVibration.mRepeat; final int uid = mVibration.mUid; final int usageHint = mVibration.mUsageHint; int index = 0; long duration = 0; while (!mDone) { // add off-time duration to any accumulated on-time duration if (index < len) { duration += pattern[index++]; } // sleep until it is time to start the vibrator delay(duration); if (mDone) { break; } if (index < len) { // read on-time duration and start the vibrator // duration is saved for delay() at top of loop duration = pattern[index++]; if (duration > 0) { VibratorService.this.doVibratorOn(duration, uid, usageHint); } } else { if (repeat < 0) { break; } else { index = repeat; duration = 0; } } } mWakeLock.release(); } synchronized (mVibrations) { if (mThread == this) { mThread = null; } if (!mDone) { // If this vibration finished naturally, start the next // vibration. mVibrations.remove(mVibration); unlinkVibration(mVibration); startNextVibrationLocked(); } } } } BroadcastReceiver mIntentReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) { synchronized (mVibrations) { // When the system is entering a non-interactive state, we want // to cancel vibrations in case a misbehaving app has abandoned // them. However it may happen that the system is currently playing // haptic feedback as part of the transition. So we don't cancel // system vibrations. if (mCurrentVibration != null && !mCurrentVibration.isSystemHapticFeedback()) { doCancelVibrateLocked(); } // Clear all remaining vibrations. Iterator it = mVibrations.iterator(); while (it.hasNext()) { Vibration vibration = it.next(); if (vibration != mCurrentVibration) { unlinkVibration(vibration); it.remove(); } } } } } }; }