|
|
|
Contents: Motion commands, Setting the LEDs, Detecting completion, Moving the head, Effector names
The robot's operating system updates the states of all the effectors (servos, motors, LEDs, etc.) every few milliseconds. Each update is called a "frame", and can accommodate simultaneous changes to any number of effectors. On the AIBO, updates occur every 8 milliseconds and frames are buffered four at a time, so the application must have a new buffer available every 32 milliseconds; other robots may use different update intervals. In Tekkotsu these buffers of frames are produced by the MotionManager, whose job is to execute a collection of simultaneously active MotionCommands (MCs) of various types every few milliseconds. The results of these MotionCommands are assembled into a buffer that is passed to the operating system (Aperios for the AIBO, or Linux for other robots).
Suppose we want the robot to blink its LEDs on and off at a rate of once per second. What we need is a MotionCommand that will calculate new states for the LEDs each time the MotionManager asks for an update. LedMC, a subclass of both MotionCommand and LedEngine, performs this service. If we create an instance of LedMC, tell it the frequency at which to blink the LEDs, and add it to the MotionManager's list of active MCs, then it will do all the work for us. There's just one catch: our application is running in the Main process, while the MotionManager runs in a separate Motion process. This is necessary to assure that potentially lengthy computations taking place in Main don't prevent Motion from running every few milliseconds. So how can we communicate with our MotionCommand while at the same time making it available to the MotionManager?
The solution is to construct MotionCommands in a memory region that is shared by both processes. Because we have continuous access to the MotionCommand, we can change its parameters even while it's active, to tell it to do different things. But it's dangerous to modify a MotionCommand while the MotionManager is in the midst of invoking it. Therefore, Tekkotsu provides a mutual exclusion mechanism called an MMAccessor that temporarily locks out the MotionManager when we need to invoke a MotionCommand's member functions from within Main. Whenever we want to call such functions, we must lock down the MotionCommand by creating an MMAccessor first. Destroying the MMAccessor unlocks the MotionCommand.
There is one remaining wrinkle to the story. When a MotionCommand is passed to the MotionManager, it is assigned a unique ID called an MC_ID that identifies it within the MotionManager's active list. To lock the MotionCommand, we must pass this MC_ID value to the MMAccessor constructor. The MC_ID is also used when we tell the MotionManager to remove this MotionCommand from its active list. So the MC_ID must be saved somewhere. Normally it is kept in a protected data member within the Behavior instance so it can be shared by the doStart, doStop, and doEvent methods.
To summarize: MotionCommands must be instantiated in shared memory. An MC_ID, which is typically stored locally in the Behavior (not in shared memory), uniquely identifies the MotionCommand within the MotionManager's active list. Certain member functions of the MotionCommand will be called repeatedly from within the Motion process, by the MotionManager, to compute updated effector states. An MMAccessor, created in Main using the MC_ID, must be used to lock down an active MotionCommand so we can safely call its member functions from within the Main process. This allows us to modify the MotionCommand's parameters on the fly.
Exercise: Setting Up a MotionCommand to Cycle the LEDs We will use a LedMC to make some LEDs blink, and increase the rate when a button is pressed. On the AIBO will use the face LEDs; for the Qwerkbot we'll use all the LEDs. First, we must create a data member to store the MC_ID of the MotionCommand we'll be creating, and initialize it in the DstBehavior constructor.
We use the global variable
The expresion doStop() removes the LedMC instance from the Motion Manager's active list.
Inside doEvent, we will change the parameters of the running LedMC to alter its blink rate. To do this safely, we must first instantiate an MMAccessor, which we'll call leds_acc, using leds_id. This locks down the MotionCommand. Then we use the accessor to call the LedEngine::cycle method to change the blink period to 250 milliseconds for rapid blinking, or 1000 milliseconds to go back to slow blinking. The blink amplitude of 100.0 causes the LEDs to jump from full off to full on, or vice versa, instead of fading up or down gradually. (The cycle() function computes a sine wave. The LED amplitudes are bounded between 0 and 1.0, so multiplying the sine wave by 100.0 means the positive half of the wave will hit the 1.0 limit very quickly, producing a step function.)
Once again we are using a bit of syntactic sugar: MMAccessor overloads
the
Compile this code and boot the robot. Unpause it, and then activate
ButtonFlash. The MotionCommand you created is now active, but no
lights should be blinking because the default LedMC parameters tell it
to do nothing. Now press a button and the call to |
cycle
, but it will terminate automatically if the
parameters were set by flash
or cflash
,
which flashes the LEDs once.) A MotionCommand can signal that a
motion has completed by posting a status event whose source ID is
equal to the MotionCommand's MC_ID.
Explore more:
state->outputs[HeadOffset+TiltOffset]
. You will need
to include Shared/WordlState.h to do this.
RobotInfo::MaxOutputSpeed[]
. The software
limits on range of motion of each joint are stored in an array of
arrays named RobotInfo::outputRanges[][]
, which was
referenced in the head motion example in the preceding section. The
hard mechanical limits on joint motion are stored in a similar array
named RobotInfo::mechanicalLimits[][]
. The tables all
use the same indexing scheme. The convention for referring to a joint
in Tekkotsu is to specify its "offset" into these tables. Rather than
writing actual integers, Tekkotsu provides enum types so you can
express these offsets symbolically.
Here is how the numbering scheme works: Joints are grouped together in
the table based on the robot's anatomy. Thus, all the leg joints have
consecutive offsets, as do all the head joints, all the tail joints,
and so on. The order of the legs is defined by
RobotInfo::LegOrder_t
, whose values are 0 to 3, defined
symbolically as LFrLegOrder
, RFrLegOrder
,
LBkLegOrder
, and RBkLegOrder
. The number of
joints per leg is given by the constant
RobotInfo::NumLegJoints
, which is equal to 3 for current
AIBO models. There are also symbols defined for each leg offset
individually. So:
LFrLegOffset == LegOffset + LFrLegOrder * NumLegJoints
RFrLegOffset == LegOffset + RFrLegOrder * NumLegJoints
LBkLegOffset == LegOffset + LBkLegOrder * NumLegJoints
RBkLegOffset == LegOffset + RBkLegOrder * NumLegJoints
The order of joints within a leg is defined by
RobotInfo::REKOffset_t
, whose values are 0 to 2, defined
symbolically as RotatorOffset
,
ElevatorOffset
, and KneeOffset
. (Note: the
rotator and elevator joints are both located in the hip/shoulder; the
rotator moves the leg forward or back, while the elevator moves it in
or out.)
To refer to a specific leg joint, we start with the offset for the leg
the joint is on, and add in the offset for the desired joint relative
to that leg. So, for example, the offset for the right back knee is
given by: RBkLegOffset + KneeOffset
.
A similar convention applies to the head. The beginning of the head
joints is specified by RobotInfo::HeadOffset
. The joint
order is defined by RobotInfo::TPROffset_t
, which stands
for Tilt, Pan, and Roll. (Note: the ERS-7 head does not roll.
Instead, it has a second tilt joint called "nod". And the Qwerkbot
has only tilt and pan.) The TPROffset values for the head joints are
TiltOffset
, PanOffset
, and
RollOffset
(or NodOffset
for the ERS-7.)
So, for example, to access information about the head tilt joint we
would use the index value HeadOffset+TiltOffset
.
Joints are always referred to by their full indices, with one exception: the HeadPointerMC motion command, because it deals exclusively with the head, refers to joints by their TPROffset without adding in the value of HeadOffset. So if you want to set the head's pan angle using a HeadPointerMC, you would write something like:
head_acc->setJointValue(PanOffset, panvalue)
But if you wanted to set the pan angle as part of adjusting the
robot's entire body posture, you would use a PostureMC rather than a
HeadPointerMC, and you would refer to the pan joint using the full
offset calculation:
posture_acc->setOutputCmd(HeadOffset+PanOffset, panvalue)
The PostureMC motion command is discussed further in the chapter on Posture and Motion Sequences.
|
|
|