blob: 425815e2fffd2eade2c3b2636ddd892f6df0ea16 [file] [log] [blame]
Austin Schuh812d0d12021-11-04 20:16:48 -07001// Copyright (c) FIRST and other WPILib contributors.
2// Open Source Software; you can modify and/or share it under the terms of
3// the WPILib BSD license file in the root directory of this project.
Brian Silverman8fce7482020-01-05 13:18:21 -08004
5#include "frc/ADXL362.h"
6
7#include <hal/FRCUsageReporting.h>
James Kuszmaulcf324122023-01-14 14:07:17 -08008#include <networktables/DoubleTopic.h>
Austin Schuh812d0d12021-11-04 20:16:48 -07009#include <networktables/NTSendableBuilder.h>
10#include <wpi/sendable/SendableRegistry.h>
Brian Silverman8fce7482020-01-05 13:18:21 -080011
Austin Schuh812d0d12021-11-04 20:16:48 -070012#include "frc/Errors.h"
Brian Silverman8fce7482020-01-05 13:18:21 -080013
14using namespace frc;
15
16static constexpr int kRegWrite = 0x0A;
17static constexpr int kRegRead = 0x0B;
18
19static constexpr int kPartIdRegister = 0x02;
20static constexpr int kDataRegister = 0x0E;
21static constexpr int kFilterCtlRegister = 0x2C;
22static constexpr int kPowerCtlRegister = 0x2D;
23
24// static constexpr int kFilterCtl_Range2G = 0x00;
25// static constexpr int kFilterCtl_Range4G = 0x40;
26// static constexpr int kFilterCtl_Range8G = 0x80;
27static constexpr int kFilterCtl_ODR_100Hz = 0x03;
28
29static constexpr int kPowerCtl_UltraLowNoise = 0x20;
30// static constexpr int kPowerCtl_AutoSleep = 0x04;
31static constexpr int kPowerCtl_Measure = 0x02;
32
33ADXL362::ADXL362(Range range) : ADXL362(SPI::Port::kOnboardCS1, range) {}
34
35ADXL362::ADXL362(SPI::Port port, Range range)
Austin Schuh812d0d12021-11-04 20:16:48 -070036 : m_spi(port), m_simDevice("Accel:ADXL362", port) {
Brian Silverman8fce7482020-01-05 13:18:21 -080037 if (m_simDevice) {
Austin Schuh812d0d12021-11-04 20:16:48 -070038 m_simRange = m_simDevice.CreateEnumDouble("range", hal::SimDevice::kOutput,
39 {"2G", "4G", "8G", "16G"},
40 {2.0, 4.0, 8.0, 16.0}, 0);
41 m_simX = m_simDevice.CreateDouble("x", hal::SimDevice::kInput, 0.0);
42 m_simY = m_simDevice.CreateDouble("y", hal::SimDevice::kInput, 0.0);
43 m_simZ = m_simDevice.CreateDouble("z", hal::SimDevice::kInput, 0.0);
Brian Silverman8fce7482020-01-05 13:18:21 -080044 }
45
46 m_spi.SetClockRate(3000000);
James Kuszmaulcf324122023-01-14 14:07:17 -080047 m_spi.SetMode(frc::SPI::Mode::kMode3);
Brian Silverman8fce7482020-01-05 13:18:21 -080048 m_spi.SetChipSelectActiveLow();
49
50 uint8_t commands[3];
51 if (!m_simDevice) {
52 // Validate the part ID
53 commands[0] = kRegRead;
54 commands[1] = kPartIdRegister;
55 commands[2] = 0;
56 m_spi.Transaction(commands, commands, 3);
57 if (commands[2] != 0xF2) {
James Kuszmaulcf324122023-01-14 14:07:17 -080058 FRC_ReportError(err::Error, "could not find ADXL362");
Brian Silverman8fce7482020-01-05 13:18:21 -080059 m_gsPerLSB = 0.0;
60 return;
61 }
62 }
63
64 SetRange(range);
65
66 // Turn on the measurements
67 commands[0] = kRegWrite;
68 commands[1] = kPowerCtlRegister;
69 commands[2] = kPowerCtl_Measure | kPowerCtl_UltraLowNoise;
70 m_spi.Write(commands, 3);
71
72 HAL_Report(HALUsageReporting::kResourceType_ADXL362, port + 1);
73
Austin Schuh812d0d12021-11-04 20:16:48 -070074 wpi::SendableRegistry::AddLW(this, "ADXL362", port);
75}
76
77SPI::Port ADXL362::GetSpiPort() const {
78 return m_spi.GetPort();
Brian Silverman8fce7482020-01-05 13:18:21 -080079}
80
81void ADXL362::SetRange(Range range) {
Austin Schuh812d0d12021-11-04 20:16:48 -070082 if (m_gsPerLSB == 0.0) {
83 return;
84 }
Brian Silverman8fce7482020-01-05 13:18:21 -080085
86 uint8_t commands[3];
87
88 switch (range) {
89 case kRange_2G:
90 m_gsPerLSB = 0.001;
91 break;
92 case kRange_4G:
93 m_gsPerLSB = 0.002;
94 break;
95 case kRange_8G:
Brian Silverman8fce7482020-01-05 13:18:21 -080096 m_gsPerLSB = 0.004;
97 break;
98 }
99
100 // Specify the data format to read
101 commands[0] = kRegWrite;
102 commands[1] = kFilterCtlRegister;
103 commands[2] =
104 kFilterCtl_ODR_100Hz | static_cast<uint8_t>((range & 0x03) << 6);
105 m_spi.Write(commands, 3);
106
Austin Schuh812d0d12021-11-04 20:16:48 -0700107 if (m_simRange) {
108 m_simRange.Set(range);
109 }
Brian Silverman8fce7482020-01-05 13:18:21 -0800110}
111
Austin Schuh812d0d12021-11-04 20:16:48 -0700112double ADXL362::GetX() {
113 return GetAcceleration(kAxis_X);
114}
Brian Silverman8fce7482020-01-05 13:18:21 -0800115
Austin Schuh812d0d12021-11-04 20:16:48 -0700116double ADXL362::GetY() {
117 return GetAcceleration(kAxis_Y);
118}
Brian Silverman8fce7482020-01-05 13:18:21 -0800119
Austin Schuh812d0d12021-11-04 20:16:48 -0700120double ADXL362::GetZ() {
121 return GetAcceleration(kAxis_Z);
122}
Brian Silverman8fce7482020-01-05 13:18:21 -0800123
124double ADXL362::GetAcceleration(ADXL362::Axes axis) {
Austin Schuh812d0d12021-11-04 20:16:48 -0700125 if (m_gsPerLSB == 0.0) {
126 return 0.0;
127 }
Brian Silverman8fce7482020-01-05 13:18:21 -0800128
Austin Schuh812d0d12021-11-04 20:16:48 -0700129 if (axis == kAxis_X && m_simX) {
130 return m_simX.Get();
131 }
132 if (axis == kAxis_Y && m_simY) {
133 return m_simY.Get();
134 }
135 if (axis == kAxis_Z && m_simZ) {
136 return m_simZ.Get();
137 }
Brian Silverman8fce7482020-01-05 13:18:21 -0800138
139 uint8_t buffer[4];
140 uint8_t command[4] = {0, 0, 0, 0};
141 command[0] = kRegRead;
142 command[1] = kDataRegister + static_cast<uint8_t>(axis);
143 m_spi.Transaction(command, buffer, 4);
144
145 // Sensor is little endian... swap bytes
146 int16_t rawAccel = buffer[3] << 8 | buffer[2];
147 return rawAccel * m_gsPerLSB;
148}
149
150ADXL362::AllAxes ADXL362::GetAccelerations() {
151 AllAxes data;
152 if (m_gsPerLSB == 0.0) {
153 data.XAxis = data.YAxis = data.ZAxis = 0.0;
154 return data;
155 }
156 if (m_simX && m_simY && m_simZ) {
157 data.XAxis = m_simX.Get();
158 data.YAxis = m_simY.Get();
159 data.ZAxis = m_simZ.Get();
160 return data;
161 }
162
163 uint8_t dataBuffer[8] = {0, 0, 0, 0, 0, 0, 0, 0};
164 int16_t rawData[3];
165
166 // Select the data address.
167 dataBuffer[0] = kRegRead;
168 dataBuffer[1] = kDataRegister;
169 m_spi.Transaction(dataBuffer, dataBuffer, 8);
170
171 for (int i = 0; i < 3; i++) {
172 // Sensor is little endian... swap bytes
173 rawData[i] = dataBuffer[i * 2 + 3] << 8 | dataBuffer[i * 2 + 2];
174 }
175
176 data.XAxis = rawData[0] * m_gsPerLSB;
177 data.YAxis = rawData[1] * m_gsPerLSB;
178 data.ZAxis = rawData[2] * m_gsPerLSB;
179
180 return data;
181}
182
Austin Schuh812d0d12021-11-04 20:16:48 -0700183void ADXL362::InitSendable(nt::NTSendableBuilder& builder) {
Brian Silverman8fce7482020-01-05 13:18:21 -0800184 builder.SetSmartDashboardType("3AxisAccelerometer");
James Kuszmaulcf324122023-01-14 14:07:17 -0800185 builder.SetUpdateTable(
186 [this, x = nt::DoubleTopic{builder.GetTopic("X")}.Publish(),
187 y = nt::DoubleTopic{builder.GetTopic("Y")}.Publish(),
188 z = nt::DoubleTopic{builder.GetTopic("Z")}.Publish()]() mutable {
189 auto data = GetAccelerations();
190 x.Set(data.XAxis);
191 y.Set(data.YAxis);
192 z.Set(data.ZAxis);
193 });
Brian Silverman8fce7482020-01-05 13:18:21 -0800194}