Mechanical Components

Mechanical Components

Chassis

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.

Chassis CAD

CAD Drawings of Chassis

Mechanical Components

Motor Accessories

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.

Motor Accessories CAD

CAD Drawings of Motor Accessories

Mechanical Components

Cascade Lift

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.

Cascade lift CAD

CAD Drawings of Cascade Lift

Mechanical Components

Magnet Servo

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.

Lift Servo CAD

CAD Drawings of Magnet Servos

Mechanical Components

IR Sensor Mount

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.

IR Sensor Mount CAD

CAD Drawing of IR Sensor Mount

IR Sensor Mount CAD Drawing
Mechanical Components

Battery Holders

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.

Batter Holder CAD

CAD Drawings of Battery Holders

Electrical Components

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.

Leader/Follower KiCADSubsystems CAD
Electrical Components

Block Diagram

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!

Block Diagram from Figma
Electrical Components

Signal Conditioning

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.  

KiCAD diagram of Signal Conditioning Circuit
Electrical Components

Leader Subsystem

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.

KiCAD diagram of Leader Subsystem

PIC32MX170F256B

  • In charge of gameplay State Machine
  • Sent commands and received information to the Follower via SPI
  • Controlled the Lifting Stepper Motor
  • Controlled Magnet Servo and Gameplay Indicator Servo using Output Compare
  • Controlled electromagnet
  • Commanded LED Gameplay Indicator
  • Microcontroller

5V Electromagnet with 2.5 Kg Holding Force

  • Main function was to pick up and put down the CRATEs
  • Sourced current from a N-Type MOSFET to create a magnetic field to grab CRATEs. This was designed to safely acquire the current necessary for the electromagnet function while limiting current drawn from the leader.
  • KiCAD diagram for EM
  • Electromagnet

Arcade Style Game Start Button

  • Began the game sequence when pushed
  • KiCAD diagram for Game Start

LED Diode

  • The LED Diode Indicates Gameplay.
  • Red LED diode

N Channel Mosfet

  • Used as a gate to provide power to the LED Diodes and Electromagnet
  • An N Channel Mosfet

Game Indicator Servo Motor (FM90)

  • This servo waved a flag indicating blue or green depending on which side the bot was on
  • a game indicator Servo

Magnet Lift Servo (HS-322HD Standard Heavy Duty Servo)

  • This servo was used to rotate the electromagnet 90 degrees after it picked up a CRATE so that it could deposit it into the vertical towers.
  • This servo wasn’t specifically used in the lifting mechanism, rather it compliments the cascade lift by changing the orientation of the CRATE so it is easy to deposit the CRATE into the tower.
  • a Standard Heavy Duty Servo

Lifting Stepper Motor (Mercury Motors SM-42BYG011-25)

  • The lifting stepper motor raised and lowered the cascade lift so that we could lift our CRATE to reach the tower.
  • This motor was connected to the leader with its own off-the-shelf motor driver that included an H-Bridge.
  • Stepper Motor
Electrical Components

Follower Subsystem

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.

Follower KiCAD

Tape Sensors

  • The tape sensors are infrared sensors that lie on the bottom of the bot. These analog sensors post a numerical value that corresponds to how much light reflects back into them. The lowest values correspond to being located on the black tape. This feedback, paired with a driving algorithm, allowed us to maintain the bot’s location on the tape.
  • a KiCAD diagram and a Tape Sensor

Drive Motors (33GN2732) 

  • The drive motors were used to drive the wheels of the bot. We elected not to incorporate encoders into these motors because all feedback was included in the line following algorithm. The Drive Motors were driven with the Output Compare modules of the PIC 32 and included motor drivers with an H-Bridge and flyback diodes to protect against current surges.
  • A Drive Motor

Software Components

SoftWare Components – State Machine
Top Level SM
Top Level SMStage One SMStage Two SMStage Three SMDrive Service SMPayload SMTower SMTower Detection SM

Click below to view pseudocode, source code, and header files.

SoftWare Components – Drive Service Pseudo Code
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.


SoftWare Components –  DRIVE Service Source Code
/****************************************************************************
 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 ------------------------------*/
SoftWare Components –  DRIVE Service Header 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 */
SoftWare Components –  Follower SPI Service Pseudo code
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
SoftWare Components – Follower SPI Service Source Code
// 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
}
SoftWare Components – Follower SPI Service Header file
#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 */
SoftWare Components – Communication Service pseudo Code
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
SoftWare Components – Communication Service source Code
// 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;
}
SoftWare Components – Communication Service Header file
#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 */
SoftWare Components – HSM Pseudo Code
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
SoftWare Components – HSM Source Code
/****************************************************************************
 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);
//}
SoftWare Components – HSM Header File
/****************************************************************************
 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 */
SoftWare Components – IR Service Pseudo Code


/*----------------------------- 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;
}
SoftWare Components – IR Service Source Code
#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;
}
SoftWare Components – IR Service Header File
#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 */
SoftWare Components – Tower HSM Pseudo Code
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.

SoftWare Components – Tower HSM Source Code
/****************************************************************************
 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;
}
SoftWare Components – TOWER HSM Header file
/****************************************************************************

  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 */
SoftWare Components – Top HSM Pseudo Code
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
SoftWare Components – TOP HSM Source Code
/****************************************************************************
 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");
}
SoftWare Components – TOP HSM Header File
/****************************************************************************
 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 */