The chassis is a pretty straightforward design, implementing three stages to carry all of the necessary components. The bottom stage of the chassis is used to secure the ball bearing in place, so that three points of contact are established between the robot and the ground. The middle stage is used to house the motor components and the cascade lift mechanism. The top stage is primarily used to house electrical components, power supplies, and wiring. The platforms are all laser cut from ⅛” thick duron. Furthermore, the top and middle platforms are connected via four 3-D printed legs, using PLA. These materials are selected to be lightweight, cost-efficient, and optimal for iterative prototyping.
With the use of 2 DC motors to drive the robot and a stepper motor to raise and lower the cascade lift, we designed and 3-D printed multiple mounting components from PLA. Some of the components were used to fasten the motors to the chassis, and others were used to stabilize the shafts from the respective motors to reduce cantilevering.
The cascade lift mechanism is a 2-stage mechanism that is driven by the stepper motor to raise the magnetized CRATE to the designated level. With various lift stages secured to linear shafts, the entire assembly is connected by a series of open-loop pulleys that are of fixed length, which means that driving the motor to move one stage up will gradually elevate the further stages, creating a continuous movement that gradually raises the CRATE along the vertical axis. The design of the cascade was chosen to fit within the 32x32 cm size constraint at the start of the game round, as the collapsed mechanism possesses the advantage of being spatially conservative.
Considering that the magnet strength is not equivalent across the different faces of the CRATEs, we overcame this problem by implementing a servo motor. When the robot wants to dispense a CRATE into the STACK, it would rotate the CRATE outwards 90º, then rotating the motor back to the original starting position to implement shear force to disengage the magnet.
The IR sensor mount is used to hold the PHOTO_NPN phototransistor at the height level of the IR beacon, in order to best measure the IR frequency emitted from the beacon positioned in the STACKs. The sensor mount also bears a hole at the front of the design, ensuring that the arcade-style game button can be inset for easy access. To best avoid the interference of ambient light, we 3-D printed this component with black PLA.
In order to reduce battery interference and to best organize the electrical components of the robot, we 3-D printed holders to house the 12V batteries as well as the portable battery that was connected to the power distribution board.
The KiCAD Schematic of the entire electrical system is as follows. Note that the follower is in the purple subsystem and the leader is in the pink subsystem. Many repetitive circuits are denoted as flags and are shown below the main KiCAD Schematic.
The design of the electrical system was intentional such that the leader PIC32 microcontroller had most of the game play functionality that was critical in advancing the play of the game. For example, the entire lifting mechanism control was placed on the leader so that we could easily flow through lifting commands without needing to use SPI to confirm that we were in the correct location. On the other hand, we planned to put most of the driving control on the follower because we wanted to keep the electrical noise from the drive motors isolated away from the leader, and we could easily plan on driving to nodes without much input from the leader. The follower could just line follow and turn on its own!
The Signal Conditioning Circuit was designed to be as simple as possible. We determined through experimentation that a high pass filter and a low pass filter were unnecessary for this project because we received a good, accurate signal with a simple op amp and comparator system. It was crucial to include the op amp because it amplified the IR signal to be read from 5 feet away. The gain resistor (R7 in the diagram above) was experimentally determined with the IR beacon and an oscilloscope. Secondly, The comparator was included to provide a hysteresis band.
This subsystem is in charge of keeping track of where the bot is during the game… i.e. it controls the flow of gameplay. As we accomplish one task, this subsystem can move us to the next. The leader is truly the brains behind the whole operation and it was given most of the critical parts of the project because we didn’t want any issues with SPI timing to stall our gameplay progress.
The Follower subsystem is mainly tasked with the driving motor subsystem, meaning that it receives commands to travel to a specified location. It then uses information from the tape sensors to line follow or perform a turn and then reorient itself on a new line. Once this subsystem completes an action, it posts a finished command to the leader through SPI so that it can receive the next drive command.
Click below to view pseudocode, source code, and header files.
STATE_MACHINE DriveService:
//Create module variables for the tape sensors
static int32_t TapeSenseA[1]; //this is the example from one analog sensor
Initialize: weightedsum, Normalsum, Tape tilt //variables to init
Init all the pins correctly in the init function:
Init each pin of the tape sensor with the ADC stuff //the next line is an example
ADC_ConfigAutoScan(BIT9HI); //example from previous project
STATE WaitingForCommand:
- Wait for an event from the Communication Service:
- EVENT MoveToNode → Transition to STATE MovingToNode
// need to set duty cycle here so that it doesn’t
// interfere with the motor controller
Set dc_LeftMotor =1
Set dc_RightMotor =1
Start the tape timer
- EVENT BeaconAlign → Transition to STATE AligningWithBeacon
- EVENT DriveF_Full → Transition to STATE DrivingForward
- EVENT DriveR_Full → Transition to STATE DrivingReverse
- (Other events ignored)
//this state drives forward until the tape sensors recognize a full line. \
// This means we’ve reached an intersection and can then choose to drive
// forward again or turn towards a block or a tower
STATE MovingToNode:
//this state is a line following state
//motors should come in with a predetermined duty cycle
// this next line runs the motors at whatever duty cycle was input prevously
OC2RS = PR_VAL*dutyCycle_LeftMotor; //oc left motor
OC3RS = PR_VAL*dutyCycle_RightMotor;
LATBbits.LATB11 = 0; //polarity right motor LOW
LATBbits.LATB4 = 0; // polarity left motor LOW
Set and start tape timer 50ms //need to play around with the rate of timer time out
//tape timer is used with a control law to monitor tape sensor and change duty cycle of motors
-EVENT TapeTimer TIMEOUT
//do some version of the multiread command for each tape sensor
ADC_MultiRead(TapeSenseA); // (x5) this reads a single tape sensor and resets the value
ADC_MultiRead(TapeSenseB);
…… (x5) //and vary what you call in the function … aka PotRead Changes
//now set a value to each PotRead[0] value
A= TapeSenseA[0]
B= TapeSenseB[0] … //continue for each tape sense val
//now need to scale these tape sensor values into a weighted sum
WeightedSum= -2*A-B+D+2E;
//now compute the normal sum of the tape readings
NormalSum= A+B+C+D+E;
//if we divide WS/NS we can get a value between -1 and 1
// which tells us how off of center we are…
// Note 0 means that we are dead center
//we can also use the normal sum to know when we get to a node…
// ie the sum is greater than a threshold amount
TapeTilt= WeightedSum/NormalSum;
// Using Tilt to directly control the Line Following without needing a PI Controller
// if tilted slightly left we need to slow down the right motor
If (tilt < 0):
dc_leftMotor = 100;
dc_rightMotor= 100+100*TapeTilt; // 1*100*TapeTilt is a scaled negative value
End if
// if tilted slightly right, we need to slow down left motor
If (tilt >0):
dc_rightMotor = 100;
dc_leftMotor = 100-100*TapeTilt;
STATE NodeTurnLeft
Switch:
//drive forward for a bit until turn timer goes off
EVENT NodeTurn Init event
Init TurnTimer (50ms?)
Set duty cycle of left and right to be 0.5
Use OCRS to start motors to drive forward
// now we start the turning movement
//now the turn timer times out
EVENT Timer timeout event && Event param = TurnTimer
lastTapeReadAB= 0 //note we only care about the left two sensors bc they will
//trigger first on a left turn!
Set DC of left motor to zero
Set DC of Right Motor to 0.5 //high
Init LeftTurnTapeTimer
Break switch
// checking to see if falling below threshold value
EVENT Timer timeout event && Event param = LeftTurnTapeTimer
If lastTapeReadAB= 0 //this means the first timeout
Read all tape sensors // I think i need to do ADC() command
TapeReadAB= TapeA+TapeB //setting a previous value
lastTapeReadAB= TapeReadAB; //setting previous value equal to tape
Init LeftTurnTapeTimer
End if
//this means that we crossed the tape and now the AB went from tape to white
If (lastTapeReadAB < ABThreshold && TapeReadAB > ABThreshold)
Post found tape event to DriveService2
End if
//if a timeout occurs and we haven’t hit our critical tape found state just read and
// calculate the values again
else
Read all tape sensors // I think I need to do ADC() command
lastTapeReadAB= TapeReadAB //resets the last value
TapeReadAB= TapeA + TapeB //calculates new value
Init LeftTurnTapeTimer
End else
Break switch
//starts drive bot forward to fully get on the tape
EVENT: FoundTapeEvent
Set duty cycle of both motors to 0.5 //high
Set ocr stuff to drive motors forward
Init foundTapeTurnTimer
Break
//drive until sensor D triggers (this is the sensor on the far side)
EVENT: Timer timeout event && Event param = FoundTapeTurnTimer
TapeD_read= Read tape Sensor D // using ADC readings
//if the D sensor reads that we are on black tape, exit state
If (TapeD_read < D_Threshold)
//stop the bot from driving and lets it wait for the next command
Set duty cycle of both motors to zero //low
Set ocr stuff to implement new duty cycle
//this means that we are more or less aligned with the tape
//need to exit this state and move on to the next command in the HSM
// the next command needs to be some type of drive forward command
// see previous code on how to do this!
End if
//else if we aren’t on the tape keep driving forward and restart the timer
Else
Init FoundTapeTurnTimer
End else
Break
STATE TurnRight:
// Do the exact same thing as the left turn, except switch which motor turns
// and incorporate the opposite sensors to reorient the bot on the line
// this state just turns the bot until it finds the beacon
// then it will call a left turn command to reorient itself on the line
STATE AligningWithBeacon:
- Set Looking4Beacon = 1.
- Rotate motors clockwise to search for beacon.
- Wait for:
- EVENT BeaconFound:
- Stop motors.
- Set Looking4Beacon = 0.
- Transition to STATE WaitingForCommand.
STATE DrivingReverse:
- (Logic to be implemented later)
- Transition to STATE WaitingForCommand after execution.
/****************************************************************************
Module
TestHarnessService0.c
Revision
1.0.1
Description
This is the first service for the Test Harness under the
Gen2 Events and Services Framework.
Notes
History
When Who What/Why
-------------- --- --------
10/26/17 18:26 jec moves definition of ALL_BITS to ES_Port.h
10/19/17 21:28 jec meaningless change to test updating
10/19/17 18:42 jec removed referennces to driverlib and programmed the
ports directly
08/21/17 21:44 jec modified LED blink routine to only modify bit 3 so that
I can test the new new framework debugging lines on PF1-2
08/16/17 14:13 jec corrected ONE_SEC constant to match Tiva tick rate
11/02/13 17:21 jec added exercise of the event deferral/recall module
08/05/13 20:33 jec converted to test harness service
01/16/12 09:58 jec began conversion from TemplateFSM.c
****************************************************************************/
/*----------------------------- Include Files -----------------------------*/
// This module
#include "../ProjectHeaders/DriveService2.h"
// debugging printf()
// Hardware
#include <xc.h>
//#include <proc/p32mx170f256b.h>
// Event & Services Framework
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "ES_DeferRecall.h"
#include "ES_Port.h"
#include "terminal.h"
#include "dbprintf.h"
#include "FollowerSPIService.h"
#include "PIC32_AD_Lib.h"
#include "DriveService2.h"
/*----------------------------- Module Defines ----------------------------*/
#define PR_VAL 24999 //CHANGE THIS TO CALCULATED PERIOD
#define Time45 1000
#define Time90 2000
#define LeftPolarity PORTBbits.RB12;
#define RightPolarity PORTBbits.RB15;
#define TAPE_PINS (BIT4HI | BIT5HI | BIT12HI | BIT11HI | BIT9HI)
#define LOWER_BOUND_DUTY 0.1
#define UPPER_BOUND_DUTY 0.4
#define LOWER_BOUND_BLOCK_FINDING 0.15
#define UPPER_BOUNS_BLOCK_FINDING 0.35
/*---------------------------- Module Functions ---------------------------*/
/* prototypes for private functions for this service.They should be functions
relevant to the behavior of this service
*/
/*---------------------------- Module Variables ---------------------------*/
// with the introduction of Gen2, we need a module level Priority variable
static uint8_t MyPriority;
static uint8_t Looking4Tape = 0;
static uint8_t Looking4Beacon = 0;
float dutyCycle_LeftMotor = 0;
float dutyCycle_RightMotor = 0;
static int32_t WeightedSum, NormalSum;
static float TapeTilt;
static uint32_t ADC_Results[5]; // Array to store ADC readings
static uint8_t TurnPseudoState;
static uint32_t lastTapeReadAB;
static uint32_t TapeReadAB;
static uint32_t AB_Threshold = 600;
static uint32_t TapeD_Read;
static uint32_t D_Threshold = 250;
static uint32_t TimeoutCounter = 0;
static uint8_t CurrRA2State = 1;
static uint8_t PrevRA2State = 1;
static DriveServiceState_t CurrentState = WaitingForCommand;
/*------------------------------ Module Code ------------------------------*/
/****************************************************************************
Function
InitDriveService
Parameters
uint8_t : the priorty of this service
Returns
bool, false if error in initialization, true otherwise
Description
Saves away the priority, and does any
other required initialization for this service
Notes
Author
J. Edward Carryer, 01/16/12, 10:00
****************************************************************************/
bool InitDriveService2(uint8_t Priority)
{
ES_Event_t ThisEvent;
MyPriority = Priority;
// Left Motor Driver
// TRISBbits.TRISB10 = 1;//encoder input
TRISBbits.TRISB5 = 0;//output compare output
// Left Polarity output to motor
TRISBbits.TRISB8 = 0;
// Right Motor Driver
// TRISBbits.TRISB8 = 1; //encoder input
TRISBbits.TRISB9 = 0; //output compare output
// Right Polarity; pin 22, RB11
TRISBbits.TRISB11 = 0;
// testing tape sensor inputs
//tape sensor 1 pin 7, RB3
TRISBbits.TRISB3=1; //input
ANSELBbits.ANSB3 = 1; //analog
//tape sensor 2, RB2, pin 6
TRISBbits.TRISB2 = 1; //input
ANSELBbits.ANSB2 =1;
//tape sensor 3: pin 24, rb13
TRISBbits.TRISB13=1; //input
ANSELBbits.ANSB13= 1; //analog
//Tape sensor 4: pin 26, rb15
TRISBbits.TRISB15=1; //input
ANSELBbits.ANSB15= 1; //analog
//Tape sensor 5: pin 23, rb12
TRISBbits.TRISB12=1; //input
ANSELBbits.ANSB12= 1; //analog
/*************************************************************************/
T2CONbits.ON = 0;
//2. Clear the TCS control bit (TxCON<1> = 0) to select the internal PBCLK source.
T2CONbits.TCS = 0;
//3. Select the desired timer input clock prescale.
T2CONbits.TCKPS = 0b010; // 1:4
T2CONbits.TGATE = 0;
//4. Load/Clear the timer register TMRx.
TMR2 = 0;
//5. Load the period register PRx with the desired 16-bit match value.
PR2 = PR_VAL; // 20ms period
//6. Set the ON control bit (TxCON<15> = 1) to enable the timer.
T2CONbits.ON = 1;
// init OC2R register (MOTOR 1)
OC2CONbits.ON = 0; // Turn off the OC1 when performing the setup
OC2CONbits.OCM = 0b110;
// set timer 2 as the clock source
OC2CONbits.OCTSEL = 0;
OC2CONbits.OC32 = 0;
OC2CONbits.OCSIDL = 0;
OC2R = 0; // Initialize primary Compare register
OC2RS = 12500; // Initialize secondary Compare register
RPB5R = 0b0101; // pin 14
OC2CONbits.ON = 1;
// init OC3R register (MOTOR 1)
OC3CONbits.ON = 0; // Turn off the OC2 when performing the setup
OC3CONbits.OCM = 0b110;
// set timer 2 as the clock source
OC3CONbits.OCTSEL = 0;
OC3CONbits.OC32 = 0;
OC3CONbits.OCSIDL = 0;
OC3R = 0; // Initialize primary Compare register
OC3RS = 12500; // Initialize secondary Compare register
RPB9R = 0b0101; // pin 18
OC3CONbits.ON = 1;
Looking4Tape = 0;
//init with the motor turning off
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
OC2RS = 0;
OC3RS = 0;
/*************************************************************************/
// ADC initialization...
ANSELBbits.ANSB2 = 1; // RB2 as analog input (AN2)
ANSELBbits.ANSB3 = 1; // RB3 as analog input (AN3)
ANSELBbits.ANSB13 = 1; // RB13 as analog input (AN13)
ANSELBbits.ANSB15 = 1; // RB15 as analog input (AN15)
ANSELBbits.ANSB12 = 1; // RB12 as analog input (AN12)
TRISBbits.TRISB2 = 1; // Set RB2 as input
TRISBbits.TRISB3 = 1; // Set RB3 as input
TRISBbits.TRISB13 = 1; // Set RB13 as input
TRISBbits.TRISB15 = 1; // Set RB15 as input
TRISBbits.TRISB12 = 1; // Set RB12 as input
DB_printf("configuring tape pins...\n");
// Configure ADC to scan these inputs
if (!ADC_ConfigAutoScan(TAPE_PINS)) {
while (1); // Halt execution if configuration fails
}
DB_printf("configured...\n");
/*************************************************************************/
// Hall Effect Initialization
// Configure RA3 as a digital input
// TRISAbits.TRISA3 = 1; // Set RA3 as input
// ES_Event_t InitEvent;
// InitEvent.EventType = ES_TIMEOUT;
// PostDriveService(InitEvent);
/********************************************
in here you write your initialization code
*******************************************/
// post the initial transition event
ThisEvent.EventType = ES_INIT;
if (ES_PostToService(MyPriority, ThisEvent) == true)
{
return true;
}
else
{
return false;
}
}
/****************************************************************************
Function
PostDriveService
Parameters
EF_Event_t ThisEvent ,the event to post to the queue
Returns
bool false if the Enqueue operation failed, true otherwise
Description
Posts an event to this state machine's queue
Notes
Author
J. Edward Carryer, 10/23/11, 19:25
****************************************************************************/
bool PostDriveService2(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
/****************************************************************************
Function
RunTemplateService
Parameters
ES_Event_t : the event to process
Returns
ES_Event, ES_NO_EVENT if no error ES_ERROR otherwise
Description
add your description here
Notes
Author
J. Edward Carryer, 01/15/12, 15:23
****************************************************************************/
ES_Event_t RunDriveService2(ES_Event_t ThisEvent) {
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // Assume no errors
switch (CurrentState) {
case WaitingForCommand:
{
if (ThisEvent.EventType == MoveToNode) {
DB_printf("Moving to node!\n");
dutyCycle_LeftMotor = 0.2;
dutyCycle_RightMotor = 0.2;
// Set motors to move
OC2RS = PR_VAL * dutyCycle_LeftMotor; // Left motor speed
OC3RS = PR_VAL * dutyCycle_RightMotor; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
ES_Timer_InitTimer(TURN_TIMER, 20); // Placeholder for tape sensor logic
// Transition to MovingToNode state
CurrentState = MovingToNode;
}
else if (ThisEvent.EventType == BeaconAlign) {
DB_printf("Aligning with beacon!\n");
Looking4Beacon = 1;
// //trying to spin the motor with latA4
// dutyCycle_LeftMotor = 0.7;
// dutyCycle_RightMotor = 0.3;
// OC2RS = PR_VAL * dutyCycle_LeftMotor;
// OC3RS = PR_VAL * dutyCycle_RightMotor;
// LATBbits.LATB11 = 0;
// LATBbits.LATB8 = 1;
// Rotate CW or basically a right turn
// Transition to AligningWithBeacon state
ES_Event_t dummyEvent;
dummyEvent.EventType = BeaconAlign; // to kick us into the beacon finding state
PostDriveService2(dummyEvent);
CurrentState = AligningWithBeacon;
}
else if (ThisEvent.EventType == MoveForwardTimed) {
// start a framework timer for the appropriate time
ES_Timer_InitTimer(TURN_TIMER, ThisEvent.EventParam);
// set the duty cycles
OC2RS = 0.2;
OC3RS = 0.2;
// move to the new state
CurrentState = MovingForwardTimed;
}
else if (ThisEvent.EventType == TurnLeft) {
DB_printf("initiating left turn...\n");
//set the duty cycle to drive the robot forward before the turn
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
OC2RS = PR_VAL * 0.6; // Left motor speed
OC3RS = PR_VAL * 0.6; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
ES_Timer_InitTimer(TURN_TIMER, 1000); // lever 1
// move to the new state
CurrentState = TurningLeft;
TurnPseudoState = 0;
}
else if (ThisEvent.EventType == DriveReverse) {
DB_printf("driving in reverse...\n");
ES_Timer_InitTimer(TURN_TIMER, 1000);
//set the duty cycle to drive the robot forward before the turn
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
OC2RS = PR_VAL * 0.7; // Left motor speed
OC3RS = PR_VAL * 0.7; // Right motor speed
LATBbits.LATB11 = 1; // Right motor polarity LOW
LATBbits.LATB8 = 1; // Left motor polarity LOW
// move to the new state
CurrentState = TimedReverse;
}
else if (ThisEvent.EventType == BeaconLeft) {
DB_printf("initiating left turn at the beacon...\n");
//set the duty cycle to drive the robot forward before the turn
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
OC2RS = PR_VAL * 0.5; // Left motor speed
OC3RS = PR_VAL * 0.5; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
ES_Timer_InitTimer(TURN_TIMER, 650); // lever 1
// move to the new state
CurrentState = TurningLeft;
TurnPseudoState = 0;
}
else if (ThisEvent.EventType == TurnRight) {
DB_printf("initiating right turn...\n");
//set the duty cycle to drive the robot forward before the turn
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
OC2RS = PR_VAL * 0.5; // Left motor speed
OC3RS = PR_VAL * 0.5; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
ES_Timer_InitTimer(TURN_TIMER, 750); // lever 1
// move to the new state
CurrentState = TurningRight;
TurnPseudoState = 0;
}
else if (ThisEvent.EventType == TurnFull) {
DB_printf("initiating full turn...\n");
//set the duty cycle to drive the robot forward before the turn
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
OC2RS = PR_VAL * 0.7; // Left motor speed
OC3RS = PR_VAL * 0.7; // Right motor speed
LATBbits.LATB11 = 1; // Right motor polarity HIGH
LATBbits.LATB8 = 1; // Left motor polarity HIGH
ES_Timer_InitTimer(TURN_TIMER, 1000);
// move to the new state
CurrentState = TurningFull;
TurnPseudoState = 0;
}
else if (ThisEvent.EventType == MoveToBlock) {
DB_printf("looking for block...\n");
dutyCycle_LeftMotor = 0.2;
dutyCycle_RightMotor = 0.2;
// Set motors to move
OC2RS = PR_VAL * dutyCycle_LeftMotor; // Left motor speed
OC3RS = PR_VAL * dutyCycle_RightMotor; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
ES_Timer_InitTimer(TURN_TIMER, 200); // Placeholder for tape sensor logic
// Transition to MovingToNode state
CurrentState = LookingForBlock;
}
else if (ThisEvent.EventType == InsertBlock) {
DB_printf("inserting the block...\n");
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
dutyCycle_LeftMotor = 0.4;
dutyCycle_RightMotor = 0.44;
OC2RS = PR_VAL * dutyCycle_LeftMotor; // Left motor speed
OC3RS = PR_VAL * dutyCycle_RightMotor; // Right motor speed
ES_Timer_StopTimer(TURN_TIMER);
ES_Timer_InitTimer(TURN_TIMER, 600);
CurrentState = DroppingBlock;
}
else if (ThisEvent.EventType == Stop){
//init with the motor turning off
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
OC2RS = 0;
OC3RS = 0;
// Notify system that turn is complete
ES_Timer_StopTimer(TURN_TIMER);
ES_Timer_InitTimer(TURN_TIMER, 200);
CurrentState = DroppingBlock;
}
else if (ThisEvent.EventType == TurnLeftBeacon) {
DB_printf("new turn beacon test...\n");
OC2RS = PR_VAL * 0.65; // Left motor speed
OC3RS = PR_VAL * 0.35; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 1; // Left motor polarity HIGH
CurrentState = TurningLeftBeacon;
TurnPseudoState = 0;
ES_Timer_StopTimer(TURN_TIMER);
ES_Timer_InitTimer(TURN_TIMER, 20);
TimeoutCounter = 0;
}
}
break;
case TurningLeftBeacon:
{
ADC_MultiRead(ADC_Results);
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER){
if (TurnPseudoState == 0) {
if (TimeoutCounter > 50) {
if (ADC_Results[4] < 250) {
DB_printf("finished the new turn test...\n");
TurnPseudoState = 1;
ES_Timer_InitTimer(TURN_TIMER, 5);
} else {
ES_Timer_InitTimer(TURN_TIMER, 5);
}
} else {
TimeoutCounter++;
ES_Timer_InitTimer(TURN_TIMER, 5);
}
} else if (TurnPseudoState == 1) {
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity HIGH
OC2RS = 0;
OC3RS = 0;
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
TurnPseudoState = 0;
}
}
}
break;
case DroppingBlock:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER) {
DB_printf("done inserting the block...\n");
OC2RS = 0;
OC3RS = 0;
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
}
}
break;
case LookingForBlock:
{
switch (ThisEvent.EventType) {
case ES_TIMEOUT:
{
if (ThisEvent.EventParam == TURN_TIMER) {
// Read current state of RA3
CurrRA2State = PORTAbits.RA3;
DB_printf("current pin state: %d\n", CurrRA2State);
// If RA2 transitioned from HIGH to LOW, exit state
if (CurrRA2State == 0) {
DB_printf("Block detected, exiting state!\n");
// Create and post event indicating the block has been found
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
OC2RS = 0;
OC3RS = 0;
TimeoutCounter = 0;
break;
}
// Update previous state for next iteration
PrevRA2State = CurrRA2State;
// Read tape sensors
ADC_MultiRead(ADC_Results);
// Compute weighted sum and normal sum
WeightedSum = (-2 * ADC_Results[1]) + (-1 * ADC_Results[0]) + (1 * ADC_Results[3]) + (2 * ADC_Results[2]);
NormalSum = ADC_Results[0] + ADC_Results[1] + ADC_Results[2] + ADC_Results[3] + ADC_Results[4];
// Compute tilt value
if (NormalSum != 0) {
TapeTilt = (float)WeightedSum / (float)NormalSum;
} else {
TapeTilt = 0;
}
// Adjust duty cycles based on tilt
if (TapeTilt < 0) {
dutyCycle_LeftMotor = UPPER_BOUNS_BLOCK_FINDING;
dutyCycle_RightMotor = (LOWER_BOUND_BLOCK_FINDING + ((UPPER_BOUNS_BLOCK_FINDING - LOWER_BOUND_BLOCK_FINDING) * (100 + 100 * TapeTilt) / 100));
} else if (TapeTilt > 0) {
dutyCycle_LeftMotor = (LOWER_BOUND_BLOCK_FINDING + ((UPPER_BOUNS_BLOCK_FINDING - LOWER_BOUND_BLOCK_FINDING) * (100 - 100 * TapeTilt) / 100));
dutyCycle_RightMotor = UPPER_BOUNS_BLOCK_FINDING;
} else {
dutyCycle_LeftMotor = UPPER_BOUNS_BLOCK_FINDING;
dutyCycle_RightMotor = UPPER_BOUNS_BLOCK_FINDING;
}
// Apply new motor speeds
OC2RS = PR_VAL * dutyCycle_LeftMotor;
OC3RS = PR_VAL * dutyCycle_RightMotor;
// Restart tape sensor timer
ES_Timer_InitTimer(TURN_TIMER, 200);
}
}
break;
default:
break;
}
}
break;
case MovingToNode:
{
switch (ThisEvent.EventType) {
case ES_TIMEOUT:
{
if (ThisEvent.EventParam == TURN_TIMER) {
// outer right = adc[2]
// middle right = adc[3]
// middle = adc[4]
// middle left = adc[0]
// outer left = adc[1]
// Read tape sensors
ADC_MultiRead(ADC_Results);
// ADC_Results[3] = ADC_Results[3] * 2; //DEBUG: middle right sensor is behaving far worse than others
// Assign values for weighted sum
// -2 * far_left - middle_left + middle_right + 2*far_right
WeightedSum = (-2 * ADC_Results[1]) + (-1 * ADC_Results[0]) + (1 * ADC_Results[3]) + (2 * ADC_Results[2]);
NormalSum = ADC_Results[0] + ADC_Results[1] + ADC_Results[2] + ADC_Results[3] + ADC_Results[4];
//DB_printf("Results 1: %u\n", ADC_Results[1]);
if (WeightedSum > 0) {
//DB_printf("weighted sum: %d\n", WeightedSum);
} else {
//DB_printf("weighted sum: -%d\n", (-WeightedSum));
}
//DB_printf("normal sum: %d\n", NormalSum);
// Compute tilt value
if (NormalSum != 0) {
TapeTilt = (float)WeightedSum/(float)NormalSum;
//DB_printf("tape tilt: %d\n", (int)(TapeTilt * 100));
} else{
TapeTilt = 0;
//DB_printf("divide by zero in tape tilt calculation...");
// how do we handle divide by zero?
}
if (TapeTilt < 0) {
//DB_printf("too far left!\n");
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = (LOWER_BOUND_DUTY + ((UPPER_BOUND_DUTY - LOWER_BOUND_DUTY) * (100 + 100 * TapeTilt) / 100));
uint32_t dc_left = dutyCycle_LeftMotor * 100;
uint32_t dc_right = dutyCycle_RightMotor * 100;
// DB_printf("tape tilt negative...\n");
// DB_printf("left duty cycle %d\n", dc_left);
// DB_printf("right duty cycle %d\n", dc_right);
} else if (TapeTilt > 0) {
//DB_printf("too far right!\n");
dutyCycle_LeftMotor = (LOWER_BOUND_DUTY + ((UPPER_BOUND_DUTY - LOWER_BOUND_DUTY) * (100 - 100 * TapeTilt) / 100));
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
uint32_t dc_left = dutyCycle_LeftMotor * 100;
uint32_t dc_right = dutyCycle_RightMotor * 100;
// DB_printf("tape tilt positive...\n");
// DB_printf("left duty cycle %d\n", dc_left);
// DB_printf("right duty cycle %d\n", dc_right);
} else {
// centered
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
//DB_printf("centered!\n");
}
// Apply new motor speeds
OC2RS = PR_VAL * dutyCycle_LeftMotor;
OC3RS = PR_VAL * dutyCycle_RightMotor;
// Check to see if we have reached a node...
// if the two right or two left are reading tape then we have reached the node...
uint32_t threshold = 350;
if (TimeoutCounter > 500) {
if ((ADC_Results[0] < threshold && ADC_Results[1] < threshold) ||
(ADC_Results[2] < threshold && ADC_Results[3] < threshold)) {//changing threshold of right side as we fine tune bad sensor
// DB_printf("We have reached the node!\n");
// DB_printf("ADC 1 reading: %d\n", ADC_Results[0]);
// DB_printf("ADC 2 reading: %d\n", ADC_Results[1]);
// DB_printf("ADC 3 reading: %d\n", ADC_Results[2]);
// DB_printf("ADC 4 reading: %d\n", ADC_Results[3]);
// DB_printf("ADC 5 reading: %d\n", ADC_Results[4]);
// Create and post event indicating the node has been reached
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
OC2RS = 0;
OC3RS = 0;
TimeoutCounter = 0;
} else {
// Restart tape sensor timer
ES_Timer_InitTimer(TURN_TIMER, 5);
// DB_printf("outer left: %d\n", ADC_Results[1]);
// DB_printf("middle left: %d\n", ADC_Results[0]);
// DB_printf("middle: %d\n", ADC_Results[4]);
// DB_printf("middle right: %d\n", ADC_Results[3]);
// DB_printf("far right: %d\n", ADC_Results[2]);
if (WeightedSum < 0) {
uint16_t summm = WeightedSum * 100;
//DB_printf("weighted sum: -%d\n", summm);
} else {
uint16_t summm = WeightedSum * 100;
// DB_printf("weighted sum: %d\n", summm);
}
}
} else {
TimeoutCounter = TimeoutCounter + 1;
ES_Timer_InitTimer(TURN_TIMER, 5); // TODO: fine tune measuring speed.
// DB_printf("outer left: %d\n", ADC_Results[1]);
// DB_printf("middle left: %d\n", ADC_Results[0]);
// DB_printf("middle: %d\n", ADC_Results[4]);
// DB_printf("middle right: %d\n", ADC_Results[3]);
// DB_printf("far right: %d\n", ADC_Results[2]);
}
}
}
break;
default:
break;
}
}
break;
case TurningFull:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER) {
// Phase 1: Initiate In-Place Turn
if (TurnPseudoState == 0) {
DB_printf("Starting in-place turn...\n");
// Set both motors to spin at the same speed but in opposite directions
OC2RS = 0.5 * PR_VAL; // Left motor speed
OC3RS = 0.5 * PR_VAL; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW (forward)
LATBbits.LATB8 = 1; // Left motor polarity HIGH (backward)
// Start checking tape sensor after some time
ES_Timer_InitTimer(TURN_TIMER, 5);
TurnPseudoState = 1;
}
// Phase 2: Check Tape Sensor for Turn Completion
else if (TurnPseudoState == 1) {
ADC_MultiRead(ADC_Results);
if (TimeoutCounter > 200) {
if (ADC_Results[4] < 250) {
DB_printf("Turn complete, proceeding to stop...\n");
TurnPseudoState = 2;
ES_Timer_InitTimer(TURN_TIMER, 5);
} else {
ES_Timer_InitTimer(TURN_TIMER, 5);
}
} else {
TimeoutCounter++;
ES_Timer_InitTimer(TURN_TIMER, 5);
}
}
// Phase 3: Stop and Transition Back to WaitingForCommand
else if (TurnPseudoState == 2) {
DB_printf("Completed full turn, stopping motors.\n");
// Stop the motors
OC2RS = 0;
OC3RS = 0;
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
// Notify system that turn is complete
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand
CurrentState = WaitingForCommand;
TimeoutCounter = 0;
TurnPseudoState = 0;
}
}
}
break;
case TurningLeft:
{
// outer right = adc[2]
// middle right = adc[3]
// middle = adc[4]
// middle left = adc[0]
// outer left = adc[1]
// on this timeout, we have finished driving forward...
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 0){
DB_printf("DriveForward timer in left turn timed out!\n");
lastTapeReadAB= 0; //initializing the last read variable
//TODO: CHECK TURN MOVES WHEELS IN CORRECT DIRECTIONS
//start the turn movement
dutyCycle_LeftMotor = 1-UPPER_BOUND_DUTY;
dutyCycle_RightMotor = UPPER_BOUND_DUTY;
// OC2RS = PR_VAL * dutyCycle_LeftMotor; // Left motor speed
OC2RS = 0.7 * PR_VAL;
OC3RS = 0.3 * PR_VAL; // Right motor speed
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 1; // Left motor polarity HIGH
//every time this times out we'll read the tape sensors
ES_Timer_InitTimer(TURN_TIMER, 5);
TurnPseudoState = 1;
} // end if statement
//start measuring tape sensors (we are now in pseudo-state 1...)
else if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 1)
{
// check for the middle sensor to trigger below a threshold
// when this happens we are done turning...
ADC_MultiRead(ADC_Results);
if (TimeoutCounter > 400) {
if (ADC_Results[4] < 300) {
DB_printf("starting pseudo state 2...\n");
TurnPseudoState = 2;
ES_Timer_InitTimer(TURN_TIMER, 150);
//return current
} else {
ES_Timer_InitTimer(TURN_TIMER, 5);
}
} else {
TimeoutCounter = TimeoutCounter + 1;
DB_printf("middle: %d\n", ADC_Results[4]);
ES_Timer_InitTimer(TURN_TIMER, 5);
DB_printf("timeout counter : %d\n", TimeoutCounter);
}
} // end active turning movement
//need to get more on tape for the first turn in the sequence (ethan's code)
//continue turning until the fourth tape sensor triggers
// else if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 2)
// {
// // check for the middle sensor to trigger below a threshold
// // when this happens we are done turning...
// ADC_MultiRead(ADC_Results);
// if (TimeoutCounter > 200) {
// if (ADC_Results[3] < 400) {
// DB_printf("starting pseudo state 2...\n");
// TurnPseudoState = 3;
// ES_Timer_InitTimer(TURN_TIMER, 300);
// //return current
// } else {
// ES_Timer_InitTimer(TURN_TIMER, 5);
// }
//
// } else {
// TimeoutCounter = TimeoutCounter + 1;
// DB_printf("middle: %d\n", ADC_Results[4]);
// ES_Timer_InitTimer(TURN_TIMER, 5);
// }
// } // end active turning movement
else if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 2) {
DB_printf("reached pseudo-state 2 end...\n");
// we have finished turning...
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
TimeoutCounter = 0;
TurnPseudoState = 2;
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
OC2RS = 0;
OC3RS = 0;
}
}
break;
case TurningRight:
{
// On this timeout, we have finished driving forward...
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 0) {
DB_printf("DriveForward timer in turn timed out!\n");
lastTapeReadAB = 0; // Initializing the last read variable
// TODO: CHECK TURN MOVES WHEELS IN CORRECT DIRECTIONS
// Start the turn movement
dutyCycle_RightMotor = 1 - UPPER_BOUND_DUTY;
dutyCycle_LeftMotor = UPPER_BOUND_DUTY;
// OC3RS = PR_VAL * dutyCycle_RightMotor; // Right motor speed
OC3RS = 0.6 * PR_VAL;
OC2RS = 0.3 * PR_VAL; // Left motor speed
LATBbits.LATB11 = 1; // Right motor polarity HIGH
LATBbits.LATB8 = 0; // Left motor polarity LOW
// Every time this times out, we'll read the tape sensors
ES_Timer_InitTimer(TURN_TIMER, 5);
TurnPseudoState = 1;
}
// Start measuring tape sensors (we are now in pseudo-state 1...)
else if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 1) {
// Check for the middle sensor to trigger below a threshold
// When this happens, we are done turning...
ADC_MultiRead(ADC_Results);
if (TimeoutCounter > 300) {
if (ADC_Results[4] < 300) {
DB_printf("starting pseudo state 2...\n");
TurnPseudoState = 2;
ES_Timer_InitTimer(TURN_TIMER, 5);
} else {
ES_Timer_InitTimer(TURN_TIMER, 5);
}
} else {
TimeoutCounter = TimeoutCounter + 1;
ES_Timer_InitTimer(TURN_TIMER, 5);
}
}
// Reached pseudo-state 2 (end of turn)
else if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER && TurnPseudoState == 2) {
DB_printf("reached pseudo-state 2 end...\n");
// We have finished turning...
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
TimeoutCounter = 0;
TurnPseudoState = 0;
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
OC2RS = 0;
OC3RS = 0;
}
}
break;
case MovingForwardTimed:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER) {
OC2RS = 0;
OC3RS = 0;
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
CurrentState = WaitingForCommand;
DB_printf("finished timed movement!\n");
}
}
break;
case TimedReverse:{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER) {
OC2RS = 0;
OC3RS = 0;
ES_Event_t CommEvent;
CommEvent.EventType = ES_DriveServiceFinished;
PostFollowerSPIService(CommEvent);
CurrentState = WaitingForCommand;
DB_printf("finished timed reverse!\n");
}
}
case AligningWithBeacon:
{
dutyCycle_LeftMotor = 0.3;
dutyCycle_RightMotor = 0.7;
OC2RS = PR_VAL * dutyCycle_LeftMotor;
OC3RS = PR_VAL * dutyCycle_RightMotor;
LATBbits.LATB11 = 1;
LATBbits.LATB8 = 0;
//when this timer times out we will spin the other direction to find the beacon
ES_Timer_InitTimer(TURN_TIMER, 10000);
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == TURN_TIMER){
//this should be a right turn now
DB_printf("Beacon Align, Changing Direction\n");
OC3RS = 0.3 * PR_VAL;
OC2RS = 0.7 * PR_VAL; // Left motor speed
LATBbits.LATB11 = 0; // Right motor polarity HIGH
LATBbits.LATB8 = 1; // Left motor polarity LOW
}
if (ThisEvent.EventType == ES_BEACONFOUND) {
DB_printf("Beacon found! Stopping motors.\n");
//init with the motor turning off
LATBbits.LATB11 = 0; // Right motor polarity LOW
LATBbits.LATB8 = 0; // Left motor polarity LOW
OC2RS = 0;
OC3RS = 0;
// Transition back to WaitingForCommand state
CurrentState = WaitingForCommand;
}
}
break;
default:
DB_printf("Error: Invalid state in DriveService FSM!\n");
CurrentState = WaitingForCommand;
break;
}
return ReturnEvent;
}
/*------------------------------- Footnotes -------------------------------*/
/*------------------------------ End of file ------------------------------*/
#ifndef DriveService2_H
#define DriveService2_H
#include <stdint.h>
#include <stdbool.h>
#include "ES_Events.h"
#include "ES_Port.h" // needed for definition of REENTRANT
// Public Function Prototypes
typedef enum {
WaitingForCommand,
AligningWithBeacon,
MovingToNode,
MovingForwardTimed,
TurningLeft,
TurningRight,
TurningFull,
LookingForBlock,
DroppingBlock,
TurningLeftBeacon,
TimedReverse
} DriveServiceState_t;
bool InitDriveService2(uint8_t Priority);
bool PostDriveService2(ES_Event_t ThisEvent);
ES_Event_t RunDriveService2(ES_Event_t ThisEvent);
#endif /* ServTemplate_H */
DEFINE MODULE FollowerSPIService
INITIALIZE:
Define CurrentState as WaitingCommand
Define MyPriority as the priority of this service
Define NewCommand as 0 (volatile)
Define CommandToSend as 0 (default response message)
FUNCTION InitFollowerSPIService(Priority):
Set MyPriority to Priority
PRINT "Initializing Follower SPI Service..."
Set CurrentState to WaitingCommand
DISABLE SPI1
// Configure SPI1 for SLAVE mode
DISABLE analog functionality on SPI pins
Set SPI1 pins (SDI, SDO, SS, SCK)
Configure SPI1 registers:
- Set as SLAVE
- Clock idle high
- Data changes on falling edge
- Enable SS control
- Use 8-bit mode
- Enable enhanced buffer mode
CLEAR SPI buffer and overflow flag
ENABLE SPI1 receive interrupt
ENABLE SPI1
ENABLE global interrupts
CREATE Event ES_INIT
RETURN Post Event to Service Queue
FUNCTION PostFollowerSPIService(Event):
Post Event to Service Queue
RETURN True if successful, False otherwise
FUNCTION RunFollowerSPIService(CurrentEvent):
DEFINE ReturnEvent as ES_NO_EVENT
SWITCH(CurrentState):
CASE WaitingCommand:
SWITCH(CurrentEvent.Type):
CASE ES_INIT:
PRINT "Follower SPI Initialized"
BREAK
CASE ES_LEADERSENT: // Leader sent a command
READ receivedData from Event.Param
PRINT "Received SPI Data:", receivedData
CREATE Event NewEvent
IF receivedData == 10: // ALIGN_BEACON
PRINT "Aligning with beacon"
Set CurrentState to WaitingBeaconAlign
NewEvent.Type = BeaconAlign
PostDriveService(NewEvent)
Set CommandToSend = COMMAND_RECEIVED
ELSE IF receivedData == 20: // ALIGN_TAPE
PRINT "Aligning with tape"
Set CurrentState to CompletingAction
Start Timer POLL_TIMER (2s)
Set CommandToSend = COMMAND_RECEIVED
ELSE IF receivedData == 30: // MOVE_TO_NODE
PRINT "Moving to node"
Set CurrentState to CompletingAction
Start Timer POLL_TIMER (2s)
NewEvent.Type = MoveToNode
PostDriveService(NewEvent)
Set CommandToSend = COMMAND_RECEIVED
ELSE IF receivedData == 40: // TURN
PRINT "Turning"
Set CurrentState to CompletingAction
Start Timer POLL_TIMER (2s)
Set CommandToSend = COMMAND_RECEIVED
ELSE IF receivedData == 50: // DROP_BLOCK
PRINT "Dropping block"
Set CurrentState to CompletingAction
Set CommandToSend = COMMAND_RECEIVED
BREAK
CASE WaitingBeaconAlign:
SWITCH(CurrentEvent.Type):
CASE ES_LEADERSENT: // Check if leader sent beacon alignment update
READ receivedData from Event.Param
IF receivedData == 60: // Beacon Found
PRINT "Beacon Found"
NewEvent.Type = ES_BEACONFOUND
PostDriveService(NewEvent)
Set CommandToSend = 0
Set CurrentState to WaitingCommand
BREAK
CASE CompletingAction:
SWITCH(CurrentEvent.Type):
CASE ES_DriveServiceFinished: // Drive service completed action
PRINT "Task Completed! Sending response to leader."
Set CommandToSend = COMMAND_COMPLETED
Set CurrentState to WaitingCommand
BREAK
RETURN ReturnEvent
FUNCTION ISR SPI1_RX_ISR():
IF SPI1 receive buffer is NOT empty:
READ NewCommand from SPI1BUF
CREATE Event ES_LEADERSENT
Set Event.Param to NewCommand
Post Event to FollowerSPIService
WRITE CommandToSend to SPI1BUF (send response)
CLEAR SPI1RX interrupt flag
// Hardware
#include <xc.h>
// Event & Services Framework
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "ES_DeferRecall.h"
#include "ES_Port.h"
#include "terminal.h"
#include "dbprintf.h"
#include <sys/attribs.h>
#include "DriveService.h"
#include "FollowerSPIService.h"
/*----------------------------- Module Defines ----------------------------*/
// These times assume a 10ms/tick timing
#define ONE_SEC 1000
#define HALF_SEC (ONE_SEC / 2)
#define TWO_SEC (ONE_SEC * 2)
#define FIVE_SEC (ONE_SEC * 5)
#define COMMAND_RECIEVED 1
#define COMMAND_COMPLETED 2
/*---------------------------- Module Variables ---------------------------*/
// Module-level Priority variable
static uint8_t MyPriority;
static volatile uint8_t NewCommand = 0;
static uint8_t CommandToSend = 0;
static FollowerSPIState_t CurrentState;
/*------------------------------ Module Code ------------------------------*/
bool InitFollowerSPIService(uint8_t Priority)
{
ES_Event_t ThisEvent;
MyPriority = Priority;
clrScrn();
DB_printf("Init Function for SPI (Follower) with Interrupts\n");
CurrentState = WaitingCommand;
__builtin_disable_interrupts();
SPI1CONbits.ON = 0; // Disable SPI1 before configuring
// Disable analog functionality on SPI pins
ANSELAbits.ANSA1 = 0; // SDI (RA1)
ANSELAbits.ANSA0 = 0; // SS (RA0)
ANSELBbits.ANSB14 = 0; // SCK (RB14)
// Set SPI pins
TRISAbits.TRISA1 = 1; // SDI (RA1) as input
SDI1R = 0b0000; // Map RA1 to SPI1 SDI
TRISAbits.TRISA2 = 0; // Set RA2 (SDO) as output
RPA2R = 0b0011; // Map RA2 pin 9 to SPI1 SDO
TRISAbits.TRISA0 = 1; // SS (RA0) as input
SS1R = 0b0000;
TRISBbits.TRISB14 = 1; // SCK (RB14) as input
// Configure SPI1 as SLAVE
SPI1CONbits.MSTEN = 0; // Slave mode
SPI1CONbits.CKP = 1; // Clock idle high
SPI1CONbits.CKE = 0; // Data changes on falling edge, sampled on rising edge
SPI1CONbits.SSEN = 1; // Enable SS control
SPI1CONbits.MODE16 = 0; // 8-bit mode
SPI1CONbits.MODE32 = 0;
SPI1CONbits.ENHBUF = 0; // Enable enhanced buffer mode (FIFO)
// Clear SPI buffer and overflow flag
volatile uint8_t dummy = SPI1BUF;
SPI1STATbits.SPIROV = 0;
// Configure and enable SPI1 receive interrupt
SPI1CONbits.SRXISEL= 0b01; //interrupt to trigger as soon as the buffer receives data after having no data
//clearing interrupt flag
IFS1CLR= _IFS1_SPI1RXIF_MASK;
//setting priority
IPC7bits.SPI1IP= 7; // highest priority
// enabling interrupt flag
IEC1SET = _IEC1_SPI1RXIE_MASK;
SPI1CONbits.ON = 1; // Enable SPI1
__builtin_enable_interrupts();
// Post an initial transition event
ThisEvent.EventType = ES_INIT;
return ES_PostToService(MyPriority, ThisEvent);
}
bool PostFollowerSPIService(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
ES_Event_t RunFollowerSPIService(ES_Event_t ThisEvent)
{
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // Default return
switch (CurrentState)
{
case WaitingCommand:
{
switch (ThisEvent.EventType)
{
case ES_INIT:
{
DB_printf("\rES_INIT received in Follower SPI Service %d\r\n", MyPriority);
}
break;
case ES_LEADERSENT: // Leader sent data
{
uint8_t receivedData = ThisEvent.EventParam;
//DB_printf("Follower Received SPI Data: %d\n", receivedData);
ES_Event_t NewEvent;
NewEvent.EventType = ES_GameCommand; // Default event type
if (receivedData == 10) // ALIGN_BEACON
{
DB_printf("Command: Aligning with beacon\n");
//ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = WaitingBeaconAlign;
//NewEvent.EventType = BeaconAlign; // Default event type
NewEvent.EventType = BeaconAlign;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 20) // ALIGN_TAPE
{
DB_printf("Command: Aligning with tape\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
//PostDriveService(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 30) // MOVE_TO_NODE
{
DB_printf("Command: Moving to node\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = MoveToNode;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 40) // TURN
{
DB_printf("Command: Turning\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
//PostDriveService(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 50) // DROP_BLOCK
{
DB_printf("Command: Dropping block\n");
//ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
//PostDriveService(NewEvent);
CommandToSend = 1;
}
else if (receivedData <= 100 && receivedData >= 110) {
uint16_t DriveTime = 10 * (receivedData - 100);
DB_printf("Command: Drive forward for %d milliseconds\n", DriveTime);
CurrentState = CompletingAction;
NewEvent.EventType = MoveForwardTimed;
NewEvent.EventParam = DriveTime;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 55) {
DB_printf("Command: turn left\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = TurnLeft;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 56) {
DB_printf("Command: turn right\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = TurnRight;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 57) {
DB_printf("Command: turn full\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = TurnFull;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 58) {
DB_printf("Command: moving to block\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = MoveToBlock;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 59) {
DB_printf("Command: turn left at the beacon\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = BeaconLeft;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 75) {
DB_printf("Command: inserting block\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = InsertBlock;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 77) {
DB_printf("Command: inserting block\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = TurnLeftBeacon;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 79) {
DB_printf("Command: driving in reverse\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = DriveReverse;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
else if (receivedData == 78){
DB_printf("Command: stop\n");
ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
CurrentState = CompletingAction;
NewEvent.EventType = Stop;
PostDriveService2(NewEvent);
CommandToSend = 1;
}
}
break;
}
}
break;
case WaitingBeaconAlign:
{
switch (ThisEvent.EventType)
{
case ES_LEADERSENT: // Leader sent data
{
uint8_t receivedData = ThisEvent.EventParam;
//DB_printf("Follower Received SPI Data: %d\n", receivedData);
ES_Event_t NewEvent;
NewEvent.EventType = ES_GameCommand; // Default event type
if (receivedData == 60) // Beacon Found
{
DB_printf("SPI Received: Beacon Found\n");
//ES_Timer_InitTimer(POLL_TIMER, TWO_SEC);
NewEvent.EventType = ES_BEACONFOUND;
PostDriveService2(NewEvent);
CommandToSend = 0;
CurrentState = WaitingCommand;
}
}
break;
}
}
break;
case CompletingAction:
{
switch (ThisEvent.EventType)
{
// case ES_TIMEOUT: // Task finished, send response to leader
// {
// DB_printf("Task Completed! Sending Response to Leader.\n");
// CommandToSend = 200; // Load response message
// CurrentState = WaitingCommand;
// }
// break;
case ES_DriveServiceFinished: // Task finished, send response to leader
{
DB_printf("Task Completed! Sending response to leader.\n");
CommandToSend = COMMAND_COMPLETED; // Load response message
CurrentState = WaitingCommand;
}
break;
}
}
break;
default:
break;
}
return ReturnEvent;
}
/*------------------------------ ISR ------------------------------*/
void __ISR(_SPI_1_VECTOR, IPL7SOFT) SPI1_RX_ISR(void) {
if (1 == SPI1STATbits.SPIRBF){
NewCommand = SPI1BUF;
// Read out the RxBuffer and put it into the newCommand variable
ES_Event_t NewCmdEvent;
NewCmdEvent.EventType = ES_LEADERSENT;
NewCmdEvent.EventParam = NewCommand;
PostFollowerSPIService(NewCmdEvent);
SPI1BUF = CommandToSend;
// uint32_t dummy = SPI1BUF;
// dummy = SPI1BUF;
}
IFS1bits.SPI1RXIF = 0; // clearing the interrupt flag
}
#ifndef FollowerSPIService_H
#define FollowerSPIService_H
#include <stdint.h>
#include <stdbool.h>
#include "ES_Events.h"
#include "ES_Port.h" // needed for definition of REENTRANT
// Public Function Prototypes
typedef enum
{
WaitingCommand,
CompletingAction,
WaitingBeaconAlign,
} FollowerSPIState_t;
bool InitFollowerSPIService(uint8_t Priority);
bool PostFollowerSPIService(ES_Event_t ThisEvent);
ES_Event_t RunFollowerSPIService(ES_Event_t ThisEvent);
#endif /* ServTemplate_H */
DEFINE MODULE CommunicationService
INITIALIZE:
Define CurrentState as AwaitingGameCommand
Define MyPriority as the priority of this service
Define DeferralQueue with capacity 3 + 1
Define dataToSend as 120 (default SPI command byte)
Define beacon_counter as 0
Define LeaderLooking4Beacon as 0 (tracking beacon alignment state)
FUNCTION InitCommunicationService(Priority):
Set MyPriority to Priority
Clear screen for debugging
// SPI Configuration
DISABLE SPI1
DISABLE analog functionality on SPI pins
Configure SPI1 pins (SDI, SDO, SS, CLK)
Configure SPI1 registers for Master Mode
Enable SPI1
// Timer Configuration for Input Capture
DISABLE Timer3
Set Timer3 period
Configure Timer3 prescaler
ENABLE Timer3
// Initialize deferral queue
Initialize DeferralQueue(DeferralQueue, ARRAY_SIZE(DeferralQueue))
// Post initial transition event
CREATE Event ES_INIT
Post Event to Service Queue
RETURN True if successful, False otherwise
FUNCTION PostCommunicationService(Event):
Post Event to Service Queue
RETURN True if successful, False otherwise
FUNCTION RunCommunicationService(CurrentEvent):
DEFINE ReturnEvent as ES_NO_EVENT
SWITCH(CurrentState):
CASE AwaitingGameCommand:
SWITCH(CurrentEvent.Type):
CASE ES_INIT:
PRINT "ES_INIT received in Communication Service"
BREAK
CASE EV_ALIGN_BEACON:
PRINT "Aligning with beacon..."
IF LeaderLooking4Beacon == 0:
Set dataToSend to 10
Set CurrentState to SendingCommand
Set LeaderLooking4Beacon to 1
Start Timer COM_TIMER (200ms)
BREAK
CASE EV_ALIGN_TAPE:
PRINT "Aligning with tape..."
Set dataToSend to 20
Set CurrentState to SendingCommand
Start Timer COM_TIMER (200ms)
BREAK
CASE EV_MOVE_TO_NODE:
PRINT "Moving to node..."
Set dataToSend to 30
Set CurrentState to SendingCommand
Start Timer COM_TIMER (200ms)
BREAK
CASE EV_TURN:
PRINT "Turning..."
Set dataToSend to 40
Set CurrentState to SendingCommand
Start Timer COM_TIMER (200ms)
BREAK
CASE EV_DROP_BLOCK:
PRINT "Dropping block..."
Set dataToSend to 50
Set CurrentState to SendingCommand
Start Timer COM_TIMER (200ms)
BREAK
CASE SendingCommand:
IF CurrentEvent.Type == ES_TIMEOUT AND CurrentEvent.Param == COM_TIMER:
Start Timer COM_TIMER (200ms)
// Ensure SPI buffer is ready
WAIT UNTIL SPI buffer is not full
// Send SPI data
Write SPI1BUF = dataToSend
// Check if response received
IF SPI Receive Buffer is NOT empty:
READ receivedData from SPI1BUF
IF receivedData == COMMAND_RECIEVED:
Set dataToSend to 0 (reset command)
IF LeaderLooking4Beacon == 1:
Start Timer COM_TIMER (5000ms)
Set CurrentState to AligningBeacon
Set LeaderLooking4Beacon to 0
ELSE IF LeaderLooking4Beacon == 2:
Set LeaderLooking4Beacon to 0
Set CurrentState to AwaitingGameCommand
CREATE Event ES_TaskAccomplished
Post Event to MasterSM
ELSE:
Set CurrentState to AwaitingActionCompletion
PRINT "Follower received command"
BREAK
CASE AligningBeacon:
PRINT "Communication Service: Aligning with beacon..."
// Post beacon alignment event
CREATE Event EV_ALIGN_BEACON with Param = 1
Post Event to IRService
IF CurrentEvent.Type == EV_ALIGN_BEACON:
PRINT "Beacon found..."
Set dataToSend to 60
Start Timer COM_TIMER (200ms)
Set CurrentState to SendingCommand
Set LeaderLooking4Beacon to 2
BREAK
CASE AwaitingActionCompletion:
Stop Timer COM_TIMER
IF CurrentEvent.Type == ES_TIMEOUT AND CurrentEvent.Param == COM_TIMER:
Start Timer COM_TIMER (200ms)
// Ensure SPI buffer is ready
WAIT UNTIL SPI buffer is not full
// Send SPI data
Write SPI1BUF = dataToSend
// Check if response received
IF SPI Receive Buffer is NOT empty:
READ receivedData from SPI1BUF
IF receivedData == COMMAND_COMPLETED:
Set dataToSend to 0 (reset command)
CREATE Event ES_TaskAccomplished
Post Event to MasterSM
Set CurrentState to AwaitingGameCommand
PRINT "Follower has completed task"
RETURN ReturnEvent
// Hardware
#include <xc.h>
//#include <proc/p32mx170f256b.h>
// Event & Services Framework
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "ES_DeferRecall.h"
#include "ES_Port.h"
#include "terminal.h"
#include "dbprintf.h"
#include <sys/attribs.h>
#include "GameHSM.h"
#include "CommunicationService.h"
#include "IRService.h"
#include "HSMTemplate.h"
#include "TopHSMTemplate.h"
/*----------------------------- Module Defines ----------------------------*/
// these times assume a 10.000mS/tick timing
#define ONE_SEC 1000
#define HALF_SEC (ONE_SEC / 2)
#define TWO_SEC (ONE_SEC * 2)
#define FIVE_SEC (ONE_SEC * 5)
#define ENTER_POST ((MyPriority<<3)|0) // TODO: what are these variables
#define ENTER_RUN ((MyPriority<<3)|1)
#define ENTER_TIMEOUT ((MyPriority<<3)|2)
#define PB_PS_FREQ 20000000 / 4 // PB clock frequency / prescaler
#define COMMAND_RECIEVED 1
#define COMMAND_COMPLETED 2
// LeaderLooking4Beacon variables
#define NOT_LOOKING 0
#define ALIGNING 1
#define BEACON_FOUND 2
/*---------------------------- Module Functions ---------------------------*/
/* prototypes for private functions for this service.They should be functions
relevant to the behavior of this service
*/
/*---------------------------- Module Variables ---------------------------*/
// with the introduction of Gen2, we need a module level Priority variable
static uint8_t MyPriority;
// add a deferral queue for up to 3 pending deferrals +1 to allow for overhead
static ES_Event_t DeferralQueue[3 + 1];
static volatile uint8_t NewCommand = 0;
//beacon sensing vars
static volatile uint16_t DummyCounter = 0;
volatile static uint32_t RolloverCounter;
volatile static uint32_t BeaconTimePassed;
static volatile uint32_t BeaconPeriod = 0;
static volatile uint32_t BeaconFreq = 0;
static volatile uint32_t BeaconFreq_Last;
static CommunicationServiceState_t CurrentState = AwaitingGameCommand;
static uint8_t dataToSend = 120; // Default SPI command byte
static uint8_t beacon_counter = 0;
static uint8_t LeaderLooking4Beacon = NOT_LOOKING;
typedef union {
uint32_t fullTime;
uint16_t byBytes[2];
}beaconInterval;
/*------------------------------ Module Code ------------------------------*/
bool InitCommunicationService(uint8_t Priority)
{
ES_Event_t ThisEvent;
MyPriority = Priority;
// When doing testing, it is useful to announce just which program
// is running.
clrScrn();
/********************************************
in here you write your initialization code
*******************************************/
SPI1CONbits.ON = 0; // Disable SPI1
// Disable analog functionality on SPI pins
ANSELAbits.ANSA1 = 0; // Disable analog on RA1 (SDI)
// ANSELAbits.ANSA2 = 0; // Disable analog on RA2 (SDO)
ANSELAbits.ANSA0 = 0; // Disable analog on RA0 (SS)
ANSELBbits.ANSB14 = 0; // Disable analog on RB14 (CLK)
// PPS -> SPI mapping
TRISAbits.TRISA1 = 1; // Set RA1 (SDI) as input
SDI1R = 0b0000; // Map RA1 pin 3 to SPI1 SDI
TRISAbits.TRISA2 = 0; // Set RA2 (SDO) as output
RPA2R = 0b0011; // Map RA2 pin 9 to SPI1 SDO
TRISAbits.TRISA0 = 0; // Set RA0 (SS) as output
RPA0R = 0b0011; // map RA0 pin 2 to SPI1 SS. RB3 is pin 7
TRISBbits.TRISB14 = 0; // Set RB14 /25 as output for SCK
TRISBbits.TRISB13 = 1;
ANSELBbits.ANSB13 = 0;
// SPI1 configuration
__builtin_disable_interrupts();
SPI1CONbits.ON = 0; // Disable SPI1 to allow configuration changes
volatile uint8_t dummy = SPI1BUF; // Clear any existing data in the SPI receive buffer
SPI1CONbits.ENHBUF = 0; // Turn enhanced buffer mode ON
SPI1BRG = 50; // clock frequency set to RECALC ***************************************************************
SPI1STATbits.SPIROV = 0; // Clear the SPI receive overflow flag
SPI1CONbits.MSSEN = 1; // Automatically drive the SS pin
SPI1CONbits.MCLKSEL = 0; // Use PBCLK as clock source
SPI1CONbits.ENHBUF = 0; // Enable enhanced buffer mode (FIFO)
SPI1CONbits.MSTEN = 1; // Enable master mode
SPI1CONbits.CKE = 0; // Data changes on falling edge, sampled on rising edge
SPI1CONbits.CKP = 1; // Clock is idle high
//additional stuff Ethan added (PLS CHECK THIS)
SPI1CONbits.FRMPOL=0; //we did this in lecture... means frame pulse (cs) is active low
SPI1CON2bits.AUDEN= 0; //no audio,, did in lec
SPI1CONbits.MODE16= 0;
SPI1CONbits.MODE32= 0; // mode16 and mode32 =0 means we enable 8 bit transfers
//INITIALIZE TIMER FOR INPUT CAPTURE
T3CONbits.ON=0; //disable timer
PR3= 0XFFFF; //setting period for timer 2
T3CONbits.TCS= 0; //use internal PBclk
T3CONbits.TCKPS= 0b010; //CHECK THIS PRESCALER 1:4 FOR INPUT CAPTURE?
T3CONbits.TGATE= 0; //disable gated time accumulation
TMR3=0; //set first timer read to be zero
// //init interrupts for timer 3 THIS IS ESSENTIALLY THE ROLLOVER INTERRUPT
IFS0CLR= _IFS0_T3IF_MASK; //clear T2IF in IFSx register
IPC3bits.T3IP= 6; //set timer to have second highest priority, see 7-2 for map
IEC0SET= _IEC0_T3IE_MASK; //set T2IF in IFSx register
T3CONbits.ON= 1; //enable timer
SPI1CONbits.ON = 1; // Re-enable SPI1
__builtin_enable_interrupts(); //enabling interrupts after initializing spi
// initialize deferral queue for testing Deferral function
ES_InitDeferralQueueWith(DeferralQueue, ARRAY_SIZE(DeferralQueue));
// initialize LED drive for testing/debug output
//setting first val of totaltime to be zero (for first period calc)
// TotalTime=0;
// post the initial transition event
ThisEvent.EventType = ES_INIT;
if (ES_PostToService(MyPriority, ThisEvent) == true)
{
return true;
}
else
{
return false;
}
}
bool PostCommunicationService(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
ES_Event_t RunCommunicationService(ES_Event_t ThisEvent)
{
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // Assume no errors
switch (CurrentState)
{
case AwaitingGameCommand:
{
switch (ThisEvent.EventType)
{
case ES_INIT:
{
//ES_Timer_InitTimer(COM_TIMER, HALF_SEC);
DB_printf("\rES_INIT received in Communication Service %d\r\n", MyPriority);
}
break;
case EV_ALIGN_BEACON:
{
DB_printf("aligning with beacon...\n");
//ES_Timer_InitTimer(COM_TIMER, HALF_SEC);
if (LeaderLooking4Beacon == NOT_LOOKING){ //0 means we want to look for beacon again
//DB_printf("aligning with beacon...\n");
dataToSend = 10;
CurrentState = SendingCommand;
LeaderLooking4Beacon= ALIGNING; // TODO: do we need?
ES_Timer_InitTimer(COM_TIMER, 200);
}
}
break;
case EV_ALIGN_TAPE:
{
DB_printf("aligning with tape...\n");
dataToSend = 20;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_MOVE_TO_NODE:
{
DB_printf("moving to node...\n");
dataToSend = 30;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
// case EV_TURN:
// {
// DB_printf("turning...\n");
// dataToSend = 40;
// CurrentState = SendingCommand;
// ES_Timer_InitTimer(COM_TIMER, 200);
// }
// break;
case EV_DROP_BLOCK:
{
DB_printf("dropping block...\n");
dataToSend = 50;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_TURN_LEFT:
{
DB_printf("turning left...\n");
dataToSend = 55;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_TURN_RIGHT:
{
DB_printf("turning right...\n");
dataToSend = 56;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_TURN_FULL:
{
DB_printf("turning full...\n");
dataToSend = 57;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_MOVE_TO_BLOCK:
{
DB_printf("move toblock in comm service...\n");
dataToSend = 58;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_BEACON_LEFT:
{
DB_printf("turning left at the beacon...\n");
dataToSend = 59;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_TOWER_RAISE:
{
DB_printf("raising tower...\n");
CurrentState = AwaitingActionCompletion;
PostTowerFSM(ThisEvent);
}
break;
case EV_TOWER_LOWER:
{
DB_printf("lowering tower...\n");
CurrentState = AwaitingActionCompletion;
PostTowerFSM(ThisEvent);
}
break;
case EV_MAGNET_UP:
{
DB_printf("magnet up...\n");
CurrentState = AwaitingActionCompletion;
PostTowerFSM(ThisEvent);
}
break;
case EV_MAGNET_DOWN:
{
DB_printf("magnet down...\n");
CurrentState = AwaitingActionCompletion;
PostTowerFSM(ThisEvent);
}
break;
case EV_INSERT_BLOCK:
{
DB_printf("placing the block...\n");
dataToSend = 75;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_BEACON_LEFT_TURN:
{
DB_printf("placing the block...\n");
dataToSend = 77;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_DRIVE_REVERSE:
{
DB_printf("driving in reverse...\n");
dataToSend = 79;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER, 200);
}
break;
case EV_STOP:
{
DB_printf("stop drive \n");
dataToSend= 78;
CurrentState = SendingCommand;
ES_Timer_InitTimer(COM_TIMER,200);
}
break;
default:
break;
}
}
break;
case SendingCommand:
{
if (ThisEvent.EventType == ES_TIMEOUT)
// TODO: Add timer parameter to check for what kind of timeout
{
if (ThisEvent.EventParam == COM_TIMER)
{
ES_Timer_InitTimer(COM_TIMER, 200);
// Ensure SPI transmit buffer is empty before writing new data
while (SPI1STATbits.SPITBF); // Wait until transmit buffer is not full
// Transmit SPI data
SPI1BUF = dataToSend;
// Check received data
if (!SPI1STATbits.SPIRBE) // If receive buffer is NOT empty
{
uint8_t receivedData = SPI1BUF; // Read received byte
//DB_printf("received data: %d\n", receivedData);
if (receivedData == COMMAND_RECIEVED) // If the follower received DROP_BLOCK command
{
dataToSend = 0; // Reset to a default/neutral command
//ES_Timer_StopTimer(COM_TIMER);
if (LeaderLooking4Beacon == ALIGNING) {
ES_Timer_InitTimer(COM_TIMER, 5000);
CurrentState = AligningBeacon;
LeaderLooking4Beacon = NOT_LOOKING;
} else if (LeaderLooking4Beacon == BEACON_FOUND){
LeaderLooking4Beacon = NOT_LOOKING;
CurrentState = AwaitingGameCommand;
ES_Event_t GameEvent;
GameEvent.EventType = ES_TaskAccomplished;
PostMasterSM(GameEvent);
} else{
CurrentState = AwaitingActionCompletion;
}
DB_printf("follower received command\n");
}
}
}
}
}
break;
case AligningBeacon:
{
DB_printf("Com Service, aligning with beacon...\n");
// TO DO ****** SEE IF THIS WORKS!!
//attempt to make it work if already aligned to beacon
ES_Event_t NewEvent;
NewEvent.EventType = EV_ALIGN_BEACON;
NewEvent.EventParam= 1; // this works now
PostIRService(NewEvent);
if (ThisEvent.EventType == EV_ALIGN_BEACON) // posted by ISR
{
DB_printf("we found the beacon...\n");
dataToSend = 60;
ES_Timer_InitTimer(COM_TIMER, 200);
CurrentState = SendingCommand;
LeaderLooking4Beacon = BEACON_FOUND;
}
}
break;
case AwaitingActionCompletion:
{
ES_Timer_StopTimer(COM_TIMER);
if (ThisEvent.EventType == ES_TIMEOUT)
{
if (ThisEvent.EventParam == COM_TIMER)
{
ES_Timer_InitTimer(COM_TIMER, 200);
// Ensure SPI transmit buffer is empty before writing new data
while (SPI1STATbits.SPITBF); // Wait until transmit buffer is not full
// Transmit SPI data
SPI1BUF = dataToSend;
// Check received data
if (!SPI1STATbits.SPIRBE) // If receive buffer is NOT empty
{
uint8_t receivedData = SPI1BUF; // Read received byte
//DB_printf("received data: %d\n", receivedData);
if (receivedData == COMMAND_COMPLETED) //
{
dataToSend = 0; // Reset to a default/neutral command
ES_Event_t NewEvent;
NewEvent.EventType = ES_TaskAccomplished;
PostMasterSM(NewEvent);
CurrentState = AwaitingGameCommand;
}
}
}
}
if (ThisEvent.EventType == ES_TOWER_FINISHED) {
DB_printf("finished processing tower event...\n");
ES_Event_t NewEvent;
NewEvent.EventType = ES_TaskAccomplished;
PostMasterSM(NewEvent);
CurrentState = AwaitingGameCommand;
}
}
break;
}
return ReturnEvent;
}
#ifndef CommunicationService_H
#define CommunicationService_H
#include <stdint.h>
#include <stdbool.h>
#include "ES_Events.h"
#include "ES_Port.h" // needed for definition of REENTRANT
// Public Function Prototypes
typedef enum
{
AwaitingGameCommand,
SendingCommand,
AwaitingActionCompletion,
AligningBeacon,
} CommunicationServiceState_t;
bool InitCommunicationService(uint8_t Priority);
bool PostCommunicationService(ES_Event_t ThisEvent);
ES_Event_t RunCommunicationService(ES_Event_t ThisEvent);
#endif /* ServTemplate_H */
DEFINE MODULE StageOne
INITIALIZE:
Define counter = 0
Define task_sequence as list of events to be processed
Define task_params as corresponding task parameters
Define length_task_sequence as the number of tasks in task_sequence
FUNCTION RunStageOne(CurrentEvent):
DEFINE MakeTransition as False
DEFINE EntryEventKind as ES_ENTRY
DEFINE ReturnEvent as CurrentEvent
PRINT "Enter RunStageOne..."
IF CurrentEvent.Type == ES_ENTRY:
PRINT "Sending first task from stage 1..."
PRINT "Counter is", counter
PRINT "First Task is", task_sequence[counter]
CREATE Event taskEvent with EventType = task_sequence[counter]
POST taskEvent to CommunicationService
Increment counter by 1
ELSE IF CurrentEvent.Type == ES_TaskAccomplished:
IF counter < length_task_sequence:
CREATE Event taskEvent with EventType = task_sequence[counter]
POST taskEvent to CommunicationService
Increment counter by 1
ELSE:
PRINT "Stage should be completed..."
CREATE Event GameEvent with EventType = EV_STAGE_COMPLETE
POST GameEvent to MasterSM
RETURN ReturnEvent
FUNCTION StartStageOne(CurrentEvent):
PRINT "Starting Stage 1..."
CALL RunStageOne(CurrentEvent)
FUNCTION QueryStageOne():
RETURN CurrentState
/****************************************************************************
Module
HSMTemplate.c
Revision
2.0.1
Description
This is a template file for implementing state machines.
Notes
History
When Who What/Why
-------------- --- --------
02/27/17 09:48 jec another correction to re-assign both CurrentEvent
and ReturnEvent to the result of the During function
this eliminates the need for the prior fix and allows
the during function to-remap an event that will be
processed at a higher level.
02/20/17 10:14 jec correction to Run function to correctly assign
ReturnEvent in the situation where a lower level
machine consumed an event.
02/03/16 12:38 jec updated comments to reflect changes made in '14 & '15
converted unsigned char to bool where appropriate
spelling changes on true (was True) to match standard
removed local var used for debugger visibility in 'C32
commented out references to Start & RunLowerLevelSM so
that this can compile.
02/07/13 21:00 jec corrections to return variable (should have been
ReturnEvent, not CurrentEvent) and several EV_xxx
event names that were left over from the old version
02/08/12 09:56 jec revisions for the Events and Services Framework Gen2
02/13/10 14:29 jec revised Start and run to add new kind of entry function
to make implementing history entry cleaner
02/13/10 12:29 jec added NewEvent local variable to During function and
comments about using either it or Event as the return
02/11/10 15:54 jec more revised comments, removing last comment in during
function that belongs in the run function
02/09/10 17:21 jec updated comments about internal transitions on During funtion
02/18/09 10:14 jec removed redundant call to RunLowerlevelSM in EV_Entry
processing in During function
02/20/07 21:37 jec converted to use enumerated type for events & states
02/13/05 19:38 jec added support for self-transitions, reworked
to eliminate repeated transition code
02/11/05 16:54 jec converted to implment hierarchy explicitly
02/25/03 10:32 jec converted to take a passed event parameter
02/18/99 10:19 jec built template from MasterMachine.c
02/14/99 10:34 jec Began Coding
****************************************************************************/
/*----------------------------- Include Files -----------------------------*/
// Basic includes for a program using the Events and Services Framework
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "CommunicationService.h"
/* include header files for this state machine as well as any machines at the
next lower level in the hierarchy that are sub-machines to this machine
*/
#include "HSMTemplate.h"
#include "TopHSMTemplate.h"
/*----------------------------- Module Defines ----------------------------*/
// define constants for the states for this machine
// and any other local defines
#define ENTRY_STATE STATE_ZERO
/*---------------------------- Module Functions ---------------------------*/
/* prototypes for private functions for this machine, things like during
functions, entry & exit functions.They should be functions relevant to the
behavior of this state machine
*/
/*---------------------------- Module Variables ---------------------------*/
// everybody needs a state variable, you may need others as well
static uint8_t counter = 0;
// Define the sequence of commands
static const ES_EventType_t task_sequence[] = {
EV_ALIGN_BEACON, // Works
// EV_DRIVE_REVERSE,
EV_BEACON_LEFT_TURN,
EV_MOVE_TO_NODE, // Works
EV_TURN_RIGHT,
EV_MOVE_TO_NODE,
EV_STOP,
//EV_TOWER_RAISE,
EV_MAGNET_UP,
EV_INSERT_BLOCK,
EV_MAGNET_DOWN,
EV_TURN_FULL,
EV_MOVE_TO_NODE,
EV_TURN_RIGHT,
EV_MOVE_TO_NODE,
EV_TURN_LEFT,
EV_TURN_LEFT,
EV_MOVE_TO_NODE,
EV_TURN_FULL,
EV_MOVE_TO_NODE,
EV_TURN_LEFT,
EV_TURN_LEFT,
EV_TURN_LEFT,
EV_TURN_LEFT,
EV_TURN_LEFT,
EV_STOP
// EV_MOVE_TO_BLOCK,
// EV_TURN_FULL,
// EV_TURN_LEFT
// EV_TOWER_RAISE,
// EV_MAGNET_UP
};
// List of parameters (for turn angles, 0 otherwise)
static const int task_params[] = {
// 0, 0, 0, 0, 0, 0, 90, 0, 180, 0, 0, 90, 0
//for checkoff 3
0, 0, 10
};
int length_task_sequence = sizeof(task_sequence) / sizeof(task_sequence[0]); //this is the number of events in the task sequence
/*------------------------------ Module Code ------------------------------*/
/****************************************************************************
Function
RunTemplateSM
Parameters
ES_Event_t: the event to process
Returns
ES_Event_t: an event to return
Description
add your description here
Notes
uses nested switch/case to implement the machine.
Author
J. Edward Carryer, 2/11/05, 10:45AM
****************************************************************************/
ES_Event_t RunStageOne( ES_Event_t CurrentEvent )
{
bool MakeTransition = false;/* are we making a state transition? */
ES_Event_t EntryEventKind = { ES_ENTRY, 0 };// default to normal entry to new state
ES_Event_t ReturnEvent = CurrentEvent; // assume we are not consuming event
if (CurrentEvent.EventType == ES_ENTRY) {
ES_Event_t taskEvent;
taskEvent.EventType = task_sequence[counter];
PostCommunicationService(taskEvent);
counter = counter + 1; //sets counter to one
}
if (CurrentEvent.EventType == ES_TaskAccomplished){
if (counter < length_task_sequence) {
ES_Event_t taskEvent;
taskEvent.EventType = task_sequence[counter];
PostCommunicationService(taskEvent);
counter = counter + 1;
} else {
DB_printf("stage should be completed...\n");
ES_Event_t GameEvent;
GameEvent.EventType = EV_STAGE_COMPLETE;
PostMasterSM(GameEvent);
}
}
// switch ( CurrentState )
// {
// case STATE_ONE : // If current state is state one
// // Execute During function for state one. ES_ENTRY & ES_EXIT are
// // processed here allow the lower level state machines to re-map
// // or consume the event
// ReturnEvent = CurrentEvent = DuringStateOne(CurrentEvent);
// //process any events
// if ( CurrentEvent.EventType != ES_NO_EVENT ) //If an event is active
// {
// switch (CurrentEvent.EventType)
// {
// case ES_LOCK : //If event is event one
// // Execute action function for state one : event one
// NextState = STATE_TWO;//Decide what the next state will be
// // for internal transitions, skip changing MakeTransition
// MakeTransition = true; //mark that we are taking a transition
// // if transitioning to a state with history change kind of entry
// EntryEventKind.EventType = ES_ENTRY_HISTORY;
// // optionally, consume or re-map this event for the upper
// // level state machine
// ReturnEvent.EventType = ES_NO_EVENT;
// break;
// // repeat cases as required for relevant events
// }
// }
// break;
// // repeat state pattern as required for other states
// }
// // If we are making a state transition
// if (MakeTransition == true)
// {
// // Execute exit function for current state
// CurrentEvent.EventType = ES_EXIT;
// RunTemplateSM(CurrentEvent);
//
// CurrentState = NextState; //Modify state variable
//
// // Execute entry function for new state
// // this defaults to ES_ENTRY
// RunTemplateSM(EntryEventKind);
// }
return(ReturnEvent);
}
/****************************************************************************
Function
StartTemplateSM
Parameters
None
Returns
None
Description
Does any required initialization for this state machine
Notes
Author
J. Edward Carryer, 2/18/99, 10:38AM
****************************************************************************/
void StartStageOne ( ES_Event_t CurrentEvent )
{
// to implement entry to a history state or directly to a substate
// you can modify the initialization of the CurrentState variable
// otherwise just start in the entry state every time the state machine
// is started
DB_printf("starting stage 1...\n");
RunStageOne(CurrentEvent);
}
/****************************************************************************
Function
QueryTemplateSM
Parameters
None
Returns
TemplateState_t The current state of the Template state machine
Description
returns the current state of the Template state machine
Notes
Author
J. Edward Carryer, 2/11/05, 10:38AM
****************************************************************************/
//TemplateState_t QueryStageOne ( void )
//{
// return(CurrentState);
//}
/****************************************************************************
Template header file for Hierarchical Sate Machines AKA StateCharts
02/08/12 adjsutments for use with the Events and Services Framework Gen2
3/17/09 Fixed prototpyes to use Event_t
****************************************************************************/
#ifndef HSMTemplate_H
#define HSMTemplate_H
// typedefs for the states
// State definitions for use with the query function
// typedef enum { EXECUTING } TemplateState_t ;
// Public Function Prototypes
ES_Event_t RunStageOne( ES_Event_t CurrentEvent );
void StartStageOne ( ES_Event_t CurrentEvent );
// TemplateState_t QueryStageOne ( void );
#endif /*SHMTemplate_H */
/*----------------------------- Module Defines ----------------------------*/
// these times assume a 10.000mS/tick timing
#define ONE_SEC 1000
#define HALF_SEC (ONE_SEC / 2)
#define TWO_SEC (ONE_SEC * 2)
#define FIVE_SEC (ONE_SEC * 5)
#define ENTER_POST ((MyPriority<<3)|0)
#define ENTER_RUN ((MyPriority<<3)|1)
#define ENTER_TIMEOUT ((MyPriority<<3)|2)
#define PB_PS_FREQ 20000000 / 4 // PB clock frequency / prescaler
#define PR_VAL 24999 // 20ms period TO DO: CHECK THIS VALUE
#define GREEN 0
#define BLUE 1
//#define TEST_INT_POST
//#define BLINK LED
/*---------------------------- Module Functions ---------------------------*/
void TurnOnRegionIndicator(uint8_t region);
void InitRegionIndicator(void);
/*---------------------------- Module Variables ---------------------------*/
// with the introduction of Gen2, we need a module level Priority variable
static uint8_t MyPriority;
// add a deferral queue for up to 3 pending deferrals +1 to allow for overhead
static ES_Event_t DeferralQueue[3 + 1];
//beacon sensing vars
include all variables here
//Initial init function for the IR Service
bool InitIRService(uint8_t Priority)
{
//Initialize Timer 3 for Output Compare
//Initialize Input Capture 5 for IR Beacon
//Enable Interrupts and set priority number (7)
InitRegionIndicator();
ThisEvent.EventType = ES_INIT;
if (ES_PostToService(MyPriority, ThisEvent) == true)
{
return true;
}
else
{
return false;
}
}
bool PostIRService(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
ES_Event_t RunIRService(ES_Event_t ThisEvent)
{
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // assume no errors
static char DeferredChar = '1';
//switch statement for all events possible
switch (ThisEvent.EventType)
//init case
case ES_INIT:
{
//do nothing
}
break;
//If the Leader is actively looking for the beacon, it will send this service thie event EV_Align_Beacon
case EV_ALIGN_BEACON:
{
if event param ==1: //means the comm service wants to look for a beacon
set InterruptEnable = 0;
set BeaconFreq_Last_Sent= 0; //basically resets values so we can run the ISR again
end if
else: //means
//check to see if we have a frequency that was equal to the last we sent as an event
// this means we are getting a repetitive signal so just reset the isr and search for beacon again
if (BeaconFreq == BeaconFreq_Last_Sent){
// do nothing
set isr_counter = 0;
end nested if
else: //means that this frequency is different than the one we last sent
reset BeaconFreqLast_Sent to BeaconFreq
// now let's turn on the region indicator
Post event to Communication service that the beacon is aligned
Include event parameter that is the beacon time passed so that we can calculate frequency and thus which side we are on
reset the isr so we can run it again
end else statement
//
end else statement
return ReturnEvent;
}
// This function is now the ISR that determines the frequency of the tower IR
void __ISR(_INPUT_CAPTURE_5_VECTOR, IPL7SOFT) BCON__SENSE_ISR(void) {
disable interrupts
set the lastTime variable to be zero
declare a union called realTime
write the buffer value to a variable
store buffer value into realTime union
clear the interrupt flag
if the captured time is greater than 0x800: //this means that the timer interrupt hasn't been dealt with yet
increment rollover counter plus one
clear roll over interrupt with mask
combine rollover counter with captured time to create a uint32
calculate beacontimepassed by subtracting the realtime.fulltime - lasttime
set beaconFreq to pb_ps frequency divided by BeaconTimePassed
end if
// now we need to calculate which tower we are sensing
//discretize tower IR data into distinct buckets
if (2700 <= BeaconFreq) {
BeaconPeriod = 300;
BeaconFreq = 3333;
// beacon G, green
Region = GREEN;
}
else if (1750 <= BeaconFreq) {
BeaconPeriod = 500;
BeaconFreq = 2000;
// beacon L, blue
Region = BLUE;
}
else if (1250 <= BeaconFreq) {
BeaconPeriod = 700;
BeaconFreq = 1427;
// beacon B, blue
Region = BLUE;
}
else {
BeaconPeriod = 1100;
BeaconFreq = 909;
// beacon R, green
Region = GREEN;
}
end discretization if statements
if isr_counter == 7 && InterruptEnable == 0:
// ensures that we only post to the IR service every 100 times the ISR fires
// instead of every time
post beacon align event with beacontimePassed parameter to IRService
set lastTime = realTime.fullTime;
end if
else{
increment isr_counter++;
end else
clear interrupt mask again // just in case
IFS0CLR = _IFS0_IC5IF_MASK;
__builtin_enable_interrupts();
} //end isr
//TIMER ROLLOVER ISR
void __ISR(_TIMER_3_VECTOR, IPL6SOFT) RolloverISR(void){
__builtin_disable_interrupts();
if flag is still pending:
RolloverCounter++;
//clear roll-over interrupt (timer 3)
end if statement
__builtin_enable_interrupts(); //via ed's code
}
// InitRegionIndicator() to set up pins correctly
void InitRegionIndicator()
{
// configure pin 10, RA3 to output
TRISAbits.TRISA3 = 0;
T2CONbits.ON = 0;
//2. Clear the TCS control bit (TxCON<1> = 0) to select the internal PBCLK source.
T2CONbits.TCS = 0;
//3. Select the desired timer input clock prescale.
T2CONbits.TCKPS = 0b011; // 1:8
T2CONbits.TGATE = 0;
//4. Load/Clear the timer register TMRx.
TMR2 = 0;
//5. Load the period register PRx with the desired 16-bit match value.
PR2 = PR_VAL; // 20ms period. PR_VAL is 6500
//6. Set the ON control bit (TxCON<15> = 1) to enable the timer.
T2CONbits.ON = 1;
// init OC3R register to control the servo
OC3CONbits.ON = 0; // Turn off the OC1 when performing the setup
OC3CONbits.OCM = 0b110; // PWM mode
// set timer 2 as the clock source
RPA3R = 0b0101; // assign OC3 to pin 10
OC3CONbits.OCTSEL = 0;
OC3CONbits.OC32 = 0;
OC3CONbits.OCSIDL = 0;
OC3R = 0; // Initialize primary Compare register
OC3RS = 0; // Initialize secondary Compare register
OC3CONbits.ON = 1;
OC3RS = 4000;
return;
}
// private function to control the servo motor.
void TurnOnRegionIndicator(uint8_t region){
if(region == BLUE){
OC3RS = 5500; // 1 ms pulse to move to zero
}else if(region == GREEN ){
DB_printf("This is Supposed to Spin\n");
OC3RS = 1500; // 2 ms pulse to move to 180 degrees
}else {
OC3RS = 2500; // otherwise set to 90 degrees
}
return;
}
#include "../ProjectHeaders/IRService.h"
// Hardware
#include <xc.h>
//#include <proc/p32mx170f256b.h>
// Event & Services Framework
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "ES_DeferRecall.h"
#include "ES_Port.h"
#include "terminal.h"
#include "dbprintf.h"
#include <sys/attribs.h>
#include "IRService.h"
#include "CommunicationService.h"
#include "TopHSMTemplate.h"
/*----------------------------- Module Defines ----------------------------*/
// these times assume a 10.000mS/tick timing
#define ONE_SEC 1000
#define HALF_SEC (ONE_SEC / 2)
#define TWO_SEC (ONE_SEC * 2)
#define FIVE_SEC (ONE_SEC * 5)
#define ENTER_POST ((MyPriority<<3)|0)
#define ENTER_RUN ((MyPriority<<3)|1)
#define ENTER_TIMEOUT ((MyPriority<<3)|2)
#define PB_PS_FREQ 20000000 / 4 // PB clock frequency / prescaler
#define PR_VAL 24999 // 20ms period TO DO: CHECK THIS VALUE
#define GREEN 0
#define BLUE 1
//#define TEST_INT_POST
//#define BLINK LED
/*---------------------------- Module Functions ---------------------------*/
void TurnOnRegionIndicator(uint8_t region);
void InitRegionIndicator(void);
/*---------------------------- Module Variables ---------------------------*/
// with the introduction of Gen2, we need a module level Priority variable
static uint8_t MyPriority;
// add a deferral queue for up to 3 pending deferrals +1 to allow for overhead
static ES_Event_t DeferralQueue[3 + 1];
//beacon sensing vars
volatile static uint32_t RolloverCounter;
volatile static uint32_t BeaconTimePassed = 0;
static volatile uint32_t BeaconPeriod = 0;
static volatile uint32_t BeaconFreq = 0;
static uint32_t BeaconFreq_Last_Sent= 0;
static uint8_t beacon_counter = 0;
static volatile uint8_t isr_counter= 0;
volatile static uint8_t InterruptEnable = 1;
volatile static uint8_t Region = BLUE;
typedef union {
uint32_t fullTime;
uint16_t byBytes[2];
}beaconInterval;
bool InitIRService(uint8_t Priority)
{
ES_Event_t ThisEvent;
MyPriority = Priority;
DB_printf("initializing the IR service\n");
//INITIALIZE TIMER FOR INPUT CAPTURE
T3CONbits.ON=0; //disable timer
PR3= 0XFFFF; //setting period for timer 2
T3CONbits.TCS= 0; //use internal PBclk
T3CONbits.TCKPS= 0b010; //CHECK THIS PRESCALER 1:4 FOR INPUT CAPTURE?
T3CONbits.TGATE= 0; //disable gated time accumulation
TMR3=0; //set first timer read to be zero
//init interrupts for timer 3 THIS IS ESSENTIALLY THE ROLLOVER INTERRUPT
IFS0CLR= _IFS0_T3IF_MASK; //clear T2IF in IFSx register
IPC3bits.T3IP= 6; //set timer to have second highest priority, see 7-2 for map
IEC0SET= _IEC0_T3IE_MASK; //set T2IF in IFSx register
T3CONbits.ON= 1; //enable timer
//INITIALIZE INPUT CAPTURE REGISTER
IC5CONbits.ON=0; //disable input capture
IC5CONbits.ICM= 0b011; //captures on every rising edge
IC5CONbits.ICTMR= 0; //Timer 3 is used for input capture
// IC5CONbits.C32=0; //set to 16 bit timer resource capture
IC5CONbits.ICI= 00; //interrupt on every capture
IC5R= 0b0011; //maps pin RB13 to IC5 port
IFS0CLR= _IFS0_IC5IF_MASK; // clear any interrupt flags
IPC5bits.IC5IP= 7; //sets interrupt priority number
IEC0SET= _IEC0_IC5IE_MASK; //enables interrupts to occur IC3
IC5CONbits.ON=1; //enable input capture
// TODO: ummm i suppose i will ignore this for now
ES_InitDeferralQueueWith(DeferralQueue, ARRAY_SIZE(DeferralQueue));
InitRegionIndicator();
ThisEvent.EventType = ES_INIT;
if (ES_PostToService(MyPriority, ThisEvent) == true)
{
return true;
}
else
{
return false;
}
}
bool PostIRService(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
ES_Event_t RunIRService(ES_Event_t ThisEvent)
{
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // assume no errors
static char DeferredChar = '1';
switch (ThisEvent.EventType)
{
case ES_INIT:
{
DB_printf("\rES_INIT IR service %d\r\n", MyPriority);
}
break;
case EV_ALIGN_BEACON:
{// TODO check this if statement//
DB_printf("\r case EV_Align_Beacon in IR Service %d\r\n", MyPriority);
if (ThisEvent.EventParam == 1){ //this means the comm service wants to look for beacon again
InterruptEnable = 0;
BeaconFreq_Last_Sent = 0; //basically resetting this value so we can run search isr again
DB_printf("getting into here...\n");
}
else
{
//check to see if we have a frequency that was equal to the last we sent as an event
if (BeaconFreq == BeaconFreq_Last_Sent){
// do nothing
isr_counter = 0; //reset the isr counter so it can go back to looking for a beacon
DB_printf("\r Beacon Found, same frequency so do nothing %d\r\n", BeaconFreq);
}
else { //means that this frequency is different than the one we last sent
BeaconFreq_Last_Sent = BeaconFreq; // resetting the value that we last sent
// BeaconFreq_Last_Sent = 0;
// now let's turn on the region indicator
if (InterruptEnable == 0) {
TurnOnRegionIndicator(Region);
ES_Event_t CommEvent;
CommEvent.EventType = EV_ALIGN_BEACON;
CommEvent.EventParam = BeaconTimePassed; //**TOD** SEE IF THIS WOULD BE BETER SENT AS A FREQUENCY
PostCommunicationService(CommEvent);
isr_counter= 0; //reset counter so isr can run again
DB_printf("\r New Beacon Found, frequency %d\r\n", BeaconFreq);
InterruptEnable = 1;
}
}
//
}
// now let's turn on the region indicator
// TurnOnRegionIndicator(Region);
break;
}
break;
// ONLY FOR TESTING SERVO MOTOR
case ES_NEW_KEY:{
if ('b' == ThisEvent.EventParam)
{
ES_Event_t NewEvent;
DB_printf("beacon triggered\n");
NewEvent.EventType = EV_ALIGN_BEACON;
NewEvent.EventParam = BeaconTimePassed;
PostIRService(NewEvent);
}
}
break;
default:
{}
break;
}
return ReturnEvent;
}
/***************************************************************************
private functions
***************************************************************************/
void __ISR(_INPUT_CAPTURE_5_VECTOR, IPL7SOFT) BCON__SENSE_ISR(void) {
__builtin_disable_interrupts();
static uint32_t lastTime = 0;
//static uint8_t counter = 0;
beaconInterval realTime; // declare a new union called realTime.
//first set vars and read buffer
// LastCapturedTime= TotalTime; //copies previous capture time value
volatile uint16_t CapturedTime = IC5BUF; //writes buffer value to a volatile var. TODO: this is not volatile
realTime.byBytes[0] = CapturedTime; // store buffer value into realTime union.
IFS0CLR= _IFS0_IC5IF_MASK; // clear any interrupt flags
if (IFS0bits.T3IF && CapturedTime < 0x8000){
//this means that the timer interrupt hasn't been dealt with yet
RolloverCounter++; //increment rollover counter
IFS0CLR= _IFS0_T3IF_MASK; //clear roll-over interrupt (timer 3)
}
realTime.byBytes[1] = RolloverCounter;
//combine roll_over counter with CapturedTime to create a uint32 (totalTime)
// TotalTime= ((uint32_t)RolloverCounter << 16)|CapturedTime;
// BeaconTimePassed = TotalTime- LastCapturedTime-65536;
BeaconTimePassed = realTime.fullTime - lastTime; // fullTime is the full 32 bits.
// BeaconFreq_Last= BeaconFreq; // resetting last frequency variable
BeaconFreq = PB_PS_FREQ / BeaconTimePassed;
//calculating which tower we are sensing
if (2700 <= BeaconFreq) {
BeaconPeriod = 300;
BeaconFreq = 3333;
// beacon G, green
Region = GREEN;
}
else if (1750 <= BeaconFreq) {
BeaconPeriod = 500;
BeaconFreq = 2000;
// beacon L, blue
Region = BLUE;
}
else if (1250 <= BeaconFreq) {
BeaconPeriod = 700;
BeaconFreq = 1427;
// beacon B, blue
Region = BLUE;
}
else {
BeaconPeriod = 1100;
BeaconFreq = 909;
// beacon R, green
Region = GREEN;
}
if (isr_counter == 7 && InterruptEnable == 0) {
// ensures that we only post to the IR service every 100 times the ISR fires
// instead of every time
ES_Event_t NewEvent;
NewEvent.EventType = EV_ALIGN_BEACON;
NewEvent.EventParam = BeaconTimePassed;
PostIRService(NewEvent);
lastTime = realTime.fullTime;
//isr_counter = isr_counter + 1;
//} else if (isr_counter == 101){ // TODO: huh? why are we adding one instead of setting it back to zero
} else{
isr_counter++;
}
while (IC5CONbits.ICBNE != 0) {
uint16_t dummyclear = IC5BUF;
}
IFS0CLR = _IFS0_IC5IF_MASK;
__builtin_enable_interrupts();
}
//TIMER ROLLOVER ISR
void __ISR(_TIMER_3_VECTOR, IPL6SOFT) RolloverISR(void){
__builtin_disable_interrupts();
if(IFS0bits.T3IF == 1){
//if the flag is still pending
RolloverCounter++;
IFS0CLR= _IFS0_T3IF_MASK; //clear roll-over interrupt (timer 3)
}
__builtin_enable_interrupts(); //via ed's code
}
// InitRegionIndicator() to set up pins correctly
void InitRegionIndicator()
{
// configure pin 10, RA3 to output
TRISAbits.TRISA3 = 0;
T2CONbits.ON = 0;
//2. Clear the TCS control bit (TxCON<1> = 0) to select the internal PBCLK source.
T2CONbits.TCS = 0;
//3. Select the desired timer input clock prescale.
T2CONbits.TCKPS = 0b011; // 1:8
T2CONbits.TGATE = 0;
//4. Load/Clear the timer register TMRx.
TMR2 = 0;
//5. Load the period register PRx with the desired 16-bit match value.
PR2 = PR_VAL; // 20ms period. PR_VAL is 6500
//6. Set the ON control bit (TxCON<15> = 1) to enable the timer.
T2CONbits.ON = 1;
// init OC3R register to control the servo
OC3CONbits.ON = 0; // Turn off the OC1 when performing the setup
OC3CONbits.OCM = 0b110; // PWM mode
// set timer 2 as the clock source
RPA3R = 0b0101; // assign OC3 to pin 10
OC3CONbits.OCTSEL = 0;
OC3CONbits.OC32 = 0;
OC3CONbits.OCSIDL = 0;
OC3R = 0; // Initialize primary Compare register
OC3RS = 0; // Initialize secondary Compare register
OC3CONbits.ON = 1;
OC3RS = 4000;
return;
}
// private function to control the servo motor.
void TurnOnRegionIndicator(uint8_t region){
if(region == BLUE){
OC3RS = 5500; // 1 ms pulse to move to zero
}else if(region == GREEN ){
DB_printf("This is Supposed to Spin\n");
OC3RS = 1500; // 2 ms pulse to move to 180 degrees
}else {
OC3RS = 2500; // otherwise set to 90 degrees
}
return;
}
#ifndef IRService_H
#define IRService_H
#include <stdint.h>
#include <stdbool.h>
#include "ES_Events.h"
#include "ES_Port.h" // needed for definition of REENTRANT
// Public Function Prototypes
bool InitIRService(uint8_t Priority);
bool PostIRService(ES_Event_t ThisEvent);
ES_Event_t RunIRService(ES_Event_t ThisEvent);
#endif /* ServTemplate_H */
Pseudocode for TowerFSM
Initialization (InitTowerFSM)
Configure Timer2 for PWM output (20ms period).
Initialize Output Compare (OC5) for controlling the servo.
Configure GPIO Pins for motor and sensor inputs/outputs.
Set initial state to IdleState.
Post an initialization event (ES_INIT).
Event Posting (PostTowerFSM)
Adds an event to the FSM’s queue.
State Machine Execution (RunTowerFSM)
State: IdleState
If the event posted is…
EV_TOWER_RAISE → Move tower up one level,
controlled via a timer→ Transition to MovingUpState.
EV_TOWER_LOWER → Move tower down one level → Transition
to MovingDownState.
EV_MAGNET_UP → Rotate magnet servo up by 90 degrees→
Transition to RotatingMagnetUp.
EV_MAGNET_DOWN → Rotate magnet down by 90 degrees →
Transition to RotatingMagnetDown.
State: MovingUpState
If the timer associated with the stepper motor (responsible for raising the tower) times out,
Move to the next step in the full-step drive sequence.
If sensor detects threshold, enter timeout mode.
If timeout limit reached, stop movement → Return to IdleState.
Restart the stepper motor timer
State: MovingDownState
Every timer tick (ES_TIMEOUT)
Move the stepper motor to the previous step in the full-step drive sequence.
If sensor detects threshold, enter timeout mode.
If timeout limit reached, stop movement → Return to IdleState.
Restart the stepper motor timer.
State: RotatingMagnetUp / RotatingMagnetDown
On ES_TIMEOUT event associated with the servo rotating the electromagnet:
Stop PWM signal.
Return to IdleState.
Query Function (QueryTowerFSM)
Returns the current state of the FSM.
/****************************************************************************
Module
TowerFSM.c
Revision
1.0.1
Description
This is a template file for implementing flat state machines under the
Gen2 Events and Services Framework.
Notes
History
When Who What/Why
-------------- --- --------
01/15/12 11:12 jec revisions for Gen2 framework
11/07/11 11:26 jec made the queue static
10/30/11 17:59 jec fixed references to CurrentEvent in RunTemplateSM()
10/23/11 18:20 jec began conversion from SMTemplate.c (02/20/07 rev)
****************************************************************************/
/*----------------------------- Include Files -----------------------------*/
/* include header files for this state machine as well as any machines at the
next lower level in the hierarchy that are sub-machines to this machine
*/
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "TowerFSM.h"
#include <xc.h>
#include <sys/attribs.h>
#include <PIC32_AD_Lib.h>
/*----------------------------- Module Defines ----------------------------*/
#define LIFT 1
#define LOWER 0
#define STEP_TIME 7
//#define STEP_DELAY 100
#define LEVEL1 1000 // Adjust these based on testing
#define LEVEL2 5000
#define LEVEL3 10000
#define TIMEOUT_LIMIT_UP 1900
#define TIMEOUT_LIMIT_DOWN 100
#define PR_VAL 24999 // 20ms period TO DO: CHECK THIS VALUE
#define SENSOR_THRESHOLD 400 // TODO: change this value appropriately.
/*---------------------------- Module Functions ---------------------------*/
/* prototypes for private functions for this machine.They should be functions
relevant to the behavior of this state machine
*/
/*---------------------------- Module Variables ---------------------------*/
// everybody needs a state variable, you may need others as well.
// type of state variable should match htat of enum in header file
static TemplateState_t CurrentState;
// with the introduction of Gen2, we need a module level Priority var as well
static uint8_t MyPriority;
static uint32_t ADC_Results[0]; // Array to store ADC readings
static uint8_t levelDir = 0;
static uint16_t currentStep = 0;
static uint16_t counter = 0;
static uint16_t target = 0;
static const uint8_t fullStepSequence[4][4] = {
{1, 0, 0, 1}, // Step 1
{0, 1, 0, 1}, // Step 2
{0, 1, 1, 0}, // Step 3
{1, 0, 1, 0}, // Step 4
};
static uint8_t pseudoState = 1;
static uint32_t timeoutCounter = 0;
/*------------------------------ Module Code ------------------------------*/
/****************************************************************************
Function
InitTowerFSM
Parameters
uint8_t : the priorty of this service
Returns
bool, false if error in initialization, true otherwise
Description
Saves away the priority, sets up the initial transition and does any
other required initialization for this state machine
Notes
Author
J. Edward Carryer, 10/23/11, 18:55
****************************************************************************/
bool InitTowerFSM(uint8_t Priority)
{
// set up the timer for PWM OC stuff
// configure pin 12, RA4 to output
// ***TODO*** map to the correct pin.
TRISAbits.TRISA4 = 0;
T2CONbits.ON = 0;
//2. Clear the TCS control bit (TxCON<1> = 0) to select the internal PBCLK source.
T2CONbits.TCS = 0;
//3. Select the desired timer input clock prescale.
T2CONbits.TCKPS = 0b011; // 1:8
T2CONbits.TGATE = 0;
//4. Load/Clear the timer register TMRx.
TMR2 = 0;
//5. Load the period register PRx with the desired 16-bit match value.
PR2 = PR_VAL; // 20ms period. PR_VAL is 6500
//6. Set the ON control bit (TxCON<15> = 1) to enable the timer.
T2CONbits.ON = 1;
// init OC5R register to control the servo
OC5CONbits.ON = 0; // Turn off the OC1 when performing the setup
OC5CONbits.OCM = 0b110; // PWM mode
// set timer 2 as the clock source
RPA4R = 0b0110; // assign OC5 to pin 14
OC5CONbits.OCTSEL = 0;
OC5CONbits.OC32 = 0;
OC5CONbits.OCSIDL = 0;
OC5R = 0; // Initialize primary Compare register
OC5RS = 0; // Initialize secondary Compare register
OC5CONbits.ON = 1;
ES_Event_t ThisEvent;
MyPriority = Priority;
TRISBbits.TRISB4 = 0;
TRISBbits.TRISB5 = 0;
TRISBbits.TRISB9 = 0;
TRISBbits.TRISB10 = 0;
ANSELBbits.ANSB15 = 0; // Disable analog function on RB15
TRISBbits.TRISB15 = 1; // Set RB15 as input
// put us into the Initial PseudoState
CurrentState = IdleState;
// post the initial transition event
ThisEvent.EventType = ES_INIT;
DB_printf("This is Initializing the Tower FSM\n");
if (ES_PostToService(MyPriority, ThisEvent) == true)
{
return true;
}
else
{
return false;
}
}
/****************************************************************************
Function
PostTowerFSM
Parameters
EF_Event_t ThisEvent , the event to post to the queue
Returns
boolean False if the Enqueue operation failed, True otherwise
Description
Posts an event to this state machine's queue
Notes
Author
J. Edward Carryer, 10/23/11, 19:25
****************************************************************************/
bool PostTowerFSM(ES_Event_t ThisEvent)
{
return ES_PostToService(MyPriority, ThisEvent);
}
/****************************************************************************
Function
RunTowerFSM
Parameters
ES_Event_t : the event to process
Returns
ES_Event_t, ES_NO_EVENT if no error ES_ERROR otherwise
Description
add your description here
Notes
uses nested switch/case to implement the machine.
Author
J. Edward Carryer, 01/15/12, 15:23
****************************************************************************/
ES_Event_t RunTowerFSM(ES_Event_t ThisEvent)
{
ES_Event_t ReturnEvent;
ReturnEvent.EventType = ES_NO_EVENT; // assume no errors
switch (CurrentState)
{
case IdleState:
{
switch (ThisEvent.EventType)
{
case EV_TOWER_RAISE:
{
DB_printf("Raising the tower...\n");
timeoutCounter = 0; // Counter for stopping after ADC threshold
pseudoState = 1; // Start in normal stepping mode
ES_Timer_InitTimer(MOTOR_TIMER, STEP_TIME);
CurrentState = MovingUpState;
}
break;
case EV_TOWER_LOWER:
{
DB_printf("Lowering the tower...\n");
levelDir = LOWER; // Set direction for lowering
timeoutCounter = 0; // Counter for stopping after ADC threshold
pseudoState = 0; // Start in normal stepping mode
ES_Timer_InitTimer(MOTOR_TIMER, STEP_TIME);
CurrentState = MovingDownState;
}
break;
case EV_MAGNET_UP:
{
DB_printf("Magnet moving up...\n");
OC5RS = 1550; // Set servo to 90 degrees
ES_Timer_StopTimer(MOTOR_TIMER);
ES_Timer_InitTimer(MOTOR_TIMER, 3000); // Wait 100ms
CurrentState = RotatingMagnetUp;
}
break;
case EV_MAGNET_DOWN:
{
DB_printf("Magnet moving down...\n");
OC5RS = 3500; // Set servo to 180 degrees
ES_Timer_StopTimer(MOTOR_TIMER);
ES_Timer_InitTimer(MOTOR_TIMER, 2000); // Wait 100ms
CurrentState = RotatingMagnetDown;
}
break;
default:
break;
}
}
break;
//---------------------------------- Moving Up ----------------------------------
case MovingUpState:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == MOTOR_TIMER)
{
// Continue stepping through the stepper sequence
LATBbits.LATB4 = fullStepSequence[currentStep][0];
LATBbits.LATB5 = fullStepSequence[currentStep][1];
LATBbits.LATB9 = fullStepSequence[currentStep][2];
LATBbits.LATB10 = fullStepSequence[currentStep][3];
if (currentStep == 3) {
currentStep = 0;
} else {
currentStep++;
}
if (pseudoState == 0) {
// Normal stepping, check ADC threshold
if (PORTBbits.RB15 == 0) {
DB_printf("Threshold reached, switching to timeout mode.\n");
pseudoState = 1; // Enter timeout mode
}
}
else if (pseudoState == 1) {
// Increment timeout counter while still stepping
timeoutCounter++;
DB_printf("Timeout counter: %d\n", timeoutCounter);
if (timeoutCounter > TIMEOUT_LIMIT_UP) {
DB_printf("Stopping movement (UP).\n");
CurrentState = IdleState;
ES_Event_t TowerEvent;
TowerEvent.EventType = ES_TOWER_FINISHED;
PostCommunicationService(TowerEvent);
}
}
// Continue stepping motor
ES_Timer_InitTimer(MOTOR_TIMER, STEP_TIME);
}
}
break;
//---------------------------------- Moving Down ----------------------------------
case MovingDownState:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == MOTOR_TIMER)
{
ADC_MultiRead(ADC_Results);
uint16_t sensorValue = ADC_Results[0]; // only read 1 pin...
DB_printf("ADC Reading: %d\n", sensorValue);
// Continue stepping through the stepper sequence
LATBbits.LATB4 = fullStepSequence[currentStep][0];
LATBbits.LATB5 = fullStepSequence[currentStep][1];
LATBbits.LATB9 = fullStepSequence[currentStep][2];
LATBbits.LATB10 = fullStepSequence[currentStep][3];
if (currentStep == 0) {
currentStep = 3;
} else {
currentStep--;
}
if (pseudoState == 0) {
// Normal stepping, check ADC threshold
if (sensorValue < SENSOR_THRESHOLD) {
DB_printf("Threshold reached, switching to timeout mode.\n");
pseudoState = 1; // Enter timeout mode
}
}
else if (pseudoState == 1) {
// Increment timeout counter while still stepping
timeoutCounter++;
DB_printf("Timeout counter: %d\n", timeoutCounter);
if (timeoutCounter > TIMEOUT_LIMIT_DOWN) {
DB_printf("Stopping movement (DOWN).\n");
// CurrentState = IdleState;
// ES_Event_t TowerEvent;
// TowerEvent.EventType = ES_TOWER_FINISHED;
// PostCommunicationService(TowerEvent);
// break;
}
}
// Continue stepping motor
ES_Timer_InitTimer(MOTOR_TIMER, STEP_TIME);
}
}
break;
case RotatingMagnetUp:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == MOTOR_TIMER)
{
DB_printf("Finished rotating magnet up, turning OC5 off...\n");
OC5RS = 0; // Turn off PWM output
CurrentState = IdleState;
ES_Event_t TowerEvent;
TowerEvent.EventType = ES_TOWER_FINISHED;
PostCommunicationService(TowerEvent);
}
}
break;
//---------------------------------- Rotating Magnet Down ----------------------------------
case RotatingMagnetDown:
{
if (ThisEvent.EventType == ES_TIMEOUT && ThisEvent.EventParam == MOTOR_TIMER)
{
DB_printf("Finished rotating magnet down, turning OC5 off...\n");
OC5RS = 0; // Turn off PWM output
CurrentState = IdleState;
ES_Event_t TowerEvent;
TowerEvent.EventType = ES_TOWER_FINISHED;
PostCommunicationService(TowerEvent);
}
}
break;
default:
{
}
break;// end switch on Current State
}
// break;
return ReturnEvent;
}
/****************************************************************************
Function
QueryTemplateSM
Parameters
None
Returns
TemplateState_t The current state of the Template state machine
Description
returns the current state of the Template state machine
Notes
Author
J. Edward Carryer, 10/23/11, 19:21
****************************************************************************/
TemplateState_t QueryTemplateFSM(void)
{
return CurrentState;
}
/****************************************************************************
Header file for template Flat Sate Machine
based on the Gen2 Events and Services Framework
****************************************************************************/
#ifndef FSMTower_H
#define FSMTower_H
// Event Definitions
#include "ES_Configure.h" /* gets us event definitions */
#include "ES_Types.h" /* gets bool type for returns */
// typedefs for the states
// State definitions for use with the query function
typedef enum
{
InitPState, IdleState, MovingUpState, MovingDownState, RotatingMagnetUp, RotatingMagnetDown
}TemplateState_t;
// Public Function Prototypes
bool InitTowerFSM(uint8_t Priority);
bool PostTowerFSM(ES_Event_t ThisEvent);
ES_Event_t RunTowerFSM(ES_Event_t ThisEvent);
TemplateState_t QueryTowerSM(void);
void StartStepperMotor(uint8_t levelDir);
void StopStepperMotor(void);
void StepperTest();
#endif /* FSMTower_H */
DEFINE MODULE GameHSM
INITIALIZE:
Define CurrentState as STATE_ONE
Define MyPriority as the priority of this service
FUNCTION InitMasterSM(Priority):
Set MyPriority to Priority
Create an Event (ES_ENTRY)
Call StartMasterSM(Event)
RETURN True
FUNCTION PostMasterSM(Event):
Post Event to service queue using MyPriority
RETURN True if successful, False otherwise
FUNCTION RunMasterSM(CurrentEvent):
SET MakeTransition to False
SET NextState to CurrentState
DEFINE EntryEventKind as ES_ENTRY
DEFINE ReturnEvent as ES_NO_EVENT
SWITCH(CurrentState):
CASE STATE_ONE:
CurrentEvent ← DuringStateOne(CurrentEvent)
IF (CurrentEvent is not ES_NO_EVENT):
SWITCH(CurrentEvent.Type):
CASE EV_STAGE_COMPLETE:
IF CurrentState == STATE_ONE:
PRINT "Moving to stage 2..."
NextState ← STATE_TWO
MakeTransition ← True
ELSE IF CurrentState == STATE_TWO:
NextState ← STATE_THREE
MakeTransition ← True
EntryEventKind ← ES_ENTRY_HISTORY
BREAK
CASE STATE_TWO:
PRINT "We are now in state 2..."
// Repeat pattern for additional states
IF MakeTransition == True:
CREATE Event ES_EXIT
CALL RunMasterSM(ES_EXIT) // Exit current state
SET CurrentState to NextState
CALL RunMasterSM(EntryEventKind) // Enter new state
RETURN ReturnEvent
FUNCTION StartMasterSM(CurrentEvent):
SET CurrentState to STATE_ONE
CALL RunMasterSM(CurrentEvent)
FUNCTION QueryTopHSMTemplateSM():
RETURN CurrentState
FUNCTION DuringStateOne(Event):
DEFINE ReturnEvent as Event
IF Event.Type == ES_ENTRY or ES_ENTRY_HISTORY:
CALL StartStageOne(Event)
ELSE IF Event.Type == ES_EXIT:
CALL RunStageOne(Event)
ELSE:
CALL RunStageOne(Event)
RETURN ReturnEvent
/****************************************************************************
Module
TopHSMTemplate.c
Revision
2.0.1
Description
This is a template for the top level Hierarchical state machine
Notes
History
When Who What/Why
-------------- --- --------
02/20/17 14:30 jec updated to remove sample of consuming an event. We
always want to return ES_NO_EVENT at the top level
unless there is a non-recoverable error at the
framework level
02/03/16 15:27 jec updated comments to reflect small changes made in '14 & '15
converted unsigned char to bool where appropriate
spelling changes on true (was True) to match standard
removed local var used for debugger visibility in 'C32
removed Microwave specific code and replaced with generic
02/08/12 01:39 jec converted from MW_MasterMachine.c
02/06/12 22:02 jec converted to Gen 2 Events and Services Framework
02/13/10 11:54 jec converted During functions to return Event_t
so that they match the template
02/21/07 17:04 jec converted to pass Event_t to Start...()
02/20/07 21:37 jec converted to use enumerated type for events
02/21/05 15:03 jec Began Coding
****************************************************************************/
/*----------------------------- Include Files -----------------------------*/
/* include header files for this state machine as well as any machines at the
next lower level in the hierarchy that are sub-machines to this machine
*/
#include "ES_Configure.h"
#include "ES_Framework.h"
#include "TopHSMTemplate.h"
#include "dbprintf.h"
#include "HSMTemplate.h"
#include "CommunicationService.h"
/*----------------------------- Module Defines ----------------------------*/
#define SIXTY_SEC 60000
#define EIGHTEEN_SEC 18000
#define TEN_SEC 10000
/*---------------------------- Module Functions ---------------------------*/
static ES_Event_t DuringStateOne( ES_Event_t Event);
void InitLED(void);
/*---------------------------- Module Variables ---------------------------*/
// everybody needs a state variable, though if the top level state machine
// is just a single state container for orthogonal regions, you could get
// away without it
static MasterState_t CurrentState;
// with the introduction of Gen2, we need a module level Priority var as well
static uint8_t MyPriority;
static uint8_t timer_counter = 0; // this is for the game timer ,2m 18s
/*------------------------------ Module Code ------------------------------*/
/****************************************************************************
Function
InitMasterSM
Parameters
uint8_t : the priorty of this service
Returns
boolean, False if error in initialization, True otherwise
Description
Saves away the priority, and starts
the top level state machine
Notes
Author
J. Edward Carryer, 02/06/12, 22:06
****************************************************************************/
bool InitMasterSM ( uint8_t Priority )
{
ES_Event_t ThisEvent;
MyPriority = Priority; // save our priority
ThisEvent.EventType = ES_ENTRY;
// Start the Master State machine
ES_Timer_InitTimer(BUTTON_TIMER, 100);
InitLED(); //initializes the game LED
StartMasterSM( ThisEvent );
// initialize the button
ANSELBbits.ANSB12 = 0; // Disable analog function (set RB12 as digital)
TRISBbits.TRISB12 = 1; // Set RB12 as an input
CNPDBbits.CNPDB12 = 1; // enable the pull-down resistor
return true;
}
/****************************************************************************
Function
PostMasterSM
Parameters
ES_Event_t ThisEvent , the event to post to the queue
Returns
boolean False if the post operation failed, True otherwise
Description
Posts an event to this state machine's queue
Notes
Author
J. Edward Carryer, 10/23/11, 19:25
****************************************************************************/
bool PostMasterSM( ES_Event_t ThisEvent )
{
return ES_PostToService( MyPriority, ThisEvent);
}
/****************************************************************************
Function
RunMasterSM
Parameters
ES_Event: the event to process
Returns
ES_Event: an event to return
Description
the run function for the top level state machine
Notes
uses nested switch/case to implement the machine.
Author
J. Edward Carryer, 02/06/12, 22:09
****************************************************************************/
ES_Event_t RunMasterSM( ES_Event_t CurrentEvent )
{
bool MakeTransition = false;/* are we making a state transition? */
MasterState_t NextState = CurrentState;
ES_Event_t EntryEventKind = { ES_ENTRY, 0 };// default to normal entry to new state
ES_Event_t ReturnEvent = { ES_NO_EVENT, 0 }; // assume no error
switch ( CurrentState )
{
case INIT_STATE :
{
if ( CurrentEvent.EventType == ES_TIMEOUT && CurrentEvent.EventParam == BUTTON_TIMER) {
if (PORTBbits.RB12 == 1) // this means we've clicked the button
{
LATBbits.LATB11 = 1; // turn this boi on (LED game indicator)
ES_Timer_InitTimer(GAME_TIMER, SIXTY_SEC); // this is the first time the GAME_TIMER gets initialized
CurrentState = STATE_ONE;
DB_printf("starting the game!\n");
ES_Event_t FinalEvent;
FinalEvent.EventType = ES_ENTRY;
PostMasterSM(FinalEvent);
} else {
ES_Timer_InitTimer(BUTTON_TIMER, 100);
}
}
}
break;
case END_GAME :
{
LATBbits.LATB11 = 0; // turn this boi off (led game indicator)
DB_printf("enter end game (turn LED off)\n");
//post a stop to drive service
}
break;
case STATE_ONE :
{// If current state is state one
// Execute During function for state one. ES_ENTRY & ES_EXIT are
// processed here allow the lowere level state machines to re-map
// or consume the event
CurrentEvent = DuringStateOne(CurrentEvent);
//process any events
if ( CurrentEvent.EventType != ES_NO_EVENT ) //If an event is active
{
switch (CurrentEvent.EventType)
{
case EV_STAGE_COMPLETE:
if (CurrentState == STATE_ONE) {
DB_printf("stage complete...\n");
//NextState = STATE_TWO;
//MakeTransition = true;
} else if (CurrentState == STATE_TWO) {
NextState = STATE_THREE;
MakeTransition = true;
}
// Execute action function for state one : event one
//NextState = STATE_TWO;//Decide what the next state will be
// for internal transitions, skip changing MakeTransition
//MakeTransition = true; //mark that we are taking a transition
// if transitioning to a state with history change kind of entry
//EntryEventKind.EventType = ES_ENTRY_HISTORY;
break;
// repeat cases as required for relevant events
}
}
//This is our game timer Logic... we have a one minute timer that will timeout twice, then set an 18 second timer that will end the game
if ( CurrentEvent.EventType == ES_TIMEOUT && CurrentEvent.EventParam == GAME_TIMER) //If an event is active
{
if(timer_counter < 1) // timer_counter starts as 0
{
ES_Timer_InitTimer(GAME_TIMER, SIXTY_SEC); // one minute timeout
DB_printf("One minute...\n");
}
else if(timer_counter == 1)
{
ES_Timer_InitTimer(GAME_TIMER, EIGHTEEN_SEC); // 18 second timeout
DB_printf("two minute ...\n");
}
else if(timer_counter == 2){
DB_printf("game over...\n");
LATBbits.LATB11 = 0; // turn off the LED indicator
CurrentState = END_GAME; //move to end game state
ES_Event_t FinalEvent;
FinalEvent.EventType = ES_GAME_OVER;
PostMasterSM(FinalEvent); // TODO: add ES_GAME_OVER case and turn off the drive motors.
}
timer_counter++;
}
}
break;
case STATE_TWO:
DB_printf("we are now in state 2...\n");
// repeat state pattern as required for other states
}
// If we are making a state transition
if (MakeTransition == true)
{
// Execute exit function for current state
CurrentEvent.EventType = ES_EXIT;
RunMasterSM(CurrentEvent);
CurrentState = NextState; //Modify state variable
// Execute entry function for new state
// this defaults to ES_ENTRY
RunMasterSM(EntryEventKind);
}
// in the absence of an error the top level state machine should
// always return ES_NO_EVENT, which we initialized at the top of func
return(ReturnEvent);
}
/****************************************************************************
Function
StartMasterSM
Parameters
ES_Event CurrentEvent
Returns
nothing
Description
Does any required initialization for this state machine
Notes
Author
J. Edward Carryer, 02/06/12, 22:15
****************************************************************************/
void StartMasterSM ( ES_Event_t CurrentEvent )
{
// if there is more than 1 state to the top level machine you will need
// to initialize the state variable
DB_printf("initializing the game...\n");
CurrentState = INIT_STATE;
ES_Timer_InitTimer(GAME_TIMER, 1000);
// now we need to let the Run function init the lower level state machines
// use LocalEvent to keep the compiler from complaining about unused var
RunMasterSM(CurrentEvent);
return;
}
MasterState_t QueryTopHSMTemplateSM ( void )
{
return(CurrentState);
}
static ES_Event_t DuringStateOne( ES_Event_t Event)
{
ES_Event_t ReturnEvent = Event; // assme no re-mapping or comsumption
// process ES_ENTRY, ES_ENTRY_HISTORY & ES_EXIT events
if ( (Event.EventType == ES_ENTRY) || (Event.EventType == ES_ENTRY_HISTORY) )
{
StartStageOne(Event);
}
else if ( Event.EventType == ES_EXIT )
{
RunStageOne(Event);
}else
{
RunStageOne(Event);
}
return(ReturnEvent);
}
//init function for gametime LED
void InitLED(void){
//RB15, PIN 26
TRISBbits.TRISB11 = 0; // set to output
LATBbits.LATB11 = 0; // turn this boi on
DB_printf("enter init led function (turn LED on)\n");
}
/****************************************************************************
Template header file for Hierarchical Sate Machines AKA StateCharts
****************************************************************************/
#ifndef TopHSMTemplate_H
#define TopHSMTemplate_H
// State definitions for use with the query function
typedef enum { INIT_STATE, STATE_ONE, STATE_TWO, STATE_THREE, END_GAME } MasterState_t ;
// Public Function Prototypes
ES_Event_t RunMasterSM( ES_Event_t CurrentEvent );
void StartMasterSM ( ES_Event_t CurrentEvent );
bool PostMasterSM( ES_Event_t ThisEvent );
bool InitMasterSM ( uint8_t Priority );
#endif /*TopHSMTemplate_H */