blob: fc69e359da5cf667a82ed29fd79117fb4cb77f83 [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/DigitalOutput.h"
6
7#include <limits>
Brian Silverman8fce7482020-01-05 13:18:21 -08008
9#include <hal/DIO.h>
10#include <hal/FRCUsageReporting.h>
11#include <hal/HALBase.h>
12#include <hal/Ports.h>
Austin Schuh812d0d12021-11-04 20:16:48 -070013#include <wpi/StackTrace.h>
14#include <wpi/sendable/SendableBuilder.h>
15#include <wpi/sendable/SendableRegistry.h>
Brian Silverman8fce7482020-01-05 13:18:21 -080016
Austin Schuh812d0d12021-11-04 20:16:48 -070017#include "frc/Errors.h"
Brian Silverman8fce7482020-01-05 13:18:21 -080018#include "frc/SensorUtil.h"
Brian Silverman8fce7482020-01-05 13:18:21 -080019
20using namespace frc;
21
22DigitalOutput::DigitalOutput(int channel) {
23 m_pwmGenerator = HAL_kInvalidHandle;
24 if (!SensorUtil::CheckDigitalChannel(channel)) {
Austin Schuh812d0d12021-11-04 20:16:48 -070025 throw FRC_MakeError(err::ChannelIndexOutOfRange, "Channel {}", channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080026 }
27 m_channel = channel;
28
29 int32_t status = 0;
Austin Schuh812d0d12021-11-04 20:16:48 -070030 std::string stackTrace = wpi::GetStackTrace(1);
31 m_handle = HAL_InitializeDIOPort(HAL_GetPort(channel), false,
32 stackTrace.c_str(), &status);
33 FRC_CheckErrorStatus(status, "Channel {}", channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080034
35 HAL_Report(HALUsageReporting::kResourceType_DigitalOutput, channel + 1);
Austin Schuh812d0d12021-11-04 20:16:48 -070036 wpi::SendableRegistry::AddLW(this, "DigitalOutput", channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080037}
38
39DigitalOutput::~DigitalOutput() {
Brian Silverman8fce7482020-01-05 13:18:21 -080040 // Disable the PWM in case it was running.
Austin Schuh812d0d12021-11-04 20:16:48 -070041 try {
42 DisablePWM();
43 } catch (const RuntimeError& e) {
44 e.Report();
45 }
Brian Silverman8fce7482020-01-05 13:18:21 -080046
47 HAL_FreeDIOPort(m_handle);
48}
49
50void DigitalOutput::Set(bool value) {
Brian Silverman8fce7482020-01-05 13:18:21 -080051 int32_t status = 0;
52 HAL_SetDIO(m_handle, value, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -070053 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080054}
55
56bool DigitalOutput::Get() const {
Brian Silverman8fce7482020-01-05 13:18:21 -080057 int32_t status = 0;
58 bool val = HAL_GetDIO(m_handle, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -070059 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080060 return val;
61}
62
Austin Schuh812d0d12021-11-04 20:16:48 -070063HAL_Handle DigitalOutput::GetPortHandleForRouting() const {
64 return m_handle;
Brian Silverman8fce7482020-01-05 13:18:21 -080065}
66
Austin Schuh812d0d12021-11-04 20:16:48 -070067AnalogTriggerType DigitalOutput::GetAnalogTriggerTypeForRouting() const {
68 return static_cast<AnalogTriggerType>(0);
69}
Brian Silverman8fce7482020-01-05 13:18:21 -080070
Austin Schuh812d0d12021-11-04 20:16:48 -070071bool DigitalOutput::IsAnalogTrigger() const {
72 return false;
73}
74
75int DigitalOutput::GetChannel() const {
76 return m_channel;
77}
Brian Silverman8fce7482020-01-05 13:18:21 -080078
James Kuszmaulcf324122023-01-14 14:07:17 -080079void DigitalOutput::Pulse(units::second_t pulseLength) {
Brian Silverman8fce7482020-01-05 13:18:21 -080080 int32_t status = 0;
James Kuszmaulcf324122023-01-14 14:07:17 -080081 HAL_Pulse(m_handle, pulseLength.to<double>(), &status);
Austin Schuh812d0d12021-11-04 20:16:48 -070082 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080083}
84
85bool DigitalOutput::IsPulsing() const {
Brian Silverman8fce7482020-01-05 13:18:21 -080086 int32_t status = 0;
87 bool value = HAL_IsPulsing(m_handle, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -070088 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080089 return value;
90}
91
92void DigitalOutput::SetPWMRate(double rate) {
Brian Silverman8fce7482020-01-05 13:18:21 -080093 int32_t status = 0;
94 HAL_SetDigitalPWMRate(rate, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -070095 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -080096}
97
James Kuszmaulcf324122023-01-14 14:07:17 -080098void DigitalOutput::EnablePPS(double dutyCycle) {
99 if (m_pwmGenerator != HAL_kInvalidHandle) {
100 return;
101 }
102
103 int32_t status = 0;
104
105 m_pwmGenerator = HAL_AllocateDigitalPWM(&status);
106 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
107
108 HAL_SetDigitalPWMPPS(m_pwmGenerator, dutyCycle, &status);
109 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
110
111 HAL_SetDigitalPWMOutputChannel(m_pwmGenerator, m_channel, &status);
112 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
113}
114
Brian Silverman8fce7482020-01-05 13:18:21 -0800115void DigitalOutput::EnablePWM(double initialDutyCycle) {
Austin Schuh812d0d12021-11-04 20:16:48 -0700116 if (m_pwmGenerator != HAL_kInvalidHandle) {
117 return;
118 }
Brian Silverman8fce7482020-01-05 13:18:21 -0800119
120 int32_t status = 0;
121
Brian Silverman8fce7482020-01-05 13:18:21 -0800122 m_pwmGenerator = HAL_AllocateDigitalPWM(&status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700123 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800124
Brian Silverman8fce7482020-01-05 13:18:21 -0800125 HAL_SetDigitalPWMDutyCycle(m_pwmGenerator, initialDutyCycle, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700126 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800127
Brian Silverman8fce7482020-01-05 13:18:21 -0800128 HAL_SetDigitalPWMOutputChannel(m_pwmGenerator, m_channel, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700129 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800130}
131
132void DigitalOutput::DisablePWM() {
Austin Schuh812d0d12021-11-04 20:16:48 -0700133 if (m_pwmGenerator == HAL_kInvalidHandle) {
134 return;
135 }
Brian Silverman8fce7482020-01-05 13:18:21 -0800136
137 int32_t status = 0;
138
139 // Disable the output by routing to a dead bit.
140 HAL_SetDigitalPWMOutputChannel(m_pwmGenerator, SensorUtil::kDigitalChannels,
141 &status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700142 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800143
144 HAL_FreeDigitalPWM(m_pwmGenerator, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700145 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800146
147 m_pwmGenerator = HAL_kInvalidHandle;
148}
149
150void DigitalOutput::UpdateDutyCycle(double dutyCycle) {
Brian Silverman8fce7482020-01-05 13:18:21 -0800151 int32_t status = 0;
152 HAL_SetDigitalPWMDutyCycle(m_pwmGenerator, dutyCycle, &status);
Austin Schuh812d0d12021-11-04 20:16:48 -0700153 FRC_CheckErrorStatus(status, "Channel {}", m_channel);
Brian Silverman8fce7482020-01-05 13:18:21 -0800154}
155
156void DigitalOutput::SetSimDevice(HAL_SimDeviceHandle device) {
157 HAL_SetDIOSimDevice(m_handle, device);
158}
159
Austin Schuh812d0d12021-11-04 20:16:48 -0700160void DigitalOutput::InitSendable(wpi::SendableBuilder& builder) {
Brian Silverman8fce7482020-01-05 13:18:21 -0800161 builder.SetSmartDashboardType("Digital Output");
Austin Schuh1e69f942020-11-14 15:06:14 -0800162 builder.AddBooleanProperty(
James Kuszmaulcf324122023-01-14 14:07:17 -0800163 "Value", [=, this] { return Get(); },
164 [=, this](bool value) { Set(value); });
Brian Silverman8fce7482020-01-05 13:18:21 -0800165}