mirror of
https://github.com/DCC-EX/CommandStation-EX.git
synced 2024-12-24 21:21:24 +01:00
1bb7b5cc77
When writing to a PWM device (servo or LED for example), it is possible to request the target position in the call, or to ask for a SET or RESET position. In the latter case, the positions corresponding to SET and RESET must be known, i.e. preconfigured. Defaults were assigned for this, but because the correct values will depend on the hardware device being driven, the defaults have been removed. In addition, the <T> command, when defining a servo turnout, now configures the PWM positions (not required by <T> commands, but desirable for consistency with other commands).
268 lines
10 KiB
C++
268 lines
10 KiB
C++
/*
|
|
* © 2021, Neil McKechnie. All rights reserved.
|
|
*
|
|
* This file is part of DCC++EX API
|
|
*
|
|
* This is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* It is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with CommandStation. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "IODevice.h"
|
|
#include "I2CManager.h"
|
|
#include "DIAG.h"
|
|
|
|
// REGISTER ADDRESSES
|
|
static const byte PCA9685_MODE1=0x00; // Mode Register
|
|
static const byte PCA9685_FIRST_SERVO=0x06; /** low byte first servo register ON*/
|
|
static const byte PCA9685_PRESCALE=0xFE; /** Prescale register for PWM output frequency */
|
|
// MODE1 bits
|
|
static const byte MODE1_SLEEP=0x10; /**< Low power mode. Oscillator off */
|
|
static const byte MODE1_AI=0x20; /**< Auto-Increment enabled */
|
|
static const byte MODE1_RESTART=0x80; /**< Restart enabled */
|
|
|
|
static const float FREQUENCY_OSCILLATOR=25000000.0; /** Accurate enough for our purposes */
|
|
static const uint8_t PRESCALE_50HZ = (uint8_t)(((FREQUENCY_OSCILLATOR / (50.0 * 4096.0)) + 0.5) - 1);
|
|
static const uint32_t MAX_I2C_SPEED = 1000000L; // PCA9685 rated up to 1MHz I2C clock speed
|
|
|
|
// Predeclare helper function
|
|
static void writeRegister(byte address, byte reg, byte value);
|
|
|
|
// Create device driver instance.
|
|
void PCA9685::create(VPIN firstVpin, int nPins, uint8_t I2CAddress) {
|
|
new PCA9685(firstVpin, nPins, I2CAddress);
|
|
}
|
|
|
|
// Configure a port on the PCA9685.
|
|
bool PCA9685::_configure(VPIN vpin, ConfigTypeEnum configType, int paramCount, int params[]) {
|
|
if (configType != CONFIGURE_SERVO) return false;
|
|
if (paramCount != 5) return false;
|
|
#ifdef DIAG_IO
|
|
DIAG(F("PCA9685 Configure VPIN:%d Apos:%d Ipos:%d Profile:%d Duration:%d state:%d"),
|
|
vpin, params[0], params[1], params[2], params[3], params[4]);
|
|
#endif
|
|
|
|
int8_t pin = vpin - _firstVpin;
|
|
struct ServoData *s = _servoData[pin];
|
|
if (s == NULL) {
|
|
_servoData[pin] = (struct ServoData *)calloc(1, sizeof(struct ServoData));
|
|
s = _servoData[pin];
|
|
if (!s) return false; // Check for failed memory allocation
|
|
}
|
|
|
|
s->activePosition = params[0];
|
|
s->inactivePosition = params[1];
|
|
s->profile = params[2];
|
|
s->duration = params[3];
|
|
int state = params[4];
|
|
|
|
if (state != -1) {
|
|
// Position servo to initial state
|
|
_writeAnalogue(vpin, state ? s->activePosition : s->inactivePosition, 0, 0);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// Constructor
|
|
PCA9685::PCA9685(VPIN firstVpin, int nPins, uint8_t I2CAddress) {
|
|
_firstVpin = firstVpin;
|
|
_nPins = min(nPins, 16);
|
|
_I2CAddress = I2CAddress;
|
|
// To save RAM, space for servo configuration is not allocated unless a pin is used.
|
|
// Initialise the pointers to NULL.
|
|
for (int i=0; i<_nPins; i++)
|
|
_servoData[i] = NULL;
|
|
|
|
addDevice(this);
|
|
|
|
// Initialise structure used for setting pulse rate
|
|
requestBlock.setWriteParams(_I2CAddress, outputBuffer, sizeof(outputBuffer));
|
|
}
|
|
|
|
// Device-specific initialisation
|
|
void PCA9685::_begin() {
|
|
I2CManager.begin();
|
|
I2CManager.setClock(1000000); // Nominally able to run up to 1MHz on I2C
|
|
// In reality, other devices including the Arduino will limit
|
|
// the clock speed to a lower rate.
|
|
|
|
// Initialise I/O module here.
|
|
if (I2CManager.exists(_I2CAddress)) {
|
|
writeRegister(_I2CAddress, PCA9685_MODE1, MODE1_SLEEP | MODE1_AI);
|
|
writeRegister(_I2CAddress, PCA9685_PRESCALE, PRESCALE_50HZ); // 50Hz clock, 20ms pulse period.
|
|
writeRegister(_I2CAddress, PCA9685_MODE1, MODE1_AI);
|
|
writeRegister(_I2CAddress, PCA9685_MODE1, MODE1_RESTART | MODE1_AI);
|
|
// In theory, we should wait 500us before sending any other commands to each device, to allow
|
|
// the PWM oscillator to get running. However, we don't do any specific wait, as there's
|
|
// plenty of other stuff to do before we will send a command.
|
|
#if defined(DIAG_IO)
|
|
_display();
|
|
#endif
|
|
}
|
|
}
|
|
|
|
// Device-specific write function, invoked from IODevice::write().
|
|
// For this function, the configured profile is used.
|
|
void PCA9685::_write(VPIN vpin, int value) {
|
|
#ifdef DIAG_IO
|
|
DIAG(F("PCA9685 Write Vpin:%d Value:%d"), vpin, value);
|
|
#endif
|
|
int pin = vpin - _firstVpin;
|
|
if (value) value = 1;
|
|
|
|
struct ServoData *s = _servoData[pin];
|
|
if (s != NULL) {
|
|
// Use configured parameters
|
|
_writeAnalogue(vpin, value ? s->activePosition : s->inactivePosition, s->profile, s->duration);
|
|
} // else { /* ignorethe request */ }
|
|
}
|
|
|
|
// Device-specific writeAnalogue function, invoked from IODevice::writeAnalogue().
|
|
// Profile is as follows:
|
|
// Bit 7: 0=Set PWM to 0% to power off servo motor when finished
|
|
// 1=Keep PWM pulses on (better when using PWM to drive an LED)
|
|
// Bits 6-0: 0 Use specified duration (defaults to 0 deciseconds)
|
|
// 1 (Fast) Move servo in 0.5 seconds
|
|
// 2 (Medium) Move servo in 1.0 seconds
|
|
// 3 (Slow) Move servo in 2.0 seconds
|
|
// 4 (Bounce) Servo 'bounces' at extremes.
|
|
//
|
|
void PCA9685::_writeAnalogue(VPIN vpin, int value, uint8_t profile, uint16_t duration) {
|
|
#ifdef DIAG_IO
|
|
DIAG(F("PCA9685 WriteAnalogue Vpin:%d Value:%d Profile:%d Duration:%d"),
|
|
vpin, value, profile, duration);
|
|
#endif
|
|
int pin = vpin - _firstVpin;
|
|
if (value > 4095) value = 4095;
|
|
else if (value < 0) value = 0;
|
|
|
|
struct ServoData *s = _servoData[pin];
|
|
if (s == NULL) {
|
|
// Servo pin not configured, so configure now using defaults
|
|
s = _servoData[pin] = (struct ServoData *) calloc(sizeof(struct ServoData), 1);
|
|
if (s == NULL) return; // Check for memory allocation failure
|
|
s->activePosition = 0;
|
|
s->inactivePosition = 0;
|
|
s->currentPosition = value;
|
|
s->profile = Instant; // Use instant profile (but not this time)
|
|
}
|
|
|
|
// Animated profile. Initiate the appropriate action.
|
|
s->currentProfile = profile;
|
|
uint8_t profileValue = profile & ~NoPowerOff; // Mask off 'don't-power-off' bit.
|
|
s->numSteps = profileValue==Fast ? 10 : // 0.5 seconds
|
|
profileValue==Medium ? 20 : // 1.0 seconds
|
|
profileValue==Slow ? 40 : // 2.0 seconds
|
|
profileValue==Bounce ? sizeof(_bounceProfile)-1 : // ~ 1.5 seconds
|
|
duration * 2 + 1; // Convert from deciseconds (100ms) to refresh cycles (50ms)
|
|
s->stepNumber = 0;
|
|
s->toPosition = value;
|
|
s->fromPosition = s->currentPosition;
|
|
}
|
|
|
|
// _isBusy returns true if the device is currently in executing an animation,
|
|
// changing the output over a period of time.
|
|
bool PCA9685::_isBusy(VPIN vpin) {
|
|
int pin = vpin - _firstVpin;
|
|
struct ServoData *s = _servoData[pin];
|
|
if (s == NULL)
|
|
return false; // No structure means no animation!
|
|
else
|
|
return (s->stepNumber < s->numSteps);
|
|
}
|
|
|
|
void PCA9685::_loop(unsigned long currentMicros) {
|
|
if (currentMicros - _lastRefreshTime >= refreshInterval * 1000) {
|
|
for (int pin=0; pin<_nPins; pin++) {
|
|
updatePosition(pin);
|
|
}
|
|
_lastRefreshTime = currentMicros;
|
|
}
|
|
}
|
|
|
|
// Private function to reposition servo
|
|
// TODO: Could calculate step number from elapsed time, to allow for erratic loop timing.
|
|
void PCA9685::updatePosition(uint8_t pin) {
|
|
struct ServoData *s = _servoData[pin];
|
|
|
|
if (s == NULL) return; // No pin configuration/state data
|
|
|
|
if (s->numSteps == 0) return; // No animation in progress
|
|
|
|
if (s->stepNumber == 0 && s->fromPosition == s->toPosition) {
|
|
// Go straight to end of sequence, output final position.
|
|
s->stepNumber = s->numSteps-1;
|
|
}
|
|
|
|
if (s->stepNumber < s->numSteps) {
|
|
// Animation in progress, reposition servo
|
|
s->stepNumber++;
|
|
if ((s->currentProfile & ~NoPowerOff) == Bounce) {
|
|
// Retrieve step positions from array in flash
|
|
byte profileValue = GETFLASH(&_bounceProfile[s->stepNumber]);
|
|
s->currentPosition = map(profileValue, 0, 100, s->fromPosition, s->toPosition);
|
|
} else {
|
|
// All other profiles - calculate step by linear interpolation between from and to positions.
|
|
s->currentPosition = map(s->stepNumber, 0, s->numSteps, s->fromPosition, s->toPosition);
|
|
}
|
|
// Send servo command
|
|
writeDevice(pin, s->currentPosition);
|
|
} else if (s->stepNumber < s->numSteps + _catchupSteps) {
|
|
// We've finished animation, wait a little to allow servo to catch up
|
|
s->stepNumber++;
|
|
} else if (s->stepNumber == s->numSteps + _catchupSteps
|
|
&& s->currentPosition != 0) {
|
|
#ifdef IO_SWITCH_OFF_SERVO
|
|
if ((s->currentProfile & NoPowerOff) == 0) {
|
|
// Wait has finished, so switch off PWM to prevent annoying servo buzz
|
|
writeDevice(pin, 0);
|
|
}
|
|
#endif
|
|
s->numSteps = 0; // Done now.
|
|
}
|
|
}
|
|
|
|
// writeDevice takes a pin in range 0 to _nPins-1 within the device, and a value
|
|
// between 0 and 4095 for the PWM mark-to-period ratio, with 4095 being 100%.
|
|
void PCA9685::writeDevice(uint8_t pin, int value) {
|
|
#ifdef DIAG_IO
|
|
DIAG(F("PCA9685 I2C:x%x WriteDevice Pin:%d Value:%d"), _I2CAddress, pin, value);
|
|
#endif
|
|
// Wait for previous request to complete
|
|
requestBlock.wait();
|
|
// Set up new request.
|
|
outputBuffer[0] = PCA9685_FIRST_SERVO + 4 * pin;
|
|
outputBuffer[1] = 0;
|
|
outputBuffer[2] = (value == 4095 ? 0x10 : 0); // 4095=full on
|
|
outputBuffer[3] = value & 0xff;
|
|
outputBuffer[4] = value >> 8;
|
|
I2CManager.queueRequest(&requestBlock);
|
|
}
|
|
|
|
// Display details of this device.
|
|
void PCA9685::_display() {
|
|
DIAG(F("PCA9685 I2C:x%x Configured on Vpins:%d-%d"), _I2CAddress, (int)_firstVpin,
|
|
(int)_firstVpin+_nPins-1);
|
|
}
|
|
|
|
// Internal helper function for this device
|
|
static void writeRegister(byte address, byte reg, byte value) {
|
|
I2CManager.write(address, 2, reg, value);
|
|
}
|
|
|
|
// Profile for a bouncing signal or turnout
|
|
// The profile below is in the range 0-100% and should be combined with the desired limits
|
|
// of the servo set by _activePosition and _inactivePosition. The profile is symmetrical here,
|
|
// i.e. the bounce is the same on the down action as on the up action. First entry isn't used.
|
|
const byte FLASH PCA9685::_bounceProfile[30] =
|
|
{0,2,3,7,13,33,50,83,100,83,75,70,65,60,60,65,74,84,100,83,75,70,70,72,75,80,87,92,97,100};
|