Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2010 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 | |
| 17 | #ifndef ANDROID_SENSOR_SERVICE_H |
| 18 | #define ANDROID_SENSOR_SERVICE_H |
| 19 | |
| 20 | #include <stdint.h> |
| 21 | #include <sys/types.h> |
| 22 | |
| 23 | #include <utils/Vector.h> |
| 24 | #include <utils/SortedVector.h> |
| 25 | #include <utils/KeyedVector.h> |
| 26 | #include <utils/threads.h> |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 27 | #include <utils/AndroidThreads.h> |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 28 | #include <utils/RefBase.h> |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 29 | #include <utils/Looper.h> |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 30 | |
| 31 | #include <binder/BinderService.h> |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 32 | |
| 33 | #include <gui/Sensor.h> |
Mathias Agopian | b398927 | 2011-10-20 18:42:02 -0700 | [diff] [blame] | 34 | #include <gui/BitTube.h> |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 35 | #include <gui/ISensorServer.h> |
| 36 | #include <gui/ISensorEventConnection.h> |
| 37 | |
Mathias Agopian | f001c92 | 2010-11-11 17:58:51 -0800 | [diff] [blame] | 38 | #include "SensorInterface.h" |
| 39 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 40 | // --------------------------------------------------------------------------- |
| 41 | |
Mathias Agopian | a1b7db9 | 2011-05-27 16:23:58 -0700 | [diff] [blame] | 42 | #define DEBUG_CONNECTIONS false |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 43 | // Max size is 100 KB which is enough to accept a batch of about 1000 events. |
| 44 | #define MAX_SOCKET_BUFFER_SIZE_BATCHED 100 * 1024 |
| 45 | // For older HALs which don't support batching, use a smaller socket buffer size. |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 46 | #define SOCKET_BUFFER_SIZE_NON_BATCHED 4 * 1024 |
Mathias Agopian | a1b7db9 | 2011-05-27 16:23:58 -0700 | [diff] [blame] | 47 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 48 | struct sensors_poll_device_t; |
| 49 | struct sensors_module_t; |
| 50 | |
| 51 | namespace android { |
| 52 | // --------------------------------------------------------------------------- |
| 53 | |
| 54 | class SensorService : |
| 55 | public BinderService<SensorService>, |
| 56 | public BnSensorServer, |
| 57 | protected Thread |
| 58 | { |
Mathias Agopian | b6df7d0 | 2013-05-09 14:53:35 -0700 | [diff] [blame] | 59 | friend class BinderService<SensorService>; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 60 | |
Mathias Agopian | b6df7d0 | 2013-05-09 14:53:35 -0700 | [diff] [blame] | 61 | static const char* WAKE_LOCK_NAME; |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 62 | |
Mathias Agopian | b6df7d0 | 2013-05-09 14:53:35 -0700 | [diff] [blame] | 63 | static char const* getServiceName() ANDROID_API { return "sensorservice"; } |
| 64 | SensorService() ANDROID_API; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 65 | virtual ~SensorService(); |
| 66 | |
| 67 | virtual void onFirstRef(); |
| 68 | |
| 69 | // Thread interface |
| 70 | virtual bool threadLoop(); |
| 71 | |
| 72 | // ISensorServer interface |
| 73 | virtual Vector<Sensor> getSensorList(); |
| 74 | virtual sp<ISensorEventConnection> createSensorEventConnection(); |
| 75 | virtual status_t dump(int fd, const Vector<String16>& args); |
| 76 | |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 77 | class SensorEventConnection : public BnSensorEventConnection, public LooperCallback { |
| 78 | friend class SensorService; |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 79 | virtual ~SensorEventConnection(); |
| 80 | virtual void onFirstRef(); |
Mathias Agopian | b398927 | 2011-10-20 18:42:02 -0700 | [diff] [blame] | 81 | virtual sp<BitTube> getSensorChannel() const; |
Aravind Akella | 724d91d | 2013-06-27 12:04:23 -0700 | [diff] [blame] | 82 | virtual status_t enableDisable(int handle, bool enabled, nsecs_t samplingPeriodNs, |
| 83 | nsecs_t maxBatchReportLatencyNs, int reservedFlags); |
| 84 | virtual status_t setEventRate(int handle, nsecs_t samplingPeriodNs); |
Aravind Akella | 701166d | 2013-10-08 14:59:26 -0700 | [diff] [blame] | 85 | virtual status_t flush(); |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 86 | // Count the number of flush complete events which are about to be dropped in the buffer. |
| 87 | // Increment mPendingFlushEventsToSend in mSensorInfo. These flush complete events will be |
| 88 | // sent separately before the next batch of events. |
Aravind Akella | c551eac | 2013-10-14 17:04:42 -0700 | [diff] [blame] | 89 | void countFlushCompleteEventsLocked(sensors_event_t* scratch, int numEventsDropped); |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 90 | |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 91 | // Check if there are any wake up events in the buffer. If yes, return the index of the |
| 92 | // first wake_up sensor event in the buffer else return -1. This wake_up sensor event will |
| 93 | // have the flag WAKE_UP_SENSOR_EVENT_NEEDS_ACK set. Exactly one event per packet will have |
| 94 | // the wake_up flag set. SOCK_SEQPACKET ensures that either the entire packet is read or |
| 95 | // dropped. |
| 96 | int findWakeUpSensorEventLocked(sensors_event_t const* scratch, int count); |
| 97 | |
| 98 | // Send pending flush_complete events. There may have been flush_complete_events that are |
| 99 | // dropped which need to be sent separately before other events. On older HALs (1_0) this |
| 100 | // method emulates the behavior of flush(). |
| 101 | void sendPendingFlushEventsLocked(); |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 102 | |
| 103 | // Writes events from mEventCache to the socket. |
| 104 | void writeToSocketFromCacheLocked(); |
| 105 | |
| 106 | // Compute the approximate cache size from the FIFO sizes of various sensors registered for |
| 107 | // this connection. Wake up and non-wake up sensors have separate FIFOs but FIFO may be |
| 108 | // shared amongst wake-up sensors and non-wake up sensors. |
| 109 | int computeMaxCacheSizeLocked() const; |
| 110 | |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 111 | // When more sensors register, the maximum cache size desired may change. Compute max cache |
| 112 | // size, reallocate memory and copy over events from the older cache. |
| 113 | void reAllocateCacheLocked(sensors_event_t const* scratch, int count); |
| 114 | |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 115 | // LooperCallback method. If there is data to read on this fd, it is an ack from the |
| 116 | // app that it has read events from a wake up sensor, decrement mWakeLockRefCount. |
| 117 | // If this fd is available for writing send the data from the cache. |
| 118 | virtual int handleEvent(int fd, int events, void* data); |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 119 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 120 | sp<SensorService> const mService; |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 121 | sp<BitTube> mChannel; |
Mathias Agopian | 5307d17 | 2012-09-18 17:02:43 -0700 | [diff] [blame] | 122 | uid_t mUid; |
Mathias Agopian | 71d7a5c | 2010-11-14 20:55:25 -0800 | [diff] [blame] | 123 | mutable Mutex mConnectionLock; |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 124 | // Number of events from wake up sensors which are still pending and haven't been delivered |
| 125 | // to the corresponding application. It is incremented by one unit for each write to the |
| 126 | // socket. |
| 127 | int mWakeLockRefCount; |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 128 | |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 129 | struct FlushInfo { |
| 130 | // The number of flush complete events dropped for this sensor is stored here. |
| 131 | // They are sent separately before the next batch of events. |
| 132 | int mPendingFlushEventsToSend; |
| 133 | // Every activate is preceded by a flush. Only after the first flush complete is |
| 134 | // received, the events for the sensor are sent on that *connection*. |
| 135 | bool mFirstFlushPending; |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 136 | FlushInfo() : mPendingFlushEventsToSend(0), mFirstFlushPending(false) {} |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 137 | }; |
| 138 | // protected by SensorService::mLock. Key for this vector is the sensor handle. |
| 139 | KeyedVector<int, FlushInfo> mSensorInfo; |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 140 | sensors_event_t *mEventCache; |
| 141 | int mCacheSize, mMaxCacheSize; |
| 142 | |
| 143 | #if DEBUG_CONNECTIONS |
| 144 | int mEventsReceived, mEventsSent, mEventsSentFromCache; |
Aravind Akella | e74baf6 | 2014-08-21 12:28:35 -0700 | [diff] [blame^] | 145 | int mTotalAcksNeeded, mTotalAcksReceived; |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 146 | #endif |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 147 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 148 | public: |
Mathias Agopian | 5307d17 | 2012-09-18 17:02:43 -0700 | [diff] [blame] | 149 | SensorEventConnection(const sp<SensorService>& service, uid_t uid); |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 150 | |
Mathias Agopian | cf51001 | 2010-07-22 16:18:10 -0700 | [diff] [blame] | 151 | status_t sendEvents(sensors_event_t const* buffer, size_t count, |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 152 | sensors_event_t* scratch); |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 153 | bool hasSensor(int32_t handle) const; |
| 154 | bool hasAnySensor() const; |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 155 | bool addSensor(int32_t handle); |
| 156 | bool removeSensor(int32_t handle); |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 157 | void setFirstFlushPending(int32_t handle, bool value); |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 158 | void dump(String8& result); |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 159 | bool needsWakeLock(); |
Mathias Agopian | 5307d17 | 2012-09-18 17:02:43 -0700 | [diff] [blame] | 160 | |
| 161 | uid_t getUid() const { return mUid; } |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 162 | }; |
| 163 | |
| 164 | class SensorRecord { |
| 165 | SortedVector< wp<SensorEventConnection> > mConnections; |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 166 | // A queue of all flush() calls made on this sensor. Flush complete events will be |
| 167 | // sent in this order. |
| 168 | Vector< wp<SensorEventConnection> > mPendingFlushConnections; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 169 | public: |
| 170 | SensorRecord(const sp<SensorEventConnection>& connection); |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 171 | bool addConnection(const sp<SensorEventConnection>& connection); |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 172 | bool removeConnection(const wp<SensorEventConnection>& connection); |
| 173 | size_t getNumConnections() const { return mConnections.size(); } |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 174 | |
| 175 | void addPendingFlushConnection(const sp<SensorEventConnection>& connection); |
| 176 | void removeFirstPendingFlushConnection(); |
| 177 | SensorEventConnection * getFirstPendingFlushConnection(); |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 178 | }; |
| 179 | |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 180 | class SensorEventAckReceiver : public Thread { |
| 181 | sp<SensorService> const mService; |
| 182 | public: |
| 183 | virtual bool threadLoop(); |
| 184 | SensorEventAckReceiver(const sp<SensorService>& service): mService(service) {} |
| 185 | }; |
| 186 | |
Mathias Agopian | f001c92 | 2010-11-11 17:58:51 -0800 | [diff] [blame] | 187 | String8 getSensorName(int handle) const; |
Aravind Akella | b4099e7 | 2013-10-15 15:43:10 -0700 | [diff] [blame] | 188 | bool isVirtualSensor(int handle) const; |
Aravind Akella | 7001804 | 2014-04-07 22:52:37 +0000 | [diff] [blame] | 189 | Sensor getSensorFromHandle(int handle) const; |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 190 | bool isWakeUpSensor(int type) const; |
| 191 | void recordLastValueLocked(const sensors_event_t* buffer, size_t count); |
Mathias Agopian | f001c92 | 2010-11-11 17:58:51 -0800 | [diff] [blame] | 192 | static void sortEventBuffer(sensors_event_t* buffer, size_t count); |
Mathias Agopian | 0319306 | 2013-05-10 19:32:39 -0700 | [diff] [blame] | 193 | Sensor registerSensor(SensorInterface* sensor); |
| 194 | Sensor registerVirtualSensor(SensorInterface* sensor); |
Mathias Agopian | ac9a96d | 2013-07-12 02:01:16 -0700 | [diff] [blame] | 195 | status_t cleanupWithoutDisable( |
| 196 | const sp<SensorEventConnection>& connection, int handle); |
| 197 | status_t cleanupWithoutDisableLocked( |
| 198 | const sp<SensorEventConnection>& connection, int handle); |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 199 | void cleanupAutoDisabledSensorLocked(const sp<SensorEventConnection>& connection, |
Mathias Agopian | b6df7d0 | 2013-05-09 14:53:35 -0700 | [diff] [blame] | 200 | sensors_event_t const* buffer, const int count); |
Aravind Akella | 7001804 | 2014-04-07 22:52:37 +0000 | [diff] [blame] | 201 | static bool canAccessSensor(const Sensor& sensor); |
| 202 | static bool verifyCanAccessSensor(const Sensor& sensor, const char* operation); |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 203 | // SensorService acquires a partial wakelock for delivering events from wake up sensors. This |
| 204 | // method checks whether all the events from these wake up sensors have been delivered to the |
| 205 | // corresponding applications, if yes the wakelock is released. |
| 206 | void checkWakeLockState(); |
| 207 | void checkWakeLockStateLocked(); |
| 208 | bool isWakeUpSensorEvent(const sensors_event_t& event) const; |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 209 | |
Aravind Akella | 6c2664a | 2014-08-13 12:24:50 -0700 | [diff] [blame] | 210 | SensorRecord * getSensorRecord(int handle); |
| 211 | |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 212 | sp<Looper> getLooper() const; |
| 213 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 214 | // constants |
| 215 | Vector<Sensor> mSensorList; |
Mathias Agopian | 3326486 | 2012-06-28 19:46:54 -0700 | [diff] [blame] | 216 | Vector<Sensor> mUserSensorListDebug; |
Mathias Agopian | 010e422 | 2011-06-08 20:06:50 -0700 | [diff] [blame] | 217 | Vector<Sensor> mUserSensorList; |
Mathias Agopian | f001c92 | 2010-11-11 17:58:51 -0800 | [diff] [blame] | 218 | DefaultKeyedVector<int, SensorInterface*> mSensorMap; |
| 219 | Vector<SensorInterface *> mVirtualSensorList; |
Mathias Agopian | 50df295 | 2010-07-19 19:09:10 -0700 | [diff] [blame] | 220 | status_t mInitCheck; |
Aravind Akella | 4c8b951 | 2013-09-05 17:03:38 -0700 | [diff] [blame] | 221 | size_t mSocketBufferSize; |
Aravind Akella | 56ae426 | 2014-07-10 16:01:10 -0700 | [diff] [blame] | 222 | sp<Looper> mLooper; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 223 | |
| 224 | // protected by mLock |
| 225 | mutable Mutex mLock; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 226 | DefaultKeyedVector<int, SensorRecord*> mActiveSensors; |
Mathias Agopian | f001c92 | 2010-11-11 17:58:51 -0800 | [diff] [blame] | 227 | DefaultKeyedVector<int, SensorInterface*> mActiveVirtualSensors; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 228 | SortedVector< wp<SensorEventConnection> > mActiveConnections; |
Aravind Akella | 9a844cf | 2014-02-11 18:58:52 -0800 | [diff] [blame] | 229 | bool mWakeLockAcquired; |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 230 | |
Mathias Agopian | 3560fb2 | 2010-07-22 21:24:39 -0700 | [diff] [blame] | 231 | // The size of this vector is constant, only the items are mutable |
| 232 | KeyedVector<int32_t, sensors_event_t> mLastEventSeen; |
| 233 | |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 234 | public: |
Mathias Agopian | db5b4bc | 2011-02-03 14:52:47 -0800 | [diff] [blame] | 235 | void cleanupConnection(SensorEventConnection* connection); |
Aravind Akella | 724d91d | 2013-06-27 12:04:23 -0700 | [diff] [blame] | 236 | status_t enable(const sp<SensorEventConnection>& connection, int handle, |
| 237 | nsecs_t samplingPeriodNs, nsecs_t maxBatchReportLatencyNs, int reservedFlags); |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 238 | status_t disable(const sp<SensorEventConnection>& connection, int handle); |
Mathias Agopian | 7c1c531 | 2010-07-21 15:59:50 -0700 | [diff] [blame] | 239 | status_t setEventRate(const sp<SensorEventConnection>& connection, int handle, nsecs_t ns); |
Aravind Akella | 724d91d | 2013-06-27 12:04:23 -0700 | [diff] [blame] | 240 | status_t flushSensor(const sp<SensorEventConnection>& connection, int handle); |
Mathias Agopian | fc32881 | 2010-07-14 23:41:37 -0700 | [diff] [blame] | 241 | }; |
| 242 | |
| 243 | // --------------------------------------------------------------------------- |
| 244 | }; // namespace android |
| 245 | |
| 246 | #endif // ANDROID_SENSOR_SERVICE_H |