mirror of
https://github.com/wpilibsuite/allwpilib
synced 2026-06-21 01:01:43 +00:00
This is the changes made by Patrick Plenefisch converting the native code to use CMake and the CMake Maven Plugin, as opposed to the native Maven plugin. This is to allow for compatibility with newer versions of the GCC toolchain. All the cpp sources were moved from maven style directories to cpp style directories for CMake. Change-Id: I67f5e3608948f37c83b0990d232105a3784f8593
300 lines
9.7 KiB
C++
300 lines
9.7 KiB
C++
/*----------------------------------------------------------------------------*/
|
|
/* Copyright (c) FIRST 2008. All Rights Reserved. */
|
|
/* Open Source Software - may be modified and shared by FRC teams. The code */
|
|
/* must be accompanied by the FIRST BSD license file in $(WIND_BASE)/WPILib. */
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
#include "AnalogModule.h"
|
|
#include "Timer.h"
|
|
#include "WPIErrors.h"
|
|
|
|
const long AnalogModule::kTimebase; ///< 40 MHz clock
|
|
const long AnalogModule::kDefaultOversampleBits;
|
|
const long AnalogModule::kDefaultAverageBits;
|
|
constexpr float AnalogModule::kDefaultSampleRate;
|
|
|
|
/**
|
|
* Get an instance of an Analog Module.
|
|
*
|
|
* Singleton analog module creation where a module is allocated on the first use
|
|
* and the same module is returned on subsequent uses.
|
|
*
|
|
* @param moduleNumber The analog module to get (1 or 2).
|
|
* @return A pointer to the AnalogModule.
|
|
*/
|
|
AnalogModule* AnalogModule::GetInstance(uint8_t moduleNumber)
|
|
{
|
|
if (checkAnalogModule(moduleNumber))
|
|
{
|
|
return (AnalogModule*)GetModule(nLoadOut::kModuleType_Analog, moduleNumber);
|
|
}
|
|
|
|
// If this wasn't caught before now, make sure we say what's wrong before we crash
|
|
char buf[64];
|
|
snprintf(buf, 64, "Analog Module %d", moduleNumber);
|
|
wpi_setGlobalWPIErrorWithContext(ModuleIndexOutOfRange, buf);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* Create a new instance of an analog module.
|
|
*
|
|
* Create an instance of the analog module object. Initialize all the parameters
|
|
* to reasonable values on start.
|
|
* Setting a global value on an analog module can be done only once unless subsequent
|
|
* values are set the previously set value.
|
|
* Analog modules are a singleton, so the constructor is never called outside of this class.
|
|
*
|
|
* @param moduleNumber The analog module to create (1 or 2).
|
|
*/
|
|
AnalogModule::AnalogModule(uint8_t moduleNumber)
|
|
: Module(nLoadOut::kModuleType_Analog, moduleNumber)
|
|
, m_moduleNumber(moduleNumber)
|
|
, m_ports()
|
|
{
|
|
for (uint32_t i = 0; i < kAnalogChannels; i++)
|
|
{
|
|
void* port = getPortWithModule(moduleNumber, i+1);
|
|
int32_t status = 0;
|
|
m_ports[i] = initializeAnalogPort(port, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
}
|
|
|
|
AddToSingletonList();
|
|
}
|
|
|
|
/**
|
|
* Destructor for AnalogModule.
|
|
*/
|
|
AnalogModule::~AnalogModule()
|
|
{
|
|
|
|
}
|
|
|
|
/**
|
|
* Set the sample rate on the module.
|
|
*
|
|
* This is a global setting for the module and effects all channels.
|
|
*
|
|
* @param samplesPerSecond The number of samples per channel per second.
|
|
*/
|
|
void AnalogModule::SetSampleRate(float samplesPerSecond)
|
|
{
|
|
int32_t status = 0;
|
|
setAnalogSampleRateWithModule(m_moduleNumber, samplesPerSecond, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
}
|
|
|
|
/**
|
|
* Get the current sample rate on the module.
|
|
*
|
|
* This assumes one entry in the scan list.
|
|
* This is a global setting for the module and effects all channels.
|
|
*
|
|
* @return Sample rate.
|
|
*/
|
|
float AnalogModule::GetSampleRate()
|
|
{
|
|
int32_t status = 0;
|
|
float sampleRate = getAnalogSampleRateWithModule(m_moduleNumber, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return sampleRate;
|
|
}
|
|
|
|
/**
|
|
* Set the number of averaging bits.
|
|
*
|
|
* This sets the number of averaging bits. The actual number of averaged samples is 2**bits.
|
|
* Use averaging to improve the stability of your measurement at the expense of sampling rate.
|
|
* The averaging is done automatically in the FPGA.
|
|
*
|
|
* @param channel Analog channel to configure.
|
|
* @param bits Number of bits to average.
|
|
*/
|
|
void AnalogModule::SetAverageBits(uint32_t channel, uint32_t bits)
|
|
{
|
|
int32_t status = 0;
|
|
setAnalogAverageBits(m_ports[channel-1], bits, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
}
|
|
|
|
/**
|
|
* Get the number of averaging bits.
|
|
*
|
|
* This gets the number of averaging bits from the FPGA. The actual number of averaged samples is 2**bits.
|
|
* The averaging is done automatically in the FPGA.
|
|
*
|
|
* @param channel Channel to address.
|
|
* @return Bits to average.
|
|
*/
|
|
uint32_t AnalogModule::GetAverageBits(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t averageBits = getAnalogAverageBits(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return averageBits;
|
|
}
|
|
|
|
/**
|
|
* Set the number of oversample bits.
|
|
*
|
|
* This sets the number of oversample bits. The actual number of oversampled values is 2**bits.
|
|
* Use oversampling to improve the resolution of your measurements at the expense of sampling rate.
|
|
* The oversampling is done automatically in the FPGA.
|
|
*
|
|
* @param channel Analog channel to configure.
|
|
* @param bits Number of bits to oversample.
|
|
*/
|
|
void AnalogModule::SetOversampleBits(uint32_t channel, uint32_t bits)
|
|
{
|
|
int32_t status = 0;
|
|
setAnalogOversampleBits(m_ports[channel-1], bits, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
}
|
|
|
|
/**
|
|
* Get the number of oversample bits.
|
|
*
|
|
* This gets the number of oversample bits from the FPGA. The actual number of oversampled values is
|
|
* 2**bits. The oversampling is done automatically in the FPGA.
|
|
*
|
|
* @param channel Channel to address.
|
|
* @return Bits to oversample.
|
|
*/
|
|
uint32_t AnalogModule::GetOversampleBits(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t oversampleBits = getAnalogOversampleBits(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return oversampleBits;
|
|
}
|
|
|
|
/**
|
|
* Get a sample straight from the channel on this module.
|
|
*
|
|
* The sample is a 12-bit value representing the -10V to 10V range of the A/D converter in the module.
|
|
* The units are in A/D converter codes. Use GetVoltage() to get the analog value in calibrated units.
|
|
*
|
|
* @return A sample straight from the channel on this module.
|
|
*/
|
|
int16_t AnalogModule::GetValue(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int16_t value = getAnalogValue(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Get a sample from the output of the oversample and average engine for the channel.
|
|
*
|
|
* The sample is 12-bit + the value configured in SetOversampleBits().
|
|
* The value configured in SetAverageBits() will cause this value to be averaged 2**bits number of samples.
|
|
* This is not a sliding window. The sample will not change until 2**(OversamplBits + AverageBits) samples
|
|
* have been acquired from the module on this channel.
|
|
* Use GetAverageVoltage() to get the analog value in calibrated units.
|
|
*
|
|
* @param channel Channel number to read.
|
|
* @return A sample from the oversample and average engine for the channel.
|
|
*/
|
|
int32_t AnalogModule::GetAverageValue(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t value = getAnalogAverageValue(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Convert a voltage to a raw value for a specified channel.
|
|
*
|
|
* This process depends on the calibration of each channel, so the channel
|
|
* must be specified.
|
|
*
|
|
* @todo This assumes raw values. Oversampling not supported as is.
|
|
*
|
|
* @param channel The channel to convert for.
|
|
* @param voltage The voltage to convert.
|
|
* @return The raw value for the channel.
|
|
*/
|
|
int32_t AnalogModule::VoltsToValue(int32_t channel, float voltage)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t value = getAnalogVoltsToValue(m_ports[channel-1], voltage, &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return value;
|
|
}
|
|
|
|
/**
|
|
* Get a scaled sample straight from the channel on this module.
|
|
*
|
|
* The value is scaled to units of Volts using the calibrated scaling data from GetLSBWeight() and GetOffset().
|
|
*
|
|
* @param channel The channel to read.
|
|
* @return A scaled sample straight from the channel on this module.
|
|
*/
|
|
float AnalogModule::GetVoltage(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
float voltage = getAnalogVoltage(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return voltage;
|
|
}
|
|
|
|
/**
|
|
* Get a scaled sample from the output of the oversample and average engine for the channel.
|
|
*
|
|
* The value is scaled to units of Volts using the calibrated scaling data from GetLSBWeight() and GetOffset().
|
|
* Using oversampling will cause this value to be higher resolution, but it will update more slowly.
|
|
* Using averaging will cause this value to be more stable, but it will update more slowly.
|
|
*
|
|
* @param channel The channel to read.
|
|
* @return A scaled sample from the output of the oversample and average engine for the channel.
|
|
*/
|
|
float AnalogModule::GetAverageVoltage(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
float voltage = getAnalogAverageVoltage(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return voltage;
|
|
}
|
|
|
|
/**
|
|
* Get the factory scaling least significant bit weight constant.
|
|
* The least significant bit weight constant for the channel that was calibrated in
|
|
* manufacturing and stored in an eeprom in the module.
|
|
*
|
|
* Volts = ((LSB_Weight * 1e-9) * raw) - (Offset * 1e-9)
|
|
*
|
|
* @param channel The channel to get calibration data for.
|
|
* @return Least significant bit weight.
|
|
*/
|
|
uint32_t AnalogModule::GetLSBWeight(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t lsbWeight = getAnalogLSBWeight(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return lsbWeight;
|
|
}
|
|
|
|
/**
|
|
* Get the factory scaling offset constant.
|
|
* The offset constant for the channel that was calibrated in manufacturing and stored
|
|
* in an eeprom in the module.
|
|
*
|
|
* Volts = ((LSB_Weight * 1e-9) * raw) - (Offset * 1e-9)
|
|
*
|
|
* @param channel The channel to get calibration data for.
|
|
* @return Offset constant.
|
|
*/
|
|
int32_t AnalogModule::GetOffset(uint32_t channel)
|
|
{
|
|
int32_t status = 0;
|
|
int32_t offset = getAnalogOffset(m_ports[channel-1], &status);
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
return offset;
|
|
}
|
|
|
|
|