blob: 07048a49154334174658900ceddd63bcdf75610d [file] [log] [blame]
Mike Lockwood3a322132009-11-24 00:30:52 -05001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Adam Lesinski182f73f2013-12-05 16:48:06 -080017package com.android.server.lights;
18
19import com.android.server.SystemService;
Ruben Brunkc7be3be2016-04-01 17:07:51 -070020import com.android.server.vr.VrManagerService;
Mike Lockwood3a322132009-11-24 00:30:52 -050021
Ruben Brunk49506e02016-04-18 18:10:47 -070022import android.app.ActivityManager;
Mike Lockwood3a322132009-11-24 00:30:52 -050023import android.content.Context;
24import android.os.Handler;
25import android.os.Message;
Ruben Brunkc7be3be2016-04-01 17:07:51 -070026import android.os.RemoteException;
Jeff Brown3edf5272014-08-14 19:25:14 -070027import android.os.Trace;
Ruben Brunk49506e02016-04-18 18:10:47 -070028import android.os.UserHandle;
29import android.provider.Settings;
Ruben Brunkc7be3be2016-04-01 17:07:51 -070030import android.service.vr.IVrManager;
31import android.service.vr.IVrStateCallbacks;
Joe Onorato8a9b2202010-02-26 18:56:32 -080032import android.util.Slog;
Mike Lockwood3a322132009-11-24 00:30:52 -050033
Adam Lesinski182f73f2013-12-05 16:48:06 -080034public class LightsService extends SystemService {
35 static final String TAG = "LightsService";
36 static final boolean DEBUG = false;
Mike Lockwood3a322132009-11-24 00:30:52 -050037
Adam Lesinski182f73f2013-12-05 16:48:06 -080038 final LightImpl mLights[] = new LightImpl[LightsManager.LIGHT_ID_COUNT];
Ruben Brunk49506e02016-04-18 18:10:47 -070039 private boolean mVrModeEnabled;
Mike Lockwood3a322132009-11-24 00:30:52 -050040
Adam Lesinski182f73f2013-12-05 16:48:06 -080041 private final class LightImpl extends Light {
Mike Lockwood3a322132009-11-24 00:30:52 -050042
Adam Lesinski182f73f2013-12-05 16:48:06 -080043 private LightImpl(int id) {
Mike Lockwood3cb67a32009-11-27 14:25:58 -050044 mId = id;
45 }
46
Adam Lesinski182f73f2013-12-05 16:48:06 -080047 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050048 public void setBrightness(int brightness) {
49 setBrightness(brightness, BRIGHTNESS_MODE_USER);
50 }
51
Adam Lesinski182f73f2013-12-05 16:48:06 -080052 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050053 public void setBrightness(int brightness, int brightnessMode) {
54 synchronized (this) {
55 int color = brightness & 0x000000ff;
56 color = 0xff000000 | (color << 16) | (color << 8) | color;
57 setLightLocked(color, LIGHT_FLASH_NONE, 0, 0, brightnessMode);
58 }
59 }
60
Adam Lesinski182f73f2013-12-05 16:48:06 -080061 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050062 public void setColor(int color) {
63 synchronized (this) {
64 setLightLocked(color, LIGHT_FLASH_NONE, 0, 0, 0);
65 }
66 }
67
Adam Lesinski182f73f2013-12-05 16:48:06 -080068 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050069 public void setFlashing(int color, int mode, int onMS, int offMS) {
70 synchronized (this) {
71 setLightLocked(color, mode, onMS, offMS, BRIGHTNESS_MODE_USER);
72 }
73 }
74
Adam Lesinski182f73f2013-12-05 16:48:06 -080075 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050076 public void pulse() {
Mike Lockwood670f9322010-01-20 12:13:36 -050077 pulse(0x00ffffff, 7);
78 }
79
Adam Lesinski182f73f2013-12-05 16:48:06 -080080 @Override
Mike Lockwood670f9322010-01-20 12:13:36 -050081 public void pulse(int color, int onMS) {
Mike Lockwood3cb67a32009-11-27 14:25:58 -050082 synchronized (this) {
Ruben Brunk8ad3f202015-12-28 14:45:11 -080083 if (mBrightnessMode == BRIGHTNESS_MODE_LOW_PERSISTENCE) {
84 return;
85 }
Mike Lockwood3cb67a32009-11-27 14:25:58 -050086 if (mColor == 0 && !mFlashing) {
Mike Lockwood670f9322010-01-20 12:13:36 -050087 setLightLocked(color, LIGHT_FLASH_HARDWARE, onMS, 1000, BRIGHTNESS_MODE_USER);
Oskar Anderoe8467192013-12-03 17:41:27 +010088 mColor = 0;
Mike Lockwood670f9322010-01-20 12:13:36 -050089 mH.sendMessageDelayed(Message.obtain(mH, 1, this), onMS);
Mike Lockwood3cb67a32009-11-27 14:25:58 -050090 }
91 }
92 }
93
Adam Lesinski182f73f2013-12-05 16:48:06 -080094 @Override
Mike Lockwood3cb67a32009-11-27 14:25:58 -050095 public void turnOff() {
96 synchronized (this) {
97 setLightLocked(0, LIGHT_FLASH_NONE, 0, 0, 0);
98 }
99 }
100
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800101 void enableLowPersistence() {
102 synchronized(this) {
103 setLightLocked(0, LIGHT_FLASH_NONE, 0, 0, BRIGHTNESS_MODE_LOW_PERSISTENCE);
104 mLocked = true;
105 }
106 }
107
108 void disableLowPersistence() {
109 synchronized(this) {
110 mLocked = false;
111 setLightLocked(mLastColor, LIGHT_FLASH_NONE, 0, 0, mLastBrightnessMode);
112 }
113 }
114
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500115 private void stopFlashing() {
116 synchronized (this) {
117 setLightLocked(mColor, LIGHT_FLASH_NONE, 0, 0, BRIGHTNESS_MODE_USER);
118 }
119 }
120
121 private void setLightLocked(int color, int mode, int onMS, int offMS, int brightnessMode) {
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800122 if (!mLocked && (color != mColor || mode != mMode || onMS != mOnMS || offMS != mOffMS ||
123 mBrightnessMode != brightnessMode)) {
Dianne Hackbornff801ec2011-01-22 18:05:38 -0800124 if (DEBUG) Slog.v(TAG, "setLight #" + mId + ": color=#"
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800125 + Integer.toHexString(color) + ": brightnessMode=" + brightnessMode);
126 mLastColor = mColor;
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500127 mColor = color;
128 mMode = mode;
129 mOnMS = onMS;
130 mOffMS = offMS;
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800131 mLastBrightnessMode = mBrightnessMode;
132 mBrightnessMode = brightnessMode;
Jeff Brownfaec22c82015-04-10 12:58:52 -0700133 Trace.traceBegin(Trace.TRACE_TAG_POWER, "setLight(" + mId + ", 0x"
134 + Integer.toHexString(color) + ")");
Jeff Brown3edf5272014-08-14 19:25:14 -0700135 try {
136 setLight_native(mNativePointer, mId, color, mode, onMS, offMS, brightnessMode);
137 } finally {
138 Trace.traceEnd(Trace.TRACE_TAG_POWER);
139 }
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500140 }
141 }
142
143 private int mId;
144 private int mColor;
145 private int mMode;
146 private int mOnMS;
147 private int mOffMS;
148 private boolean mFlashing;
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800149 private int mBrightnessMode;
150 private int mLastBrightnessMode;
151 private int mLastColor;
152 private boolean mLocked;
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500153 }
Mike Lockwood3a322132009-11-24 00:30:52 -0500154
Jeff Brownb880d882014-02-10 19:47:07 -0800155 public LightsService(Context context) {
156 super(context);
157
Mike Lockwood3a322132009-11-24 00:30:52 -0500158 mNativePointer = init_native();
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500159
Adam Lesinski182f73f2013-12-05 16:48:06 -0800160 for (int i = 0; i < LightsManager.LIGHT_ID_COUNT; i++) {
161 mLights[i] = new LightImpl(i);
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500162 }
Mike Lockwood3a322132009-11-24 00:30:52 -0500163 }
164
Adam Lesinski182f73f2013-12-05 16:48:06 -0800165 @Override
166 public void onStart() {
Adam Lesinski182f73f2013-12-05 16:48:06 -0800167 publishLocalService(LightsManager.class, mService);
168 }
169
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800170 @Override
171 public void onBootPhase(int phase) {
172 if (phase == PHASE_SYSTEM_SERVICES_READY) {
Ruben Brunkc7be3be2016-04-01 17:07:51 -0700173 IVrManager vrManager =
174 (IVrManager) getBinderService(VrManagerService.VR_MANAGER_BINDER_SERVICE);
175 try {
176 vrManager.registerListener(mVrStateCallbacks);
177 } catch (RemoteException e) {
178 Slog.e(TAG, "Failed to register VR mode state listener: " + e);
179 }
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800180 }
181 }
182
Ruben Brunk49506e02016-04-18 18:10:47 -0700183 private int getVrDisplayMode() {
184 int currentUser = ActivityManager.getCurrentUser();
185 return Settings.Secure.getIntForUser(getContext().getContentResolver(),
186 Settings.Secure.VR_DISPLAY_MODE,
187 /*default*/Settings.Secure.VR_DISPLAY_MODE_LOW_PERSISTENCE,
188 currentUser);
189 }
190
Ruben Brunkc7be3be2016-04-01 17:07:51 -0700191 private final IVrStateCallbacks mVrStateCallbacks = new IVrStateCallbacks.Stub() {
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800192 @Override
Ruben Brunkc7be3be2016-04-01 17:07:51 -0700193 public void onVrStateChanged(boolean enabled) throws RemoteException {
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800194 LightImpl l = mLights[LightsManager.LIGHT_ID_BACKLIGHT];
Ruben Brunk49506e02016-04-18 18:10:47 -0700195 int vrDisplayMode = getVrDisplayMode();
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800196
Ruben Brunk49506e02016-04-18 18:10:47 -0700197 // User leaves VR mode before altering display settings.
198 if (enabled && vrDisplayMode == Settings.Secure.VR_DISPLAY_MODE_LOW_PERSISTENCE) {
199 if (!mVrModeEnabled) {
200 if (DEBUG)
201 Slog.v(TAG, "VR mode enabled, setting brightness to low persistence");
202 l.enableLowPersistence();
203 mVrModeEnabled = true;
204 }
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800205 } else {
Ruben Brunk49506e02016-04-18 18:10:47 -0700206 if (mVrModeEnabled) {
207 if (DEBUG) Slog.v(TAG, "VR mode disabled, resetting brightnes");
208 l.disableLowPersistence();
209 mVrModeEnabled = false;
210 }
Ruben Brunk8ad3f202015-12-28 14:45:11 -0800211 }
212 }
213 };
214
Adam Lesinski182f73f2013-12-05 16:48:06 -0800215 private final LightsManager mService = new LightsManager() {
216 @Override
Jeff Brown5d6443b2015-04-10 20:15:01 -0700217 public Light getLight(int id) {
Adam Lesinski182f73f2013-12-05 16:48:06 -0800218 if (id < LIGHT_ID_COUNT) {
219 return mLights[id];
220 } else {
221 return null;
222 }
223 }
224 };
225
Jeff Brownb880d882014-02-10 19:47:07 -0800226 @Override
Mike Lockwood3a322132009-11-24 00:30:52 -0500227 protected void finalize() throws Throwable {
228 finalize_native(mNativePointer);
229 super.finalize();
230 }
231
Mike Lockwood3a322132009-11-24 00:30:52 -0500232 private Handler mH = new Handler() {
233 @Override
234 public void handleMessage(Message msg) {
Adam Lesinski182f73f2013-12-05 16:48:06 -0800235 LightImpl light = (LightImpl)msg.obj;
Mike Lockwood3cb67a32009-11-27 14:25:58 -0500236 light.stopFlashing();
Mike Lockwood3a322132009-11-24 00:30:52 -0500237 }
238 };
239
Ashok Bhat7e2a9dc2014-01-02 19:43:30 +0000240 private static native long init_native();
241 private static native void finalize_native(long ptr);
Mike Lockwood3a322132009-11-24 00:30:52 -0500242
Narayan Kamath3f7b8d02014-01-08 12:19:40 +0000243 static native void setLight_native(long ptr, int light, int color, int mode,
Mike Lockwood3a322132009-11-24 00:30:52 -0500244 int onMS, int offMS, int brightnessMode);
245
Ashok Bhat7e2a9dc2014-01-02 19:43:30 +0000246 private long mNativePointer;
Mike Lockwood3a322132009-11-24 00:30:52 -0500247}