001// Copyright (c) FIRST and other WPILib contributors. 002// Open Source Software; you can modify and/or share it under the terms of 003// the WPILib BSD license file in the root directory of this project. 004 005package edu.wpi.first.wpilibj; 006 007import edu.wpi.first.hal.FRCNetComm.tResourceType; 008import edu.wpi.first.hal.HAL; 009import edu.wpi.first.hal.SimDevice; 010import edu.wpi.first.hal.SimDouble; 011import edu.wpi.first.hal.SimEnum; 012import edu.wpi.first.networktables.DoublePublisher; 013import edu.wpi.first.networktables.DoubleTopic; 014import edu.wpi.first.networktables.NTSendable; 015import edu.wpi.first.networktables.NTSendableBuilder; 016import edu.wpi.first.util.sendable.SendableRegistry; 017import java.nio.ByteBuffer; 018import java.nio.ByteOrder; 019 020/** 021 * ADXL362 SPI Accelerometer. 022 * 023 * <p>This class allows access to an Analog Devices ADXL362 3-axis accelerometer. 024 */ 025public class ADXL362 implements NTSendable, AutoCloseable { 026 private static final byte kRegWrite = 0x0A; 027 private static final byte kRegRead = 0x0B; 028 029 private static final byte kPartIdRegister = 0x02; 030 private static final byte kDataRegister = 0x0E; 031 private static final byte kFilterCtlRegister = 0x2C; 032 private static final byte kPowerCtlRegister = 0x2D; 033 034 private static final byte kFilterCtl_Range2G = 0x00; 035 private static final byte kFilterCtl_Range4G = 0x40; 036 private static final byte kFilterCtl_Range8G = (byte) 0x80; 037 private static final byte kFilterCtl_ODR_100Hz = 0x03; 038 039 private static final byte kPowerCtl_UltraLowNoise = 0x20; 040 041 // @SuppressWarnings("PMD.UnusedPrivateField") 042 // private static final byte kPowerCtl_AutoSleep = 0x04; 043 044 private static final byte kPowerCtl_Measure = 0x02; 045 046 /** Accelerometer range. */ 047 public enum Range { 048 /** 2 Gs max. */ 049 k2G, 050 /** 4 Gs max. */ 051 k4G, 052 /** 8 Gs max. */ 053 k8G 054 } 055 056 /** Accelerometer axes. */ 057 public enum Axes { 058 /** X axis. */ 059 kX((byte) 0x00), 060 /** Y axis. */ 061 kY((byte) 0x02), 062 /** Z axis. */ 063 kZ((byte) 0x04); 064 065 /** Axis value. */ 066 public final byte value; 067 068 Axes(byte value) { 069 this.value = value; 070 } 071 } 072 073 /** Container type for accelerations from all axes. */ 074 @SuppressWarnings("MemberName") 075 public static class AllAxes { 076 /** Acceleration along the X axis in g-forces. */ 077 public double XAxis; 078 079 /** Acceleration along the Y axis in g-forces. */ 080 public double YAxis; 081 082 /** Acceleration along the Z axis in g-forces. */ 083 public double ZAxis; 084 085 /** Default constructor. */ 086 public AllAxes() {} 087 } 088 089 private SPI m_spi; 090 091 private SimDevice m_simDevice; 092 private SimEnum m_simRange; 093 private SimDouble m_simX; 094 private SimDouble m_simY; 095 private SimDouble m_simZ; 096 097 private double m_gsPerLSB; 098 099 /** 100 * Constructor. Uses the onboard CS1. 101 * 102 * @param range The range (+ or -) that the accelerometer will measure. 103 */ 104 public ADXL362(Range range) { 105 this(SPI.Port.kOnboardCS1, range); 106 } 107 108 /** 109 * Constructor. 110 * 111 * @param port The SPI port that the accelerometer is connected to 112 * @param range The range (+ or -) that the accelerometer will measure. 113 */ 114 @SuppressWarnings("this-escape") 115 public ADXL362(SPI.Port port, Range range) { 116 m_spi = new SPI(port); 117 118 // simulation 119 m_simDevice = SimDevice.create("Accel:ADXL362", port.value); 120 if (m_simDevice != null) { 121 m_simRange = 122 m_simDevice.createEnumDouble( 123 "range", 124 SimDevice.Direction.kOutput, 125 new String[] {"2G", "4G", "8G", "16G"}, 126 new double[] {2.0, 4.0, 8.0, 16.0}, 127 0); 128 m_simX = m_simDevice.createDouble("x", SimDevice.Direction.kInput, 0.0); 129 m_simY = m_simDevice.createDouble("y", SimDevice.Direction.kInput, 0.0); 130 m_simZ = m_simDevice.createDouble("z", SimDevice.Direction.kInput, 0.0); 131 } 132 133 m_spi.setClockRate(3000000); 134 m_spi.setMode(SPI.Mode.kMode3); 135 m_spi.setChipSelectActiveLow(); 136 137 ByteBuffer transferBuffer = ByteBuffer.allocate(3); 138 if (m_simDevice == null) { 139 // Validate the part ID 140 transferBuffer.put(0, kRegRead); 141 transferBuffer.put(1, kPartIdRegister); 142 m_spi.transaction(transferBuffer, transferBuffer, 3); 143 if (transferBuffer.get(2) != (byte) 0xF2) { 144 m_spi.close(); 145 m_spi = null; 146 DriverStation.reportError("could not find ADXL362 on SPI port " + port.value, false); 147 return; 148 } 149 } 150 151 setRange(range); 152 153 // Turn on the measurements 154 transferBuffer.put(0, kRegWrite); 155 transferBuffer.put(1, kPowerCtlRegister); 156 transferBuffer.put(2, (byte) (kPowerCtl_Measure | kPowerCtl_UltraLowNoise)); 157 m_spi.write(transferBuffer, 3); 158 159 HAL.report(tResourceType.kResourceType_ADXL362, port.value + 1); 160 SendableRegistry.addLW(this, "ADXL362", port.value); 161 } 162 163 /** 164 * Returns the SPI port. 165 * 166 * @return The SPI port. 167 */ 168 public int getPort() { 169 return m_spi.getPort(); 170 } 171 172 @Override 173 public void close() { 174 SendableRegistry.remove(this); 175 if (m_spi != null) { 176 m_spi.close(); 177 m_spi = null; 178 } 179 if (m_simDevice != null) { 180 m_simDevice.close(); 181 m_simDevice = null; 182 } 183 } 184 185 /** 186 * Set the measuring range of the accelerometer. 187 * 188 * @param range The maximum acceleration, positive or negative, that the accelerometer will 189 * measure. 190 */ 191 public final void setRange(Range range) { 192 if (m_spi == null) { 193 return; 194 } 195 196 final byte value = 197 switch (range) { 198 case k2G -> { 199 m_gsPerLSB = 0.001; 200 yield kFilterCtl_Range2G; 201 } 202 case k4G -> { 203 m_gsPerLSB = 0.002; 204 yield kFilterCtl_Range4G; 205 } 206 case k8G -> { 207 m_gsPerLSB = 0.004; 208 yield kFilterCtl_Range8G; 209 } 210 }; 211 212 // Specify the data format to read 213 byte[] commands = 214 new byte[] {kRegWrite, kFilterCtlRegister, (byte) (kFilterCtl_ODR_100Hz | value)}; 215 m_spi.write(commands, commands.length); 216 217 if (m_simRange != null) { 218 m_simRange.set(value); 219 } 220 } 221 222 /** 223 * Returns the acceleration along the X axis in g-forces. 224 * 225 * @return The acceleration along the X axis in g-forces. 226 */ 227 public double getX() { 228 return getAcceleration(Axes.kX); 229 } 230 231 /** 232 * Returns the acceleration along the Y axis in g-forces. 233 * 234 * @return The acceleration along the Y axis in g-forces. 235 */ 236 public double getY() { 237 return getAcceleration(Axes.kY); 238 } 239 240 /** 241 * Returns the acceleration along the Z axis in g-forces. 242 * 243 * @return The acceleration along the Z axis in g-forces. 244 */ 245 public double getZ() { 246 return getAcceleration(Axes.kZ); 247 } 248 249 /** 250 * Get the acceleration of one axis in Gs. 251 * 252 * @param axis The axis to read from. 253 * @return Acceleration of the ADXL362 in Gs. 254 */ 255 public double getAcceleration(ADXL362.Axes axis) { 256 if (axis == Axes.kX && m_simX != null) { 257 return m_simX.get(); 258 } 259 if (axis == Axes.kY && m_simY != null) { 260 return m_simY.get(); 261 } 262 if (axis == Axes.kZ && m_simZ != null) { 263 return m_simZ.get(); 264 } 265 if (m_spi == null) { 266 return 0.0; 267 } 268 ByteBuffer transferBuffer = ByteBuffer.allocate(4); 269 transferBuffer.put(0, kRegRead); 270 transferBuffer.put(1, (byte) (kDataRegister + axis.value)); 271 m_spi.transaction(transferBuffer, transferBuffer, 4); 272 // Sensor is little endian 273 transferBuffer.order(ByteOrder.LITTLE_ENDIAN); 274 275 return transferBuffer.getShort(2) * m_gsPerLSB; 276 } 277 278 /** 279 * Get the acceleration of all axes in Gs. 280 * 281 * @return An object containing the acceleration measured on each axis of the ADXL362 in Gs. 282 */ 283 public ADXL362.AllAxes getAccelerations() { 284 ADXL362.AllAxes data = new ADXL362.AllAxes(); 285 if (m_simX != null && m_simY != null && m_simZ != null) { 286 data.XAxis = m_simX.get(); 287 data.YAxis = m_simY.get(); 288 data.ZAxis = m_simZ.get(); 289 return data; 290 } 291 if (m_spi != null) { 292 ByteBuffer dataBuffer = ByteBuffer.allocate(8); 293 // Select the data address. 294 dataBuffer.put(0, kRegRead); 295 dataBuffer.put(1, kDataRegister); 296 m_spi.transaction(dataBuffer, dataBuffer, 8); 297 // Sensor is little endian... swap bytes 298 dataBuffer.order(ByteOrder.LITTLE_ENDIAN); 299 300 data.XAxis = dataBuffer.getShort(2) * m_gsPerLSB; 301 data.YAxis = dataBuffer.getShort(4) * m_gsPerLSB; 302 data.ZAxis = dataBuffer.getShort(6) * m_gsPerLSB; 303 } 304 return data; 305 } 306 307 @Override 308 public void initSendable(NTSendableBuilder builder) { 309 builder.setSmartDashboardType("3AxisAccelerometer"); 310 DoublePublisher pubX = new DoubleTopic(builder.getTopic("X")).publish(); 311 DoublePublisher pubY = new DoubleTopic(builder.getTopic("Y")).publish(); 312 DoublePublisher pubZ = new DoubleTopic(builder.getTopic("Z")).publish(); 313 builder.addCloseable(pubX); 314 builder.addCloseable(pubY); 315 builder.addCloseable(pubZ); 316 builder.setUpdateTable( 317 () -> { 318 AllAxes data = getAccelerations(); 319 pubX.set(data.XAxis); 320 pubY.set(data.YAxis); 321 pubZ.set(data.ZAxis); 322 }); 323 } 324}