forked from epics_driver_modules/motorBase
1004 lines
38 KiB
C++
1004 lines
38 KiB
C++
/*
|
|
FILENAME... XPSMotorDriver.cpp
|
|
USAGE... Newport XPS EPICS asyn motor device driver
|
|
|
|
Version: $Revision: 19717 $
|
|
Modified By: $Author: sluiter $
|
|
Last Modified: $Date: 2009-12-09 10:21:24 -0600 (Wed, 09 Dec 2009) $
|
|
HeadURL: $URL: https://subversion.xor.aps.anl.gov/synApps/trunk/support/motor/vstub/motorApp/NewportSrc/XPSMotorDriver.cpp $
|
|
*/
|
|
|
|
/*
|
|
Original Author: Mark Rivers
|
|
*/
|
|
|
|
/*
|
|
Copyright (c) 2005 University of Chicago and the Regents of the University of
|
|
California. All rights reserved.
|
|
|
|
synApps is distributed subject to the following license conditions:
|
|
SOFTWARE LICENSE AGREEMENT
|
|
Software: synApps
|
|
Versions: Release 4-5 and higher.
|
|
|
|
1. The "Software", below, refers to synApps (in either source code, or
|
|
binary form and accompanying documentation). Each licensee is addressed
|
|
as "you" or "Licensee."
|
|
|
|
2. The copyright holders shown above and their third-party licensors hereby
|
|
grant Licensee a royalty-free nonexclusive license, subject to the
|
|
limitations stated herein and U.S. Government license rights.
|
|
|
|
3. You may modify and make a copy or copies of the Software for use within
|
|
your organization, if you meet the following conditions:
|
|
1. Copies in source code must include the copyright notice and this
|
|
Software License Agreement.
|
|
2. Copies in binary form must include the copyright notice and this
|
|
Software License Agreement in the documentation and/or other
|
|
materials provided with the copy.
|
|
|
|
4. You may modify a copy or copies of the Software or any portion of it, thus
|
|
forming a work based on the Software, and distribute copies of such work
|
|
outside your organization, if you meet all of the following conditions:
|
|
1. Copies in source code must include the copyright notice and this
|
|
Software License Agreement;
|
|
2. Copies in binary form must include the copyright notice and this
|
|
Software License Agreement in the documentation and/or other
|
|
materials provided with the copy;
|
|
3. Modified copies and works based on the Software must carry
|
|
prominent notices stating that you changed specified portions of
|
|
the Software.
|
|
|
|
5. Portions of the Software resulted from work developed under a
|
|
U.S. Government contract and are subject to the following license:
|
|
the Government is granted for itself and others acting on its behalf a
|
|
paid-up, nonexclusive, irrevocable worldwide license in this computer
|
|
software to reproduce, prepare derivative works, and perform publicly and
|
|
display publicly.
|
|
|
|
6. WARRANTY DISCLAIMER. THE SOFTWARE IS SUPPLIED "AS IS" WITHOUT WARRANTY OF
|
|
ANY KIND. THE COPYRIGHT HOLDERS, THEIR THIRD PARTY LICENSORS, THE UNITED
|
|
STATES, THE UNITED STATES DEPARTMENT OF ENERGY, AND THEIR EMPLOYEES: (1)
|
|
DISCLAIM ANY WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO
|
|
ANY IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
|
PURPOSE, TITLE OR NON-INFRINGEMENT, (2) DO NOT ASSUME ANY LEGAL LIABILITY
|
|
OR RESPONSIBILITY FOR THE ACCURACY, COMPLETENESS, OR USEFULNESS OF THE
|
|
SOFTWARE, (3) DO NOT REPRESENT THAT USE OF THE SOFTWARE WOULD NOT
|
|
INFRINGE PRIVATELY OWNED RIGHTS, (4) DO NOT WARRANT THAT THE SOFTWARE WILL
|
|
FUNCTION UNINTERRUPTED, THAT IT IS ERROR-FREE OR THAT ANY ERRORS WILL BE
|
|
CORRECTED.
|
|
|
|
7. LIMITATION OF LIABILITY. IN NO EVENT WILL THE COPYRIGHT HOLDERS, THEIR
|
|
THIRD PARTY LICENSORS, THE UNITED STATES, THE UNITED STATES DEPARTMENT OF
|
|
ENERGY, OR THEIR EMPLOYEES: BE LIABLE FOR ANY INDIRECT, INCIDENTAL,
|
|
CONSEQUENTIAL, SPECIAL OR PUNITIVE DAMAGES OF ANY KIND OR NATURE,
|
|
INCLUDING BUT NOT LIMITED TO LOSS OF PROFITS OR LOSS OF DATA, FOR ANY
|
|
REASON WHATSOEVER, WHETHER SUCH LIABILITY IS ASSERTED ON THE BASIS OF
|
|
CONTRACT, TORT (INCLUDING NEGLIGENCE OR STRICT LIABILITY), OR OTHERWISE,
|
|
EVEN IF ANY OF SAID PARTIES HAS BEEN WARNED OF THE POSSIBILITY OF SUCH
|
|
LOSS OR DAMAGES.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
#include <epicsTime.h>
|
|
#include <epicsThread.h>
|
|
#include <epicsExport.h>
|
|
#include <epicsString.h>
|
|
#include <iocsh.h>
|
|
|
|
#include "XPSController.h"
|
|
#include "XPS_C8_drivers.h"
|
|
#include "xps_ftp.h"
|
|
|
|
static const char *driverName = "XPSController";
|
|
|
|
/** Struct for a list of strings describing the different corrector types possible on the XPS.*/
|
|
typedef struct {
|
|
char *PIPosition;
|
|
char *PIDFFVelocity;
|
|
char *PIDFFAcceleration;
|
|
char *PIDDualFFVoltage;
|
|
char *NoCorrector;
|
|
} CorrectorTypes_t;
|
|
|
|
const static CorrectorTypes_t CorrectorTypes = {
|
|
"PositionerCorrectorPIPosition",
|
|
"PositionerCorrectorPIDFFVelocity",
|
|
"PositionerCorrectorPIDFFAcceleration",
|
|
"PositionerCorrectorPIDDualFFVoltage",
|
|
"NoCorrector"
|
|
};
|
|
|
|
/** This is controlled via the XPSEnableSetPosition function (available via the IOC shell). */
|
|
static int doSetPosition = 1;
|
|
|
|
/**
|
|
* Parameter to control the sleep time used when setting position.
|
|
* A function called XPSSetPosSleepTime(int) (millisec parameter)
|
|
* is available in the IOC shell to control this.
|
|
*/
|
|
static double setPosSleepTime = 0.5;
|
|
|
|
/* Constants used for FTP to the XPS */
|
|
#define TRAJECTORY_DIRECTORY "/Admin/public/Trajectories"
|
|
#define MAX_FILENAME_LEN 256
|
|
#define MAX_MESSAGE_LEN 256
|
|
#define TRAJECTORY_FILE "TrajectoryScan.trj"
|
|
#define DEFAULT_FTP_USERNAME "Administrator"
|
|
#define DEFAULT_FTP_PASSWORD "Administrator"
|
|
#define DEFAULT_PROFILE_GROUPNAME "Group1"
|
|
|
|
/* The maximum size of the item names in gathering, e.g. "GROUP2.POSITIONER1.CurrentPosition" */
|
|
#define MAX_GATHERING_AXIS_STRING 60
|
|
/* Number of items per axis */
|
|
#define NUM_GATHERING_ITEMS 2
|
|
/* Total length of gathering configuration string */
|
|
#define MAX_GATHERING_STRING MAX_GATHERING_AXIS_STRING * NUM_GATHERING_ITEMS * MAX_AXES
|
|
// Maximum number of bytes that GatheringDataMultipleLinesGet() can return
|
|
#define GATHERING_MAX_READ_LEN 65536
|
|
|
|
/** Deadband to use for the velocity comparison with zero. */
|
|
#define XPS_VELOCITY_DEADBAND 0.0000001
|
|
|
|
#define XPSC8_END_OF_RUN_MINUS 0x80000100
|
|
#define XPSC8_END_OF_RUN_PLUS 0x80000200
|
|
|
|
#define TCP_TIMEOUT 2.0
|
|
#define MAX(a,b) ((a)>(b)? (a): (b))
|
|
#define MIN(a,b) ((a)<(b)? (a): (b))
|
|
|
|
|
|
XPSController::XPSController(const char *portName, const char *IPAddress, int IPPort,
|
|
int numAxes, double movingPollPeriod, double idlePollPeriod)
|
|
: asynMotorController(portName, numAxes, NUM_XPS_PARAMS,
|
|
asynInt32Mask | asynFloat64Mask | asynUInt32DigitalMask,
|
|
asynInt32Mask | asynFloat64Mask | asynUInt32DigitalMask,
|
|
ASYN_CANBLOCK | ASYN_MULTIDEVICE,
|
|
1, // autoconnect
|
|
0, 0) // Default priority and stack size
|
|
{
|
|
static const char *functionName = "XPSController";
|
|
|
|
IPAddress_ = epicsStrDup(IPAddress);
|
|
IPPort_ = IPPort;
|
|
pAxes_ = (XPSAxis **)(asynMotorController::pAxes_);
|
|
|
|
// Create controller-specific parameters
|
|
createParam(XPSMinJerkString, asynParamFloat64, &XPSMinJerk_);
|
|
createParam(XPSMaxJerkString, asynParamFloat64, &XPSMaxJerk_);
|
|
createParam(XPSStatusString, asynParamInt32, &XPSStatus_);
|
|
|
|
// This socket is used for polling by the controller and all axes
|
|
pollSocket_ = TCP_ConnectToServer((char *)IPAddress, IPPort, TCP_TIMEOUT);
|
|
if (pollSocket_ < 0) {
|
|
printf("%s:%s: error calling TCP_ConnectToServer for pollSocket\n",
|
|
driverName, functionName);
|
|
}
|
|
|
|
// This socket is used for moving motors during profile moves
|
|
// Each axis also has its own moveSocket
|
|
moveSocket_ = TCP_ConnectToServer((char *)IPAddress, IPPort, TCP_TIMEOUT);
|
|
if (moveSocket_ < 0) {
|
|
printf("%s:%s: error calling TCP_ConnectToServer for moveSocket\n",
|
|
driverName, functionName);
|
|
}
|
|
|
|
// Set a default username and password for ftp
|
|
ftpUsername_ = epicsStrDup(DEFAULT_FTP_USERNAME);
|
|
ftpPassword_ = epicsStrDup(DEFAULT_FTP_PASSWORD);
|
|
profileGroupName_ = epicsStrDup(DEFAULT_PROFILE_GROUPNAME);
|
|
|
|
FirmwareVersionGet(pollSocket_, firmwareVersion_);
|
|
|
|
/* Create the poller thread for this controller
|
|
* NOTE: at this point the axis objects don't yet exist, but the poller tolerates this */
|
|
startPoller(movingPollPeriod, idlePollPeriod, 10);
|
|
}
|
|
|
|
void XPSController::report(FILE *fp, int level)
|
|
{
|
|
int axis;
|
|
XPSAxis *pAxis;
|
|
|
|
fprintf(fp, "XPS motor driver %s, numAxes=%d, firmware version=%s, moving poll period=%f, idle poll period=%f\n",
|
|
this->portName, numAxes_, firmwareVersion_, movingPollPeriod_, idlePollPeriod_);
|
|
|
|
if (level > 0) {
|
|
for (axis=0; axis<numAxes_; axis++) {
|
|
pAxis = getAxis(axis);
|
|
fprintf(fp, " axis %d\n"
|
|
" name = %s\n"
|
|
" step size = %g\n"
|
|
" poll socket = %d, moveSocket = %d\n"
|
|
" status = %d\n",
|
|
pAxis->axisNo_,
|
|
pAxis->positionerName_,
|
|
pAxis->stepSize_,
|
|
pAxis->pollSocket_, pAxis->moveSocket_,
|
|
pAxis->axisStatus_);
|
|
}
|
|
}
|
|
|
|
// Call the base class method
|
|
asynMotorController::report(fp, level);
|
|
}
|
|
|
|
/**
|
|
* Perform a deferred move (a coordinated group move) on all the axes in a group.
|
|
* @param pController Pointer to XPSController structure.
|
|
* @param groupName Pointer to string naming the group on which to perform the group move.
|
|
* @return motor driver status code.
|
|
*/
|
|
asynStatus XPSController::processDeferredMovesInGroup(char *groupName)
|
|
{
|
|
double positions[XPS_MAX_AXES];
|
|
int positions_index = 0;
|
|
int first_loop = 1;
|
|
int axis = 0;
|
|
int NbPositioners = 0;
|
|
int relativeMove = 0;
|
|
int status = 0;
|
|
XPSAxis *pAxis=NULL;
|
|
|
|
/* Loop over all axes in this controller. */
|
|
for (axis=0; axis<numAxes_; axis++) {
|
|
pAxis = getAxis(axis);
|
|
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"Executing deferred move on XPS: %s, Group: %s\n",
|
|
this->portName, groupName);
|
|
|
|
/* Ignore axes in other groups. */
|
|
if (!strcmp(pAxis->groupName_, groupName)) {
|
|
if (first_loop) {
|
|
/* Get the number of axes in this group. */
|
|
NbPositioners = pAxis->isInGroup();
|
|
first_loop = 0;
|
|
}
|
|
|
|
/* Set relative flag for the actual move at the end of the funtion.*/
|
|
if (pAxis->deferredRelative_) {
|
|
relativeMove = 1;
|
|
}
|
|
|
|
/* Build position buffer.*/
|
|
if (pAxis->deferredMove_) {
|
|
positions[positions_index] =
|
|
pAxis->deferredRelative_ ? (pAxis->setpointPosition_ + pAxis->deferredPosition_) : pAxis->deferredPosition_;
|
|
} else {
|
|
positions[positions_index] =
|
|
pAxis->deferredRelative_ ? 0 : pAxis->setpointPosition_;
|
|
}
|
|
|
|
/* Reset deferred flag. */
|
|
/* We need to do this for the XPS, because we cannot do partial group moves. Every axis
|
|
in the group will be included the next time we do a group move. */
|
|
pAxis->deferredMove_ = 0;
|
|
|
|
/* Next axis in this group. */
|
|
positions_index++;
|
|
}
|
|
}
|
|
|
|
/* Send the group move command. */
|
|
if (relativeMove) {
|
|
status = GroupMoveRelative(pAxis->moveSocket_,
|
|
groupName,
|
|
NbPositioners,
|
|
positions);
|
|
} else {
|
|
status = GroupMoveAbsolute(pAxis->moveSocket_,
|
|
groupName,
|
|
NbPositioners,
|
|
positions);
|
|
}
|
|
|
|
if (status!=0) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"Error peforming GroupMoveAbsolute/Relative in processDeferredMovesInGroup. XPS Return code: %d\n",
|
|
status);
|
|
return asynError;
|
|
}
|
|
|
|
return asynSuccess;
|
|
|
|
}
|
|
|
|
/**
|
|
* Process deferred moves for a controller and groups.
|
|
* This function calculates which unique groups in the controller
|
|
* and passes the controller pointer and group name to processDeferredMovesInGroup.
|
|
* @return motor driver status code.
|
|
*/
|
|
asynStatus XPSController::processDeferredMoves()
|
|
{
|
|
asynStatus status = asynError;
|
|
int axis = 0;
|
|
int i = 0;
|
|
int dealWith = 0;
|
|
/* Array to cache up to XPS_MAX_AXES group names. Don't initialise to null */
|
|
char *groupNames[XPS_MAX_AXES];
|
|
char *blankGroupName = " ";
|
|
XPSAxis *pAxis;
|
|
|
|
/* Clear group name cache. */
|
|
for (i=0; i<XPS_MAX_AXES; i++) {
|
|
groupNames[i] = blankGroupName;
|
|
}
|
|
|
|
/* Loop over axes, testing for unique groups. */
|
|
for (axis=0; axis<numAxes_; axis++) {
|
|
pAxis = getAxis(axis);
|
|
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"Processing deferred moves on XPS: %s\n", this->portName);
|
|
|
|
/* Call processDeferredMovesInGroup only once for each group on this controller.
|
|
Positioners in the same group may not be adjacent in list, so we have to test for this. */
|
|
for (i=0; i<XPS_MAX_AXES; i++) {
|
|
if (strcmp(pAxis->groupName_, groupNames[i])) {
|
|
dealWith++;
|
|
groupNames[i] = pAxis->groupName_;
|
|
}
|
|
}
|
|
if (dealWith == XPS_MAX_AXES) {
|
|
dealWith = 0;
|
|
/* Group name was not in cache, so deal with this group. */
|
|
status = this->processDeferredMovesInGroup(pAxis->groupName_);
|
|
}
|
|
/* Next axis, and potentially next group.*/
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
asynStatus XPSController::writeFloat64(asynUser *pasynUser, epicsFloat64 value)
|
|
{
|
|
int function = pasynUser->reason;
|
|
int status = asynSuccess;
|
|
XPSAxis *pAxis = this->getAxis(pasynUser);
|
|
double deviceValue;
|
|
static const char *functionName = "writeFloat64";
|
|
|
|
/* Set the parameter and readback in the parameter library. */
|
|
status = pAxis->setDoubleParam(function, value);
|
|
|
|
if (function == motorResolution_)
|
|
{
|
|
pAxis->stepSize_ = value;
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: Set XPS %s, axis %d stepSize to %f\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, value);
|
|
}
|
|
else if (function == motorLowLimit_)
|
|
{
|
|
deviceValue = value*pAxis->stepSize_;
|
|
/* We need to read the current highLimit because otherwise we could be setting it to an invalid value */
|
|
status = PositionerUserTravelLimitsGet(pAxis->pollSocket_,
|
|
pAxis->positionerName_,
|
|
&pAxis->lowLimit_, &pAxis->highLimit_);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: motorAxisSetDouble[%s,%d]: error performing PositionerUserTravelLimitsGet for lowLim status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, status);
|
|
goto done;
|
|
}
|
|
status = PositionerUserTravelLimitsSet(pAxis->pollSocket_,
|
|
pAxis->positionerName_,
|
|
deviceValue, pAxis->highLimit_);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: motorAxisSetDouble[%s,%d]: error performing PositionerUserTravelLimitsSet for lowLim=%f status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, deviceValue, status);
|
|
goto done;
|
|
}
|
|
pAxis->lowLimit_ = deviceValue;
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: Set XPS %s, axis %d low limit to %f\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, deviceValue);
|
|
status = asynSuccess;
|
|
} else if (function == motorHighLimit_)
|
|
{
|
|
deviceValue = value*pAxis->stepSize_;
|
|
/* We need to read the current highLimit because otherwise we could be setting it to an invalid value */
|
|
status = PositionerUserTravelLimitsGet(pAxis->pollSocket_,
|
|
pAxis->positionerName_,
|
|
&pAxis->lowLimit_, &pAxis->highLimit_);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: motorAxisSetDouble[%s,%d]: error performing PositionerUserTravelLimitsGet for highLim status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, status);
|
|
goto done;
|
|
}
|
|
status = PositionerUserTravelLimitsSet(pAxis->pollSocket_,
|
|
pAxis->positionerName_,
|
|
pAxis->lowLimit_, deviceValue);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: motorAxisSetDouble[%s,%d]: error performing PositionerUserTravelLimitsSet for highLim=%f status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, deviceValue, status);
|
|
goto done;
|
|
}
|
|
pAxis->highLimit_ = deviceValue;
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: Set XPS %s, axis %d high limit to %f\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, deviceValue);
|
|
status = asynSuccess;
|
|
} else if (function == motorPgain_)
|
|
{
|
|
status = pAxis->setPID(&value, 0);
|
|
} else if (function == motorIgain_)
|
|
{
|
|
status = pAxis->setPID(&value, 1);
|
|
} else if (function == motorDgain_)
|
|
{
|
|
status = pAxis->setPID(&value, 2);
|
|
} else {
|
|
/* Call base class method */
|
|
status = asynMotorController::writeFloat64(pasynUser, value);
|
|
}
|
|
done:
|
|
return (asynStatus)status;
|
|
}
|
|
|
|
|
|
asynStatus XPSController::writeInt32(asynUser *pasynUser, epicsInt32 value)
|
|
{
|
|
int function = pasynUser->reason;
|
|
int status = asynSuccess;
|
|
XPSAxis *pAxis = this->getAxis(pasynUser);
|
|
static const char *functionName = "writeInt32";
|
|
|
|
/* Set the parameter and readback in the parameter library. This may be overwritten when we read back the
|
|
* status at the end, but that's OK */
|
|
status = pAxis->setIntegerParam(function, value);
|
|
|
|
if (function == motorSetClosedLoop_)
|
|
{
|
|
if (value) {
|
|
status = GroupMotionEnable(pAxis->pollSocket_, pAxis->groupName_);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: [%s,%d]: error calling GroupMotionEnable status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, status);
|
|
} else {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: motorAxisSetInteger set XPS %s, axis %d closed loop enable\n",
|
|
driverName, functionName, portName, pAxis->axisNo_);
|
|
}
|
|
} else {
|
|
status = GroupMotionDisable(pAxis->pollSocket_, pAxis->groupName_);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: [%s,%d]: error calling GroupMotionDisable status=%d\n",
|
|
driverName, functionName, portName, pAxis->axisNo_, status);
|
|
} else {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: motorAxisSetInteger set XPS %s, axis %d closed loop disable\n",
|
|
driverName, functionName, portName, pAxis->axisNo_);
|
|
}
|
|
}
|
|
} else if (function == motorDeferMoves_)
|
|
{
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s%s: Setting deferred move mode on XPS %s to %d\n",
|
|
driverName, functionName, portName, value);
|
|
if (value == 0.0 && this->movesDeferred_ != 0) {
|
|
status = this->processDeferredMoves();
|
|
}
|
|
this->movesDeferred_ = value;
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s%s: Deferred moved failed on XPS %s, status=%d\n",
|
|
driverName, functionName, portName, status);
|
|
status = asynError;
|
|
}
|
|
} else {
|
|
/* Call base class method */
|
|
status = asynMotorController::writeInt32(pasynUser, value);
|
|
}
|
|
return (asynStatus)status;
|
|
}
|
|
|
|
/** Returns a pointer to an XPSAxis object.
|
|
* Returns NULL if the axis number encoded in pasynUser is invalid.
|
|
* \param[in] pasynUser asynUser structure that encodes the axis index number. */
|
|
XPSAxis* XPSController::getAxis(asynUser *pasynUser)
|
|
{
|
|
int axisNo;
|
|
|
|
getAddress(pasynUser, &axisNo);
|
|
return getAxis(axisNo);
|
|
}
|
|
|
|
/** Returns a pointer to an XPSAxis object.
|
|
* Returns NULL if the axis number is invalid.
|
|
* \param[in] axisNo Axis index number. */
|
|
XPSAxis* XPSController::getAxis(int axisNo)
|
|
{
|
|
if ((axisNo < 0) || (axisNo >= numAxes_)) return NULL;
|
|
return pAxes_[axisNo];
|
|
}
|
|
|
|
/* Function to build, install and verify trajectory */
|
|
asynStatus XPSController::buildProfile()
|
|
{
|
|
FILE *trajFile;
|
|
int i, j;
|
|
int status;
|
|
int nPoints;
|
|
int nElements;
|
|
double trajVel;
|
|
double D0, D1, T0, T1;
|
|
int ftpSocket;
|
|
char fileName[MAX_FILENAME_LEN];
|
|
char buildMessage[MAX_MESSAGE_LEN];
|
|
int buildStatus;
|
|
double distance;
|
|
double maxVelocity[XPS_MAX_AXES];
|
|
double maxAcceleration[XPS_MAX_AXES];
|
|
double maxVelocityActual;
|
|
double maxAccelerationActual;
|
|
double minPositionActual, maxPositionActual;
|
|
double minProfile, maxProfile;
|
|
double lowLimit, highLimit;
|
|
double minJerkTime[XPS_MAX_AXES], maxJerkTime[XPS_MAX_AXES];
|
|
double preTimeMax, postTimeMax;
|
|
double preVelocity[XPS_MAX_AXES], postVelocity[XPS_MAX_AXES];
|
|
double preDistance[XPS_MAX_AXES], postDistance[XPS_MAX_AXES];
|
|
double time;
|
|
int moveAxis[XPS_MAX_AXES];
|
|
static const char *functionName = "buildProfile";
|
|
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: entry\n",
|
|
driverName, functionName);
|
|
|
|
setIntegerParam(profileBuildState_, PROFILE_BUILD_BUSY);
|
|
setIntegerParam(profileBuildStatus_, PROFILE_STATUS_UNDEFINED);
|
|
strcpy(buildMessage, "");
|
|
callParamCallbacks();
|
|
|
|
/* We create trajectories with an extra element at the beginning and at the end.
|
|
* The distance and time of the first element is defined so that the motors will
|
|
* accelerate from 0 to the velocity of the first "real" element at their
|
|
* maximum allowed acceleration.
|
|
* Similarly, the distance and time of last element is defined so that the
|
|
* motors will decelerate from the velocity of the last "real" element to 0
|
|
* at the maximum allowed acceleration. */
|
|
|
|
/* Compute the velocity of each motor during the first real trajectory element,
|
|
* and the time required to reach this velocity. */
|
|
preTimeMax = 0.;
|
|
postTimeMax = 0.;
|
|
getIntegerParam(profileNumPoints_, &nPoints);
|
|
|
|
/* Zero values since axes may not be used */
|
|
for (j=0; j<numAxes_; j++) {
|
|
preVelocity[j] = 0.;
|
|
postVelocity[j] = 0.;
|
|
getIntegerParam(j, profileUseAxis_, &moveAxis[j]);
|
|
}
|
|
for (j=0; j<numAxes_; j++) {
|
|
if (!moveAxis[j]) continue;
|
|
status = PositionerSGammaParametersGet(pollSocket_, pAxes_[j]->positionerName_,
|
|
&maxVelocity[j], &maxAcceleration[j],
|
|
&minJerkTime[j], &maxJerkTime[j]);
|
|
if (status) {
|
|
sprintf(buildMessage, "Error calling positionerSGammaParametersSet, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
|
|
/* The calculation using maxAcceleration read from controller below
|
|
* is "correct" but subject to roundoff errors when sending ASCII commands
|
|
* to XPS. Reduce acceleration 10% to account for this. */
|
|
maxAcceleration[j] *= 0.9;
|
|
|
|
/* Note: the preDistance and postDistance numbers computed here are
|
|
* in user coordinates, not XPS coordinates, because they are used for
|
|
* EPICS moves at the start and end of the scan */
|
|
distance = pAxes_[j]->profilePositions_[1] - pAxes_[j]->profilePositions_[0];
|
|
preVelocity[j] = distance/profileTimes_[0];
|
|
time = fabs(preVelocity[j]) / maxAcceleration[j];
|
|
preTimeMax = MAX(preTimeMax, time);
|
|
distance = pAxes_[j]->profilePositions_[nPoints-1] -
|
|
pAxes_[j]->profilePositions_[nPoints-2];
|
|
postVelocity[j] = distance/profileTimes_[nPoints-1];
|
|
time = fabs(postVelocity[j]) / maxAcceleration[j];
|
|
postTimeMax = MAX(postTimeMax, time);
|
|
}
|
|
|
|
/* Compute the distance that each motor moves during its acceleration phase.
|
|
* Only move it this far. */
|
|
for (j=0; j<numAxes_; j++) {
|
|
preDistance[j] = 0.5 * preVelocity[j] * preTimeMax;
|
|
postDistance[j] = 0.5 * postVelocity[j] * postTimeMax;
|
|
}
|
|
|
|
/* Create the profile file */
|
|
trajFile = fopen (TRAJECTORY_FILE, "w");
|
|
|
|
/* Create the initial acceleration element */
|
|
fprintf(trajFile,"%f", preTimeMax);
|
|
for (j=0; j<numAxes_; j++) {
|
|
fprintf(trajFile,", %f, %f", preDistance[j], preVelocity[j]);
|
|
}
|
|
fprintf(trajFile,"\n");
|
|
|
|
/* The number of profile elements in the file is nPoints-1 */
|
|
nElements = nPoints - 1;
|
|
for (i=0; i<nElements; i++) {
|
|
T0 = profileTimes_[i];
|
|
if (i < nElements-1)
|
|
T1 = profileTimes_[i+1];
|
|
else
|
|
T1 = T0;
|
|
for (j=0; j<numAxes_; j++) {
|
|
D0 = pAxes_[j]->profilePositions_[i+1] -
|
|
pAxes_[j]->profilePositions_[i];
|
|
if (i < nElements-1)
|
|
D1 = pAxes_[j]->profilePositions_[i+2] -
|
|
pAxes_[j]->profilePositions_[i+1];
|
|
else
|
|
D1 = D0;
|
|
|
|
/* Average either side of the point */
|
|
trajVel = ((D0 + D1) / (T0 + T1));
|
|
if (!moveAxis[j]) {
|
|
D0 = 0.0; /* Axis turned off*/
|
|
trajVel = 0.0;
|
|
}
|
|
|
|
if (j == 0) fprintf(trajFile,"%f", profileTimes_[i]);
|
|
fprintf(trajFile,", %f, %f",D0,trajVel);
|
|
if (j == (numAxes_-1)) fprintf(trajFile,"\n");
|
|
}
|
|
}
|
|
|
|
/* Create the final acceleration element. Final velocity must be 0. */
|
|
fprintf(trajFile,"%f", postTimeMax);
|
|
for (j=0; j<numAxes_; j++) {
|
|
fprintf(trajFile,", %f, %f", postDistance[j], 0.);
|
|
}
|
|
fprintf(trajFile,"\n");
|
|
fclose (trajFile);
|
|
|
|
/* FTP the trajectory file from the local directory to the XPS */
|
|
status = ftpConnect(IPAddress_, ftpUsername_, ftpPassword_, &ftpSocket);
|
|
if (status) {
|
|
sprintf(buildMessage, "Error calling ftpConnect, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
status = ftpChangeDir(ftpSocket, TRAJECTORY_DIRECTORY);
|
|
if (status) {
|
|
sprintf(buildMessage, "Error calling ftpChangeDir, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
status = ftpStoreFile(ftpSocket, TRAJECTORY_FILE);
|
|
if (status) {
|
|
sprintf(buildMessage, "Error calling ftpStoreFile, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
status = ftpDisconnect(ftpSocket);
|
|
if (status) {
|
|
sprintf(buildMessage, "Error calling ftpDisconnect, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
|
|
/* Verify trajectory */
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: calling MultipleAxesPVTVerification(%d, %s, %s)\n",
|
|
driverName, functionName, pollSocket_, profileGroupName_, TRAJECTORY_FILE);
|
|
status = MultipleAxesPVTVerification(pollSocket_, profileGroupName_, TRAJECTORY_FILE);
|
|
|
|
switch (-status) {
|
|
case 0:
|
|
strcpy(buildMessage, " ");
|
|
break;
|
|
case 69:
|
|
strcpy(buildMessage, "Acceleration Too High");
|
|
break;
|
|
case 68:
|
|
strcpy(buildMessage, "Velocity Too High");
|
|
break;
|
|
case 70:
|
|
strcpy(buildMessage, "Final Velocity Non Zero");
|
|
break;
|
|
case 75:
|
|
strcpy(buildMessage, "Negative or Null Delta Time");
|
|
break;
|
|
default:
|
|
sprintf(buildMessage, "Unknown trajectory verify error=%d", status);
|
|
break;
|
|
}
|
|
|
|
/* Read dynamic parameters*/
|
|
for (j=0; j<numAxes_; j++) {
|
|
maxVelocityActual = 0;
|
|
maxAccelerationActual = 0;
|
|
status = MultipleAxesPVTVerificationResultGet(pollSocket_,
|
|
pAxes_[j]->positionerName_, fileName,
|
|
&minPositionActual, &maxPositionActual,
|
|
&maxVelocityActual, &maxAccelerationActual);
|
|
if (status) {
|
|
sprintf(buildMessage, "MultipleAxesPVTVerificationResultGet error for axis %s, status=%d\n",
|
|
pAxes_[j]->positionerName_, status);
|
|
goto done;
|
|
}
|
|
/* Check that the trajectory won't exceed the software limits
|
|
* The XPS does not check this because the trajectory is defined in relative moves and it does
|
|
* not know where we will be in absolute coordinates when we execute the trajectory */
|
|
status = PositionerUserTravelLimitsGet(pollSocket_,
|
|
pAxes_[j]->positionerName_,
|
|
&lowLimit,
|
|
&highLimit);
|
|
minProfile = pAxes_[j]->profilePositions_[0] + minPositionActual;
|
|
if (minProfile < lowLimit) {
|
|
sprintf(buildMessage, "Low soft limit violation for axis %s, position=%f, limit=%f\n",
|
|
pAxes_[j]->positionerName_, minProfile, lowLimit);
|
|
goto done;
|
|
}
|
|
maxProfile = pAxes_[j]->profilePositions_[0] + maxPositionActual;
|
|
if (maxProfile > lowLimit) {
|
|
sprintf(buildMessage, "High soft limit violation for axis %s, position=%f, limit=%f\n",
|
|
pAxes_[j]->positionerName_, maxProfile, highLimit);
|
|
goto done;
|
|
}
|
|
}
|
|
done:
|
|
buildStatus = status ? PROFILE_STATUS_FAILURE : PROFILE_STATUS_SUCCESS;
|
|
setIntegerParam(profileBuildStatus_, buildStatus);
|
|
setStringParam(profileBuildMessage_, buildMessage);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: %s\n",
|
|
driverName, functionName, buildMessage);
|
|
}
|
|
/* Clear build command. This is a "busy" record, don't want to do this until build is complete. */
|
|
setIntegerParam(profileBuild_, 0);
|
|
setIntegerParam(profileBuildState_, PROFILE_BUILD_DONE);
|
|
callParamCallbacks();
|
|
return status ? asynError : asynSuccess;
|
|
}
|
|
|
|
/* Function to execute trajectory */
|
|
asynStatus XPSController::executeProfile()
|
|
{
|
|
return asynSuccess;
|
|
}
|
|
|
|
/* Function to readback trajectory */
|
|
asynStatus XPSController::readbackProfile()
|
|
{
|
|
char readbackMessage[MAX_MESSAGE_LEN];
|
|
int numPulses;
|
|
char* buffer=NULL;
|
|
char* bptr, *tptr;
|
|
int currentSamples, maxSamples;
|
|
double setpointPosition, actualPosition;
|
|
int readbackStatus;
|
|
int status;
|
|
int i, j;
|
|
int nitems;
|
|
int numRead=0, numInBuffer, numChars;
|
|
int moveAxis[XPS_MAX_AXES];
|
|
static const char *functionName = "buildProfile";
|
|
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: entry\n",
|
|
driverName, functionName);
|
|
|
|
setIntegerParam(profileBuildState_, PROFILE_BUILD_BUSY);
|
|
setIntegerParam(profileBuildStatus_, PROFILE_STATUS_UNDEFINED);
|
|
strcpy(readbackMessage, "");
|
|
callParamCallbacks();
|
|
|
|
status = getIntegerParam(profileNumPulses_, &numPulses);
|
|
|
|
/* Erase the readback and error arrays */
|
|
for (j=0; j<numAxes_; j++) {
|
|
memset(pAxes_[j]->profileReadbacks_, 0, maxProfilePoints_*sizeof(double));
|
|
memset(pAxes_[j]->profileFollowingErrors_, 0, maxProfilePoints_*sizeof(double));
|
|
getIntegerParam(j, profileUseAxis_, &moveAxis[j]);
|
|
}
|
|
/* Read the number of lines of gathering */
|
|
status = GatheringCurrentNumberGet(pollSocket_, ¤tSamples, &maxSamples);
|
|
if (status != 0) {
|
|
sprintf(readbackMessage, "Error calling GatherCurrentNumberGet, status=%d\n", status);
|
|
goto done;
|
|
}
|
|
if (currentSamples != numPulses) {
|
|
sprintf(readbackMessage, "Error, numPulses=%d, currentSamples=%d\n", numPulses, currentSamples);
|
|
goto done;
|
|
}
|
|
buffer = (char *)calloc(GATHERING_MAX_READ_LEN, sizeof(char));
|
|
numInBuffer = 0;
|
|
for (numRead=0; numRead<currentSamples;) {
|
|
/* Read the next buffer */
|
|
/* Try to read all the remaining points */
|
|
status = -1;
|
|
numInBuffer = currentSamples - numRead;
|
|
while (status && (numInBuffer > 0)) {
|
|
status = GatheringDataMultipleLinesGet(pollSocket_, numRead, numInBuffer, buffer);
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_FLOW,
|
|
"%s:%s: GatheringDataMultipleLinesGet, status=%d, numInBuffer=%d\n",
|
|
driverName, functionName, status, numInBuffer);
|
|
if (status) numInBuffer /= 2;
|
|
}
|
|
if (numInBuffer == 0) {
|
|
sprintf(readbackMessage, "Error reading gathering data, numInBuffer = 0\n");
|
|
goto done;
|
|
}
|
|
bptr = buffer;
|
|
for (i=0; i<numInBuffer; i++) {
|
|
/* Find the next \n and replace with null */
|
|
tptr = strstr(bptr, "\n");
|
|
if (tptr) *tptr = 0;
|
|
for (j=0; j<numAxes_; j++) {
|
|
if (!moveAxis[j]) continue;
|
|
nitems = sscanf(bptr, "%lf;%lf%n",
|
|
&setpointPosition, &actualPosition, &numChars);
|
|
bptr += numChars+1;
|
|
if (nitems != NUM_GATHERING_ITEMS) {
|
|
sprintf(readbackMessage, "Error reading Gathering.dat file, nitems=%d, should be %d\n",
|
|
nitems, NUM_GATHERING_ITEMS);
|
|
goto done;
|
|
}
|
|
// Note, these positions are in controller units, need to be converted to user units
|
|
pAxes_[j]->profileFollowingErrors_[numRead] = actualPosition - setpointPosition;
|
|
pAxes_[j]->profileReadbacks_[numRead] = actualPosition;
|
|
}
|
|
numRead++;
|
|
bptr = tptr + 1;
|
|
}
|
|
}
|
|
|
|
done:
|
|
if (buffer) free(buffer);
|
|
/* Call the base class method that converts from controller to user units and posts the arrays */
|
|
asynMotorController::readbackProfile();
|
|
readbackStatus = status ? PROFILE_STATUS_FAILURE : PROFILE_STATUS_SUCCESS;
|
|
setIntegerParam(profileReadbackStatus_, readbackStatus);
|
|
setStringParam(profileReadbackMessage_, readbackMessage);
|
|
setIntegerParam(profileActualPulses_, numRead);
|
|
setIntegerParam(profileNumReadbacks_, numRead);
|
|
if (status) {
|
|
asynPrint(this->pasynUserSelf, ASYN_TRACE_ERROR,
|
|
"%s:%s: %s\n",
|
|
driverName, functionName, readbackMessage);
|
|
}
|
|
/* Clear readback command. This is a "busy" record, don't want to do this until build is complete. */
|
|
setIntegerParam(profileReadback_, 0);
|
|
setIntegerParam(profileReadbackState_, PROFILE_READBACK_DONE);
|
|
callParamCallbacks();
|
|
return status ? asynError : asynSuccess;
|
|
}
|
|
|
|
|
|
|
|
/** The following functions have C linkage, and can be called directly or from iocsh */
|
|
extern "C" {
|
|
/**
|
|
* Function to enable/disable the write down of position to the
|
|
* XPS controller. Call this function at IOC shell.
|
|
* @param setPos 0=disable, 1=enable
|
|
*/
|
|
/* void XPSEnableSetPosition(int setPos)
|
|
{
|
|
doSetPosition = setPos;
|
|
}
|
|
*/
|
|
/**
|
|
* Function to set the threadSleep time used when setting the XPS position.
|
|
* The sleep is performed after the axes are initialised, to take account of any
|
|
* post initialisation wobble.
|
|
* @param posSleep The time in miliseconds to sleep.
|
|
*/
|
|
/* void XPSSetPosSleepTime(int posSleep)
|
|
{
|
|
setPosSleepTime = (double)posSleep / 1000.0;
|
|
}
|
|
*/
|
|
asynStatus XPSCreate(const char *portName, const char *IPAddress, int IPPort,
|
|
int numAxes, int movingPollPeriod, int idlePollPeriod)
|
|
{
|
|
XPSController *pXPSController
|
|
= new XPSController(portName, IPAddress, IPPort, numAxes, movingPollPeriod/1000., idlePollPeriod/1000.);
|
|
pXPSController = NULL;
|
|
return(asynSuccess);
|
|
}
|
|
|
|
asynStatus XPSCreateAxis(const char *XPSName, /* specify which controller by port name */
|
|
int axis, /* axis number 0-7 */
|
|
const char *positionerName, /* groupName.positionerName e.g. Diffractometer.Phi */
|
|
int stepsPerUnit) /* steps per user unit */
|
|
{
|
|
XPSController *pC;
|
|
XPSAxis *pAxis;
|
|
static const char *functionName = "XPSCreateAxis";
|
|
|
|
pC = (XPSController*) findAsynPortDriver(XPSName);
|
|
if (!pC) {
|
|
printf("%s:%s: Error port %s not found\n",
|
|
driverName, functionName, XPSName);
|
|
return asynError;
|
|
}
|
|
pAxis = new XPSAxis(pC, axis, positionerName, 1./stepsPerUnit);
|
|
pAxis = NULL;
|
|
return(asynSuccess);
|
|
}
|
|
|
|
/* Code for iocsh registration */
|
|
|
|
/* XPSCreate */
|
|
static const iocshArg XPSCreateArg0 = {"Controller port name", iocshArgString};
|
|
static const iocshArg XPSCreateArg1 = {"IP address", iocshArgString};
|
|
static const iocshArg XPSCreateArg2 = {"IP port", iocshArgInt};
|
|
static const iocshArg XPSCreateArg3 = {"Number of axes", iocshArgInt};
|
|
static const iocshArg XPSCreateArg4 = {"Moving poll rate (ms)", iocshArgInt};
|
|
static const iocshArg XPSCreateArg5 = {"Idle poll rate (ms)", iocshArgInt};
|
|
static const iocshArg * const XPSCreateArgs[6] = {&XPSCreateArg0,
|
|
&XPSCreateArg1,
|
|
&XPSCreateArg2,
|
|
&XPSCreateArg2,
|
|
&XPSCreateArg4,
|
|
&XPSCreateArg5};
|
|
static const iocshFuncDef createXPS = {"XPSCreate", 6, XPSCreateArgs};
|
|
static void createXPSCallFunc(const iocshArgBuf *args)
|
|
{
|
|
XPSCreate(args[0].sval, args[1].sval, args[2].ival,
|
|
args[3].ival, args[4].ival, args[5].ival);
|
|
}
|
|
|
|
|
|
/* XPSCreateAxis */
|
|
static const iocshArg XPSCreateAxisArg0 = {"Controller port name", iocshArgString};
|
|
static const iocshArg XPSCreateAxisArg1 = {"Axis number", iocshArgInt};
|
|
static const iocshArg XPSCreateAxisArg2 = {"Axis name", iocshArgString};
|
|
static const iocshArg XPSCreateAxisArg3 = {"Steps per unit", iocshArgInt};
|
|
static const iocshArg * const XPSCreateAxisArgs[4] = {&XPSCreateAxisArg0,
|
|
&XPSCreateAxisArg1,
|
|
&XPSCreateAxisArg2,
|
|
&XPSCreateAxisArg3};
|
|
static const iocshFuncDef createXPSAxis = {"XPSCreateAxis", 4, XPSCreateAxisArgs};
|
|
|
|
static void createXPSAxisCallFunc(const iocshArgBuf *args)
|
|
{
|
|
XPSCreateAxis(args[0].sval, args[1].ival, args[2].sval, args[3].ival);
|
|
}
|
|
|
|
|
|
/* void XPSEnableSetPosition(int setPos) */
|
|
static const iocshArg XPSEnableSetPositionArg0 = {"Set Position Flag", iocshArgInt};
|
|
static const iocshArg * const XPSEnableSetPositionArgs[1] = {&XPSEnableSetPositionArg0};
|
|
static const iocshFuncDef xpsEnableSetPosition = {"XPSEnableSetPosition", 1, XPSEnableSetPositionArgs};
|
|
static void xpsEnableSetPositionCallFunc(const iocshArgBuf *args)
|
|
{
|
|
// XPSEnableSetPosition(args[0].ival);
|
|
}
|
|
|
|
/* void XPSSetPosSleepTime(int posSleep) */
|
|
static const iocshArg XPSSetPosSleepTimeArg0 = {"Set Position Sleep Time", iocshArgInt};
|
|
static const iocshArg * const XPSSetPosSleepTimeArgs[1] = {&XPSSetPosSleepTimeArg0};
|
|
static const iocshFuncDef xpsSetPosSleepTime = {"XPSSetPosSleepTime", 1, XPSSetPosSleepTimeArgs};
|
|
static void xpsSetPosSleepTimeCallFunc(const iocshArgBuf *args)
|
|
{
|
|
// XPSSetPosSleepTime(args[0].ival);
|
|
}
|
|
|
|
|
|
static void XPSRegister3(void)
|
|
{
|
|
iocshRegister(&createXPS, createXPSCallFunc);
|
|
iocshRegister(&createXPSAxis, createXPSAxisCallFunc);
|
|
// iocshRegister(&xpsEnableSetPosition, xpsEnableSetPositionCallFunc);
|
|
// iocshRegister(&xpsSetPosSleepTime, xpsSetPosSleepTimeCallFunc);
|
|
}
|
|
|
|
epicsExportRegistrar(XPSRegister3);
|
|
} // extern "C"
|