/*----------------------------------------------------------------------------*/ /* Copyright (c) FIRST 2016. 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 the root directory of */ /* the project. */ /*----------------------------------------------------------------------------*/ #include "HAL/DIO.h" #include #include "DigitalInternal.h" #include "PortsInternal.h" #include "handles/HandlesInternal.h" #include "handles/LimitedHandleResource.h" using namespace hal; // Create a mutex to protect changes to the digital output values static priority_recursive_mutex digitalDIOMutex; static LimitedHandleResource digitalPWMHandles; extern "C" { /** * Create a new instance of a digital port. */ HAL_DigitalHandle HAL_InitializeDIOPort(HAL_PortHandle port_handle, uint8_t input, int32_t* status) { initializeDigital(status); if (*status != 0) return HAL_kInvalidHandle; int16_t pin = getPortHandlePin(port_handle); if (pin == InvalidHandleIndex) { *status = PARAMETER_OUT_OF_RANGE; return HAL_kInvalidHandle; } auto handle = digitalPinHandles.Allocate(pin, HAL_HandleEnum::DIO, status); if (*status != 0) return HAL_kInvalidHandle; // failed to allocate. Pass error back. auto port = digitalPinHandles.Get(handle, HAL_HandleEnum::DIO); if (port == nullptr) { // would only occur on thread issue. *status = HAL_HANDLE_ERROR; return HAL_kInvalidHandle; } port->pin = static_cast(pin); std::lock_guard sync(digitalDIOMutex); tDIO::tOutputEnable outputEnable = digitalSystem->readOutputEnable(status); if (port->pin < kNumDigitalHeaders) { uint32_t bitToSet = 1 << port->pin; if (input) { outputEnable.Headers = outputEnable.Headers & (~bitToSet); // clear the bit for read } else { outputEnable.Headers = outputEnable.Headers | bitToSet; // set the bit for write } } else { uint32_t bitToSet = 1 << remapMXPChannel(port->pin); // Disable special functions on this pin short specialFunctions = digitalSystem->readEnableMXPSpecialFunction(status); digitalSystem->writeEnableMXPSpecialFunction(specialFunctions & ~bitToSet, status); if (input) { outputEnable.MXP = outputEnable.MXP & (~bitToSet); // clear the bit for read } else { outputEnable.MXP = outputEnable.MXP | bitToSet; // set the bit for write } } digitalSystem->writeOutputEnable(outputEnable, status); return handle; } void HAL_FreeDIOPort(HAL_DigitalHandle dio_port_handle) { // no status, so no need to check for a proper free. digitalPinHandles.Free(dio_port_handle, HAL_HandleEnum::DIO); } /** * Allocate a DO PWM Generator. * Allocate PWM generators so that they are not accidentally reused. * * @return PWM Generator handle */ HAL_DigitalPWMHandle HAL_AllocateDigitalPWM(int32_t* status) { auto handle = digitalPWMHandles.Allocate(); if (handle == HAL_kInvalidHandle) { *status = NO_AVAILABLE_RESOURCES; return HAL_kInvalidHandle; } auto id = digitalPWMHandles.Get(handle); if (id == nullptr) { // would only occur on thread issue. *status = HAL_HANDLE_ERROR; return HAL_kInvalidHandle; } *id = static_cast(getHandleIndex(handle)); return handle; } /** * Free the resource associated with a DO PWM generator. * * @param pwmGenerator The pwmGen to free that was allocated with * allocateDigitalPWM() */ void HAL_FreeDigitalPWM(HAL_DigitalPWMHandle pwmGenerator, int32_t* status) { digitalPWMHandles.Free(pwmGenerator); } /** * Change the frequency of the DO PWM generator. * * The valid range is from 0.6 Hz to 19 kHz. The frequency resolution is * logarithmic. * * @param rate The frequency to output all digital output PWM signals. */ void HAL_SetDigitalPWMRate(double rate, int32_t* status) { // Currently rounding in the log rate domain... heavy weight toward picking a // higher freq. // TODO: Round in the linear rate domain. uint8_t pwmPeriodPower = (uint8_t)( log(1.0 / (pwmSystem->readLoopTiming(status) * 0.25E-6 * rate)) / log(2.0) + 0.5); digitalSystem->writePWMPeriodPower(pwmPeriodPower, status); } /** * Configure the duty-cycle of the PWM generator * * @param pwmGenerator The generator index reserved by allocateDigitalPWM() * @param dutyCycle The percent duty cycle to output [0..1]. */ void HAL_SetDigitalPWMDutyCycle(HAL_DigitalPWMHandle pwmGenerator, double dutyCycle, int32_t* status) { auto port = digitalPWMHandles.Get(pwmGenerator); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return; } uint32_t id = *port; if (dutyCycle > 1.0) dutyCycle = 1.0; if (dutyCycle < 0.0) dutyCycle = 0.0; float rawDutyCycle = 256.0 * dutyCycle; if (rawDutyCycle > 255.5) rawDutyCycle = 255.5; { std::lock_guard sync(digitalPwmMutex); uint8_t pwmPeriodPower = digitalSystem->readPWMPeriodPower(status); if (pwmPeriodPower < 4) { // The resolution of the duty cycle drops close to the highest // frequencies. rawDutyCycle = rawDutyCycle / std::pow(2.0, 4 - pwmPeriodPower); } if (id < 4) digitalSystem->writePWMDutyCycleA(id, (uint8_t)rawDutyCycle, status); else digitalSystem->writePWMDutyCycleB(id - 4, (uint8_t)rawDutyCycle, status); } } /** * Configure which DO channel the PWM signal is output on * * @param pwmGenerator The generator index reserved by allocateDigitalPWM() * @param channel The Digital Output channel to output on */ void HAL_SetDigitalPWMOutputChannel(HAL_DigitalPWMHandle pwmGenerator, uint32_t pin, int32_t* status) { auto port = digitalPWMHandles.Get(pwmGenerator); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return; } uint32_t id = *port; if (pin >= kNumDigitalHeaders) { // if it is on the MXP pin += kMXPDigitalPWMOffset; // then to write as a digital PWM pin requires // an offset to write on the correct pin } digitalSystem->writePWMOutputSelect(id, pin, status); } /** * Write a digital I/O bit to the FPGA. * Set a single value on a digital I/O channel. * * @param channel The Digital I/O channel * @param value The state to set the digital channel (if it is configured as an * output) */ void HAL_SetDIO(HAL_DigitalHandle dio_port_handle, short value, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return; } if (value != 0 && value != 1) { if (value != 0) value = 1; } { std::lock_guard sync(digitalDIOMutex); tDIO::tDO currentDIO = digitalSystem->readDO(status); if (port->pin < kNumDigitalHeaders) { if (value == 0) { currentDIO.Headers = currentDIO.Headers & ~(1 << port->pin); } else if (value == 1) { currentDIO.Headers = currentDIO.Headers | (1 << port->pin); } } else { if (value == 0) { currentDIO.MXP = currentDIO.MXP & ~(1 << remapMXPChannel(port->pin)); } else if (value == 1) { currentDIO.MXP = currentDIO.MXP | (1 << remapMXPChannel(port->pin)); } uint32_t bitToSet = 1 << remapMXPChannel(port->pin); short specialFunctions = digitalSystem->readEnableMXPSpecialFunction(status); digitalSystem->writeEnableMXPSpecialFunction(specialFunctions & ~bitToSet, status); } digitalSystem->writeDO(currentDIO, status); } } /** * Read a digital I/O bit from the FPGA. * Get a single value from a digital I/O channel. * * @param channel The digital I/O channel * @return The state of the specified channel */ bool HAL_GetDIO(HAL_DigitalHandle dio_port_handle, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return false; } tDIO::tDI currentDIO = digitalSystem->readDI(status); // Shift 00000001 over channel-1 places. // AND it against the currentDIO // if it == 0, then return false // else return true if (port->pin < kNumDigitalHeaders) { return ((currentDIO.Headers >> port->pin) & 1) != 0; } else { // Disable special functions uint32_t bitToSet = 1 << remapMXPChannel(port->pin); short specialFunctions = digitalSystem->readEnableMXPSpecialFunction(status); digitalSystem->writeEnableMXPSpecialFunction(specialFunctions & ~bitToSet, status); return ((currentDIO.MXP >> remapMXPChannel(port->pin)) & 1) != 0; } } /** * Read the direction of a the Digital I/O lines * A 1 bit means output and a 0 bit means input. * * @param channel The digital I/O channel * @return The direction of the specified channel */ bool HAL_GetDIODirection(HAL_DigitalHandle dio_port_handle, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return false; } tDIO::tOutputEnable currentOutputEnable = digitalSystem->readOutputEnable(status); // Shift 00000001 over port->pin-1 places. // AND it against the currentOutputEnable // if it == 0, then return false // else return true if (port->pin < kNumDigitalHeaders) { return ((currentOutputEnable.Headers >> port->pin) & 1) != 0; } else { return ((currentOutputEnable.MXP >> remapMXPChannel(port->pin)) & 1) != 0; } } /** * Generate a single pulse. * Write a pulse to the specified digital output channel. There can only be a * single pulse going at any time. * * @param channel The Digital Output channel that the pulse should be output on * @param pulseLength The active length of the pulse (in seconds) */ void HAL_Pulse(HAL_DigitalHandle dio_port_handle, double pulseLength, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return; } tDIO::tPulse pulse; if (port->pin < kNumDigitalHeaders) { pulse.Headers = 1 << port->pin; } else { pulse.MXP = 1 << remapMXPChannel(port->pin); } digitalSystem->writePulseLength( (uint8_t)(1.0e9 * pulseLength / (pwmSystem->readLoopTiming(status) * 25)), status); digitalSystem->writePulse(pulse, status); } /** * Check a DIO line to see if it is currently generating a pulse. * * @return A pulse is in progress */ bool HAL_IsPulsing(HAL_DigitalHandle dio_port_handle, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return false; } tDIO::tPulse pulseRegister = digitalSystem->readPulse(status); if (port->pin < kNumDigitalHeaders) { return (pulseRegister.Headers & (1 << port->pin)) != 0; } else { return (pulseRegister.MXP & (1 << remapMXPChannel(port->pin))) != 0; } } /** * Check if any DIO line is currently generating a pulse. * * @return A pulse on some line is in progress */ bool HAL_IsAnyPulsing(int32_t* status) { tDIO::tPulse pulseRegister = digitalSystem->readPulse(status); return pulseRegister.Headers != 0 && pulseRegister.MXP != 0; } /** * Write the filter index from the FPGA. * Set the filter index used to filter out short pulses. * * @param digital_port_pointer The digital I/O channel * @param filter_index The filter index. Must be in the range 0 - 3, * where 0 means "none" and 1 - 3 means filter # filter_index - 1. */ void HAL_SetFilterSelect(HAL_DigitalHandle dio_port_handle, int filter_index, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return; } std::lock_guard sync(digitalDIOMutex); if (port->pin < kNumDigitalHeaders) { digitalSystem->writeFilterSelectHdr(port->pin, filter_index, status); } else { digitalSystem->writeFilterSelectMXP(remapMXPChannel(port->pin), filter_index, status); } } /** * Read the filter index from the FPGA. * Get the filter index used to filter out short pulses. * * @param digital_port_pointer The digital I/O channel * @return filter_index The filter index. Must be in the range 0 - 3, * where 0 means "none" and 1 - 3 means filter # filter_index - 1. */ int HAL_GetFilterSelect(HAL_DigitalHandle dio_port_handle, int32_t* status) { auto port = digitalPinHandles.Get(dio_port_handle, HAL_HandleEnum::DIO); if (port == nullptr) { *status = HAL_HANDLE_ERROR; return 0; } std::lock_guard sync(digitalDIOMutex); if (port->pin < kNumDigitalHeaders) { return digitalSystem->readFilterSelectHdr(port->pin, status); } else { return digitalSystem->readFilterSelectMXP(remapMXPChannel(port->pin), status); } } /** * Set the filter period for the specified filter index. * * Set the filter period in FPGA cycles. Even though there are 2 different * filter index domains (MXP vs HDR), ignore that distinction for now since it * compilicates the interface. That can be changed later. * * @param filter_index The filter index, 0 - 2. * @param value The number of cycles that the signal must not transition to be * counted as a transition. */ void HAL_SetFilterPeriod(int filter_index, uint32_t value, int32_t* status) { std::lock_guard sync(digitalDIOMutex); digitalSystem->writeFilterPeriodHdr(filter_index, value, status); if (*status == 0) { digitalSystem->writeFilterPeriodMXP(filter_index, value, status); } } /** * Get the filter period for the specified filter index. * * Get the filter period in FPGA cycles. Even though there are 2 different * filter index domains (MXP vs HDR), ignore that distinction for now since it * compilicates the interface. Set status to NiFpga_Status_SoftwareFault if the * filter values miss-match. * * @param filter_index The filter index, 0 - 2. * @param value The number of cycles that the signal must not transition to be * counted as a transition. */ uint32_t HAL_GetFilterPeriod(int filter_index, int32_t* status) { uint32_t hdr_period = 0; uint32_t mxp_period = 0; { std::lock_guard sync(digitalDIOMutex); hdr_period = digitalSystem->readFilterPeriodHdr(filter_index, status); if (*status == 0) { mxp_period = digitalSystem->readFilterPeriodMXP(filter_index, status); } } if (hdr_period != mxp_period) { *status = NiFpga_Status_SoftwareFault; return -1; } return hdr_period; } }