1 /* 2 * Copyright (C) 2017 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 package android.hardware; 17 18 import android.annotation.IntDef; 19 import android.os.MemoryFile; 20 21 import dalvik.system.CloseGuard; 22 23 import java.io.IOException; 24 import java.lang.annotation.Retention; 25 import java.lang.annotation.RetentionPolicy; 26 import java.nio.channels.Channel; 27 import java.util.concurrent.atomic.AtomicBoolean; 28 29 /** 30 * Class representing a sensor direct channel. Use 31 * {@link SensorManager#createDirectChannel(android.os.MemoryFile)} or 32 * {@link SensorManager#createDirectChannel(android.hardware.HardwareBuffer)} 33 * to obtain an object. The channel object can be then configured 34 * (see {@link #configure(Sensor, int)}) 35 * to start delivery of sensor events into shared memory buffer. 36 */ 37 public final class SensorDirectChannel implements Channel { 38 39 // shared memory types 40 41 /** @hide */ 42 @Retention(RetentionPolicy.SOURCE) 43 @IntDef(flag = true, prefix = { "TYPE_" }, value = { 44 TYPE_MEMORY_FILE, 45 TYPE_HARDWARE_BUFFER 46 }) 47 public @interface MemoryType {} 48 49 /** 50 * Shared memory type ashmem, wrapped in MemoryFile object. 51 * 52 * @see SensorManager#createDirectChannel(MemoryFile) 53 */ 54 public static final int TYPE_MEMORY_FILE = 1; 55 56 /** 57 * Shared memory type wrapped by HardwareBuffer object. 58 * 59 * @see SensorManager#createDirectChannel(HardwareBuffer) 60 */ 61 public static final int TYPE_HARDWARE_BUFFER = 2; 62 63 // sensor rate levels 64 65 /** @hide */ 66 @Retention(RetentionPolicy.SOURCE) 67 @IntDef(prefix = { "RATE_" }, value = { 68 RATE_STOP, 69 RATE_NORMAL, 70 RATE_FAST, 71 RATE_VERY_FAST 72 }) 73 public @interface RateLevel {} 74 75 /** 76 * Sensor stopped (no event output). 77 * 78 * @see #configure(Sensor, int) 79 */ 80 public static final int RATE_STOP = 0; 81 /** 82 * Sensor operates at nominal rate of 50Hz. 83 * 84 * The actual rate is expected to be between 55% to 220% of nominal rate, thus between 27.5Hz to 85 * 110Hz. 86 * 87 * @see #configure(Sensor, int) 88 */ 89 public static final int RATE_NORMAL = 1; //50Hz 90 /** 91 * Sensor operates at nominal rate of 200Hz. 92 * 93 * The actual rate is expected to be between 55% to 220% of nominal rate, thus between 110Hz to 94 * 440Hz. 95 * 96 * @see #configure(Sensor, int) 97 */ 98 public static final int RATE_FAST = 2; // ~200Hz 99 /** 100 * Sensor operates at nominal rate of 800Hz. 101 * 102 * The actual rate is expected to be between 55% to 220% of nominal rate, thus between 440Hz to 103 * 1760Hz. 104 * 105 * @see #configure(Sensor, int) 106 */ 107 public static final int RATE_VERY_FAST = 3; // ~800Hz 108 109 /** 110 * Determine if a channel is still valid. A channel is invalidated after {@link #close()} is 111 * called. 112 * 113 * @return <code>true</code> if channel is valid. 114 */ 115 @Override isOpen()116 public boolean isOpen() { 117 return !mClosed.get(); 118 } 119 120 /** @removed */ 121 @Deprecated isValid()122 public boolean isValid() { 123 return isOpen(); 124 } 125 126 /** 127 * Close sensor direct channel. 128 * 129 * Stop all active sensor in the channel and free sensor system resource related to channel. 130 * Shared memory used for creating the direct channel need to be closed or freed separately. 131 * 132 * @see SensorManager#createDirectChannel(MemoryFile) 133 * @see SensorManager#createDirectChannel(HardwareBuffer) 134 */ 135 @Override close()136 public void close() { 137 if (mClosed.compareAndSet(false, true)) { 138 mCloseGuard.close(); 139 // actual close action 140 mManager.destroyDirectChannel(this); 141 } 142 } 143 144 /** 145 * Configure sensor rate or stop sensor report. 146 * 147 * To start event report of a sensor, or change rate of existing report, call this function with 148 * rateLevel other than {@link android.hardware.SensorDirectChannel#RATE_STOP}. Sensor events 149 * will be added into a queue formed by the shared memory used in creation of direction channel. 150 * Each element of the queue has size of 104 bytes and represents a sensor event. Data 151 * structure of an element (all fields in little-endian): 152 * 153 * <pre> 154 * offset type name 155 * ------------------------------------------------------------------------ 156 * 0x0000 int32_t size (always 104) 157 * 0x0004 int32_t sensor report token 158 * 0x0008 int32_t type (see SensorType) 159 * 0x000C uint32_t atomic counter 160 * 0x0010 int64_t timestamp (see Event) 161 * 0x0018 float[16]/int64_t[8] data (data type depends on sensor type) 162 * 0x0058 int32_t[4] reserved (set to zero) 163 * </pre> 164 * 165 * There are no head or tail pointers. The sequence and frontier of new sensor events is 166 * determined by the atomic counter, which counts from 1 after creation of direct channel and 167 * increments 1 for each new event. Atomic counter will wrap back to 1 after it reaches 168 * UINT32_MAX, skipping value 0 to avoid confusion with uninitialized memory. The writer in 169 * sensor system will wrap around from the start of shared memory region when it reaches the 170 * end. If size of memory region is not a multiple of size of element (104 bytes), the residual 171 * is not used at the end. Function returns a positive sensor report token on success. This 172 * token can be used to differentiate sensor events from multiple sensor of the same type. For 173 * example, if there are two accelerometers in the system A and B, it is guaranteed different 174 * report tokens will be returned when starting sensor A and B. 175 * 176 * To stop a sensor, call this function with rateLevel equal {@link 177 * android.hardware.SensorDirectChannel#RATE_STOP}. If the sensor parameter is left to be null, 178 * this will stop all active sensor report associated with the direct channel specified. 179 * Function return 1 on success or 0 on failure. 180 * 181 * @param sensor A {@link android.hardware.Sensor} object to denote sensor to be operated. 182 * @param rateLevel rate level defined in {@link android.hardware.SensorDirectChannel}. 183 * @return * starting report or changing rate: positive sensor report token on success, 184 * 0 on failure; 185 * * stopping report: 1 on success, 0 on failure. 186 * @throws NullPointerException when channel is null. 187 */ configure(Sensor sensor, @RateLevel int rateLevel)188 public int configure(Sensor sensor, @RateLevel int rateLevel) { 189 return mManager.configureDirectChannelImpl(this, sensor, rateLevel); 190 } 191 192 /** @hide */ SensorDirectChannel(SensorManager manager, int id, int type, long size)193 SensorDirectChannel(SensorManager manager, int id, int type, long size) { 194 mManager = manager; 195 mNativeHandle = id; 196 mType = type; 197 mSize = size; 198 mCloseGuard.open("SensorDirectChannel"); 199 } 200 201 /** @hide */ getNativeHandle()202 int getNativeHandle() { 203 return mNativeHandle; 204 } 205 206 /** 207 * This function encode handle information in {@link android.os.MemoryFile} into a long array to 208 * be passed down to native methods. 209 * 210 * @hide */ encodeData(MemoryFile ashmem)211 static long[] encodeData(MemoryFile ashmem) { 212 int fd; 213 try { 214 fd = ashmem.getFileDescriptor().getInt$(); 215 } catch (IOException e) { 216 fd = -1; 217 } 218 return new long[] { 1 /*numFds*/, 0 /*numInts*/, fd }; 219 } 220 221 @Override finalize()222 protected void finalize() throws Throwable { 223 try { 224 if (mCloseGuard != null) { 225 mCloseGuard.warnIfOpen(); 226 } 227 228 close(); 229 } finally { 230 super.finalize(); 231 } 232 } 233 234 private final AtomicBoolean mClosed = new AtomicBoolean(); 235 private final CloseGuard mCloseGuard = CloseGuard.get(); 236 private final SensorManager mManager; 237 private final int mNativeHandle; 238 private final long mSize; 239 private final int mType; 240 } 241