Files
allwpilib/wpilibc/src/main/native/cpp/counter/Tachometer.cpp
Tyler Veness 1fc098e696 Enable log macros to work with no args (#4475)
This is enabled by the C++20 __VA_OPT__ feature.
Uses of "{}" format string were updated.
Some warning suppressions were required for older clang versions.
Also improve codegen of wpi::Logger::Log(), frc::ReportError(), and frc::MakeError();
these generate better and less redundant code if they use fmt::string_view for the
format string instead of templating on it.
2022-10-19 10:49:27 -07:00

123 lines
3.9 KiB
C++

// Copyright (c) FIRST and other WPILib contributors.
// Open Source Software; you can modify and/or share it under the terms of
// the WPILib BSD license file in the root directory of this project.
#include "frc/counter/Tachometer.h"
#include <frc/DigitalSource.h>
#include <hal/Counter.h>
#include <hal/FRCUsageReporting.h>
#include <wpi/NullDeleter.h>
#include <wpi/sendable/SendableBuilder.h>
#include "frc/Errors.h"
using namespace frc;
Tachometer::Tachometer(DigitalSource& source)
: Tachometer({&source, wpi::NullDeleter<DigitalSource>()}) {}
Tachometer::Tachometer(std::shared_ptr<DigitalSource> source) {
if (source == nullptr) {
throw FRC_MakeError(err::NullParameter, "source");
}
m_source = source;
int32_t status = 0;
HAL_SetCounterUpSource(m_handle, m_source->GetPortHandleForRouting(),
static_cast<HAL_AnalogTriggerType>(
m_source->GetAnalogTriggerTypeForRouting()),
&status);
FRC_CheckErrorStatus(status, "{}", m_index);
HAL_SetCounterUpSourceEdge(m_handle, true, false, &status);
FRC_CheckErrorStatus(status, "{}", m_index);
HAL_Report(HALUsageReporting::kResourceType_Counter, m_index + 1);
wpi::SendableRegistry::AddLW(this, "Tachometer", m_index);
}
Tachometer::~Tachometer() {
int32_t status = 0;
HAL_FreeCounter(m_handle, &status);
}
units::hertz_t Tachometer::GetFrequency() const {
auto period = GetPeriod();
if (period.to<double>() == 0) {
return units::hertz_t{0.0};
}
return 1 / period;
}
units::second_t Tachometer::GetPeriod() const {
int32_t status = 0;
double period = HAL_GetCounterPeriod(m_handle, &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
return units::second_t{period};
}
int Tachometer::GetEdgesPerRevolution() const {
return m_edgesPerRevolution;
}
void Tachometer::SetEdgesPerRevolution(int edges) {
m_edgesPerRevolution = edges;
}
units::turns_per_second_t Tachometer::GetRevolutionsPerSecond() const {
auto period = GetPeriod();
if (period.to<double>() == 0) {
return units::turns_per_second_t{0.0};
}
int edgesPerRevolution = GetEdgesPerRevolution();
if (edgesPerRevolution == 0) {
return units::turns_per_second_t{0.0};
}
auto rotationHz = ((1.0 / edgesPerRevolution) / period);
return units::turns_per_second_t{rotationHz.to<double>()};
}
units::revolutions_per_minute_t Tachometer::GetRevolutionsPerMinute() const {
return units::revolutions_per_minute_t{GetRevolutionsPerSecond()};
}
bool Tachometer::GetStopped() const {
int32_t status = 0;
bool stopped = HAL_GetCounterStopped(m_handle, &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
return stopped;
}
int Tachometer::GetSamplesToAverage() const {
int32_t status = 0;
int32_t samplesToAverage = HAL_GetCounterSamplesToAverage(m_handle, &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
return samplesToAverage;
}
void Tachometer::SetSamplesToAverage(int samples) {
int32_t status = 0;
HAL_SetCounterSamplesToAverage(m_handle, samples, &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
}
void Tachometer::SetMaxPeriod(units::second_t maxPeriod) {
int32_t status = 0;
HAL_SetCounterMaxPeriod(m_handle, maxPeriod.to<double>(), &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
}
void Tachometer::SetUpdateWhenEmpty(bool updateWhenEmpty) {
int32_t status = 0;
HAL_SetCounterUpdateWhenEmpty(m_handle, updateWhenEmpty, &status);
FRC_CheckErrorStatus(status, "Channel {}", m_source->GetChannel());
}
void Tachometer::InitSendable(wpi::SendableBuilder& builder) {
builder.SetSmartDashboardType("Tachometer");
builder.AddDoubleProperty(
"RPS", [&] { return GetRevolutionsPerSecond().to<double>(); }, nullptr);
builder.AddDoubleProperty(
"RPM", [&] { return GetRevolutionsPerMinute().to<double>(); }, nullptr);
}