- 
                Notifications
    You must be signed in to change notification settings 
- Fork 2
Collision Detection Project
This project aims to develop a system for detecting collisions within the LARMOR instrument at STFC's ISIS facility, using the IBEX control system.
Presently, experiments are executed using Genie Python scripts, which command the various axes of motion, as well as the DAE system and other instrument systems. Care must be taken to ensure that the motion axes are not allowed to collide with each other, or other parts of the instrument. This means that "soft" limits must be placed on each axis of motion, based on the equipment present on the instrument and determined by the instrument scientist.
There are certain instrument configurations and situations where setting these "soft" limits does not allow the full range of motion required, or where the limits on multiple axes become interdependent. In this case, it would be useful to re-calculate the "soft" limits dynamically, in order to improve equipment safety whilst also giving the scientists greater freedom.
This project aims to produce a system capable of:
- Developing a method for describing the geometry & motion of an instrument
- Detecting collisions which have occurred, and stopping motion
- Dynamically calculating the range of motion for each axis of motion and updating the "soft" limits for each axis
- Detecting a collision that is likely to occur in the future, due to moving multiple axes
The system must appear transparent to the instrument users, and to this end, should require minimal changes to experiment scripts.
The system will be prototyped using python, which is already used extensively as part of IBEX.
The system resides within the EPICS-inst_servers repository. Start/stop scripts are in the root, with the code under CollisionDetection.
This project assumes a development environment with a configured IBEX installation.
The system has been developed in PyCharm. For development purposes, the environment variables from the standard EPICS config_env.bat and from start_collision_detection_cmd.bat should be set in the PyCharm run configuration, with the script target set to main.py.
The system interfaces with the IBEX server using EPICS via the Channel Access protocol. Genie python is used for simple setting and getting of PVs, and the python ca module is used to monitor changes in PVs.
For collision detection, the system uses the Open Dynamics Engine, through the python module pyode. A wheel for which is located at https://www.lfd.uci.edu/~gohlke/pythonlibs/#ode
The system also produces a visual rendering of the system, for diagnostic and development purposes. This uses pygame and the pyopengl bindings, and needs glut.dll and glut32.dll, which are included in the repository and can be found here.
The system comprises four main parts:
- Collision detector
- Limit calculator
- PV server
- Graphic visualiser
Additionally the instrument geometry configuration must be described, and is loaded in from config.py.
The system uses the dial versions of the readback and limit PVs (DVAL, DRBV, DHLM, DLLM) to protect the collision detection algorithm from changes in the user coordinate system. See EPICS Motor Record.
The collision detector is responsible for stopping motion if a collision occurs. The collision detector must be executed frequently, to stop collisions as promptly as possible. The collision detector runs in a separate thread, to allow it to operate independently to the main program, and provides a CollisionDetector.collisions list for interfacing with the rest of the program.
The system uses the Open Dynamics Engine (ODE) to calculate whether any bodies have collided, using the function collide. A list of GeomBox objects are created, one for each body, each containing an ode.GeomBox object. The ode.Collide function is used to determine whether each pair of geometries has collided. The config.ignore list ensures that only the geometries of interest are analysed, reducing the computational load.
When a collision is detected, the system sends a .STOP message to each motor that is currently moving, using Genie python.
The system dynamically calculates limits by stepping each motor through its range of motion, and checking for collisions, using the auto_seek_limits function. This can be approached as a sequential search, where collide is evaluated at each step along the range of motion. Each direction is searched first with a coarse step, then with a fine step.
This works on the assumption that most collisions behave like a rectangular function, and remain collided for a significant portion of movement. Typically this is true for linear motion, but when objects are inclined or in rotary cases the duration of the rectangle function is short. This means that for any step size, the search is not guaranteed to find collision.
If the geometries of the computer model are sufficiently larger than the real-world system, the search step can be optimised.
For a given increase in size S applied to each face of the box:
modeled size = actual size + 2S
Assuming a head on collision and considering only linear movement of the seeking axis, a collision of the real world system occurs once the model has collided by at least 2S. Furthermore, taking two objects with an actual size of zero, and a modeled size of 2S, a "head-on" collision is maintained for 4S.
In the case of an inclined collision, the collision will persist for longer as the collision path through the centre of the object increases with angle.
In the case of a glancing collision, whereby the collision of the model does not infer a collision of the real world system, a collision of the model may or may not be detected, but the real-world system will never be at risk.
Therefore any search step of 4S or less will detect a real-world collision.
For movements which involve rotation however, the search step must be chosen to ensure that no point on the body moves by more than 4S in any direction. To achieve this, the system calculates the positions of each vertex of the body at each step. The magnitude of the move is calculated, and if greater than 4S, the search step can be reduced. The magnitude of the move is re-calculated and the step reduced until the step is less than or equal to 4S.
Once an appropriate step size has been found, the magnitude of the move need not be calculated further, reducing the computational load. This assumes that the movement of any body can be expressed as a linear function of the seeking axis - each subsequent step in the seeking process produces a movement of magnitude equal to the first step.
###PV Server
The system exposes some PVs for controlling operation, and getting feedback for the user through pv_server.py. The pcaspy module is used to achieve this, similarly to the BlockServer.
There is no access security in the EPICS sense. However, writing to read only PVs is ignored by the driver.
| PV Name | Access | Description | 
|---|---|---|
| AUTO_LIMIT | R/W | Automatically send new limits to all motors whenever new dynamic limits are calculated. If 0, the limits from config.pyare sent instead. Writing to this PV will cause the dynamic limits to be recalculated. | 
| AUTO_STOP | R/W | Automatically issue a .STOPcommand to any moving motors, when a collision is detected. If 0, no.STOPcommands will be sent. Writing to this PV will cause the dynamic limits to be recalculated. | 
| CALC | W | Writing any number to this PV will cause the dynamic limits to be recalculated. | 
| COARSE | R/W | The initial coarse step used when seeking limits. OVERSIZEis also updated so that the relationship4 * OVERSIZE = COARSEis maintained. Writing to this PV will cause the dynamic limits to be recalculated. | 
| COLLIDED | R | A list of values for each body. Value will be 1 is a collision has been detected on that body. | 
| FINE | R/W | The fine step used when seeking limits. Writing to this PV will cause the dynamic limits to be recalculated. | 
| HI_LIM | R | A list of the upper dynamic limits for each motor axis. | 
| LO_LIM | R | A list of the lower dynamic limits for each motor axis. | 
| MODE | R/W | A 3-bit binary number of the operating mode (see OperatingModeclass): bit 0 =AUTO_STOP, bit 1 =AUTO_LIMIT, bit 2 = close the program. Writing to this PV will cause the dynamic limits to be recalculated. | 
| MSG | R | A human readable message describing the current collision status. This could be added to the spangle banner. Typical messages include No collisions detected.,Collision expected in 0.9s - 1.0s,Collision on Name1, Name2 | 
| NAMES | R | A list of the names read from config.py | 
| OVERSIZE | R/W | The additional distance added to the faces of each bodies, to make collisions easier to detect. COARSEis also updated so that the relationship4 * OVERSIZE = COARSEis maintained. Writing to this PV will cause the dynamic limits to be recalculated. | 
| RAND | R | A random number for testing that the server is responsive | 
| SAFE | R | Value is 0 if any collisions have occurred, otherwise 1. | 
| TIME | R | The time taken to calculate the last set of dynamic limits | 
| TRAV_F | R | A list of the distance from the current position to the upper limit for each motor axis. | 
| TRAV_R | R | A list of the distance from the current position to the lower limit for each motor axis. Should always be a negative number. | 
| TRAVEL | R | A list of the distance to the closest dynamic limit for each motor axis. If the axis is at either of its limits, this will be 0. | 
The visualiser is started with the main program, and can be found in render.py.
The visualiser runs in its own thread, and iterates every 50 ms to draw each frame. It updates the graphic by monitoring the .DRBV PVs of each axis, and through a RenderParams object containing the soft limits, the collision status, and the duration of the most recent limit calculation.

The program can be controlled through the visualisation window in the same way as setting the MODE PV.
| Key | Function | 
|---|---|
| 1 | AUTO_STOPon. | 
| 2 | AUTO_STOPoff. | 
| 3 | AUTO_LIMITon. | 
| 4 | AUTO_LIMIToff. | 
The camera can also be re-positioned within the visualisation:
| Key | Function | 
|---|---|
| W | Forward | 
| A | Strafe left | 
| S | Backward | 
| D | Strafe right | 
| Q | Down | 
| E | Up | 
| Z | Rotate clockwise | 
| X | Rotate anti-clockwise | 
| Up | Pan up | 
| Down | Pan down | 
| Left | Pan left | 
| Right | Pan right | 
| Space | Reset to initial view | 
The geometry of the instrument is described in config.py. Generally the content of this file is a set of parameter lists, indexed against either the bodies in the mechanical system, or the motion axes. The various parameters are:
| Parameter | Indexed on | Description | 
|---|---|---|
| colors | Axis | A list of RGB tuples used when drawing the limits and current position of each axis, where (1, 1, 1)is white. The colours can be re-used ingeometries. Red(1, 0, 0)ought not be used as bodies are drawn in red if they have collided. | 
| geometries | Body | A list of dictcontaining thenameandsize, and optionallycolor, andpositionparameters of the body. For moving bodies, thepositionis not required, as it will be overridden whenever the system moves. Thecolordefaults to white. | 
| moves | Body | A list of functions which take a list of position values, and return a Transformationto describe the new position of each mechanical body. Alternatively (and more efficiently) a function which moves everything and yieldsTransformations at each step, which can then be iterated over in the same way as the list. | 
| pvs | Axis | A list of motor PVs for each axis. The order of these PVs is the order of the position values given to moves. | 
| hardlimits | Axis | The end limits of each axis of motion. Nominally the end of travel, though tighter limits can be imposed. The dynamically calculated limits are always within these values. | 
| ignore | N/A | A list of geometriesindex pairs which are not of interest. This is useful for bodies which are mechanically connected - we don't care if the carriage collides with it's slide. As this list can be long for a complicated moving system, this is best generated using a nestedforloop. | 
| coarse | N/A | The initial coarse limit seek step, which can be overridden by PV. | 
| fine | N/A | The initial fine limit seek step, which can be overridden by PV (but probably never needs to change from 0.5). | 
| oversize | N/A | The initial oversize parameter to apply to the bodies, which can be overriden by PV. The relationship oversize = coarse / 4should be maintained. |