Files
allwpilib/hal/src/main/native/systemcore/DutyCycle.cpp

138 lines
3.9 KiB
C++
Raw Normal View History

// 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.
2025-11-07 19:56:21 -05:00
#include "wpi/hal/DutyCycle.h"
#include <cstdio>
#include <thread>
#include "HALInitializer.hpp"
#include "PortsInternal.hpp"
#include "SmartIo.hpp"
#include "wpi/hal/ErrorHandling.hpp"
2025-11-07 19:56:21 -05:00
#include "wpi/hal/Errors.h"
#include "wpi/hal/handles/HandlesInternal.hpp"
#include "wpi/hal/monotonic_clock.hpp"
2025-11-07 20:00:05 -05:00
using namespace wpi::hal;
2025-11-07 20:00:05 -05:00
namespace wpi::hal::init {
void InitializeDutyCycle() {}
2025-11-07 20:00:05 -05:00
} // namespace wpi::hal::init
extern "C" {
2025-01-30 18:59:34 -08:00
HAL_DutyCycleHandle HAL_InitializeDutyCycle(int32_t channel,
const char* allocationLocation,
int32_t* status) {
2025-11-07 20:00:05 -05:00
wpi::hal::init::CheckInit();
2025-01-30 18:59:34 -08:00
if (channel < 0 || channel >= kNumSmartIo) {
*status = MakeErrorIndexOutOfRange(HAL_RESOURCE_OUT_OF_RANGE,
"Invalid Index for DutyCycle", 0,
kNumSmartIo, channel);
return HAL_INVALID_HANDLE;
}
auto resource = smartIoHandles->Allocate(channel, HAL_HandleEnum::DUTY_CYCLE,
"DutyCycle");
if (!resource) {
*status = resource.error();
return HAL_INVALID_HANDLE; // failed to allocate. Pass error back.
}
auto [handle, port] = *resource;
port->channel = channel;
2026-06-06 14:54:49 -07:00
*status = port->InitializeMode(MRC_SmartIOMode::MRC_SmartIOMode_PwmInput);
if (*status != 0) {
smartIoHandles->Free(handle, HAL_HandleEnum::DUTY_CYCLE);
return HAL_INVALID_HANDLE;
}
port->previousAllocation = allocationLocation ? allocationLocation : "";
return handle;
}
void HAL_FreeDutyCycle(HAL_DutyCycleHandle dutyCycleHandle) {
auto port = smartIoHandles->Get(dutyCycleHandle, HAL_HandleEnum::DUTY_CYCLE);
if (port == nullptr) {
return;
}
smartIoHandles->Free(dutyCycleHandle, HAL_HandleEnum::DUTY_CYCLE);
// Wait for no other object to hold this handle.
auto start = wpi::hal::monotonic_clock::now();
while (port.use_count() != 1) {
auto current = wpi::hal::monotonic_clock::now();
if (start + std::chrono::seconds(1) < current) {
2026-06-06 14:54:49 -07:00
std::puts("DutyCycle handle free timeout");
std::fflush(stdout);
break;
}
std::this_thread::yield();
}
}
2026-06-06 14:54:49 -07:00
void HAL_SetDutyCycleSimDevice(HAL_DutyCycleHandle handle,
HAL_SimDeviceHandle device) {}
double HAL_GetDutyCycleFrequency(HAL_DutyCycleHandle dutyCycleHandle,
int32_t* status) {
auto port = smartIoHandles->Get(dutyCycleHandle, HAL_HandleEnum::DUTY_CYCLE);
if (port == nullptr) {
*status = HAL_HANDLE_ERROR;
return 0;
}
uint16_t ret = 0;
*status = port->GetPwmInputPeriodMicroseconds(&ret);
if (ret == 0) {
return 0.0;
}
return 1000000.0 / ret;
}
double HAL_GetDutyCycleOutput(HAL_DutyCycleHandle dutyCycleHandle,
int32_t* status) {
auto port = smartIoHandles->Get(dutyCycleHandle, HAL_HandleEnum::DUTY_CYCLE);
if (port == nullptr) {
*status = HAL_HANDLE_ERROR;
return 0.0;
}
uint16_t highTime = 0;
*status = port->GetPwmInputMicroseconds(&highTime);
uint16_t period = 0;
*status = port->GetPwmInputPeriodMicroseconds(&period);
if (period == 0) {
return 0.0;
}
if (highTime >= period) {
return 1.0;
}
return static_cast<double>(highTime) / static_cast<double>(period);
}
int32_t HAL_GetDutyCycleHighTime(HAL_DutyCycleHandle dutyCycleHandle,
int32_t* status) {
auto port = smartIoHandles->Get(dutyCycleHandle, HAL_HandleEnum::DUTY_CYCLE);
if (port == nullptr) {
*status = HAL_HANDLE_ERROR;
return 0;
}
uint16_t ret = 0;
*status = port->GetPwmInputMicroseconds(&ret);
return static_cast<int32_t>(ret) * 1000;
}
} // extern "C"