blob: beb8688e4f919ce16124434d04d7373b9e97b4d1 [file] [log] [blame]
/*----------------------------------------------------------------------------*/
/* 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 "AnalogChannel.h"
#include "AnalogModule.h"
#include "NetworkCommunication/UsageReporting.h"
#include "Resource.h"
#include "WPIErrors.h"
#include "LiveWindow/LiveWindow.h"
static Resource *channels = NULL;
const UINT8 AnalogChannel::kAccumulatorModuleNumber;
const UINT32 AnalogChannel::kAccumulatorNumChannels;
const UINT32 AnalogChannel::kAccumulatorChannels[] = {1, 2};
/**
* Common initialization.
*/
void AnalogChannel::InitChannel(UINT8 moduleNumber, UINT32 channel)
{
char buf[64];
Resource::CreateResourceObject(&channels, kAnalogModules * kAnalogChannels);
if (!CheckAnalogModule(moduleNumber))
{
snprintf(buf, 64, "Analog Module %d", moduleNumber);
wpi_setWPIErrorWithContext(ModuleIndexOutOfRange, buf);
return;
}
if (!CheckAnalogChannel(channel))
{
snprintf(buf, 64, "Analog Channel %d", channel);
wpi_setWPIErrorWithContext(ChannelIndexOutOfRange, buf);
return;
}
snprintf(buf, 64, "Analog Input %d (Module: %d)", channel, moduleNumber);
if (channels->Allocate((moduleNumber - 1) * kAnalogChannels + channel - 1, buf) == ~0ul)
{
CloneError(channels);
return;
}
m_channel = channel;
m_module = AnalogModule::GetInstance(moduleNumber);
if (IsAccumulatorChannel())
{
tRioStatusCode localStatus = NiFpga_Status_Success;
m_accumulator = tAccumulator::create(channel - 1, &localStatus);
wpi_setError(localStatus);
m_accumulatorOffset=0;
}
else
{
m_accumulator = NULL;
}
LiveWindow::GetInstance()->AddActuator("AnalogChannel",channel, GetModuleNumber(), this);
nUsageReporting::report(nUsageReporting::kResourceType_AnalogChannel, channel, GetModuleNumber() - 1);
}
/**
* Construct an analog channel on a specified module.
*
* @param moduleNumber The analog module (1 or 2).
* @param channel The channel number to represent.
*/
AnalogChannel::AnalogChannel(UINT8 moduleNumber, UINT32 channel)
{
InitChannel(moduleNumber, channel);
}
/**
* Construct an analog channel on the default module.
*
* @param channel The channel number to represent.
*/
AnalogChannel::AnalogChannel(UINT32 channel)
{
InitChannel(GetDefaultAnalogModule(), channel);
}
/**
* Channel destructor.
*/
AnalogChannel::~AnalogChannel()
{
channels->Free((m_module->GetNumber() - 1) * kAnalogChannels + m_channel - 1);
}
/**
* Get the analog module that this channel is on.
* @return A pointer to the AnalogModule that this channel is on.
*/
AnalogModule *AnalogChannel::GetModule()
{
if (StatusIsFatal()) return NULL;
return m_module;
}
/**
* Get a sample straight from this channel on the 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 this channel on the module.
*/
INT16 AnalogChannel::GetValue()
{
if (StatusIsFatal()) return 0;
return m_module->GetValue(m_channel);
}
/**
* Get a sample from the output of the oversample and average engine for this 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.
* @return A sample from the oversample and average engine for this channel.
*/
INT32 AnalogChannel::GetAverageValue()
{
if (StatusIsFatal()) return 0;
return m_module->GetAverageValue(m_channel);
}
/**
* Get a scaled sample straight from this channel on the module.
* The value is scaled to units of Volts using the calibrated scaling data from GetLSBWeight() and GetOffset().
* @return A scaled sample straight from this channel on the module.
*/
float AnalogChannel::GetVoltage()
{
if (StatusIsFatal()) return 0.0f;
return m_module->GetVoltage(m_channel);
}
/**
* Get a scaled sample from the output of the oversample and average engine for this 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.
* @return A scaled sample from the output of the oversample and average engine for this channel.
*/
float AnalogChannel::GetAverageVoltage()
{
if (StatusIsFatal()) return 0.0f;
return m_module->GetAverageVoltage(m_channel);
}
/**
* 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)
*
* @return Least significant bit weight.
*/
UINT32 AnalogChannel::GetLSBWeight()
{
if (StatusIsFatal()) return 0;
return m_module->GetLSBWeight(m_channel);
}
/**
* 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)
*
* @return Offset constant.
*/
INT32 AnalogChannel::GetOffset()
{
if (StatusIsFatal()) return 0;
return m_module->GetOffset(m_channel);
}
/**
* Get the channel number.
* @return The channel number.
*/
UINT32 AnalogChannel::GetChannel()
{
if (StatusIsFatal()) return 0;
return m_channel;
}
/**
* Get the module number.
* @return The module number.
*/
UINT8 AnalogChannel::GetModuleNumber()
{
if (StatusIsFatal()) return 0;
return m_module->GetNumber();
}
/**
* 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 bits Number of bits of averaging.
*/
void AnalogChannel::SetAverageBits(UINT32 bits)
{
if (StatusIsFatal()) return;
m_module->SetAverageBits(m_channel, bits);
}
/**
* Get the number of averaging bits previously configured.
* 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.
*
* @return Number of bits of averaging previously configured.
*/
UINT32 AnalogChannel::GetAverageBits()
{
if (StatusIsFatal()) return 0;
return m_module->GetAverageBits(m_channel);
}
/**
* 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 bits Number of bits of oversampling.
*/
void AnalogChannel::SetOversampleBits(UINT32 bits)
{
if (StatusIsFatal()) return;
m_module->SetOversampleBits(m_channel, bits);
}
/**
* Get the number of oversample bits previously configured.
* 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.
*
* @return Number of bits of oversampling previously configured.
*/
UINT32 AnalogChannel::GetOversampleBits()
{
if (StatusIsFatal()) return 0;
return m_module->GetOversampleBits(m_channel);
}
/**
* Is the channel attached to an accumulator.
*
* @return The analog channel is attached to an accumulator.
*/
bool AnalogChannel::IsAccumulatorChannel()
{
if (StatusIsFatal()) return false;
if(m_module->GetNumber() != kAccumulatorModuleNumber) return false;
for (UINT32 i=0; i<kAccumulatorNumChannels; i++)
{
if (m_channel == kAccumulatorChannels[i]) return true;
}
return false;
}
/**
* Initialize the accumulator.
*/
void AnalogChannel::InitAccumulator()
{
if (StatusIsFatal()) return;
m_accumulatorOffset = 0;
SetAccumulatorCenter(0);
ResetAccumulator();
}
/**
* Set an inital value for the accumulator.
*
* This will be added to all values returned to the user.
* @param initialValue The value that the accumulator should start from when reset.
*/
void AnalogChannel::SetAccumulatorInitialValue(INT64 initialValue)
{
if (StatusIsFatal()) return;
m_accumulatorOffset = initialValue;
}
/**
* Resets the accumulator to the initial value.
*/
void AnalogChannel::ResetAccumulator()
{
if (StatusIsFatal()) return;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
m_accumulator->strobeReset(&localStatus);
wpi_setError(localStatus);
}
/**
* Set the center value of the accumulator.
*
* The center value is subtracted from each A/D value before it is added to the accumulator. This
* is used for the center value of devices like gyros and accelerometers to make integration work
* and to take the device offset into account when integrating.
*
* This center value is based on the output of the oversampled and averaged source from channel 1.
* Because of this, any non-zero oversample bits will affect the size of the value for this field.
*/
void AnalogChannel::SetAccumulatorCenter(INT32 center)
{
if (StatusIsFatal()) return;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
m_accumulator->writeCenter(center, &localStatus);
wpi_setError(localStatus);
}
/**
* Set the accumulator's deadband.
*/
void AnalogChannel::SetAccumulatorDeadband(INT32 deadband)
{
if (StatusIsFatal()) return;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
m_accumulator->writeDeadband(deadband, &localStatus);
wpi_setError(localStatus);
}
/**
* Read the accumulated value.
*
* Read the value that has been accumulating on channel 1.
* The accumulator is attached after the oversample and average engine.
*
* @return The 64-bit value accumulated since the last Reset().
*/
INT64 AnalogChannel::GetAccumulatorValue()
{
if (StatusIsFatal()) return 0;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return 0;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
INT64 value = m_accumulator->readOutput_Value(&localStatus) + m_accumulatorOffset;
wpi_setError(localStatus);
return value;
}
/**
* Read the number of accumulated values.
*
* Read the count of the accumulated values since the accumulator was last Reset().
*
* @return The number of times samples from the channel were accumulated.
*/
UINT32 AnalogChannel::GetAccumulatorCount()
{
if (StatusIsFatal()) return 0;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return 0;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
UINT32 count = m_accumulator->readOutput_Count(&localStatus);
wpi_setError(localStatus);
return count;
}
/**
* Read the accumulated value and the number of accumulated values atomically.
*
* This function reads the value and count from the FPGA atomically.
* This can be used for averaging.
*
* @param value Pointer to the 64-bit accumulated output.
* @param count Pointer to the number of accumulation cycles.
*/
void AnalogChannel::GetAccumulatorOutput(INT64 *value, UINT32 *count)
{
if (StatusIsFatal()) return;
if (m_accumulator == NULL)
{
wpi_setWPIError(NullParameter);
return;
}
if (value == NULL || count == NULL)
{
wpi_setWPIError(NullParameter);
return;
}
tRioStatusCode localStatus = NiFpga_Status_Success;
tAccumulator::tOutput output = m_accumulator->readOutput(&localStatus);
*value = output.Value + m_accumulatorOffset;
*count = output.Count;
wpi_setError(localStatus);
}
/**
* Get the Average voltage for the PID Source base object.
*
* @return The average voltage.
*/
double AnalogChannel::PIDGet()
{
if (StatusIsFatal()) return 0.0;
return GetAverageValue();
}
void AnalogChannel::UpdateTable() {
if (m_table != NULL) {
m_table->PutNumber("Value", GetAverageVoltage());
}
}
void AnalogChannel::StartLiveWindowMode() {
}
void AnalogChannel::StopLiveWindowMode() {
}
std::string AnalogChannel::GetSmartDashboardType() {
return "Analog Input";
}
void AnalogChannel::InitTable(ITable *subTable) {
m_table = subTable;
UpdateTable();
}
ITable * AnalogChannel::GetTable() {
return m_table;
}