forked from epics_driver_modules/motorBase
492 lines
16 KiB
C++
492 lines
16 KiB
C++
/*
|
|
FILENAME... SMChydraDriver.cpp
|
|
USAGE... Motor driver support for the Micos SMC hydra controller.
|
|
|
|
Note: This driver was tested with the Micos SMC hydra CM and
|
|
motor forms 0 (stepper) and 1 (linear).
|
|
|
|
*/
|
|
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <math.h>
|
|
|
|
#include <iocsh.h>
|
|
#include <epicsThread.h>
|
|
|
|
#include <asynOctetSyncIO.h>
|
|
|
|
#include "SMChydraDriver.h"
|
|
#include <epicsExport.h>
|
|
|
|
#define NINT(f) (int)((f)>0 ? (f)+0.5 : (f)-0.5)
|
|
|
|
/** Creates a new SMChydraController object.
|
|
* \param[in] portName The name of the asyn port that will be created for this driver
|
|
* \param[in] SMChydraPortName The name of the drvAsynSerialPort that was created previously to connect to the SMC hydra controller
|
|
* \param[in] numAxes The number of axes that this controller supports
|
|
* \param[in] movingPollPeriod The time between polls when any axis is moving
|
|
* \param[in] idlePollPeriod The time between polls when no axis is moving
|
|
*/
|
|
SMChydraController::SMChydraController(const char *portName, const char *SMChydraPortName, int numAxes,
|
|
double movingPollPeriod, double idlePollPeriod)
|
|
: asynMotorController(portName, numAxes, NUM_SMCHYDRA_PARAMS,
|
|
0, // No additional interfaces beyond those in base class
|
|
0, // No additional callback interfaces beyond those in base class
|
|
ASYN_CANBLOCK | ASYN_MULTIDEVICE,
|
|
1, // autoconnect
|
|
0, 0) // Default priority and stack size
|
|
{
|
|
int axis;
|
|
asynStatus status;
|
|
SMChydraAxis *pAxis;
|
|
static const char *functionName = "SMChydraController::SMChydraController";
|
|
|
|
/* Connect to SMC hydra controller */
|
|
status = pasynOctetSyncIO->connect(SMChydraPortName, 0, &pasynUserController_, NULL);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s: cannot connect to SMC hydra controller\n",
|
|
functionName);
|
|
}
|
|
for (axis=0; axis<numAxes; axis++) {
|
|
pAxis = new SMChydraAxis(this, axis);
|
|
}
|
|
|
|
startPoller(movingPollPeriod, idlePollPeriod, 2);
|
|
}
|
|
|
|
|
|
/** Creates a new SMChydraController object.
|
|
* Configuration command, called directly or from iocsh
|
|
* \param[in] portName The name of the asyn port that will be created for this driver
|
|
* \param[in] SMChydraPortName The name of the drvAsynIPPPort that was created previously to connect to the SMC hydra controller
|
|
* \param[in] numAxes The number of axes that this controller supports
|
|
* \param[in] movingPollPeriod The time in ms between polls when any axis is moving
|
|
* \param[in] idlePollPeriod The time in ms between polls when no axis is moving
|
|
*/
|
|
extern "C" int SMChydraCreateController(const char *portName, const char *SMChydraPortName, int numAxes,
|
|
int movingPollPeriod, int idlePollPeriod)
|
|
{
|
|
SMChydraController *pSMChydraController
|
|
= new SMChydraController(portName, SMChydraPortName, numAxes, movingPollPeriod/1000., idlePollPeriod/1000.);
|
|
pSMChydraController = NULL;
|
|
return(asynSuccess);
|
|
}
|
|
|
|
/** Reports on status of the driver
|
|
* \param[in] fp The file pointer on which report information will be written
|
|
* \param[in] level The level of report detail desired
|
|
*
|
|
* If details > 0 then information is printed about each axis.
|
|
* After printing controller-specific information it calls asynMotorController::report()
|
|
*/
|
|
void SMChydraController::report(FILE *fp, int level)
|
|
{
|
|
fprintf(fp, "SMC hydra motor driver %s, numAxes=%d, moving poll period=%f, idle poll period=%f\n",
|
|
this->portName, numAxes_, movingPollPeriod_, idlePollPeriod_);
|
|
|
|
// Call the base class method
|
|
asynMotorController::report(fp, level);
|
|
}
|
|
|
|
/** Returns a pointer to an SMChydraAxis object.
|
|
* Returns NULL if the axis number encoded in pasynUser is invalid.
|
|
* \param[in] pasynUser asynUser structure that encodes the axis index number. */
|
|
SMChydraAxis* SMChydraController::getAxis(asynUser *pasynUser)
|
|
{
|
|
return static_cast<SMChydraAxis*>(asynMotorController::getAxis(pasynUser));
|
|
}
|
|
|
|
/** Returns a pointer to an SMChydraAxis object.
|
|
* Returns NULL if the axis number encoded in pasynUser is invalid.
|
|
* \param[in] axisNo Axis index number. */
|
|
SMChydraAxis* SMChydraController::getAxis(int axisNo)
|
|
{
|
|
return static_cast<SMChydraAxis*>(asynMotorController::getAxis(axisNo));
|
|
}
|
|
|
|
|
|
// These are the SMChydraAxis methods
|
|
|
|
/** Creates a new SMChydraAxis object.
|
|
* \param[in] pC Pointer to the SMChydraController to which this axis belongs.
|
|
* \param[in] axisNo Index number of this axis, range 0 to pC->numAxes_-1.
|
|
*
|
|
* Initializes register numbers, etc.
|
|
*/
|
|
SMChydraAxis::SMChydraAxis(SMChydraController *pC, int axisNo)
|
|
: asynMotorAxis(pC, axisNo),
|
|
pC_(pC)
|
|
{
|
|
sprintf(pC_->outString_, "%i getmotor", (axisNo + 1));
|
|
pC_->writeReadController();
|
|
motorForm_ = atoi( (char *) &pC_->inString_ );
|
|
|
|
sprintf(pC_->outString_, "%i getpitch", (axisNo + 1));
|
|
pC_->writeReadController();
|
|
pitch_ = atof( (char *) &pC_->inString_ );
|
|
|
|
sprintf(pC_->outString_, "%i getpolepairs", (axisNo + 1));
|
|
pC_->writeReadController();
|
|
polePairs_ = atoi( (char *) &pC_->inString_ );
|
|
|
|
sprintf(pC_->outString_, "%i getclperiod", (axisNo + 1));
|
|
pC_->writeReadController();
|
|
clPeriod_ = atof( (char *) &pC_->inString_ );
|
|
|
|
switch (motorForm_)
|
|
{
|
|
case 0:
|
|
// Stepper motor
|
|
axisRes_ = pitch_ / ( 4.0 * polePairs_);
|
|
break;
|
|
|
|
case 1:
|
|
// Linear or torque motor
|
|
axisRes_ = clPeriod_;
|
|
break;
|
|
|
|
default:
|
|
// For now assume clPeriod_ works for other motor forms
|
|
axisRes_ = clPeriod_;
|
|
break;
|
|
}
|
|
|
|
/* Enable gain support so that the CNEN field can be used to send
|
|
the init command to clear a motor fault for stepper motors, even
|
|
though they lack closed-loop support. */
|
|
setIntegerParam(pC_->motorStatusGainSupport_, 1);
|
|
|
|
// Determine the travel limits (will change after homing)
|
|
sprintf(pC_->outString_, "%i getnlimit", (axisNo + 1));
|
|
pC_->writeReadController();
|
|
sscanf(pC_->inString_, "%lf %lf", &negTravelLimit_, &posTravelLimit_);
|
|
|
|
}
|
|
|
|
/** Reports on status of the axis
|
|
* \param[in] fp The file pointer on which report information will be written
|
|
* \param[in] level The level of report detail desired
|
|
*
|
|
* After printing device-specific information calls asynMotorAxis::report()
|
|
*/
|
|
void SMChydraAxis::report(FILE *fp, int level)
|
|
{
|
|
if (level > 0) {
|
|
fprintf(fp, " axis %d\n", axisNo_);
|
|
fprintf(fp, " motorForm %d\n", motorForm_);
|
|
fprintf(fp, " pitch %f\n", pitch_);
|
|
fprintf(fp, " polePairs %d\n", polePairs_);
|
|
fprintf(fp, " clPeriod %f\n", clPeriod_);
|
|
fprintf(fp, " axisRes %f\n", axisRes_);
|
|
fprintf(fp, " lowLimitConfig %d\n", lowLimitConfig_);
|
|
fprintf(fp, " highLimitConfig %d\n", highLimitConfig_);
|
|
fprintf(fp, " posTravelLimit %f\n", posTravelLimit_);
|
|
fprintf(fp, " negTravelLimit %f\n", negTravelLimit_);
|
|
}
|
|
|
|
// Call the base class method
|
|
asynMotorAxis::report(fp, level);
|
|
}
|
|
|
|
asynStatus SMChydraAxis::sendAccelAndVelocity(double acceleration, double velocity)
|
|
{
|
|
asynStatus status;
|
|
// static const char *functionName = "SMChydraAxis::sendAccelAndVelocity";
|
|
|
|
// Send the velocity
|
|
sprintf(pC_->outString_, "%f %i snv", (velocity * axisRes_), (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
|
|
// Send the acceleration
|
|
// acceleration is in units/sec/sec
|
|
sprintf(pC_->outString_, "%f %i sna", (acceleration * axisRes_), (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
|
|
asynStatus SMChydraAxis::move(double position, int relative, double baseVelocity, double slewVelocity, double acceleration)
|
|
{
|
|
asynStatus status;
|
|
// static const char *functionName = "SMChydraAxis::move";
|
|
|
|
status = sendAccelAndVelocity(acceleration, slewVelocity);
|
|
|
|
if (relative) {
|
|
sprintf(pC_->outString_, "%f %i nr", (position * axisRes_), (axisNo_ + 1));
|
|
} else {
|
|
sprintf(pC_->outString_, "%f %i nm", (position * axisRes_), (axisNo_ + 1));
|
|
}
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
asynStatus SMChydraAxis::home(double baseVelocity, double slewVelocity, double acceleration, int forwards)
|
|
{
|
|
asynStatus status;
|
|
// static const char *functionName = "SMChydraAxis::home";
|
|
|
|
status = sendAccelAndVelocity(acceleration, slewVelocity);
|
|
|
|
if (forwards) {
|
|
sprintf(pC_->outString_, "%i nrm", (axisNo_ + 1));
|
|
} else {
|
|
sprintf(pC_->outString_, "%i ncal", (axisNo_ + 1));
|
|
}
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
asynStatus SMChydraAxis::moveVelocity(double baseVelocity, double slewVelocity, double acceleration)
|
|
{
|
|
asynStatus status;
|
|
static const char *functionName = "SMChydraAxis::moveVelocity";
|
|
|
|
asynPrint(pasynUser_, ASYN_TRACE_FLOW,
|
|
"%s: baseVelocity=%f, slewVelocity=%f, acceleration=%f\n",
|
|
functionName, baseVelocity, slewVelocity, acceleration);
|
|
|
|
/* SMC hydra does not have jog command. Move to a limit*/
|
|
if (slewVelocity > 0.) {
|
|
status = sendAccelAndVelocity(acceleration, slewVelocity);
|
|
sprintf(pC_->outString_, "%f %i nm", posTravelLimit_, (axisNo_ + 1));
|
|
} else {
|
|
status = sendAccelAndVelocity(acceleration, (slewVelocity * -1.0));
|
|
sprintf(pC_->outString_, "%f %i nm", negTravelLimit_, (axisNo_ + 1));
|
|
}
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
asynStatus SMChydraAxis::stop(double acceleration )
|
|
{
|
|
asynStatus status;
|
|
//static const char *functionName = "SMChydraAxis::stop";
|
|
|
|
// Set stop deceleration (will be overridden by accel if accel is higher)
|
|
sprintf(pC_->outString_, "%f %i ssd", (acceleration * axisRes_), (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
|
|
sprintf(pC_->outString_, "%i nabort", (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
asynStatus SMChydraAxis::setPosition(double position)
|
|
{
|
|
asynStatus status;
|
|
//static const char *functionName = "SMChydraAxis::setPosition";
|
|
|
|
sprintf(pC_->outString_, "%f %i setnpos", (position * axisRes_), (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
return status;
|
|
}
|
|
|
|
asynStatus SMChydraAxis::setClosedLoop(bool closedLoop)
|
|
{
|
|
asynStatus status = asynSuccess;
|
|
//static const char *functionName = "SMChydraAxis::setClosedLoop";
|
|
|
|
switch (motorForm_)
|
|
{
|
|
case 0:
|
|
// Stepper motor
|
|
|
|
// reinit to clear a motor fault when CNEN=1
|
|
if (closedLoop)
|
|
{
|
|
sprintf(pC_->outString_, "%i init", (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
|
|
// a delay is required after the init command is sent
|
|
epicsThreadSleep(0.2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
// Linear or torque motor
|
|
|
|
// enable closed-loop control
|
|
sprintf(pC_->outString_, "%i %i setcloop", closedLoop ? 1:0, (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
|
|
// reinit so the closed-loop setting takes effect
|
|
sprintf(pC_->outString_, "%i init", (axisNo_ + 1));
|
|
status = pC_->writeController();
|
|
|
|
// a delay is required after the init command is sent
|
|
epicsThreadSleep(0.2);
|
|
|
|
break;
|
|
|
|
default:
|
|
// Do nothing
|
|
break;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
/** Polls the axis.
|
|
* This function reads the motor position, the limit status, the home status, the moving status,
|
|
* and the drive power-on status.
|
|
* It calls setIntegerParam() and setDoubleParam() for each item that it polls,
|
|
* and then calls callParamCallbacks() at the end.
|
|
* \param[out] moving A flag that is set indicating that the axis is moving (true) or done (false). */
|
|
asynStatus SMChydraAxis::poll(bool *moving)
|
|
{
|
|
int done;
|
|
int driveOn;
|
|
int lowLimit;
|
|
int highLimit;
|
|
int ignoreLowLimit;
|
|
int ignoreHighLimit;
|
|
int axisStatus=-1;
|
|
double position=0.0;
|
|
asynStatus comStatus;
|
|
|
|
static const char *functionName = "SMChydraAxis::poll";
|
|
|
|
// Read the current motor position
|
|
sprintf(pC_->outString_, "%i np", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
if (comStatus) goto skip;
|
|
// The response string is a double
|
|
position = atof( (char *) &pC_->inString_);
|
|
setDoubleParam(pC_->motorPosition_, (position / axisRes_) );
|
|
setDoubleParam(pC_->motorEncoderPosition_, (position / axisRes_) );
|
|
|
|
// Read the status of this motor
|
|
sprintf(pC_->outString_, "%i nst", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
if (comStatus) goto skip;
|
|
// The response string is an int
|
|
axisStatus = atoi( (char *) &pC_->inString_);
|
|
|
|
// Check the moving bit
|
|
done = !(axisStatus & 0x1);
|
|
setIntegerParam(pC_->motorStatusDone_, done);
|
|
setIntegerParam(pC_->motorStatusMoving_, !done);
|
|
*moving = done ? false:true;
|
|
|
|
// Read the commanded velocity and acceleration
|
|
sprintf(pC_->outString_, "%i gnv", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
|
|
sprintf(pC_->outString_, "%i gna", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
|
|
// Check the limit bit (0x40)
|
|
if (axisStatus & 0x40)
|
|
{
|
|
asynPrint(this->pasynUser_, ASYN_TRACEIO_DRIVER,
|
|
"%s: axis %i limit indicator active.\n",
|
|
functionName, (axisNo_ + 1));
|
|
|
|
// query limits?
|
|
}
|
|
|
|
// Check the e-stop bit (0x80)
|
|
if (axisStatus & 0x80)
|
|
{
|
|
asynPrint(this->pasynUser_, ASYN_TRACEIO_DRIVER,
|
|
"%s: axis %i emergency stopped.\n",
|
|
functionName, (axisNo_ + 1));
|
|
}
|
|
|
|
// Check the e-stop switch active bit (0x200)
|
|
if (axisStatus & 0x200)
|
|
{
|
|
asynPrint(this->pasynUser_, ASYN_TRACEIO_DRIVER,
|
|
"%s: axis %i emergency stop switch active.\n",
|
|
functionName, (axisNo_ + 1));
|
|
setIntegerParam(pC_->motorStatusProblem_, 1);
|
|
}
|
|
else{
|
|
setIntegerParam(pC_->motorStatusProblem_, 0);
|
|
}
|
|
|
|
// Check the device busy bit (0x400)
|
|
if (axisStatus & 0x400)
|
|
{
|
|
asynPrint(this->pasynUser_, ASYN_TRACE_ERROR,
|
|
"%s: axis %i device is busy - move commands discarded.\n",
|
|
functionName, (axisNo_ + 1));
|
|
}
|
|
|
|
// Read the limit status
|
|
// Note: calibration switch = low limit; range measure switch = high limit
|
|
// also need to read the switch confiruation to see if limits are ignored"
|
|
|
|
// Read switch confiruation
|
|
// Bit 0: polarity (0 = NO, 1 = NC)
|
|
// Bit 1: mask (0 = enabled, 1 = disabled)
|
|
sprintf(pC_->outString_, "%i getsw", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
if (comStatus) goto skip;
|
|
sscanf(pC_->inString_, "%i %i", &lowLimitConfig_, &highLimitConfig_);
|
|
ignoreLowLimit = lowLimitConfig_ & 0x2;
|
|
ignoreHighLimit = highLimitConfig_ & 0x2;
|
|
|
|
// Read status of switches 0=inactive 1=active
|
|
sprintf(pC_->outString_, "%i getswst", (axisNo_ + 1));
|
|
comStatus = pC_->writeReadController();
|
|
if (comStatus) goto skip;
|
|
// The response string is of the form "0 0"
|
|
sscanf(pC_->inString_, "%i %i", &lowLimit, &highLimit);
|
|
//
|
|
if (ignoreLowLimit)
|
|
setIntegerParam(pC_->motorStatusLowLimit_, 0);
|
|
else
|
|
setIntegerParam(pC_->motorStatusLowLimit_, lowLimit);
|
|
|
|
if (ignoreHighLimit)
|
|
setIntegerParam(pC_->motorStatusHighLimit_, 0);
|
|
else
|
|
setIntegerParam(pC_->motorStatusHighLimit_, highLimit);
|
|
|
|
/*setIntegerParam(pC_->motorStatusAtHome_, limit);*/
|
|
|
|
// Check the drive power bit (0x100)
|
|
driveOn = (axisStatus & 0x100) ? 0 : 1;
|
|
setIntegerParam(pC_->motorStatusPowerOn_, driveOn);
|
|
setIntegerParam(pC_->motorStatusProblem_, 0);
|
|
|
|
skip:
|
|
setIntegerParam(pC_->motorStatusProblem_, comStatus ? 1:0);
|
|
callParamCallbacks();
|
|
return comStatus ? asynError : asynSuccess;
|
|
}
|
|
|
|
/** Code for iocsh registration */
|
|
static const iocshArg SMChydraCreateControllerArg0 = {"Port name", iocshArgString};
|
|
static const iocshArg SMChydraCreateControllerArg1 = {"SMC hydra port name", iocshArgString};
|
|
static const iocshArg SMChydraCreateControllerArg2 = {"Number of axes", iocshArgInt};
|
|
static const iocshArg SMChydraCreateControllerArg3 = {"Moving poll period (ms)", iocshArgInt};
|
|
static const iocshArg SMChydraCreateControllerArg4 = {"Idle poll period (ms)", iocshArgInt};
|
|
static const iocshArg * const SMChydraCreateControllerArgs[] = {&SMChydraCreateControllerArg0,
|
|
&SMChydraCreateControllerArg1,
|
|
&SMChydraCreateControllerArg2,
|
|
&SMChydraCreateControllerArg3,
|
|
&SMChydraCreateControllerArg4};
|
|
static const iocshFuncDef SMChydraCreateControllerDef = {"SMChydraCreateController", 5, SMChydraCreateControllerArgs};
|
|
static void SMChydraCreateControllerCallFunc(const iocshArgBuf *args)
|
|
{
|
|
SMChydraCreateController(args[0].sval, args[1].sval, args[2].ival, args[3].ival, args[4].ival);
|
|
}
|
|
|
|
static void SMChydraRegister(void)
|
|
{
|
|
iocshRegister(&SMChydraCreateControllerDef, SMChydraCreateControllerCallFunc);
|
|
}
|
|
|
|
extern "C" {
|
|
epicsExportRegistrar(SMChydraRegister);
|
|
}
|