Skip to content

Latest commit

 

History

History
4942 lines (3998 loc) · 153 KB

BIRDS-X_SOFTWARE_DEVELOPER_MANUAL.md

File metadata and controls

4942 lines (3998 loc) · 153 KB

BIRDS-X SATELLITE SOFTWARE DEVELOPER’S MANUAL

birds x

Introduction

Document Purpose

This document is prepared to share a thorough explaination of the execution of the BIRDS-X satellite bus software.

The system block diagram of the BIRDS-X satellite is detailed in figure 1 below.

BUS SYSTEM BLOCK DIAGRAM

OBC BLOCK DIAGRAM

DETAILED OBC BLOCK DIAGRAM

The organisational file stucture of the microcontrollers in the BIRDS-X OBC board is detailed in figure 2 below.

1. START PIC

File content

The files found in the START PIC folder are detailed in Table 1 below.

File name pattern Scope Content
STRATPIC.c
StartPIC_Functions.c

This is a system where data arrives asynchronously and must be processed in real-time without losing bytes.

It uses a mix of hardware register mappings and UART (Universal Asynchronous Receiver-Transmitter) configuration to handle communication and operations.

This code sets up a hardware abstraction layer for:

  • Timer 1 Configuration: Provides named access to specific bits in the T1CON register for controlling and configuring the Timer 1 peripheral.
  • Data Communication: Configures a UART interface and declares variables and buffers for handling incoming and outgoing data.

It uses interrupts to receive data, buffers it, and provides functions to read from the buffer and process incoming data.

Let's break it down line by line:

STARTPIC_FUNCTIONS.C

Hardware Register and Bit Definitions

#byte T1CON = 0x018

This line declares a byte (8-bit value) variable T1CON mapped to the hardware memory address 0x018. This address typically corresponds to the T1CON register in a microcontroller, which is used to control Timer 1. The register allows configuration of timer settings, such as prescaling, synchronization, and enabling the timer.

#bit    TMR1ON = T1CON.0

Declares TMR1ON as a bit within the T1CON register at bit position 0. TMR1ON is used to enable or disable Timer 1.

#bit    T1SYNC = T1CON.2

Maps T1SYNC to bit position 2 of the T1CON register. This bit controls the synchronization of Timer 1 with an external clock source.

#bit    T1OSCEN = T1CON.3

Maps T1OSCEN to bit position 3. This bit enables or disables the Timer 1 oscillator.

#bit    T1CKPS0 = T1CON.4

Maps T1CKPS0 to bit position 4. This bit is part of the prescaler configuration for Timer 1.

#bit    T1CKPS1 = T1CON.5

Maps T1CKPS1 to bit position 5. This works in combination with T1CKPS0 to set the prescaler value.

#bit    TMR1CS0 = T1CON.6

Maps TMR1CS0 to bit position 6. This bit is part of the configuration for the Timer 1 clock source.

#bit    TMR1CS1 = T1CON.7

Maps TMR1CS1 to bit position 7. Along with TMR1CS0, this sets the clock source for Timer 1.

Variable and Macro Definitions

unsigned int8 RPIC_TO_SPIC_ARRAY[10];

Declares an 8-bit unsigned integer array of size 10. This is used to store data being exchanged between two communication interfaces, RPIC (Reset PIC) and SPIC (Start PIC).

#define ON   0x69
#define OFF  0x96

Defines two macros: ON and OFF, with hexadecimal values 0x69 and 0x96. These represent status codes for controlling hardware.

char POWER_LINE_STATUS;

Declares a char variable to store the status of a power line, representing whether the power is on or off.

UART Configuration and Buffer Variables

#define RP_BFR_SIZE 10

Defines the size of the RP_Buffer as 10 bytes, for storing UART data

#use rs232(UART1, baud=38400, parity=N, bits=8, stream=RPIC, ERRORS)

Configures a UART communication interface:

  • UART1: Uses UART1 hardware module.
  • baud=38400: Sets the baud rate to 38,400 bits per second.
  • parity=N: No parity bit is used.
  • bits=8: 8 data bits per frame.
  • stream=RPIC: Associates the UART stream with the name RPIC.
  • ERRORS: Enables automatic handling of UART errors.
unsigned int8 RP_Buffer[RP_BFR_SIZE];

Declares an 8-bit unsigned integer array RP_Buffer of size RP_BFR_SIZE (10). This buffer stores incoming UART data.

unsigned int16 RP_Byte_Counter = 0;

Declares a 16-bit unsigned integer RP_Byte_Counter initialized to 0. It counts the number of bytes received or transmitted.

unsigned int8 RP_Overflow = 0;

Declares an 8-bit unsigned integer RP_Overflow initialized to 0. It indicates whether the RP_Buffer has overflowed.

unsigned int16 RP_Read_Byte_counter = 0;

Declares a 16-bit unsigned integer RP_Read_Byte_counter initialized to 0. It tracks the number of bytes read from the RP_Buffer.

unsigned int8 RP_Temp_byte = 0;

Declares an 8-bit unsigned integer RP_Temp_byte initialized to 0. This is a temporary variable for storing a single byte during processing.

Interrupt Service Routine (ISR) for UART

ISR (SERIAL_ISR1()): Handles incoming UART data efficiently, storing it in a buffer.

#INT_RDA
Void SERIAL_ISR1() // MAIN PIC UART interrupt loop
{
   if( RP_Byte_Counter < RP_BFR_SIZE )
   {
      RP_Buffer[RP_Byte_Counter] = fgetc(RPIC);
      RP_Byte_Counter++;
   }
   else RP_Overflow = fgetc(RPIC);
}

INT_RDA: This is the interrupt identifier for UART receive data available. It triggers whenever the UART hardware receives a byte.

SERIAL_ISR1(): The interrupt handler (ISR) for the UART.

Operation:

  • If the buffer (RP_Buffer) has space: The received byte is read from the UART (fgetc(RPIC)) and stored in the buffer at the current position (RP_Byte_Counter). The byte counter (RP_Byte_Counter) increments.
  • If the buffer is full: The received byte is discarded by reading it (fgetc(RPIC)), and the overflow flag (RP_Overflow) is set. This prevents the UART hardware from getting stuck.

Function to Check Available Bytes

RPic_Available()

Checks buffer status.

unsigned int8 RPic_Available()
{
   return RP_Byte_Counter;
}

This function checks if there is data available to read from the buffer and returns the number of bytes currently in the buffer (RP_Byte_Counter).

Function to Read a Byte

RPic_Read()

Safely reads from the buffer, maintaining counters and avoiding overflow.

unsigned int8 RPic_Read()
{
   if (RP_Byte_Counter > 0)
   {    
      RP_Temp_byte = RP_Buffer[RP_Read_Byte_counter];
      RP_Byte_Counter--;
      RP_Read_Byte_counter++;
      if(RP_Byte_Counter == 0) RP_Read_Byte_counter = 0;
      return RP_Temp_byte; 
   }
   if (RP_Byte_Counter == 0)
   { 
      RP_Read_Byte_counter = 0;
      RP_Temp_byte = 0x00;
      return RP_Temp_byte; 
   }
}
  1. Checks if there are bytes available (RP_Byte_Counter > 0):
  • Reads the byte from RP_Buffer at the position of RP_Read_Byte_counter.
  • Decreases the byte counter(RP_Byte_Counter) and increments the read index (RP_Read_Byte_counter).
  • If the buffer is now empty (RP_Byte_Counter == 0), resets the read index (RP_Read_Byte_counter = 0).
  1. If no data is available:
  • Resets the read index and returns 0x00 as a default.

Function to Process Incoming UART Data

CHECK_UART_INCOMING_FROM_RESET_PIC()

Processes incoming messages, searches for a synchronization byte (0xAA), and stores the message in an array. It also sends part of the data to a PC for debugging or monitoring.

void CHECK_UART_INCOMING_FROM_RESET_PIC()
{
   if( Rpic_Available() )
   {
      delay_ms(10);
      for(int i=0; i<8; i++)
      {
         if( RPIC_Read() == 0xAA )
         {
            RPIC_TO_SPIC_ARRAY[0] = 0xAA;
            break;
         }
      }
      for(int i=1; i<10; i++)
      {
         RPIC_TO_SPIC_ARRAY[i] = RPIC_Read();
      }
      for(int i=0; i<3; i++)
      {
         fprintf(PC, "%X ", RPIC_TO_SPIC_ARRAY[i]);
      }
      fprintf(PC, "\n\r");
   }
}

1.Checks if data is available:

  • Calls Rpic_Available() to see if the buffer has any bytes.
  1. Waits briefly:
  • Introduces a 10 ms delay (delay_ms(10)). This might allow more data to arrive if the system is processing a multi-byte message.
  1. Searches for a synchronization byte (0xAA):
  • Loops up to 8 times, reading bytes using RPIC_Read().
  • If it finds 0xAA, it stores it in the first position of the RPIC_TO_SPIC_ARRAY array and exits the loop.
  1. Reads remaining data:
  • Fills the rest of the RPIC_TO_SPIC_ARRAY array with the next 9 bytes from the buffer.
  1. Prints the first three bytes to the PC:
  • Sends the first three bytes from RPIC_TO_SPIC_ARRAY to a PC via another UART (fprintf(PC, "%X ", ...)).
  • Ends the output with a newline and carriage return.

Initial power management

The next code defines three utility functions for managing power lines and clearing a communication array. These functions manage the power lines by controlling GPIO pins. A "low" state turns off the power, while a "high" state turns it on. Delays ensure stable transitions. Clearing the RPIC_TO_SPIC_ARRAY is essential for ensuring the integrity of data in UART-based communication. The POWER_LINE_STATUS variable is used to track whether power lines are on (ON) or off (OFF), which can be helpful for debugging or program logic.

TURN_OFF_ALL_POWER_LINES() This function disables power to all connected components by setting specific control pins to a "low" state (0V).

void TURN_OFF_ALL_POWER_LINES()
{
   output_Low(PIN_D6); // Switch enable for COM PIC
   output_Low(PIN_D7); // Switch enable for Main PIC   
   output_Low(PIN_C2); // OCP switch disable for Unreg #1
   output_Low(PIN_C5); // OCP switch disable for Unreg #2
   
   POWER_LINE_STATUS = OFF;
   fprintf(PC, "Turned OFF Power lines\n\r");
}
  • output_Low(PIN_D6); Sets the voltage of PIN_D6 to low, disabling the power line for the COM PIC.
  • output_Low(PIN_D7); Sets the voltage of PIN_D7 to low, disabling the power line for the Main PIC.
  • output_Low(PIN_C2); Disables the Over-Current Protection (OCP) switch for an unregulated power line labeled "Unreg #1".
  • output_Low(PIN_C5); Disables the OCP switch for "Unreg #2".
  • POWER_LINE_STATUS = OFF; Updates the POWER_LINE_STATUS variable to the predefined constant OFF (0x96).
  • fprintf(PC, "Turned OFF Power lines\n\r"); Sends a message over a serial connection (to a PC or debugging terminal) to confirm the action.

TURN_ON_ALL_POWER_LINES() This function enables power to all connected components by setting specific control pins to a "high" state (e.g., 3.3V or 5V).

void TURN_ON_ALL_POWER_LINES()
{
   output_High(PIN_D6);                                                         
   Delay_ms(50);
   output_High(PIN_D7);                                                         
   Delay_ms(50);
   output_High(PIN_C2);                                                         
   Delay_ms(50);
   output_High(PIN_C5);                                                         
   Delay_ms(50);
   
   POWER_LINE_STATUS = ON;
   fprintf(PC, "Turned ON Power lines\n\r");
}
  • output_High(PIN_D6); Sets the voltage of PIN_D6 to high, enabling the power line for the COM PIC.
  • Delay_ms(50); Waits for 50 milliseconds to stabilize the power line.
  • output_High(PIN_D7); Enables the power line for the Main PIC. Another 50 ms delay is added for stability.
  • output_High(PIN_C2); and output_High(PIN_C5); Enables the OCP switches for the unregulated power lines "Unreg #1" and "Unreg #2," with 50 ms delays in between.
  • POWER_LINE_STATUS = ON; Updates the POWER_LINE_STATUS variable to the predefined constant ON (likely 0x69).
  • fprintf(PC, "Turned ON Power lines\n\r"); Sends a confirmation message to a PC or debugging terminal.

CLEAR_RPIC_TO_SPIC_ARRAY() This function resets the RPIC_TO_SPIC_ARRAY to all zeros. This is used to clear stale or invalid data from the communication array.

void CLEAR_RPIC_TO_SPIC_ARRAY()
{
   for( int i = 0; i<10; i++ ) RPIC_TO_SPIC_ARRAY[i] = 0;
}
  • for( int i = 0; i<10; i++ ): Iterates through the 10 elements of the RPIC_TO_SPIC_ARRAY.
  • RPIC_TO_SPIC_ARRAY[i] = 0: Sets each element of the array to 0, effectively clearing it.

STARTPIC.C

This is the main code for the Start PIC MCU that manages power lines and checks for responses from another microcontroller (Reset PIC).

Headers and Configuration

#include <16F1789.h>

Includes the device-specific header file for the PIC16F1789 microcontroller. This provides access to hardware-specific definitions like registers, pins, and peripherals.

#fuses NOWDT, MCLR, NOBROWNOUT
  • NOWDT: Disables the Watchdog Timer (WDT).
  • MCLR: Enables the Master Clear Reset pin for manual reset.
  • NOBROWNOUT: Disables the Brown-Out Reset feature, which would reset the MCU if the supply voltage drops below a threshold.
#use delay(Crystal = 16MHz, clock = 16MHz)

Specifies that the system uses a 16 MHz crystal oscillator.

#use rs232(baud=38400, parity=N, xmit=PIN_D3, bits=8, stream=PC, force_sw)

Configures a UART communication stream on PIN_D3 for transmitting data at 38,400 baud. The force_sw option enforces software-based serial communication.

Global Variables

unsigned int16 RPIC_RESPOND_SEC_COUNTER = 0;
unsigned int16 MLC = 0;

unsigned int16 SEC_COUNT = 0;
unsigned int8 ONEHOUR_FLAG = 0x00;
  • RPIC_RESPOND_SEC_COUNTER: Tracks how long the Reset PIC takes to respond.
  • MLC: A secondary counter used in the main loop.
  • SEC_COUNT: A timer counter incrementing every 100 ms to track one-hour periods.
  • ONEHOUR_FLAG: A flag indicating whether one hour has elapsed. 0xAA means one hour has passed.
char BB, CC;

Temporary variables to read inputs from pins.

Main Function

The main function manages the system by checking for responses from the Reset PIC, turning power lines ON/OFF based on conditions and handling a 1-hour timer and ensuring the system doesn't stall.

Key Tasks:

  • Use UART communication to interact with another microcontroller.
  • Track time for system stability and response.
  • Control power lines for external components.
  • Manage external Watchdog Timer toggling.

Initialization

void main()
{
   Delay_ms(500);
   fprintf(PC, "Start PIC booting...........\n\r");
  • Delays startup by 500 ms to allow system stabilization.
  • Sends a message over the UART (serial) indicating that the Start PIC is booting.
enable_interrupts(INT_RDA);
enable_interrupts(GLOBAL);
  • Enables the Receive Data Available (RDA) interrupt for UART communication.
  • Enables global interrupts for the system.
   TURN_ON_ALL_POWER_LINES();

Powers on all connected components by calling the function TURN_ON_ALL_POWER_LINES.

   CC = input(PIN_B6);
   BB = input(PIN_B7);

Reads the digital state of pins PIN_B6 and PIN_B7 into variables CC and BB.

Let us delve further into the main loop itself to see how it works. Main Loop

   while(TRUE)
   {
      CHECK_UART_INCOMING_FROM_RESET_PIC();
  • Infinite Loop: Continuously executes system tasks.
  • CHECK_UART_INCOMING_FROM_RESET_PIC: Checks for data from the Reset PIC and processes it.
      if( ONEHOUR_FLAG == 0xAA )
      {
         if( (RPIC_TO_SPIC_ARRAY[0] == 0xAA) && (RPIC_TO_SPIC_ARRAY[1] == 0xBB) && (RPIC_TO_SPIC_ARRAY[2] == 0xCC) )
         {
            RPIC_RESPOND_SEC_COUNTER = 0;
            MLC = 0;
            if( POWER_LINE_STATUS == ON ) TURN_OFF_ALL_POWER_LINES();
            CLEAR_RPIC_TO_SPIC_ARRAY();
         }
  • Checks if one hour has passed (ONEHOUR_FLAG == 0xAA).
  • If the Reset PIC has sent a valid response (RPIC_TO_SPIC_ARRAY[0..2] match expected values), it:
    • Resets the response counter.
    • Turns off power lines if they are ON.
    • Clears the response buffer.
         if( MLC >= 50 )
         {
            MLC = 0;
            RPIC_RESPOND_SEC_COUNTER++;
         }

If MLC reaches 50 iterations, increment the response counter.

         if( RPIC_RESPOND_SEC_COUNTER >= 36 )
         {
            fprintf(PC, "Reset pic did not respond for three minute\n\r");
            RPIC_RESPOND_SEC_COUNTER = 0;
            TURN_ON_ALL_POWER_LINES();
         }

If no valid response from the Reset PIC after 3 minutes (36 counts), log a message and turn the power lines back on.

Hourly Timer

      if( ONEHOUR_FLAG == 0x00 )
      {
         SEC_COUNT++;
         if(SEC_COUNT >= 36000)     // one hour 36000
         {
            ONEHOUR_FLAG = 0xAA;
         }
      }
  • If one hour hasn't passed (ONEHOUR_FLAG == 0x00), increment SEC_COUNT every 100 ms.
  • Once SEC_COUNT reaches 36,000 (1 hour), set ONEHOUR_FLAG to 0xAA.

Watchdog Toggle

      delay_ms(100);
      Output_Toggle(PIN_B2);
  • Delays 100 ms in each iteration.
  • Toggles the state of PIN_B2 to reset an external Watchdog Timer.

Here's the flow diagram illustrating the operation of the microcontroller system based on the code. Each step is represented to show the system's initialization, main loop, and decision-making processes.

Here is a flow diagram focused on the main loop

Here's the flow diagram illustrating the operation of the microcontroller system based on the code. Each step is represented to show the system's initialization, main loop, and decision-making processes.

          START
           |
           V
          +-----------------------+
          | Initialize System     |
          | - Delay for boot time |
          | - Enable interrupts   |
          | - Turn on power lines |
          | - Set inputs for CC,  |
          |   BB                  |
          +-----------------------+
           |
           | < -----------------------------------------------------------------------------------------+
           V                                                                                            |
          +-----------------------+                                                                     |
          | Check UART Incoming   |                                                                     |
          | - Look for 0xAA       |                                                                     |
          | - Store in RPIC array |                                                                     |
          | - Print first 3 bytes |                                                                     |
          +-----------------------+                                                                     |
           |                                                                                            |
           +--------------------------+                                                                 |
+->+---->  | Is ONEHOUR_FLAG == 0xAA? |-----> NO -----> Increment SEC_COUNT                             |
|  |       +--------------------------+                        |                                        |
|  |                   |                                       V                            +----------------------------+
|  |                 YES                                Is SEC_COUNT >= 36000? ---- NO -->  | Toggle Watchdog Timer (PIN)|
|  |                 |                                        |                             +----------------------------+
|  |                 V                                        |
|  |      +-----------------------------+                     |
|  |      | Process Hourly Tasks        |                     V
|  |      | - Check RPIC array for      |<-------------------YES
|  |      |   0xAA, 0xBB, 0xCC          |
|  |      | - Reset counters if valid   |
|  |      | - Turn off power if ON      |
|  |      +-----------------------------+
|  |                 |
|  |           Response VALID
|  |                 |
|  |                 |
|  |            Reset counters
|  |      Turn OFF power lines if ON
|  |                 |
|  |                 |
|  |                 V
|  |       +-------------------------------+
|  |       | Is Main Loop Counter >= 50?   |
|  |       +-------------------------------+     
|  |                 |
|  |                YES
|  |                 |
|  |                 V
|  |      +-----------------------------------+
|  |      |           Reset MLC               |
|  |      | Increment RPIC Response Counter   |
|  |      +-----------------------------------+
|  |                 |
|  |                 |
|  |                 V
|  |      +-------------------------------+
|  NO ----| Is RPIC Response Counter>= 36? | 
|         +-------------------------------+
|                  | 
|                 YES
|                  |
|                  V
|         +-----------------------------+
|         | No RPIC response: Turn ON   |
|         | power lines, reset counter  |
|          +-----------------------------+
|                    |
|                    |
|               Increment MLC
|                    |
|                    |
+---------<----------+

2. RESET PIC

File content

The files found in the RESET PIC folder are detailed in Table 2 below.

File name pattern Scope Content
ADC_Power_Lines_Control.c satellite power management and monitoring; it controls various power lines, monitors voltages and currents, and performs periodic resets.
PIC18F67J94_registers.h
RPIC_CPIC.c
RPIC_MPIC.c
RPIC_STARTPIC.c
RTC_fun.c
ResetMain.c
ResetPIC_Functions.c

ADC_Power_Lines_Control.c

This code manages power lines, resets the system, and monitors current and voltage using analog-to-digital converter (ADC). Each function is modular and updates POWER_LINE_STATUS to track the state of the system. Functions like SYSTEM_RESET and SYSTEM_RESET_24H ensure operational stability through scheduled or manual resets.

Here is a detailed breakdown of the functionality:

Global Definitions and Variables

#define ON  1
#define OFF 0

Defines symbolic constants ON and OFF for readability, representing 1 (ON) and 0 (OFF) states.

#define BB_ON_OCP_ON   1
#define BB_ON_OCP_OFF  2
#define BB_OFF_OCP_OFF 3

Defines states for a buck-boost converter and overcurrent protection (OCP):

  • BB_ON_OCP_ON: Buck-boost ON, OCP ON.
  • BB_ON_OCP_OFF: Buck-boost ON, OCP OFF.
  • BB_OFF_OCP_OFF: Buck-boost OFF, OCP OFF.
unsigned int8 POWER_LINE_STATUS = 0;
unsigned int8 RESET_TIME = 0;
  • POWER_LINE_STATUS: 8-bit variable where each bit indicates the status of different power lines or components.
  • RESET_TIME: Tracks the time when a system reset occurred.

Functions for System Components

MP_CP_BuckBoost(): This function controls the Main PIC <-> Com PIC buck-boost converter.

void MP_CP_BuckBoost(int1 status)
{
    if (status == ON)
    {
      output_LOW(PIN_C4);
    }
    if (status == OFF)
    {
      output_HIGH(PIN_C4);
    }
    RST_EXT_WDT();
}
  • Turns the converter ON/OFF by controlling PIN_C4.
  • Calls RST_EXT_WDT() to reset an external watchdog timer (prevents system reset due to inactivity).

MainPic_Power(): This function manages power for the main microcontroller (Main PIC)

void MainPic_Power(int1 status)
{
    if (status == 1)
    {
       output_high(PIN_F5);
       BIT_SET(POWER_LINE_STATUS,7);
    }
    if (status == 0)
    {
       output_low(PIN_F5);
       BIT_CLEAR(POWER_LINE_STATUS,7);
    }
    delay_ms(50);
    RST_EXT_WDT();
}
  • Activates/deactivates power to Main PIC via PIN_F5.
  • Updates bit 7 of POWER_LINE_STATUS to reflect the Main PIC's state.
  • Adds a delay (debouncing) and resets the watchdog timer.

ComPic_Power(): This function manages power for the communication microcontroller (Com PIC)

void ComPic_Power(int1 status)
{
    if (status == 1)
    {
       output_high(PIN_F6);
       BIT_SET(POWER_LINE_STATUS,6);
    }
    if (status == 0)
    {
       output_low(PIN_F6);
       BIT_CLEAR(POWER_LINE_STATUS,6);
    }
    delay_ms(50);
    RST_EXT_WDT();
}
  • Similar to MainPic_Power, but controls PIN_F6 and updates bit 6 of POWER_LINE_STATUS.
  • Adds a delay (debouncing) and resets the watchdog timer.

_3V3Power_Line1(): This function controls a 3.3V#1 power line with overcurrent protection

void _3V3Power_Line1(int1 status )
{
    if (status == BB_ON_OCP_ON)
    {
       output_high(PIN_D1);
       Delay_ms(50);
       output_high(PIN_D4);
       BIT_SET(POWER_LINE_STATUS,5);
    }

    if (status == BB_ON_OCP_OFF)
    {
       output_high(PIN_D1);
       output_low(PIN_D4);
       BIT_CLEAR(POWER_LINE_STATUS,5);
    }

    if (status == BB_OFF_OCP_OFF)
    {
       output_low(PIN_D1);
       output_low(PIN_D4);
       BIT_CLEAR(POWER_LINE_STATUS,5);
    }
    delay_ms(50);
    RST_EXT_WDT();
}
  • Uses PIN_D1 to control the buck-boost converter and PIN_D4 to control OCP.
  • Updates bit 5 of POWER_LINE_STATUS to reflect the 3.3V#1 power line state.
  • Adds a delay (debouncing) and resets the watchdog timer.

Other Line-Control Functions

  • _3V3Power_Line2

    • Uses PIN_D2 to control the buck-boost converter and PIN_D5 to control OCP.
    • Updates bit 4 of POWER_LINE_STATUS to reflect the 3.3V#2 power line state.
    • Adds a delay (debouncing) and resets the watchdog timer.
  • _5V0Power_Line,

    • Uses PIN_D3 to control the buck-boost converter and PIN_D6 to control OCP.
    • Updates bit 3 of POWER_LINE_STATUS to reflect the 5.0V power line state.
    • Adds a delay (debouncing) and resets the watchdog timer.
  • Unreg1_Line,

    • Uses PIN_D7 to control OCP.
    • Updates bit 2 of POWER_LINE_STATUS to reflect the Unreg1 power line state.
    • Adds a delay (debouncing) and resets the watchdog timer.
  • Unreg2_Line:

    • Uses PIN_D0 to control OCP.
    • Updates bit 1 of POWER_LINE_STATUS to reflect the Unreg2 power line state.
    • Adds a delay (debouncing) and resets the watchdog timer.

System Reset Functions

SYSTEM_RESET(): Performs a manual system reset of the satellite by command from PC or GS

void SYSTEM_RESET()
{  
   int sec_c = 0;
   Fprintf(PC,"system reset by command \n\r");
   // turn off system
   MainPic_Power(OFF)                ;  Delay_ms(250) ; 
   ComPic_Power(OFF);Delay_ms(250)   ;  Delay_ms(250) ; 
   _3V3Power_Line1(BB_OFF_OCP_OFF)   ;  Delay_ms(250) ; 
   _3V3Power_Line2(BB_OFF_OCP_OFF)   ;  Delay_ms(250) ; 
   _5V0Power_Line(BB_OFF_OCP_OFF)    ;  Delay_ms(250) ; 
   Unreg1_Line(OFF);Delay_ms(250)    ;  Delay_ms(250) ; 
   Unreg2_Line(OFF);Delay_ms(250)    ;  Delay_ms(250) ;

   RESET_TIME = hour;

   for (int i = 0; i < 10; i++)
   {
       Delay_ms(500);
       RST_EXT_WDT();
       Delay_ms(500);
       RST_EXT_WDT();
       sec_c++;
       Fprintf(PC, "Waiting to turn on system again %02d Sec\n\r", sec_c);
   }

   // turn on system
   MainPic_Power(ON)                ;    Delay_ms(250) ;      
   ComPic_Power(ON)                 ;    Delay_ms(250) ;      
   _3V3Power_Line1(BB_ON_OCP_ON)    ;    Delay_ms(250) ;     
   _3V3Power_Line2(BB_ON_OCP_ON)    ;    Delay_ms(250) ;       
   _5V0Power_Line(BB_ON_OCP_ON)     ;    Delay_ms(250) ;      
   Unreg1_Line(ON)                  ;    Delay_ms(250) ;     
   Unreg2_Line(OFF)                 ;    Delay_ms(250) ;
}
  • Sequentially turns off all components.
  • Set reset time to 1 hour
  • Waits 10 seconds, resetting the watchdog timer periodically (every half second).
  • Turns all components back on after the delay.

SYSTEM_RESET_24H(): This function automates a daily reset at a specific time. Similar to SYSTEM_RESET(), but triggered when hour, minute, and second match predefined values.

ADC Measurement Functions

Each power rail or unregulated line has a sensor (e.g., a current sensor or voltage divider) connected to the analog input of the microcontroller. The microcontroller reads these analog signals and converts them into digital values using its ADC module.

The ADC readings are stored in the corresponding variables (_Raw_power_ADC_val, _3V3_1_current_ADC_val, etc.) for further processing or monitoring.

These values are used for:

  • Power system diagnostics (e.g., detecting overcurrent conditions).
  • Monitoring power health in real-time.
  • Logging and telemetry

Commented Section

The commented code below describes how certain hardware signals are being monitored and measured using analog-to-digital conversion.

// C3 = Raw power monitor enable
// A2 (AN2) = Raw voltage measure
// A1 (AN1) = 3V3-1 current measure
// A0 (AN0) = 3V3-2 current measure
// A5 (AN4) = 5V0 current measure
// A4 (AN6) = UNREG-1 current measure
// C2 (AN9) = UNREG-2 current measure

Each signal is mapped to a specific pin (or port) on the microcontroller, and the purpose of that pin is described:

  1. C3 = Raw power monitor enable
  • This pin (C3) controls a pin used to enable or disable monitoring of the raw power input. It is toggled as needed to save power and/or isolate specific circuits.
  1. A2 (AN2) = Raw voltage measure
  • Analog channel 2 (AN2) is used to measure the raw voltage level. This represents the voltage coming directly from the power source before regulation.
  1. A1 (AN1) = 3V3-1 current measure
  • Analog channel 1 (AN1) measures the current flowing through the 3.3V#1 power rail (3V3-1).
  1. A0 (AN0) = 3V3-2 current measure
  • Analog channel 0 (AN0) measures the current flowing through the 3.3V#2 power rail (3V3-2).
  1. A5 (AN4) = 5V0 current measure
  • Analog channel 4 (AN4) measures the current on the 5V power line.
  1. A4 (AN6) = UNREG-1 current measure
  • Analog channel 6 (AN6) measures the current for an unregulated power line labeled UNREG-1.
  1. C2 (AN9) = UNREG-2 current measure
  • Analog channel 9 (AN9) measures the current for another unregulated power line labeled UNREG-2.
unsigned int16 _Raw_power_ADC_val       = 0 ;          
unsigned int16 _3V3_1_current_ADC_val   = 0 ;          
unsigned int16 _3V3_2_current_ADC_val   = 0 ;          
unsigned int16 _5V0_current_ADC_val     = 0 ;          
unsigned int16 _UNREG_1_current_ADC_val = 0 ;      
unsigned int16 _UNREG_2_current_ADC_val = 0 ;

Variables

The variables declared are used to store the ADC (Analog-to-Digital Conversion) readings for the respective signals.

  • unsigned int16 _Raw_power_ADC_val = 0; Holds the ADC result for the raw power voltage measurement (from AN2).
  • unsigned int16 _3V3_1_current_ADC_val = 0; Stores the ADC result for the current measurement of 3V3-1 (from AN1).
  • unsigned int16 _3V3_2_current_ADC_val = 0; Stores the ADC result for the current measurement of 3V3-2 (from AN0).
  • unsigned int16 _5V0_current_ADC_val = 0; Holds the ADC result for the current measurement of the 5V power line (from AN4).
  • unsigned int16 _UNREG_1_current_ADC_val = 0; Stores the ADC result for the current measurement of the unregulated power line UNREG-1 (from AN6).
  • unsigned int16 _UNREG_2_current_ADC_val = 0; Stores the ADC result for the current measurement of the unregulated power line UNREG-2 (from AN9).

Current measuring functions

Measure_(): These functions measure voltages and currents for different lines with '' being replaced with;

  • Raw_voltage
  • 3V3_1_current
  • 3V3_2_current
  • 5V0_current
  • UNREG_1_current
  • UNREG_2_current
unsigned int16 Measure_Raw_voltage() // A2 (AN2) = Raw voltage measure
{
    Output_low(PIN_C3);
    Delay_us(100);

    SET_ADC_CHANNEL(2);
    delay_us(20);
    Output_high(PIN_C3);
    return READ_ADC();
}
  • The function returns a 16-bit unsigned integer. This integer represents the digital value of the voltage measured at analog channel 2 (AN2), corresponding to the raw voltage.
  • Sets PIN_C3 to a low logic level (0). In the comments, C3 is described as the "Raw power monitor enable" pin. Setting it low activates the raw voltage measurement circuit (e.g., by powering on a sensor or enabling a voltage divider).
  • Introduces a delay of 100 microseconds. This allows the circuitry connected to PIN_C3 to stabilize after being activated. Stabilization is important for accurate ADC readings, especially in systems with capacitors or sensors.
  • Configures the ADC to read from channel 2 (AN2), which is connected to the raw voltage signal.
  • Introduces a smaller delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels. This ensures the input signal is correctly sampled.
  • Sets PIN_C3 to a high logic level (1). This disables the raw voltage measurement circuit, conserving power or isolating the circuit when not in use.
  • Performs an ADC conversion and returns the digital value representing the measured raw voltage. The READ_ADC() function converts the analog signal at channel 2 into a numerical value, which corresponds to the voltage level based on the system's ADC resolution and reference voltage.

The remaining functions do not toggle pin outputs, they only set the ADC channel, delay, then return the value of the READ_ADC() function.

  • unsigned int16 Measure_3V3_1_current()
    • Configures the ADC to read from channel 1 (AN1), which is connected to the 3v3-1 current measure signal.
    • Introduces a small delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels.
    • Performs an ADC conversion and returns the digital value representing the measured 3v3-1 voltage.
  • unsigned int16 Measure_3V3_2_current()
    • Configures the ADC to read from channel 0 (AN0), which is connected to the 3v3-2 current measure signal.
    • Introduces a small delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels.
    • Performs an ADC conversion and returns the digital value representing the measured 3v3-2 voltage.
  • unsigned int16 Measure_5V0_current()
    • Configures the ADC to read from channel 4 (AN4), which is connected to the 5v0 current measure signal.
    • Introduces a small delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels.
    • Performs an ADC conversion and returns the digital value representing the measured 5v0 voltage.
  • unsigned int16 Measure_UNREG_1_current()
    • Configures the ADC to read from channel 6 (AN6), which is connected to the UNREG-1 current measure signal.
    • Introduces a small delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels.
    • Performs an ADC conversion and returns the digital value representing the measured unreg-1 voltage.
  • unsigned int16 Measure_UNREG_2_current()
    • Configures the ADC to read from channel 9 (AN9), which is connected to the UNREG-2 current measure signal.
    • Introduces a small delay (20 microseconds) to allow the ADC hardware to stabilize after switching channels.
    • Performs an ADC conversion and returns the digital value representing the measured unreg-2 voltage.

PIC18F67J94_registers.h

This block of code is a series of register and bit definitions for working with the Reset PIC microcontroller. It sets up low-level hardware access by mapping the microcontroller's memory-mapped registers and specific bits to easy-to-use variable names. It allows developers to:

  1. Configure pin directions (TRIS registers).
  2. Control output states (LAT registers).
  3. Read or write pin states (PORT registers).
  4. Manage peripherals like RTCs, timers, and interrupts.

Here's a line-by-line explanation:

Byte Definitions for TRIS Registers

A TRIS register, or tri-state register, is a control register in Microchip PIC microcontrollers that configures the direction of data flow through a port's I/O pins.

#byte TRISG = 0xF98
#byte TRISF = 0xF97
#byte TRISE = 0xF96
#byte TRISD = 0xF95
#byte TRISC = 0xF94
#byte TRISB = 0xF93
#byte TRISA = 0xF92
  • TRIS Registers are used to configure the direction of pins (input or output) for each port (A to G).
  • Setting a TRIS bit to 1 makes the corresponding pin an input; setting it to 0 makes it an output.
  • These definitions map the TRIS registers to their respective memory addresses (hex values starting at 0xF92).

Byte Definitions for LAT Registers

#byte LATG = 0xF8F
#byte LATF = 0xF8E
#byte LATE = 0xF8D
#byte LATD = 0xF8C
#byte LATC = 0xF8B
#byte LATB = 0xF8A
#byte LATA = 0xF89
  • LAT Registers (LATCH) control the output state of pins.
  • Writing to a LAT register sets the state of the corresponding port pins (1 = HIGH, 0 = LOW).
  • These definitions link LAT registers to their specific memory locations.

Bit-Level Access to TRIS Registers

#bit TRISE0 = TRISE.0
#bit TRISE1 = TRISE.1
#bit TRISE2 = TRISE.2
#bit TRISC4 = TRISC.4
#bit TRISB3 = TRISB.3
#bit TRISA0 = TRISA.0
#bit TRISA1 = TRISA.1
#bit TRISA2 = TRISA.2
#bit TRISA5 = TRISA.5
  • Individual bits within the TRIS registers are defined for ease of access.
  • Example: TRISE0 refers to bit 0 of the TRISE register (controls direction for pin E0).

Bit-Level Access to LAT Registers

#bit LATA0 = LATA.0
#bit LATA1 = LATA.1
#bit LATA2 = LATA.2
#bit LATA3 = LATA.3
  • Defines specific bits in the LAT registers, corresponding to individual pins on port A.
  • Example: LATA0 controls the state of pin A0.

Byte Definitions for PORT Registers

#byte PORTG = 0xF86
#byte PORTF = 0xF85
#byte PORTE = 0xF84
#byte PORTD = 0xF83
#byte PORTC = 0xF82
#byte PORTB = 0xF81
#byte PORTA = 0xF80

PORT Registers hold the current state of pins on each port. Reading a PORT register retrieves the pin states, and writing to a PORT register directly changes output levels (if configured as outputs).

Bit-Level Access to PORT Registers

#bit RG0 = PORTG.0
#bit RG1 = PORTG.1
#bit RG2 = PORTG.2
#bit RG3 = PORTG.3
#bit RF2 = PORTF.2
#bit RE2 = PORTE.2
#bit RC0 = PORTC.0
#bit RB3 = PORTB.3
#bit RA0 = PORTA.0
  • Bit-level definitions for individual pins on ports.
  • Example: RG0 refers to bit 0 of PORTG, representing the state of pin G0.

Real-Time Clock Control and Alarm

The RTC (Real-Time Clock) section defines control registers and bits for managing the clock and alarms.

#byte RTCCON1 = 0xF5F
#bit RTCPTR0 = RTCCON1.0
#bit RTCPTR1 = RTCCON1.1
#bit RTCOE = RTCCON1.2
#bit HALFSEC = RTCCON1.3
#bit RTCSYNC = RTCCON1.4
#bit RTCWREN = RTCCON1.5
#bit RTCEN = RTCCON1.7
  • RTCCON1 is the control register for the RTC.
    • RTCPTR0 and RTCPTR1 are specific bits within this register, for configuring the pointer to RTC data.
    • RTCOE
    • HALFSEC
    • RTCSYNC
    • RTCWREN
    • RTCEN
#byte RTCCAL = 0xF5E
#byte RTCVALH = 0xF5D
#bit    WAITE0 = RTCVALH.0
#bit    WAITE1 = RTCVALH.1
#bit    WAITM0 = RTCVALH.2
#bit    WAITM1 = RTCVALH.3
#bit    WAITM2 = RTCVALH.4
#bit    WAITM3 = RTCVALH.5
#bit    WAITB0 = RTCVALH.6
#bit    WAITB1 = RTCVALH.7
  • RTCCAL is the ... register for the RTC.
  • RTCVALH is the ... register for the RTC.
    • WAITE0 is a specific bit within the RTCVALH register, for configuring the ...
    • WAITE1
    • WAITM0
    • WAITM1
    • WAITM2
    • WAITM3
    • WAITB0
    • WAITB1
#byte RTCVALL = 0xF5C
#byte ALRMCFG = 0xF5B
#bit    ALRMPTR0 = ALRMCFG.0
#bit    ALRMPTR1 = ALRMCFG.1
#bit    AMASK0 = ALRMCFG.2
#bit    AMASK1 = ALRMCFG.3
#bit    AMASK2 = ALRMCFG.4
#bit    AMASK3 = ALRMCFG.5
#bit    CHIME = ALRMCFG.6
#bit    ALRMEN = ALRMCFG.7
  • RTCVALL is the ... register for the RTC.
  • ALRMCFG is the ... register for the Alarm.
    • ALRMPTR0 is a specific bit within the RTCVALH register, for configuring the ...
    • ALRMPTR1
    • AMASK0
    • AMASK1
    • AMASK2
    • AMASK3
    • CHIME
    • ALRMEN
#byte ALRMRPT = 0xF5A
#byte ALRMVALH = 0xF59
#byte ALRMVALL = 0xF58
#byte RTCCON2 = 0xF57
#bit    RTCSECSEL0 = RTCCON2.0
#bit    RTCSECSEL1 = RTCCON2.1
#bit    RTCCLKSEL0 = RTCCON2.2
#bit    RTCCLKSEL1 = RTCCON2.3
#bit    PWCSPRE = RTCCON2.4
#bit    PWCCPRE = RTCCON2.5
#bit    PWCPOL = RTCCON2.6
#bit    PWCEN = RTCCON2.7
  • ALRMRPT is the ... register for the RTC.
  • ALRMVALH is the ... register for the RTC.
  • ALRMVALL is the ... register for the RTC.
  • RTCCON2 is the ... register for the Alarm.
    • RTCSECSEL0 is a specific bit within the RTCCON2 register, for configuring the ...
    • RTCSECSEL1
    • RTCCLKSEL0
    • RTCCLKSEL1
    • PWCSPRE
    • PWCCPRE
    • PWCPOL
    • PWCEN

Other Control Registers

#byte EECON2 = 0xF7E
#byte OSCCON = 0xFD3
#byte T1CON = 0xFCD
#bit  T1CON7 = T1CON.7
#bit  T1CON6 = T1CON.6
#bit  SOSCEN1 = T1CON.3
#bit  TMR1ON = T1CON.0
#byte T3CON = 0xFB1
#bit  SOSCEN3 = T3CON.3
  • These are control registers for various peripherals:
    • EECON2:
    • OSCCON: Oscillator control.
    • T1CON: Timer 1 control.
      • T1CON7 refers to bit 7 of T1CON, enabling ...
      • T1CON6 refers to bit 6 of T1CON, enabling ...
      • SOSCEN1 refers to bit 3 of T1CON, enabling ...
      • TMR1ON refers to bit 0 of T1CON, enabling ...
    • T3CON: Timer 3 control.
      • SOSCEN3 refers to bit 3 of T3CON, enabling ...

Interrupt and Miscellaneous Registers

#byte PIE3 = 0xFA3
#bit RC2IE = PIE3.5
  • PIE3 is an interrupt enable register for specific peripherals.
    • RC2IE refers to bit 5 of PIE3, enabling an interrupt for the RC2 peripheral.

RPIC_CPIC.c

CHECK_UART_INCOMING_FROM_COM_PIC()

here we check main pic UART is available or not if available we save in MPIC_TO_RPIC array

void CHECK_UART_INCOMING_FROM_COM_PIC()
{
   if( CPic_Available() )
   {
      Delay_ms(100);
      for( int i = 0; i<10; i++ )
      {
         if( CPic_Read() == 0xC0 )
         {
            CPIC_TO_RPIC_ARRAY[0] = 0xC0 ;
            break;
         }
      }

      for(int i = 1; i<=30; i++)
      {
         CPIC_TO_RPIC_ARRAY[i] = CPic_Read();
      }
   }
}

PRINT_RECIVED_COMMAND_FROM_COM_PIC()

this function will print recived command from com pic

void PRINT_RECIVED_COMMAND_FROM_COM_PIC()
{
   printline();
   Fprintf(PC,"RCVD CMD FROM COM PIC >> ");
   for(int i = 0; i<20; i++)
   {
      Fprintf(PC,"%X ",CPIC_TO_RPIC_ARRAY[i]);
   }
   printline();
}

MONITOR_COM_PIC_90SEC_COMUNICATION(time)

this functions monitor the com pic. if no response within 10min reset pic will restart the com pic

void MONITOR_COM_PIC_90SEC_COMUNICATION(int time)
{
   if( CPIC_TIME_COUNTER >= 600 )
   {
      CPIC_TIME_COUNTER = 0;
      NUMOF_CPIC_RST++;
      
      ComPic_Power(0);   // turn off main pic
      for( int i = 0; i<time ; i++)
      {
         Delay_ms(1000);
        Fprintf(PC,"Waiting to turn on com Pic %02d Sec\n\r",i);
      }
      ComPic_Power(1);   // turn on main pic
   }
}

RESEPOND_TO_CPIC_90SEC_CMD()

void RESEPOND_TO_CPIC_90SEC_CMD()
{
   if(CPIC_TO_RPIC_ARRAY[1] == 0xE0)
   {
      Fprintf(PC,"90 seconds comunication command\n\r");
      
      CPIC_TIME_COUNTER = 0;             // because comunication happaned corretly
      CLEAR_DATA_ARRAY( RPIC_TO_CPIC_ARRAY, 20);
      RPIC_TO_CPIC_ARRAY[0]  = 0xC0   ;
      RPIC_TO_CPIC_ARRAY[1]  = 0xE0   ; 
      
      RPIC_TO_CPIC_ARRAY[2]  = year   ;
      RPIC_TO_CPIC_ARRAY[3]  = month  ;
      RPIC_TO_CPIC_ARRAY[4]  = day    ;
      RPIC_TO_CPIC_ARRAY[5]  = hour   ;
      RPIC_TO_CPIC_ARRAY[6]  = minute ;
      RPIC_TO_CPIC_ARRAY[7]  = second ;
      
      RPIC_TO_CPIC_ARRAY[19] = 0xC1   ;
      for(int i = 0; i<20; i++)
      {
         fputc(RPIC_TO_CPIC_ARRAY[i],CPic);
      }
      printline();
      //_______________________________________________________________________
      
   } 
}

UPDATE_RTC_BY_COM_PIC_CMD()

Void UPDATE_RTC_BY_COM_PIC_CMD()
{
   if(CPIC_TO_RPIC_ARRAY[1] == 0xEA) //MP HF
   {
      Fprintf(PC,"Reset pic RTC update command Received from COM PIC\n\r");
      
      // Acknowleging to the comand____________________________________________
      Delay_ms(10);
      CLEAR_DATA_ARRAY( RPIC_TO_CPIC_ARRAY, 20);
      RPIC_TO_CPIC_ARRAY[0]  = 0xC0   ;
      RPIC_TO_CPIC_ARRAY[1]  = 0xEA   ;
      RPIC_TO_CPIC_ARRAY[19] = 0xC1   ;
      for(int i = 0; i<20; i++)
      {
         fputc(RPIC_TO_CPIC_ARRAY[i],CPic);
      }
      //_______________________________________________________________________
      
      //Updating the RTC
      year   = CPIC_TO_RPIC_ARRAY[2]  ;
      month  = CPIC_TO_RPIC_ARRAY[3]  ;
      day    = CPIC_TO_RPIC_ARRAY[4]  ;
      hour   = CPIC_TO_RPIC_ARRAY[5]  ;
      minute = CPIC_TO_RPIC_ARRAY[6]  ;
      second = CPIC_TO_RPIC_ARRAY[7]  ;
     

      //Printing New RTC value
      Fprintf(PC,"Updated New Time >> ") ;
      Fprintf(PC,"%u-", year)        ;
      Fprintf(PC,"%u-", month)       ;
      Fprintf(PC,"%u__", day)        ;
      Fprintf(PC,"%u:", hour)        ;
      Fprintf(PC,"%u:", minute)      ;
      Fprintf(PC,"%u\n\r", second)   ;
       
   }
}

SENDING_TIME_TO_COMPIC()

void SENDING_TIME_TO_COMPIC()
{
   if(CPIC_TO_RPIC_ARRAY[1] == 0xEB)
   {
      Fprintf(PC,"SENDING_TIME_TO_COMPIC\n\r");
      
      CPIC_TIME_COUNTER = 0;             // because comunication happaned corretly
      CLEAR_DATA_ARRAY( RPIC_TO_CPIC_ARRAY, 20);
      RPIC_TO_CPIC_ARRAY[0]  = 0xC0   ;
      RPIC_TO_CPIC_ARRAY[1]  = 0xEB   ; 
      
      RPIC_TO_CPIC_ARRAY[2]  = year   ;
      RPIC_TO_CPIC_ARRAY[3]  = month  ;
      RPIC_TO_CPIC_ARRAY[4]  = day    ;
      RPIC_TO_CPIC_ARRAY[5]  = hour   ;
      RPIC_TO_CPIC_ARRAY[6]  = minute ;
      RPIC_TO_CPIC_ARRAY[7]  = second ;
      
      RPIC_TO_CPIC_ARRAY[19] = 0xC1   ;
      for(int i = 0; i<20; i++)
      {
         fputc(RPIC_TO_CPIC_ARRAY[i],CPic);
      }
      printline();
      //_______________________________________________________________________
      
   } 
}

RPIC_MPIC.c

This code is designed for robust communication and control between the main PIC and the reset PIC (RPIC), with built-in redundancy and debugging features. Each function focuses on a specific task, ensuring modularity and clarity.

CHECK_UART_INCOMING_FROM_MAIN_PIC

This function checks if data is available on the UART interface from the main PIC and saves it into an array.

void CHECK_UART_INCOMING_FROM_MAIN_PIC()
{
   if( MPic_Available() )
   {
      Delay_ms(100);
      for( int i = 0; i<10; i++ )
      {
         if( MPic_Read() == 0xA0 )
         {
            MPIC_TO_RPIC_ARRAY[0] = 0xA0 ;
            break;
         }
      }

      for(int i = 1; i<=30; i++)
      {
         MPIC_TO_RPIC_ARRAY[i] = MPic_Read();
         fprintf(PC, "%X ", MPIC_TO_RPIC_ARRAY[i]);
      }
      printline();
   }
}

if (MPic_Available()): Checks if the UART data is available from the main PIC. Delay_ms(100);: Waits for 100 milliseconds to ensure data is fully received. First for loop: Reads up to 10 bytes and looks for a specific header value (0xA0). if (MPic_Read() == 0xA0): Verifies the header byte; if found, saves it in the array at index 0 and exits the loop. Second for loop: Reads the next 30 bytes from the UART and stores them in MPIC_TO_RPIC_ARRAY. fprintf(PC, "%X ", MPIC_TO_RPIC_ARRAY[i]);: Logs each received byte to a PC for debugging. printline();: Prints a line break (likely for formatting the debug output).

PRINT_RECIVED_COMMAND_FROM_MAIN_PIC

This function prints the command received from the main PIC to the PC.

void PRINT_RECIVED_COMMAND_FROM_MAIN_PIC()
{
   printline();
   Fprintf(PC,"RCVD CMD FROM MAIN PIC >> ");
   for(int i = 0; i<10; i++)
   {
      Fprintf(PC,"%X ",MPIC_TO_RPIC_ARRAY[i]);
   }
   printline();
}

Fprintf(PC, "RCVD CMD FROM MAIN PIC >> ");: Prints a header message to the PC. for loop: Iterates over the first 10 bytes of MPIC_TO_RPIC_ARRAY and prints them in hexadecimal format.

MONITOR_MAIN_PIC_90SEC_COMUNICATION

Monitors communication with the main PIC. If there’s no response for 10 minutes, it resets the main PIC.

void MONITOR_MAIN_PIC_90SEC_COMUNICATION(int time)
{
   if( MPIC_TIME_COUNTER >= 600 )
   {
      MPIC_TIME_COUNTER = 0;
      NUMOF_MPIC_RST++;
      
      Fprintf(PC,"Hang up reset, MainPIC is turned off\n\r");
      MainPic_Power(0);   // turn off main pic
      for( int i = 0; i<time; i++)
      {
         Delay_ms(1000);
         Fprintf(PC,"Waiting to turn on Main Pic %02d Sec\n\r",i);
      }
      MainPic_Power(1);   // turn on main pic
      Fprintf(PC,"MainPIC is restarted\n\r");
   }
}

if (MPIC_TIME_COUNTER >= 600): Checks if the counter exceeds 10 minutes (assuming each increment is 1 second). Reset Procedure: MPIC_TIME_COUNTER = 0;: Resets the timer. NUMOF_MPIC_RST++;: Increments the reset counter. Logs reset activity to the PC: Prints messages indicating the PIC is being restarted. MainPic_Power(0);: Turns off the main PIC. Delay_ms(1000);: Waits for a specified duration (in seconds). MainPic_Power(1);: Turns the main PIC back on.

RESEPOND_TO_MPIC_90SEC_CMD

Responds to a 90-second communication command from the main PIC by sending housekeeping data.

void RESEPOND_TO_MPIC_90SEC_CMD()
{
   if(MPIC_TO_RPIC_ARRAY[1] == 0x7A)
   {
      Fprintf(PC,"90 seconds comunication command\n\r");
      
      // we make main pic counter zero because succesful comunication
      MPIC_TIME_COUNTER = 0;
      
      // clear RPIC_TO_MPIC array before putting new data______________________
      CLEAR_DATA_ARRAY( RPIC_TO_MPIC_ARRAY, 32 );
      
      // reading ADC values____________________________________________________
      _Raw_power_ADC_val       = Measure_Raw_voltage() ;          
      _3V3_1_current_ADC_val   = Measure_3V3_1_current() ;          
      _3V3_2_current_ADC_val   = Measure_3V3_2_current() ;
      _5V0_current_ADC_val     = Measure_5V0_current() ;          
      _UNREG_1_current_ADC_val = Measure_UNREG_1_current() ;      
      _UNREG_2_current_ADC_val = Measure_UNREG_2_current() ; 
      
      RPIC_TO_MPIC_ARRAY[0]    = 0xA0   ;     // header
      RPIC_TO_MPIC_ARRAY[1]    = 0x7A   ;
      RPIC_TO_MPIC_ARRAY[2]    = year   ;
      RPIC_TO_MPIC_ARRAY[3]    = month  ;
      RPIC_TO_MPIC_ARRAY[4]    = day    ;
      RPIC_TO_MPIC_ARRAY[5]    = hour   ;
      RPIC_TO_MPIC_ARRAY[6]    = minute ;
      RPIC_TO_MPIC_ARRAY[7]    = second ;     // sending reset pic RTC time to main pic
      
      RPIC_TO_MPIC_ARRAY[8]    = (unsigned int8)((_Raw_power_ADC_val>>8)     & 0xFF)     ;   
      RPIC_TO_MPIC_ARRAY[9]    = (unsigned int8)((_Raw_power_ADC_val)        & 0xFF)     ;   
      RPIC_TO_MPIC_ARRAY[10]   = (unsigned int8)((_3V3_1_current_ADC_val>>8) & 0xFF)     ;   
      RPIC_TO_MPIC_ARRAY[11]   = (unsigned int8)((_3V3_1_current_ADC_val)    & 0xFF)     ; 
      RPIC_TO_MPIC_ARRAY[12]   = (unsigned int8)((_3V3_2_current_ADC_val>>8) & 0xFF)     ;   
      RPIC_TO_MPIC_ARRAY[13]   = (unsigned int8)((_3V3_2_current_ADC_val)    & 0xFF)     ; 
      RPIC_TO_MPIC_ARRAY[14]   = (unsigned int8)((_5V0_current_ADC_val>>8)   & 0xFF)     ;   
      RPIC_TO_MPIC_ARRAY[15]   = (unsigned int8)((_5V0_current_ADC_val)      & 0xFF)     ;
      RPIC_TO_MPIC_ARRAY[16]   = (unsigned int8)((_UNREG_1_current_ADC_val>>8) & 0xFF)   ;   
      RPIC_TO_MPIC_ARRAY[17]   = (unsigned int8)((_UNREG_1_current_ADC_val)    & 0xFF)   ;
      RPIC_TO_MPIC_ARRAY[18]   = (unsigned int8)((_UNREG_2_current_ADC_val>>8) & 0xFF)   ;   
      RPIC_TO_MPIC_ARRAY[19]   = (unsigned int8)((_UNREG_2_current_ADC_val) )            ;
      
      RPIC_TO_MPIC_ARRAY[20]   = NUMOF_MPIC_RST ;    // number of main pic reset
      RPIC_TO_MPIC_ARRAY[21]   = NUMOF_CPIC_RST ;    // number of com pic reset
      RPIC_TO_MPIC_ARRAY[22]   = POWER_LINE_STATUS ; // power line status
      RPIC_TO_MPIC_ARRAY[23]   = (unsigned int8)((LAST_RESET_HOUR>>8) & 0xFF) ;
      RPIC_TO_MPIC_ARRAY[24]   = (unsigned int8)((LAST_RESET_HOUR)    & 0xFF) ;
      RPIC_TO_MPIC_ARRAY[25]   = 0xAA   ;  
      
      RPIC_TO_MPIC_ARRAY[31]  = 0xA1   ;  // footer  
      // sending data to main pic______________________________________________
      for(int i = 0; i<32; i++)
      {
         fputc( RPIC_TO_MPIC_ARRAY[i] , MPic);
      }
      
      // just printing sent reply to main pic__________________________________
      Fprintf(PC,"Reply sent to main pic >> ");
      for(int i = 0; i<32; i++)
      {
         Fprintf(PC,"%X ",RPIC_TO_MPIC_ARRAY[i]);
      }
      printline();
      printline();
   }
}

if (MPIC_TO_RPIC_ARRAY[1] == 0x7A): Checks if the command byte is 0x7A (90-second communication command). Reset Timer: Sets MPIC_TIME_COUNTER to 0 to indicate successful communication. Clear Array: Calls CLEAR_DATA_ARRAY to clear RPIC_TO_MPIC_ARRAY. Read ADC Values: Calls functions like Measure_Raw_voltage() to read sensor data (e.g., voltages, currents). Saves these values into the array as two bytes (high and low). Time and Metadata: Adds RTC values (year, month, etc.) and reset counters to the array. Send Data: Sends the array to the main PIC using fputc. Logs the data to the PC for debugging.

UPDATE_RTC_BY_MAIN_PIC_CMD

Updates the RTC (Real-Time Clock) based on a command from the main PIC.

Void UPDATE_RTC_BY_MAIN_PIC_CMD()
{
   if(MPIC_TO_RPIC_ARRAY[1] == 0x7B) //MP HF
   {
      Fprintf(PC,"Reset pic RTC update command Received\n\r");
      
      // Acknowleging to the comand____________________________________________
      CLEAR_DATA_ARRAY( RPIC_TO_MPIC_ARRAY, 32);
      RPIC_TO_MPIC_ARRAY[0]  = 0xA0   ;
      RPIC_TO_MPIC_ARRAY[1]  = 0x7B   ;
      RPIC_TO_MPIC_ARRAY[31] = 0xA1   ;
      for(int i = 0; i<32; i++)
      {
         fputc(RPIC_TO_MPIC_ARRAY[i],MPic);
      }
      //_______________________________________________________________________
      
      //Updating the RTC
      year   = MPIC_TO_RPIC_ARRAY[2]  ;
      month  = MPIC_TO_RPIC_ARRAY[3]  ;
      day    = MPIC_TO_RPIC_ARRAY[4]  ;
      hour   = MPIC_TO_RPIC_ARRAY[5]  ;
      minute = MPIC_TO_RPIC_ARRAY[6]  ;
      second = MPIC_TO_RPIC_ARRAY[7]  ;
     

      //Printing New RTC value
      Fprintf(PC,"Updated New Time >> ") ;
      Fprintf(PC,"%u-", year)        ;
      Fprintf(PC,"%u-", month)       ;
      Fprintf(PC,"%u__", day)        ;
      Fprintf(PC,"%u:", hour)        ;
      Fprintf(PC,"%u:", minute)      ;
      Fprintf(PC,"%u\n\r", second)   ;
       
   }
}

if (MPIC_TO_RPIC_ARRAY[1] == 0x7B): Checks for the RTC update command (0x7B). Acknowledgment: Clears RPIC_TO_MPIC_ARRAY. Sends an acknowledgment array with 0xA0 (header), 0x7B (command), and 0xA1 (footer). Update RTC: Updates the RTC variables (year, month, etc.) with the data from MPIC_TO_RPIC_ARRAY. Log Update: Prints the updated RTC time to the PC.

POWER_LINE_CONTROL_USING_MAIN_PIC_CMD

Controls power lines based on commands from the main PIC.

Void POWER_LINE_CONTROL_USING_MAIN_PIC_CMD()
{
   if(MPIC_TO_RPIC_ARRAY[1] == 0x7C) 
   {
      Fprintf(PC,"Power line control command Received from main pic\n\r");
      // Acknowleging to the comand____________________________________________
      CLEAR_DATA_ARRAY( RPIC_TO_MPIC_ARRAY, 32);
      RPIC_TO_MPIC_ARRAY[0]  = 0xA0   ;
      RPIC_TO_MPIC_ARRAY[1]  = 0x7C   ;
      RPIC_TO_MPIC_ARRAY[31] = 0xA1   ;
      for(int i = 0; i<32; i++)
      {
         fputc(RPIC_TO_MPIC_ARRAY[i],MPic);
      }
      //_______________________________________________________________________
      
      if(MPIC_TO_RPIC_ARRAY[2] == 0x01) _3V3Power_Line1(BB_ON_OCP_ON)  ;
      else _3V3Power_Line1(BB_OFF_OCP_OFF)  ;
      
      if(MPIC_TO_RPIC_ARRAY[3] == 0x01) _3V3Power_Line2(BB_ON_OCP_ON)  ;
      else _3V3Power_Line2(BB_OFF_OCP_OFF)   ;
      
      if(MPIC_TO_RPIC_ARRAY[4] == 0x01) _5V0Power_Line(BB_ON_OCP_ON)   ;
      else _5V0Power_Line(BB_OFF_OCP_OFF)  ;
      
      if(MPIC_TO_RPIC_ARRAY[5] == 0x01) Unreg2_Line(ON)         ;
      else Unreg2_Line(OFF)  ;
      
   }
}

if (MPIC_TO_RPIC_ARRAY[1] == 0x7C): Checks for the power line control command (0x7C). Acknowledgment: Clears RPIC_TO_MPIC_ARRAY and sends an acknowledgment to the main PIC. Control Logic: Reads specific bytes (e.g., MPIC_TO_RPIC_ARRAY[2]) to control power lines like _3V3Power_Line1.

RESET_SATELLITE_CMD

Resets the satellite when a command is received.

Void RESET_SATELLITE_CMD()
{
   if(MPIC_TO_RPIC_ARRAY[1] == 0x7D)
   {  
      Fprintf(PC, "Reset satellite command Received from main PIC");
      SYSTEM_RESET();
   } 
   // Acknowleging to the comand____________________________________________
   CLEAR_DATA_ARRAY( RPIC_TO_MPIC_ARRAY, 32);
   RPIC_TO_MPIC_ARRAY[0]  = 0xA0   ;
   RPIC_TO_MPIC_ARRAY[1]  = 0x7C   ;
   RPIC_TO_MPIC_ARRAY[31] = 0xA1   ;
   for(int i = 0; i<32; i++)
   {
      fputc(RPIC_TO_MPIC_ARRAY[i],MPic);
   }
}

if (MPIC_TO_RPIC_ARRAY[1] == 0x7D): Checks for the reset command (0x7D). SYSTEM_RESET();: Executes a system-wide reset. Acknowledgment: Sends an acknowledgment array back to the main PIC.

TURN_ON_UNREG_2_LINE_FOR_ANT_DEPLOYMENT

Controls the unregulated power line to deploy an antenna.

Void TURN_ON_UNREG_2_LINE_FOR_ANT_DEPLOYMENT()
{
   if(MPIC_TO_RPIC_ARRAY[1] == 0xDA) 
   {
      Fprintf(PC,"Antenna deployment command received\n\r"); 
      // Acknowleging to the comand____________________________________________
      
      CLEAR_DATA_ARRAY( RPIC_TO_MPIC_ARRAY, 32);
      RPIC_TO_MPIC_ARRAY[0]  = 0xA0   ;
      RPIC_TO_MPIC_ARRAY[1]  = 0xDA   ;
      RPIC_TO_MPIC_ARRAY[31] = 0xA1   ;
      for(int i = 0; i<32; i++)
      {
         fputc(RPIC_TO_MPIC_ARRAY[i],MPic);
      }
      //_______________________________________________________________________
      
      int sec_count = 0 ;
      Unreg2_Line(1);
      Fprintf(PC,"UNREG-2 line is turned ON \n\r") ;
            
      for(int i = 0; i<120 ;i++)  // waiting 30 seconds, we can change this
      { 
         RST_EXT_WDT();
         Delay_ms(250); 
         Fprintf(PC,"Counting deployment time %02d Sec \n\r",sec_count++) ;
      }
      
      Unreg2_Line(0);
      Fprintf(PC,"UNREG-2 line is turned OFF \n\r") ;   
   } 
}

if (MPIC_TO_RPIC_ARRAY[1] == 0xDA): Checks for the antenna deployment command (0xDA). Acknowledgment: Clears RPIC_TO_MPIC_ARRAY and sends an acknowledgment. Deployment Procedure: Turns on the unregulated power line (Unreg2_Line(1)). Waits for a specified time (e.g., 30 seconds) while keeping the watchdog timer reset (RST_EXT_WDT()). Turns off the unregulated power line after deployment.

RPIC_STARTPIC.c

This first section is commented out but nonetheless we will look at the use of it.

CHECK_UART_INCOMING_FROM_START_PIC

This function checks whether there is data available from the "START PIC" over UART and reads the data into an array if available.

//!// here we check START pic UART is available or not if available we save in MPIC_TO_RPIC array
//!void CHECK_UART_INCOMING_FROM_START_PIC()
//!{
//!   if( SPic_available() )
//!   {
//!      Delay_ms(30);
//!      for(int i = 0; i<=5; i++)
//!      {
//!         SPIC_TO_RPIC_ARRAY[i] = SPic_Read();
//!      }
//!   }
//!}

SPic_available(): This function likely checks if there is data waiting in the UART receive buffer from the "START PIC." Delay_ms(30): Introduces a 30-millisecond delay. This is likely used to ensure that the incoming data transmission is complete before reading it. Reading loop: The for loop iterates 6 times (indexes 0 to 5) to read bytes from the "START PIC" via the SPic_Read() function and store them in the global array SPIC_TO_RPIC_ARRAY.

PRINT_RECIVED_COMMAND_FROM_START_PIC

This function prints the contents of the received command from the "START PIC" to a PC console.

//!// this function will print recived command from START pic_______________________
//!void PRINT_RECIVED_COMMAND_FROM_START_PIC()
//!{
//!   printline();
//!   Fprintf(PC,"RCVD CMD FROM START PIC >> ");
//!   for(int i = 0; i<=5; i++)
//!   {
//!      Fprintf(PC,"%X ",SPIC_TO_RPIC_ARRAY[i]);
//!   }
//!   printline();
//!}

printline(): Assumes it prints a separator line or a newline for formatting purposes. Fprintf(PC, ...): Outputs formatted data to the PC console via a predefined communication channel PC. Hexadecimal printing: The loop prints each byte from SPIC_TO_RPIC_ARRAY in hexadecimal format (%X).

INFORM_WORKING_TO_START_PIC

This function appears to send a sequence of specific bytes (0xAA, 0xBB, 0xCC) to the "START PIC" microcontroller over UART, likely to signal its working status. After sending the bytes, it sets PIN_C6 to a low state, potentially as an additional signal to indicate the operation's completion.

void INFORM_WORKING_TO_START_PIC()
{
   fputc(0xAA, SPic);
   delay_ms(1);
   fputc(0xBB, SPic);
   delay_ms(1);
   fputc(0xCC, SPic);
   delay_ms(1);
   output_low(PIN_C6);
}
  1. Byte Transmission Over UART: fputc(0xAA, SPic): Sends the byte 0xAA over the UART communication channel SPic. This is repeated for 0xBB and 0xCC, with a 1-millisecond delay between transmissions.
  2. Delay Between Transmissions: delay_ms(1): Introduces a 1-millisecond delay after each byte transmission. This ensures there is enough time for the "START PIC" to process each byte before receiving the next one. Depending on the baud rate and UART settings, this delay may not be necessary and could be optimized.
  3. Pin State Change: output_low(PIN_C6): Drives the microcontroller pin C6 to a low state (grounded). This might be used to: Signal a state change to an external device or the "START PIC." Reset a connected device or trigger a specific action.

RTC_fun.c

The code provided handles functions for a basic real-time clock (RTC) and includes utilities for updating and displaying time, as well as monitoring power line status. Below is an analysis of the functions, their purpose, and suggestions for improvement.

Set_RTC Function()

Initializes or updates the RTC with specific date and time values.

void Set_RTC( char y, char mo, char d, char h, char mi, char s )
{
   year   = y  ;
   month  = mo ;
   day    = d  ;
   hour   = h  ;
   minute = mi ;
   second = s  ;
}

RTC_Function()

Updates the RTC values every second, accounting for leap years, month boundaries, and year rollover.

Void RTC_Function()
{
   if (second < 59)                                                            // updating seconds
   {
      second++;
   }
      
   else
   {
      second = 0;
      minute++;
   }
      
   if (minute >= 60)                                                           // updating minutes
   {
      minute = 0;
      hour++;
      LAST_RESET_HOUR ++ ;
   }
      
   if (hour >= 24)                                                             // updating day
   {
      hour = 0;
      day++;
   }
   
   if( (day >= 31) && (month == 4 || month == 6 || month == 9 || month == 11) )     // 30 days months
   {
      day = 1;
      month++;
   }
   
   if( (day >= 32) && (month == 1 || month == 3 || month == 5 || month == 7 || month == 8 || month == 10) )  // 31 days months
   {
      day = 1;
      month++;
   }
   
   if(year%4 == 0)
   {
      if( (day >= 30) && (month == 2) )                                           // february
      {
         day = 1;
         month++;
      } 
   }
   
   else
   {
      if( (day >= 29) && (month == 2) )                                           // february
      {
         day = 1;
         month++;
      } 
   }
   
   
   if( (day >= 32) && (month == 12) )                                          //december
   {
      day = 1;
      month = 1;
      year++;
   }
      
}

PRINT_POWER_LINE_STATUS()

Prints the status of various power lines by extracting bit information from a POWER_LINE_STATUS variable.

void PRINT_POWER_LINE_STATUS()
{
   fprintf(PC,"M=%d,", (POWER_LINE_STATUS>>7) & 0x01 ); 
   fprintf(PC,"C=%d,", (POWER_LINE_STATUS>>6) & 0x01 );
   fprintf(PC,"3=%d,", (POWER_LINE_STATUS>>5) & 0x01 );
   fprintf(PC,"3=%d,", (POWER_LINE_STATUS>>4) & 0x01 );
   fprintf(PC,"5=%d,", (POWER_LINE_STATUS>>3) & 0x01 );
   fprintf(PC,"U=%d,", (POWER_LINE_STATUS>>2) & 0x01 );
   fprintf(PC,"U=%d" , (POWER_LINE_STATUS>>1) & 0x01 );
   printline();
}

Print_RTC()

Prints the current RTC time and resets, with additional context like power line status and counters.

void Print_RTC()
{
   if(previous_second != second )
   {
      fprintf(PC,"RP ");
      fprintf(PC,"%02d", year);
      fprintf(PC,"/%02d", month);
      fprintf(PC,"/%02d", day);
      fprintf(PC," %02d", hour);
      fprintf(PC,":%02d", minute);
      fprintf(PC,":%02d ", second);                                         // 20-01-01__05:20:22
      
      fprintf(PC,"%d>",NUMOF_MPIC_RST);
      fprintf(PC,"%03Ld", MPIC_TIME_COUNTER);
      fprintf(PC," %d>",NUMOF_CPIC_RST);
      fprintf(PC,"%03Ld ", CPIC_TIME_COUNTER);
      
      PRINT_POWER_LINE_STATUS();
   } 
   previous_second = second ;
}

ResetMain.c

The code is for a reset controller for a satellite with multiple PIC microcontrollers. Its primary role is managing communication, monitoring system health, and executing commands from other microcontrollers (MAIN_PIC and COM_PIC) while maintaining time synchronization using an RTC (Real-Time Clock).

Key Functions

  • Command Parsing
    • Commands are validated by checking predefined markers (e.g., MPIC_TO_RPIC_ARRAY[0] == 0xA0). A set of command-specific functions is called based on the parsed command.
  • Clear Arrays
    • CLEAR_DATA_ARRAY() clears the command arrays after processing to avoid re-executing old commands.
  • Watchdog Reset
    • RST_EXT_WDT() ensures the external watchdog timer is reset, preventing system reset due to inactivity.

Initialization

#include <18F67J94.h>
#FUSES NOWDT,NOBROWNOUT,SOSC_DIG
#device ADC = 12
#use delay(crystal = 16MHz, clock = 16MHz)
#include <PIC18F67J94_REGISTERS.h>

#include <ResetPIC_Functions.c>
#include <ADC_Power_Lines_Control.c>
#include <RPIC_MPIC.c>
#include <RPIC_CPIC.c>
#include <RPIC_STARTPIC.c>
#include <RTC_fun.c>

unsigned int16 MLC = 0;

// we set interupt priority here , timer 1 is the highest priority_____________
#PRIORITY INT_TIMER1,INT_RDA,INT_RDA3           

// this is the timer 1 interupt loop___________________________________________
#INT_TIMER1

TIMER1_ISR()

Void TIMER1_ISR()                                                              
{ 
  set_timer1(0x8000);                                // Timer-1 preload
  RTC_Function();                                    // updating RTC
  
  MPIC_TIME_COUNTER++;                               // this is main pic comunication time counter
  CPIC_TIME_COUNTER++;                               // this is com  pic comunication time counter
}

Timer-1 Interrupt The TIMER1_ISR() function is triggered periodically by Timer-1:

Updates the RTC by calling RTC_Function(). Increments counters for monitoring communication with MAIN_PIC and COM_PIC.

settings()

// here we set the reset pic initial setups____________________________________
void settings()
{
   //output_low(PIN_C6);
   MP_CP_BuckBoost(ON)              ;      // enable MP CP buck boost conveter
   MainPic_Power(ON)                ;      // turn on main pic power
   ComPic_Power(ON)                 ;      // turn on com pic power
   _3V3Power_Line1(BB_ON_OCP_ON)    ;      // both obc and buck boost converters are ON
   _3V3Power_Line2(BB_ON_OCP_ON)    ;      // both obc and buck boost converter are OFF
   _5V0Power_Line(BB_ON_OCP_ON)     ;      // both obc and buck boost converters are OFF BB_ON_OCP_ON
   
   Unreg1_Line(ON)                  ;      // turn on unreg line 1
   Unreg2_Line(ON)                  ;      // turn off unreg line 2

   setup_timer_1(T1_EXTERNAL | T1_DIV_BY_1);         // timer-1 clock source ans prescaler value         
   SOSCEN1 = 1;                                      // enabling timer 1
   set_timer1(0x8000);                               // timer 1 preload
   
   enable_interrupts(INT_RDA2);                      // enabling com pic UART interupt
   enable_interrupts(INT_RDA3);                      // enabling main pic UART interupt
   enable_interrupts(INT_TIMER1);                    // enabling timer - 1 interupt
   enable_interrupts(GLOBAL);                        // start interupt procesing
   
   SETUP_ADC(ADC_CLOCK_INTERNAL);
   SETUP_ADC_PORTS(sAN2|sAN1|sAN0|sAN4|sAN6|sAN9);   // setting all analog ports
   
   Set_RTC(24,01,01, 00,00,06);
   MPic_flush() ;
   fprintf( PC, "Reset pic is booting.......\n\r");
}

The settings() function configures the hardware peripherals and system state:

Power Lines Control:

Enables the power to specific lines (MainPic_Power, ComPic_Power) and sets converters. Activates or deactivates unregulated power lines (Unreg1_Line, Unreg2_Line). Timer-1 Setup:

Configures Timer-1 as an external clock source with a preload value of 0x8000. Timer-1 is used for real-time operations (e.g., updating the RTC). Interrupts:

Enables UART receive interrupts for communication with MAIN_PIC and COM_PIC. Timer-1 interrupt is enabled for periodic tasks. ADC Ports:

Configures specific pins as analog inputs. RTC Setup:

Initializes the RTC with a starting date and time. Sends a boot message to the PC for debugging.

main()

The main() function contains the core operational logic:

void main()
{
   settings();
   while(true)
   {  
      // printing rtc__________________________________________________________
      Print_RTC();  
      
      // this funtion will reset the system if time is 00:00:00________________
      SYSTEM_RESET_24H();

RTC Management:

Prints the current RTC time (Print_RTC()). Resets the system at midnight (SYSTEM_RESET_24H()).

      // check uart incoming from main pic and compic__________________________
      CHECK_UART_INCOMING_FROM_MAIN_PIC()  ;
      CHECK_UART_INCOMING_FROM_COM_PIC()   ; 

UART Command Handling:

Checks for incoming UART data from MAIN_PIC and COM_PIC using dedicated functions. Processes valid command arrays from both sources and performs corresponding actions.

      // monitoring 90 second comunication of compic and mainpic_______________
      MONITOR_MAIN_PIC_90SEC_COMUNICATION(10)  ;   // 10 means how much turn off time before restart 
      MONITOR_COM_PIC_90SEC_COMUNICATION(10)   ;   // 10 means how much turn off time before restart 

90-Second Communication Monitoring:

Monitors communication with MAIN_PIC and COM_PIC. If no communication is detected for 90 seconds, resets are triggered.

      // executing main pic comands____________________________________________
      if( MPIC_TO_RPIC_ARRAY[0] == 0xA0 && MPIC_TO_RPIC_ARRAY[9] == 0xA1 )
      {
         PRINT_RECIVED_COMMAND_FROM_MAIN_PIC();
         Delay_ms(5);   // just delay 5ms before execute
         RST_EXT_WDT();
         
         RESEPOND_TO_MPIC_90SEC_CMD();                // 0x7A
         UPDATE_RTC_BY_MAIN_PIC_CMD();                // 0x7B
         POWER_LINE_CONTROL_USING_MAIN_PIC_CMD();     // 0x7C
         RESET_SATELLITE_CMD();                       // 0x7D
         TURN_ON_UNREG_2_LINE_FOR_ANT_DEPLOYMENT();   // 0xDA
         
         // finaly clear the data array received from main pic
         CLEAR_DATA_ARRAY( MPIC_TO_RPIC_ARRAY, 10 );
      }
      
      // executing com pic comands____________________________________________
      if( CPIC_TO_RPIC_ARRAY[0] == 0xC0 && CPIC_TO_RPIC_ARRAY[19] == 0xC1 )
      {
         PRINT_RECIVED_COMMAND_FROM_COM_PIC();
         Delay_ms(5);   // just delay 5ms before execute
         RST_EXT_WDT();
         
         RESEPOND_TO_CPIC_90SEC_CMD()     ;           // 0xE0
         UPDATE_RTC_BY_COM_PIC_CMD()      ;           // 0xEA
         SENDING_TIME_TO_COMPIC()         ;           // 0xEB
         
         // finaly clear the data array received from main pic
         CLEAR_DATA_ARRAY( CPIC_TO_RPIC_ARRAY, 20 );
      }

Command Execution:

Handles specific commands received from MAIN_PIC and COM_PIC. Examples include updating the RTC, controlling power lines, and resetting the satellite.

      if( (MLC%25) == 0 )  // every half sec
      {        
         Output_Toggle(PIN_F2); 
         if(MLC>=500)
         {
            INFORM_WORKING_TO_START_PIC();
            MLC = 0;
         }
      }

      Delay_ms(10);
      MLC++;
   }
}

Heartbeat Signal:

Toggles PIN_F2 every 0.5 seconds to indicate system activity. Sends a periodic message (INFORM_WORKING_TO_START_PIC()) every 5 seconds.

ResetPIC_Functions.c

Monitoring Variables

Monitoring: Ensures the health of the Main PIC and COM PIC through periodic communication checks. Provides a fail-safe mechanism to restart components if communication fails.

Communication: Buffers allow for efficient, structured data exchange between the Reset PIC, Main PIC, and COM PIC.

RTC Management: Timekeeping is crucial for scheduling operations and triggering time-based resets.

Main PIC (MPIC) Monitoring

unsigned int16 MPIC_TIME_COUNTER = 0;
unsigned int8  NUMOF_MPIC_RST = 0;
unsigned int16 CPIC_TIME_COUNTER = 0;
unsigned int8  NUMOF_CPIC_RST = 0;
unsigned int8 MPIC_TO_RPIC_ARRAY[35] ;   
unsigned int8 RPIC_TO_MPIC_ARRAY[32] ;
unsigned int8 CPIC_TO_RPIC_ARRAY[35] ;   
unsigned int8 RPIC_TO_CPIC_ARRAY[20] ;
//unsigned int8 SPIC_TO_RPIC_ARRAY[10] ;   // only 
//unsigned int8 RPIC_TO_SPIC_ARRAY[10] ;
unsigned int8 year, month, day, hour, minute, second;

MPIC_TIME_COUNTER:

A 16-bit counter incremented every second in the TIMER1_ISR interrupt. Monitors communication with the Main PIC. Reset to 0 if the Main PIC communicates successfully within 90 seconds. If no communication occurs for 10 minutes (600 seconds), the Main PIC is restarted.

NUMOF_MPIC_RST:

An 8-bit counter tracking how many times the Reset PIC has restarted the Main PIC. Useful for diagnostics and debugging.

COM PIC (CPIC) Monitoring

CPIC_TIME_COUNTER:

Similar to MPIC_TIME_COUNTER, but for monitoring the COM PIC. Incremented every second and reset upon successful communication.

NUMOF_CPIC_RST:

Tracks the number of times the Reset PIC has restarted the COM PIC.

UART Communication Buffers

Main PIC Buffers MPIC_TO_RPIC_ARRAY[35]:

A buffer for storing data received from the Main PIC. Only the first 10 bytes are actively used for communication commands.

RPIC_TO_MPIC_ARRAY[32]:

  • A buffer for storing data to be sent to the Main PIC.

COM PIC Buffers CPIC_TO_RPIC_ARRAY[35]:

  • Stores data received from the COM PIC.
  • Only the first 20 bytes are actively used for commands.

RPIC_TO_CPIC_ARRAY[20]:

  • Stores data to be sent to the COM PIC.

Commented SPI Buffers

SPIC_TO_RPIC_ARRAY and RPIC_TO_SPIC_ARRAY:

  • Commented out, possibly reserved for SPI communication with another subsystem or PIC.
  • Could be enabled later if required.

RTC Variables year, month, day, hour, minute, second:

  • Used to store and manage the RTC (Real-Time Clock) values.
  • Likely updated periodically using RTC_Function() in the Timer-1 interrupt.
#use rs232(baud=38400, parity=N, xmit=PIN_B2,  bits=8, stream = PC, errors, force_sw )  // dummy port

baud=38400:

Specifies the communication speed of 38400 bits per second. parity=N:

Indicates no parity checking is used for error detection. xmit=PIN_B2:

Assigns the transmit pin for UART communication to PIN_B2. bits=8:

Data format consists of 8 data bits per frame. stream=PC:

Defines a named stream (PC) for easier reference in the program. errors:

Enables error detection and handling during UART communication. force_sw:

Forces the use of software UART, even if the hardware UART module is available.

RST_EXT_WDT()

void RST_EXT_WDT()
{
   Output_Toggle(PIN_F2);
}

The function ensures the external watchdog timer does not expire and cause an unwanted reset. By toggling the state of PIN_F2, the watchdog timer interprets it as the system being healthy and operational.

UART port connection to main pic definitons

#define MP_BFR_SIZE 30
#pin_select TX3=PIN_E1  
#pin_select RX3=PIN_E0  
#use rs232(UART3, baud=38400, parity=N, bits=8, stream=MPic, errors) 

unsigned int8  MP_Buffer[MP_BFR_SIZE];
unsigned int16 MP_Byte_Counter = 0;
unsigned int8  MP_Overflow = 0;
unsigned int16 MP_Read_Byte_counter = 0;
unsigned int8  MP_Temp_byte = 0;

unsigned int16 LAST_RESET_HOUR = 0;
#INT_RDA3

#define MP_BFR_SIZE 30

Defines the size of the buffer MP_Buffer as 30 bytes. This buffer will temporarily store incoming UART data from UART3.

#pin_select TX3=PIN_E1 and #pin_select RX3=PIN_E0

Assigns the UART3 transmit (TX) pin to PIN_E1 and the receive (RX) pin to PIN_E0. These pin mappings are specific to microcontrollers that allow remappable pins.

#use rs232 Configures UART3 for communication: UART3: Uses the third UART module. baud=38400: Sets the baud rate to 38,400 bits per second. parity=N: No parity bit. bits=8: Uses 8 data bits. stream=MPic: Creates a stream identifier (MPic) for easier UART read/write operations. errors: Enables error checking (e.g., framing errors).

Variables

  • MP_Buffer[MP_BFR_SIZE]: An array that serves as a ring buffer to store incoming data.
  • MP_Byte_Counter: Tracks the number of bytes written into the buffer.
  • MP_Overflow: A flag to indicate if the buffer has overflowed.
  • MP_Read_Byte_counter: Tracks the number of bytes read from the buffer.
  • MP_Temp_byte: Temporarily stores a received byte for processing.
  • LAST_RESET_HOUR: Tracks the hour when the system was last reset.
  • #INT_RDA3 Declares an interrupt service routine (ISR) for the UART3 receive interrupt. When a byte is received via UART3, this interrupt is triggered, and the ISR handles the byte.

SERIAL_ISR3()

The code processes incoming bytes, stores them in a buffer, and manages the overflow condition.

Void SERIAL_ISR3()         // MAIN PIC uart interupt loop
{
   if( kbhit(MPic) )
   {
      if( MP_Byte_Counter < MP_BFR_SIZE )
      {
         MP_Buffer[MP_Byte_Counter] = fgetc(MPic);
         MP_Byte_Counter++;
      }
      else MP_Overflow = fgetc(MPic);
   }
}

Use of kbhit(MPic):

kbhit(MPic) is used to check if there is data available to read from the UART stream. This is generally a good way to avoid blocking the program while waiting for data. Storing Data in the Buffer:

The MP_Buffer[MP_Byte_Counter] = fgetc(MPic); line stores the received byte in MP_Buffer if the buffer has space (i.e., MP_Byte_Counter < MP_BFR_SIZE). Handling Buffer Overflow:

If the buffer is full (MP_Byte_Counter >= MP_BFR_SIZE), the code reads a byte into MP_Overflow to store the overflowed data. This is an interesting method, but a more typical approach would be to set a flag and handle the overflow in the main program. Inconsistent Handling of Overflow:

If an overflow occurs, the code reads the incoming byte and stores it in MP_Overflow. This seems somewhat counterintuitive because the system already has a dedicated overflow flag. Instead, it would make more sense to just set the MP_Overflow flag and leave the byte unread if there is no room in the buffer.

MPic_Available()

unsigned int8 MPic_Available()
{
   return MP_Byte_Counter ;
}

The function returns the number of bytes that have been received and stored in the MP_Buffer by the interrupt handler (SERIAL_ISR3). This is useful for other parts of your program to know if there is data available to process.

Return Value: The function returns the value of MP_Byte_Counter, which keeps track of the number of bytes that have been successfully received.

MPic_Read()

unsigned int8 MPic_Read()
{
   if (MP_Byte_Counter>0)
   {    
      MP_Temp_byte = MP_Buffer[MP_Read_Byte_counter];
      
      MP_Byte_Counter--;
      MP_Read_Byte_counter++;
      if(MP_Byte_Counter == 0) MP_Read_Byte_counter = 0;
      return MP_Temp_byte; 
   }
   
   if (MP_Byte_Counter == 0)
   { 
      MP_Read_Byte_counter = 0;
      MP_Temp_byte = 0x00;
      return MP_Temp_byte; 
   }
 
}
  • Check if Data is Available:

The function first checks if there are any bytes available in the buffer (MP_Byte_Counter > 0). If data is available, it reads a byte from MP_Buffer[MP_Read_Byte_counter] and returns it. It also decrements the byte counter (MP_Byte_Counter) and increments the read counter (MP_Read_Byte_counter).

  • Reset on Empty Buffer:

If the buffer becomes empty (i.e., MP_Byte_Counter == 0), the function resets MP_Read_Byte_counter to 0 and returns 0x00. Returns:

It returns the byte from the buffer if there is data, or 0x00 if the buffer is empty.

MPic_flush()

void MPic_flush()
{
   while( MPic_Available() ) MPic_Read() ;
}
  • MPic_Available():

This function checks how many bytes are available to be read from the buffer. If there are any bytes in the buffer (MPic_Available() returns a non-zero value), the loop continues.

  • Flush the Buffer:

The MPic_Read() function is called within the while loop to read and discard all bytes in the buffer. The loop will continue reading bytes from the buffer until the buffer is empty (i.e., MPic_Available() returns 0).

UART port connection to com pic definitons

For COM and Start pic, the functions do the same thing as above.

#define CP_BFR_SIZE 30
#pin_select TX2=PIN_G1  
#pin_select RX2=PIN_G0  
#use rs232(UART2, baud=38400, parity=N, bits=8, stream=CPic, errors) 

unsigned int8  CP_Buffer[CP_BFR_SIZE];
unsigned int16 CP_Byte_Counter = 0;
unsigned int8  CP_Overflow = 0;
unsigned int16 CP_Read_Byte_counter = 0;
unsigned int8  CP_Temp_byte = 0;

#INT_RDA2

SERIAL_ISR2()

Void SERIAL_ISR2()         // MAIN PIC uart interupt loop
{
   if( kbhit(CPic) )
   {
      if( CP_Byte_Counter < CP_BFR_SIZE )
      {
         CP_Buffer[CP_Byte_Counter] = fgetc(CPic);
         CP_Byte_Counter++;
      }
      else CP_Overflow = fgetc(CPic);
   }
}

CPic_Available()

unsigned int8 CPic_Available()
{
   return CP_Byte_Counter ;
}

CPic_Read()

unsigned int8 CPic_Read()
{
   if (CP_Byte_Counter>0)
   {    
      CP_Temp_byte = CP_Buffer[CP_Read_Byte_counter];
      
      CP_Byte_Counter--;
      CP_Read_Byte_counter++;
      if(CP_Byte_Counter == 0) CP_Read_Byte_counter = 0;
      return CP_Temp_byte; 
   }
   
   if (CP_Byte_Counter == 0)
   { 
      CP_Read_Byte_counter = 0;
      CP_Temp_byte = 0x00;
      return CP_Temp_byte; 
   }
 
}

CPic_flush()

void CPic_flush()
{
   while( CPic_Available() ) CPic_Read() ;
}

UART port connection to start pic definitons

#define SP_BFR_SIZE 5
#pin_select TX1=PIN_C6  
#pin_select RX1=PIN_C7  
#use rs232(UART1, baud=38400, parity=N, bits=8, stream=SPic, errors) 

unsigned int8  SP_Buffer[SP_BFR_SIZE];
unsigned int16 SP_Byte_Counter = 0;
unsigned int8  SP_Overflow = 0;
unsigned int16 SP_Read_Byte_counter = 0;
unsigned int8  SP_Temp_byte = 0;

#INT_RDA

SERIAL_ISR()

Void SERIAL_ISR()         // MAIN PIC uart interupt loop
{
   if( SP_Byte_Counter < SP_BFR_SIZE )
   {
      SP_Buffer[SP_Byte_Counter] = fgetc(SPic);
      SP_Byte_Counter++;
   }
   
   else SP_Overflow = fgetc(SPic);
}

SPic_Available()

unsigned int8 SPic_Available()
{
   return SP_Byte_Counter ;
}

SPic_Read()

unsigned int8 SPic_Read()
{
   if (SP_Byte_Counter>0)
   {    
      SP_Temp_byte = SP_Buffer[SP_Read_Byte_counter];
      
      SP_Byte_Counter--;
      SP_Read_Byte_counter++;
      if(SP_Byte_Counter == 0) SP_Read_Byte_counter = 0;
      return SP_Temp_byte; 
   }
   
   if (SP_Byte_Counter == 0)
   { 
      SP_Read_Byte_counter = 0;
      SP_Temp_byte = 0x00;
      return SP_Temp_byte; 
   }
 }

SPic_flush()

void SPic_flush()
{
   while( SPic_Available() ) SPic_Read() ;
}

printline()

void printline()
{
   fprintf( PC, "\n\r");
}

CLEAR_DATA_ARRAY(array, array_size)

using this function we can make any data array clear

void CLEAR_DATA_ARRAY(unsigned int8 array[], int array_size)
{   
   for(int i = 0; i < array_size; i++)
   {
      array[i] = 0 ;
   }
}

3. MAIN PIC

File content

The files found in the MAIN PIC folder are detailed in Table 3 below.

File name pattern Scope Content
BIRDS-X_Mission_Boss.h
ComPic_Settings.c
Debug.c
Definitions.c
FAB_Settings.c
Flash_Memory.c
MainPIC_Settings.c
MainPICmain.c
MEMORY_OPERATION.c
MPIC_CPIC.c
MPIC_FAB.c
MPIC_MBOSS.c
MPIC_RPIC.c
PIC18F67J94_registers.c
RESERVE_fun.c
RTC_functions.c

This program is designed to manage and control various subsystems of BIRDS-X using the PIC18F67J94 microcontroller. It includes functionalities such as antenna deployment, real-time clock (RTC) management, communication with other PICs (microcontrollers), and handling commands through UART interfaces.

The code is modular and uses external files to implement specific functionalities such as flash memory handling, RTC functions, and debugging.

MainPICmain.c

Header Files and Configuration

#include <18F67J94.h>
#FUSES NOWDT,NOBROWNOUT,SOSC_DIG
#use delay(crystal = 16MHz, clock = 16MHz)
#include <PIC18F67J94_REGISTERS.h>
#include <stdio.h>
#include <18F67J94.h>: Includes the device-specific header file.
  • FUSES: Configuration fuses set up the microcontroller's behavior:
  • NOWDT: Disables the Watchdog Timer.
  • NOBROWNOUT: Disables Brown-out Reset.
  • SOSC_DIG: Configures the secondary oscillator for digital mode.
#use delay: Sets the crystal frequency and system clock.
#include <PIC18F67J94_REGISTERS.h>: Defines registers for specific hardware components.

External Files

#include <MainPIC_Settings.c>
#include <RTC_functions.c>
#include <Flash_Memory.c>
#include <RESERVE_fun.c>
#include <MPIC_RPIC.c>
#include <MPIC_FAB.c>
#include <MPIC_MBOSS.c>
#include <MPIC_CPIC.c>
#include <Debug.c>

These files implement modular functionalities, such as:

  • RTC_functions.c: RTC setup and manipulation.
  • Flash_Memory.c: Flash memory operations.
  • Debug.c: Debugging utilities.

Interrupt Priority Configuration

#PRIORITY INT_RDA4, INT_RDA2, INT_RDA3, INT_RDA

Configures interrupt priorities for UART communication on different ports.

setting()

Void setting()
{
   enable_interrupts(INT_RDA);
   enable_interrupts(INT_RDA2);
   enable_interrupts(INT_RDA3);
   enable_interrupts(INT_RDA4);
   enable_interrupts(GLOBAL);
   
   Output_Low(PIN_A4);
   SETUP_RTC(RTC_ENABLE | RTC_CLOCK_SOSC, 0);
   Write_OBC_RTC(23, 07, 28, 00, 00, 01);
   
   OUTPUT_HIGH(PIN_C5);
   OUTPUT_HIGH(PIN_A5);
   
   output_HIGH(MBOSS_EN);
}
  • Enables UART interrupts for communication with other PICs.
  • Configures the RTC and initializes its values.
  • Sets up pins to enable or disable certain components like the On-Board Computer (OBC) and Mission Boss (MBOSS).

main()

The main() function orchestrates all operations.

Initialization

Delay_ms(1000);
setting();
fprintf(PC, "MainPIC booting...........\n\r");

Delays for 1 second, initializes the system, and sends a debug message to the PC. System Initialization and Operations

LOAD_ANTENNA_DEPLOYMENT_VALUES();
LOAD_RESERVATION_COMMANDS_ON_OFF_TIME_AT_BOOT();
LOAD_30DAY_COUNTER();
INCREACE_30DAY_COUNTER_ONCE_PER_DAY_AND_CHECK_30DAY_CW_STATUS();

Loads settings for antenna deployment, reservation commands, and counters for monitoring system status. Infinite Loop

while(true)
{
   READ_MP_RTC_AND_PRINT();
   DEPLOY_ANTENNAS_SET_1(31, 5);
   DEPLOY_ANTENNAS_SET_2(40, 5);
   COMUNICATION_WITH_RST_PIC_90SEC(90);
   CHECK_RESERVATION_COMMAND_ON_TIME_AND_OFF_TIME();
   RESTARTING_MISSIONS_AFTER_SYSTEM_RESET();
   CHECK_UART_INCOMING_FROM_DEBUG_PORT();
   CHECK_UART_INCOMING_FROM_COM_PIC();
}
  • RTC Operations: Reads and prints the RTC values.
  • Antenna Deployment: Configures antenna deployment based on time and other parameters.
  • Communication: Interacts with the Reset PIC every 90 seconds and checks incoming UART commands. Command Handling Commands from COM PIC
if(CPIC_TO_MPIC_ARRAY[0] == 0xB0 && CPIC_TO_MPIC_ARRAY[39] == 0xB1)
{
   PRINT_RECIVED_COMMAND_FROM_COM_PIC();
   Delay_ms(5);
   GIVE_COMFM_ACCESS_TO_COMPIC_FOR_DATA_DOWNLOAD();
   // Other operations...
   CLEAR_DATA_ARRAY(CPIC_TO_MPIC_ARRAY, 32);
}

Commands from Debug Port

if(DEBUG_TO_MPIC_ARRAY[0] == 0xD0 && DEBUG_TO_MPIC_ARRAY[12] == 0xD1)
{
   PRINT_RECIVED_COMMAND_FROM_DEBUG_PORT();
   Delay_ms(5);
   READ_FM_DATA_THROUGH_PC();
   // Other operations...
   CLEAR_DATA_ARRAY(DEBUG_TO_MPIC_ARRAY, 20);
}

Handles specific command sequences received through UART, performs necessary tasks, and clears the command buffer.

MainPIC_Settings.c

Header Definitions and Constants

  • Pin Configuration:
    • #define MBOSS_EN PIN_D1
    • Enables MBOSS communication pin.
    • #define LED_PIN PIN_G2 Defines LED pin control.
  • Status Flags:
    • #define Enabled 0xAA
    • #define Disabled 0xBB
  • Buffer Sizes:
    • #define PC_BFR_SIZE 50
    • #define RP_BFR_SIZE 50
    • #define CP_BFR_SIZE 50
    • #define FP_BFR_SIZE 50

Global Variables

  • Counters and Arrays for Communication: Arrays used for bidirectional communication with various subsystems (debug, reset PIC, communication PIC, FAB PIC, MBOSS PIC).
char DEBUG_TO_MPIC_ARRAY[55];
char MPIC_TO_RPIC_ARRAY[10], RPIC_TO_MPIC_ARRAY[55];
char MPIC_TO_CPIC_ARRAY[32], CPIC_TO_MPIC_ARRAY[55];
char MPIC_TO_FAB_ARRAY[32], FAB_TO_MPIC_ARRAY[55];
char MPIC_TO_MBOSS_ARRAY[40], MBOSS_TO_MPIC_ARRAY[40];
  • Antenna Deployment Variables: Flags and counters for antenna deployment tracking.
unsigned int16 ANT_SET_1_DEP_FLAG, ANT_SET_2_DEP_FLAG;
unsigned int16 ANTSET_1_COUNT, ANTSET_2_COUNT;
  • Housekeeping Data: Communication success/failure counters for Reset PIC and FAB PIC.
unsigned int32 SUCCESFULL_COMUNICATION_WITH_RST_PIC, FAILED_COMUNICATION_WITH_RST_PIC;
unsigned int32 SUCCESFULL_COMUNICATION_WITH_FAB, FAILED_COMUNICATION_WITH_FAB;
  • Real-Time Clock Variables: Tracking year, month, day, hour, minute, and second for the main PIC's RTC.
char MP_RTC_YEAR, MP_RTC_MONTH, MP_RTC_DAY;
char MP_RTC_HOUR, MP_RTC_MIN, MP_RTC_SEC;
  • Mission Flags: Used to track the status of payloads and references.
unsigned int8 APRS_REFERENSE_1_FLAG, APRS_REFERENSE_2_FLAG;
unsigned int8 APRS_PAYLOAD_1_FLAG, APRS_PAYLOAD_2_FLAG, APRS_PAYLOAD_3_FLAG, APRS_PAYLOAD_4_FLAG, APRS_PAYLOAD_5_FLAG;
unsigned int8 MISSION_STATUS;

Serial Communication Configurations

  • UART Streams and Pins: Configured for various communication ports (PC, Reset PIC, COM PIC, FAB PIC).
#use rs232(baud=115200, parity=N, UART1, bits=8, stream=PC, errors)
#use rs232(baud=38400, parity=N, UART4, bits=8, stream=RPic, errors)
#use rs232(baud=38400, parity=N, UART2, bits=8, stream=cpic, errors)
#use rs232(baud=38400, parity=N, UART3, bits=8, stream=FAB, errors)

Interrupt Service Routines (ISRs)

  • UART Receive Interrupts: Handles data reception for PC, Reset PIC, COM PIC, and FAB PIC communication.
#INT_RDA, #INT_RDA2, #INT_RDA3, #INT_RDA4
Void SERIAL_ISR1(), SERIAL_ISR2(), SERIAL_ISR3(), SERIAL_ISR4();

Functions

  • Buffer Management: Functions to check available data, read data, and flush buffers for each communication channel.
unsigned int8 PC_Available(), PC_Read(), RPic_Available(), RPic_Read(), CPic_Available(), CPic_Read(), FABPic_Available(), FABPic_Read();
void PC_flush(), RPic_flush(), CPic_flush(), FABPic_flush();
  • Data and Debug Utilities: Printing and clearing data.
void printline(), printtable(unsigned int8 table[], int table_size, int column_size);
void CLEAR_DATA_ARRAY(unsigned int8 array[], int array_size);
  • Mission Communication Utilities: Placeholder for core functions related to communication and operations:
void COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(int numof_times, int16 time_delay = 200, int16 wait_time = 70, int inc_array_length = 3);
void GIVE_COMFM_ACCESS_TO_COMPIC_FOR_DATA_DOWNLOAD();

Code Highlights

  1. Communication Handling: The code establishes a structured communication mechanism between the main PIC and subsystems like FAB PIC, Reset PIC, and external PCs, using UART streams and dedicated buffers.

  2. Antenna Deployment Logic: Includes deployment flags and counters to track and control antenna operations.

  3. Mission Status Monitoring: Tracks multiple payloads and reference systems, allowing detailed monitoring of mission objectives.

  4. Error Handling: Includes counters for communication failures and success rates, which can be used for diagnostics.

MPIC_FAB.c

CHECK_UART_INCOMING_FROM_FAB_PIC()

This function verifies and processes UART data received from the FAB PIC (Peripheral Interface Controller).

void CHECK_UART_INCOMING_FROM_FAB_PIC()
{
   if(FABPic_available())
   {
      Delay_ms(100);
      for( int i = 0; i<5; i++)
      {
         if( FABPIC_Read() == 0xFA )
         {
            FAB_TO_MPIC_ARRAY[0] = 0xFA;
            break;
         }
      }
      
      for (int i=1; i<=50; i++)
      {
         FAB_TO_MPIC_ARRAY[i] = FABPIC_Read();
      }
   }
}

Steps:

  1. Checks if there is data available from FABPic_available().
  2. Introduces a 100 ms delay (Delay_ms(100)) to allow data to accumulate in the buffer.
  3. Reads up to 5 bytes from FABPIC_Read() and looks for the start byte 0xFA. If found, it stores it as the first byte in FAB_TO_MPIC_ARRAY.
  4. Reads the next 50 bytes from the FAB PIC into FAB_TO_MPIC_ARRAY.

PRINT_RECIVED_COMMAND_FROM_FAB_PIC()

This function prints the data received from the FAB PIC.

void PRINT_RECIVED_COMMAND_FROM_FAB_PIC()
{
   Fprintf(PC,"Received command from FAB PIC >> ");
   for(int i = 0; i<35; i++)
   {
      Fprintf(PC,"%X ",FAB_TO_MPIC_ARRAY[i]);
   }
   printline();
   printline();
}

Steps:

  1. Outputs a message indicating the data is being printed: "Received comand from FAB pic >> ".
  2. Iterates through the first 35 bytes of FAB_TO_MPIC_ARRAY and prints each in hexadecimal format.
  3. Adds two blank lines for clarity (printline();).

COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE()

Handles communication with the FAB PIC, sends a command, and waits for the response.

void COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(int numof_times, int16 wait_time = 200, int16 time_delay = 70, int inc_array_length = 3)
{
   printline();
   for(int j=0; j<numof_times; j++)
   {
      Fprintf(PC,"Num of communication tries to FAB PIC = %d\n\r", j+1);
      FABPic_flush();
      for( int i = 0; i<3; i++)
      {
         fputc(MPIC_TO_FAB_ARRAY[i], FAB);
      }
      Delay_ms(wait_time);
      
      CHECK_UART_INCOMING_FROM_FAB_PIC();
      PRINT_RECIVED_COMMAND_FROM_FAB_PIC();
      
      if( FAB_TO_MPIC_ARRAY[0] == 0xFA && FAB_TO_MPIC_ARRAY[inc_array_length-1] == 0xFB )
      {
         Fprintf(PC,"FAB responded correctly\n\r");
         break;
      } 
      else
      {  
         Delay_ms(time_delay);
         Fprintf(PC,"Received wrong response from FAB PIC >> ");
         for(int i = 0; i<inc_array_length; i++)
         {
            Fprintf(PC,"%X ",FAB_TO_MPIC_ARRAY[i]);
         }
         printline();  
      }
   }
}
  • Parameters:
    • numof_times: Number of communication attempts.
    • wait_time: Delay before checking the response (default: 200 ms).
    • time_delay: Delay between retry attempts (default: 70 ms).
    • inc_array_length: Expected length of the response array (default: 3).
  • Steps:
    1. Attempts communication numof_times times.
    2. Sends the MPIC_TO_FAB_ARRAY data via the FAB UART.
    3. Waits for wait_time milliseconds, then calls CHECK_UART_INCOMING_FROM_FAB_PIC() to receive the response.
    4. Prints the received data using PRINT_RECIVED_COMMAND_FROM_FAB_PIC().
    5. Checks if the response starts with 0xFA and ends with 0xFB.
    • If valid, prints success and breaks out of the loop.
    • Otherwise, retries after a time_delay and prints the incorrect response.

_CLOSE_FAB_KILL_SWITCH()

Closes the FAB kill switch.

void _CLOSE_FAB_KILL_SWITCH()
{
   CLEAR_DATA_ARRAY(FAB_TO_MPIC_ARRAY, 10);
   MPIC_TO_FAB_ARRAY[0] = 0xFA;
   MPIC_TO_FAB_ARRAY[1] = 0xF1;
   MPIC_TO_FAB_ARRAY[2] = 0xFB;
   COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(3, 500, 200);
   
   if(FAB_TO_MPIC_ARRAY[0] == 0xFA && FAB_TO_MPIC_ARRAY[1] == 0xF1 && FAB_TO_MPIC_ARRAY[2] == 0xFB)
   {
      Fprintf(PC,"FAB kill switch successfully closed\n\r");    
   }
   else Fprintf(PC,"FAB kill switch closing not successful\n\r");
   printline();
}

Steps:

  1. Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command sequence [0xFA, 0xF1, 0xFB].
  2. Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to communicate the command.
  3. Verifies the response matches [0xFA, 0xF1, 0xFB].
  • If valid, prints a success message.
  • Otherwise, prints a failure message.

_CLOSE_OBC_KILL_SWITCH()

Closes the OBC (Onboard Computer) kill switch.

void _CLOSE_OBC_KILL_SWITCH()
{
   CLEAR_DATA_ARRAY(FAB_TO_MPIC_ARRAY, 10);
   MPIC_TO_FAB_ARRAY[0] = 0xFA;
   MPIC_TO_FAB_ARRAY[1] = 0xF2;
   MPIC_TO_FAB_ARRAY[2] = 0xFB;
   COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(3, 500, 200);
   
   if(FAB_TO_MPIC_ARRAY[0] == 0xFA && FAB_TO_MPIC_ARRAY[1] == 0xF2 && FAB_TO_MPIC_ARRAY[2] == 0xFB)
   {
      Output_high(PIN_A4);
      Delay_ms(1000);
      Output_low(PIN_A4);
      Fprintf(PC,"OBC kill switch successfully closed\n\r");
   }
   else Fprintf(PC,"OBC kill switch closing not successful\n\r");
   Output_low(PIN_A4);
   printline();
}

Steps:

  1. Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF2, 0xFB].
  2. Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
  3. Verifies the response.
  • If valid, toggles the hardware pin PIN_A4 to close the switch and prints success.
  • Otherwise, prints failure.

_OPEN_FAB_KILL_SWITCH()

Opens the FAB kill switch.

void _OPEN_FAB_KILL_SWITCH()
{
   CLEAR_DATA_ARRAY(FAB_TO_MPIC_ARRAY, 10);
   MPIC_TO_FAB_ARRAY[0] = 0xFA;
   MPIC_TO_FAB_ARRAY[1] = 0xF3;
   MPIC_TO_FAB_ARRAY[2] = 0xFB;
   COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(1, 200, 200);
   
   if(FAB_TO_MPIC_ARRAY[0] == 0xFA && FAB_TO_MPIC_ARRAY[1] == 0xF3 && FAB_TO_MPIC_ARRAY[2] == 0xFB)
   {
      Fprintf(PC,"FAB kill open command successfully sent to FAB\n\r");
   }
   else Fprintf(PC,"Communication with FAB not successful\n\r"); 
   printline();
}

Steps:

  1. Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF3, 0xFB].
  2. Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
  3. Verifies the response.
  • If valid, prints success.
  • Otherwise, prints failure.

_OPEN_OBC_KILL_SWITCH()

Opens the OBC kill switch.

void _OPEN_OBC_KILL_SWITCH()
{
   CLEAR_DATA_ARRAY(FAB_TO_MPIC_ARRAY, 10);
   MPIC_TO_FAB_ARRAY[0] = 0xFA;
   MPIC_TO_FAB_ARRAY[1] = 0xF4;
   MPIC_TO_FAB_ARRAY[2] = 0xFB;
   COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE(1, 200, 200);
   
   if(FAB_TO_MPIC_ARRAY[0] == 0xFA && FAB_TO_MPIC_ARRAY[2] == 0xFB)
   {
      Fprintf(PC,"OBC kill open command successfully sent to FAB\n\r");
      if(FAB_TO_MPIC_ARRAY[1] == 0xF4)
      {
         Output_high(PIN_A4);
         Delay_ms(1000);
         Output_low(PIN_A4);
         Fprintf(PC,"OBC kill is opened\n\r");
      }
   }
   else Fprintf(PC,"Communication with FAB not successful\n\r"); 
   Output_low(PIN_A4);
   printline();
}

Steps:

  1. Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF4, 0xFB].
  2. Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
  3. Verifies the response starts with 0xFA and ends with 0xFB.
  • If valid:
    • Checks if the second byte is 0xF4.
    • Toggles the hardware pin PIN_A4 to open the switch and prints success.
  • Otherwise, prints failure.

Supporting Logic:

  • Data Arrays:

    • FAB_TO_MPIC_ARRAY: Buffer for storing received data.
    • MPIC_TO_FAB_ARRAY: Command data to be sent.
  • Serial Communication:

    • FABPic_available(): Checks if data is available on the FAB PIC UART.
    • FABPIC_Read(): Reads a byte from the FAB PIC.
    • Fprintf(PC, ...): Outputs data to the PC console.
  • Delays:

    • Delay_ms(): Introduces a delay in milliseconds for synchronization.
  • Hardware Interaction:

    • Output_high(PIN_A4) and Output_low(PIN_A4): Toggles a hardware pin for kill switch control.

FAB_Settings.c

This code is designed for managing UART communication between a microcontroller and a subsystem (OBC - On-Board Computer). Additionally, it provides functionality to control and monitor kill switches for solar panel connections. The kill switches are toggled via specific pins, enabling or disabling power transfer.

Data Arrays for Communication

// Arrays used for communication with OBC
unsigned int8 MPIC_TO_FAB_ARRAY[12];  // Only 3 bytes are used
unsigned int8 FAB_TO_MPIC_ARRAY[50];

These arrays handle data exchange:

  • MPIC_TO_FAB_ARRAY: Stores outgoing data.
  • FAB_TO_MPIC_ARRAY: Stores incoming data from the FAB PIC.

UART Ports and Debug Communication

#use rs232(baud=19200, xmit = PIN_B7, parity=N, bits=8, stream=PC, force_sw, ERRORS)
  • Configures the UART port for communication.
  • Debug communication is established via the PC stream.

UART Buffer and Interrupt

#define MP_BFR_SIZE 10  // UART buffer size
unsigned int8 MP_Buffer[MP_BFR_SIZE];
unsigned int16 MP_Byte_Counter = 0;
unsigned int8 MP_Overflow = 0;
unsigned int16 MP_Read_Byte_counter = 0;
unsigned int8 MP_Temp_byte = 0;

SERIAL_ISR1()

#INT_RDA
Void SERIAL_ISR1()
{
   if (MP_Byte_Counter < MP_BFR_SIZE)
   {
      MP_Buffer[MP_Byte_Counter] = fgetc(MPIC);
      MP_Byte_Counter++;
   }
   else MP_Overflow = fgetc(MPIC);  // Handle overflow
}
  • Purpose: Manages UART communication via interrupt service.
  • Key Points:
    • Incoming data is stored in a circular buffer (MP_Buffer).
    • If buffer overflows, extra data is discarded.
  • Helper Functions:
    1. MPic_Available()
    • Returns the number of bytes available in the buffer.
    1. MPic_Read()
    • Reads a byte from the buffer, reducing the byte counter.
    • Resets counters when the buffer is empty.

Utility Functions

CLEAR_DATA_ARRAY()

void CLEAR_DATA_ARRAY(unsigned int8 array[], int array_size)
{
   for (int i = 0; i < array_size; i++) array[i] = 0;
}

Clears any array, setting all elements to zero.

void printline()
{
   fprintf(PC, "\n\r");
}

Prints a blank line for debug readability.

FAB_KILL_SWITCH()

void FAB_KILL_SWITCH(int status)
{
   Output_High(PIN_D4);    // Enable kill switch driver
   Delay_ms(100);
   
   if (status == CLOSE)
   {
      Output_High(PIN_D5);    
      Output_Low(PIN_D6);
      fprintf(PC, "FAB Kill Switch Closing done\n\r");
   }
   
   if (status == OPEN)
   {
      Output_Low(PIN_D5);      
      Output_High(PIN_D6); 
      fprintf(PC, "FAB Kill Switch Opening done\n\r");
   }
   
   Delay_ms(100);
   Output_Low(PIN_D4);  // Disable kill switch driver
}
  • Purpose: Controls the FAB kill switch, which disconnects or reconnects the solar panels.
  • Operation:
    • status == CLOSE: Connects solar panels.
    • status == OPEN: Disconnects solar panels.
  • Pins:
    • PIN_D4: Enables kill switch driver.
    • PIN_D5/PIN_D6: Sets the switch state.

OBC_KILL_SWITCH()

void OBC_KILL_SWITCH(int status)
{
   if (status == CLOSE)
   {
      Output_High(PIN_D2);    
      Output_Low(PIN_D1);
      fprintf(PC, "OBC Kill Switch Closing done\n\r");
   }
   
   if (status == OPEN)
   {
      Output_Low(PIN_D2);     
      Output_High(PIN_D1);
      fprintf(PC, "OBC Kill Switch Opening done\n\r");
   }
}
  • Purpose: Controls the OBC kill switch, operating similarly to the FAB kill switch.
  • Pins:
    • PIN_D2/PIN_D1: Sets the switch state.
  • Operation:
    • status == CLOSE: Connects solar panels.
    • status == OPEN: Disconnects solar panels.

How the Code Works

  • UART Communication: Handles bidirectional communication via MP_Buffer and interrupt-driven logic.
  • Kill Switch Control: The FAB and OBC kill switches allow toggling the connection between solar panels and the system.
  • Debugging: Uses fprintf to print status messages for monitoring operations.

MPIC_MBOSS.c

This code manages communication between a Mission Boss (MBOSS) system and an APRS (Automatic Packet Reporting System) setup. It includes functionality for sending commands, receiving responses, toggling system states, and handling APRS board numbers and flags.

  • Mission Boss Management: Use this code to interact with the Mission Boss system via UART communication.
  • APRS Command Handling: Ensure reliable transfer of commands and data validation.

Global Variables

int board_number;
unsigned int8 MissionBoss_flag = 0;
  • board_number: Stores the board identifier, configurable in Settings.c.
  • MissionBoss_flag: Tracks whether the Mission Boss is active (1) or inactive (0).

CHECK_UART_INCOMING_FROM_MBOSS_PIC()

void CHECK_UART_INCOMING_FROM_MBOSS_PIC(unsigned int32 looping = 100000)
{
   int MBC = 0;
   
   for (unsigned int32 i = 0; i < looping; i++)
   {
      if (kbhit(MBOSS) && MissionBoss_flag == 1)
      {
         MBOSS_TO_MPIC_ARRAY[MBC] = fgetc(MBOSS);
         MBC++;
         if (MBC >= 13) break;
      }
   }
   
   fprintf(PC, "CHECK_UART_INCOMING_FROM_MBOSS_PIC\r\n");
   fprintf(PC, "Data from MBOSS:");
   
   for (int i = 0; i < 13; i++)
   {
      fprintf(PC, "%X ", MBOSS_TO_MPIC_ARRAY[i]);
   }
   printline();
}
  • Purpose: Reads incoming UART data from MBOSS into MBOSS_TO_MPIC_ARRAY.
  • Loops for a specified count (looping) or until 13 bytes are read.
  • Prints received data to the debug interface.

PRINT_RECIVED_COMMAND_FROM_MISSION_BOSS()

void PRINT_RECIVED_COMMAND_FROM_MISSION_BOSS()
{
   fprintf(PC, "Received command from Mission Boss >> ");
   for (int i = 0; i < 40; i++)
   {
      fprintf(PC, "%X ", MBOSS_TO_MPIC_ARRAY[i]);
   }
   printline();
   printline();
}
  • Purpose: Outputs all received data from MBOSS in a human-readable format for debugging.

ACK_APRS_COMMAND_TO_COM()

void ACK_APRS_COMMAND_TO_COM(unsigned int board_number, unsigned int mission_number)
{
   CLEAR_DATA_ARRAY(MPIC_TO_CPIC_ARRAY, 32);
   MPIC_TO_CPIC_ARRAY[0]  = 0xB0;   // Header
   MPIC_TO_CPIC_ARRAY[1]  = board_number;
   MPIC_TO_CPIC_ARRAY[2]  = mission_number;
   MPIC_TO_CPIC_ARRAY[31] = 0xB1;   // Footer
   
   for (int i = 0; i < 32; i++)
   {
      fputc(MPIC_TO_CPIC_ARRAY[i], CPic);
   }
}
  • Purpose: Sends an acknowledgment to the communication PIC (CPIC) with board and mission numbers.

APRS_BOARD_IDENTIFY()

int APRS_BOARD_IDENTIFY(unsigned int a)
{
   char c[3];
   int bn;
   sprintf(c, "%X", a);
   bn = c[1] - '0';
   
   if (bn <= 7)
   {
      printf("Board Number --> %X", bn);
   }
   else
   {
      printf("Not correct board number");
   }
   return bn;
}
  • Purpose: Decodes the board number from a command and ensures it is valid.

TURN_ON_MISSIONBOSS()

void TURN_ON_MISSIONBOSS()
{
   output_high(PIN_D1);
   MissionBoss_flag = 1;
}
  • TURN_ON_MISSIONBOSS: Activates the Mission Boss system.

TURN_OFF_MISSIONBOSS()

void TURN_OFF_MISSIONBOSS()
{
   output_low(PIN_D1);
   MissionBoss_flag = 0;
}
  • TURN_OFF_MISSIONBOSS: Deactivates the Mission Boss system.

GIVE_SFM_ACCESS_TO_MISSIONBOSS()

void GIVE_SFM_ACCESS_TO_MISSIONBOSS()
{
   fprintf(PC, "GIVE_SFM_ACCESS_TO\\MISSIONBOSS\n\r");
   output_high(PIN_A5);
}

STOP_SFM_ACCESS_TO_MISSIONBOSS()

void STOP_SFM_ACCESS_TO_MISSIONBOSS()
{
   fprintf(PC, "STOP_SFM_ACCESS_TO\\MISSIONBOSS\n\r");
   output_low(PIN_A5);
}

These functions control access to the Subsystem Function Module (SFM) for the Mission Boss.

RESET_ALL_APRSBOARD_NUMBER_FLAGS()

void RESET_ALL_APRSBOARD_NUMBER_FLAGS()
{
   if (MISSION_STATUS == 0)
   {
      APRS_REFERENSE_1_FLAG = 0;
      APRS_REFERENSE_2_FLAG = 0;
      APRS_PAYLOAD_1_FLAG = 0;
      APRS_PAYLOAD_2_FLAG = 0;
      APRS_PAYLOAD_3_FLAG = 0;
      APRS_PAYLOAD_4_FLAG = 0;
      APRS_PAYLOAD_5_FLAG = 0;
   }
}

Resets all APRS board flags when the mission is not active (MISSION_STATUS == 0).

SEND_APRS_COMMAND_TO_MISSIONBOSS_THROUGH_MAIN()

void SEND_APRS_COMMAND_TO_MISSIONBOSS_THROUGH_MAIN()
{
   if (CPIC_TO_MPIC_ARRAY[1] == 0x00 && (CPIC_TO_MPIC_ARRAY[2] & 0xF0) == 0xB0)
   {
      // Acknowledge to CPIC
      CLEAR_DATA_ARRAY(MPIC_TO_CPIC_ARRAY, 32);
      MPIC_TO_CPIC_ARRAY[0]  = 0xB0;
      MPIC_TO_CPIC_ARRAY[1]  = 0xB0;
      MPIC_TO_CPIC_ARRAY[31] = 0xB1;
      
      for (int i = 0; i < 32; i++)
      {
         fputc(MPIC_TO_CPIC_ARRAY[i], CPic);
      }
      
      fprintf(PC, "Sending APRS command to Mission Boss\n\r");
      TURN_ON_MISSIONBOSS();
      delay_ms(1000);
      
      CLEAR_DATA_ARRAY(MPIC_TO_MBOSS_ARRAY, 40);
      CLEAR_DATA_ARRAY(MBOSS_TO_MPIC_ARRAY, 40);

      MPIC_TO_MBOSS_ARRAY[0] = CPIC_TO_MPIC_ARRAY[2];
      MPIC_TO_MBOSS_ARRAY[1] = CPIC_TO_MPIC_ARRAY[3];
      MPIC_TO_MBOSS_ARRAY[2] = 0xE0;

      for (int i = 3; i < 10; i++)
      {
         MPIC_TO_MBOSS_ARRAY[i] = CPIC_TO_MPIC_ARRAY[i + 1];
      }
      MPIC_TO_MBOSS_ARRAY[10] = 0xED;
      
      fprintf(PC, "APRS command:");
      fputc(0xAA, MBOSS); // Header for Mission Boss
      for (int i = 0; i < 11; i++)
      {
         fputc(MPIC_TO_MBOSS_ARRAY[i], MBOSS);
         fprintf(PC, "%X ", MPIC_TO_MBOSS_ARRAY[i]);
      }

      if (MPIC_TO_MBOSS_ARRAY[3] == 0x11) // Grant access if data transfer command
      {
         GIVE_SFM_ACCESS_TO_MISSIONBOSS();
         delay_ms(5000);
         STOP_SFM_ACCESS_TO_MISSIONBOSS();
      }
      
      printline();
      CHECK_UART_INCOMING_FROM_MBOSS_PIC(500000);
   }
}
  • Sends an APRS command to Mission Boss.
  • Grants SFM access if the command is a data transfer request.

BIRDS-X_Mission_Boss.h

Pin definition

#define sel_0 PIN_C6    //CPLD(D)
#define sel_1 PIN_C7    //CPLD(E)
#define sel_2 PIN_C2    //CPLD(F)
//#define sel_3 PIN_C3    //CPLD(G)

Debugging UART

A software UART is configured on Pin C0 with a baud rate of 19200 for debugging purposes.

#use rs232(baud=19200, parity=N, xmit=pin_C0,  bits=8, stream=DEBUG, ERRORS, force_sw) //Pin_B7 = PGD, Pin_B6 = PGC

APRS Communication (UART2):

unsigned int8 APRS_RESPONSE[210];

#define APRS_BFR_SIZE 200
// UART PC functions___________________________________________________________
#pin_select TX2 = PIN_G1   //MB1, CPLD(84)
#pin_select RX2 = PIN_G0   //MB1, CPLD(83)                                                      
#use rs232(baud=9600, parity=N, UART2, bits=8, stream=APRS, ERRORS)                     


unsigned int8  APRS_Buffer[APRS_BFR_SIZE];
unsigned int16 APRS_Byte_Counter = 0;
unsigned int8  APRS_Overflow = 0;
unsigned int16 APRS_Read_Byte_counter = 0;
unsigned int8  APRS_Temp_byte = 0;
#INT_RDA2

SERIAL_ISR2()

Void SERIAL_ISR2()                                                             // MAIN PIC uart interupt loop
{
   if( APRS_Byte_Counter < APRS_BFR_SIZE )
   {
      APRS_Buffer[APRS_Byte_Counter] = fgetc(APRS);
      APRS_Byte_Counter++;
   }
   
   else APRS_Overflow = fgetc(APRS);
}

APRS_Available()

unsigned int8 APRS_Available()
{
   return APRS_Byte_Counter ;
}

APRS_Read()

unsigned int8 APRS_Read()
{
   if (APRS_Byte_Counter>0)
   {    
      APRS_Temp_byte = APRS_Buffer[APRS_Read_Byte_counter];
      
      APRS_Byte_Counter--;
      APRS_Read_Byte_counter++;
      if(APRS_Byte_Counter == 0) APRS_Read_Byte_counter = 0;
      return APRS_Temp_byte; 
   }
   
   if (APRS_Byte_Counter == 0)
   { 
      APRS_Read_Byte_counter = 0;
      APRS_Temp_byte = 0x00;
      return APRS_Temp_byte; 
   }
 
}

APRS_flush()

void APRS_flush()
{
   while( APRS_Available() ) APRS_Read() ;
}

GET_DATA_OR_ACK_FROM_APRS()

void GET_DATA_OR_ACK_FROM_APRS()
{
   if( APRS_Available() )
   {
      delay_ms(300);
      for(int i=0; i<200; i++)
      {
         APRS_RESPONSE[i] = APRS_READ();
      }
   }
}

Configures UART communication with an Automatic Packet Reporting System (APRS) using:

  • TX2 = PIN_G1
  • RX2 = PIN_G0
  • Baud rate = 9600. The APRS buffer and its ISR (#INT_RDA2) manage incoming data:
  • Stores data in a buffer (APRS_Buffer).
  • Detects and handles buffer overflow. Helper functions:
  • APRS_Available: Returns the number of bytes available in the buffer.
  • APRS_Read: Reads bytes from the buffer sequentially.
  • APRS_flush: Clears the buffer.
  • GET_DATA_OR_ACK_FROM_APRS: Reads and stores incoming data into the APRS_RESPONSE array after ensuring data availability.

OBC Communication (UART3):

char command_data[39];

#define OBC_BFR_SIZE 20
// UART PC functions___________________________________________________________
#pin_select TX3 = PIN_E1   //RAB, CPLD(81)
#pin_select RX3 = PIN_E0   //RAB, CPLD(82)
#use rs232(baud=9600, parity=N, UART3, bits=8, stream=OBC, ERRORS)                  


unsigned int8  OBC_Buffer[OBC_BFR_SIZE];
unsigned int16 OBC_Byte_Counter = 0;
unsigned int8  OBC_Overflow = 0;
unsigned int16 OBC_Read_Byte_counter = 0;
unsigned int8  OBC_Temp_byte = 0;

int8 Flag_OBC = 0;

#INT_RDA3

SERIAL_ISR3()

Void SERIAL_ISR3()                                                             // MAIN PIC uart interupt loop
{
   if( OBC_Byte_Counter < OBC_BFR_SIZE )
   {
      OBC_Buffer[OBC_Byte_Counter] = fgetc(OBC);
      OBC_Byte_Counter++;
   }
   
   else OBC_Overflow = fgetc(OBC);
}

OBC_Available()

unsigned int8 OBC_Available()
{
   return OBC_Byte_Counter ;
}

OBC_Read()

unsigned int8 OBC_Read()
{
   if (OBC_Byte_Counter>0)
   {    
      OBC_Temp_byte = OBC_Buffer[OBC_Read_Byte_counter];
      
      OBC_Byte_Counter--;
      OBC_Read_Byte_counter++;
      if(OBC_Byte_Counter == 0) OBC_Read_Byte_counter = 0;
      return OBC_Temp_byte; 
   }
   
   if (OBC_Byte_Counter == 0)
   { 
      OBC_Read_Byte_counter = 0;
      OBC_Temp_byte = 0x00;
      return OBC_Temp_byte; 
   }
 
}

OBC_flush()

void OBC_flush()
{
   while( OBC_Available() ) OBC_Read() ;
}

char header ;

GET_COMMANDS_FROM_OBC()

void GET_COMMANDS_FROM_OBC()
{
   if( OBC_Available() )
   {
      header = OBC_READ() ;
      
      if( (0xB0 <= header) && (header <= 0xB6) )
      {    
         delay_ms(100);
         
         command_data[0] = header ;
         
         for(int i=1; i<20; i++)
         {
            command_data[i] = OBC_READ();
         }
         Flag_OBC = 1;
      }
   }
}

Configures communication with an On-Board Computer (OBC):

  • TX3 = PIN_E1
  • RX3 = PIN_E0
  • Baud rate = 9600. Similar to APRS, it uses a buffer and an ISR (#INT_RDA3) for data handling:
  • Stores OBC data in OBC_Buffer.
  • Implements helper functions (OBC_Available, OBC_Read, OBC_flush).
  • GET_COMMANDS_FROM_OBC: Processes specific header-based commands (0xB0 to 0xB6) and stores the command data.

SPI Interface for Shared Flash Memory

#use spi(MASTER, CLK = PIN_A0, DI = PIN_A1, DO = PIN_A2,  BAUD = 1000000, BITS = 8, STREAM = SHARED_FM, MODE = 0)
// CS pin = PIN_A3
unsigned int8 address[4];
unsigned int8 _data;

SHARED_FM_WRITE_ENABLE()

void SHARED_FM_WRITE_ENABLE()
{
  Output_low(Pin_A3);
  spi_xfer(SHARED_FM,0x06);                
  Output_high(Pin_A3);
  return;
}

SHARED_FM_SECTOR_ERASE()

void SHARED_FM_SECTOR_ERASE(unsigned int32 sector_address,char sector_size, unsigned int16 delay = 1000)
{
   
   address[0]  = (unsigned int8)((sector_address>>24) & 0xFF);   // 0x __ 00 00 00
   address[1]  = (unsigned int8)((sector_address>>16) & 0xFF);   // 0x 00 __ 00 00
   address[2]  = (unsigned int8)((sector_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   address[3]  = (unsigned int8)((sector_address)     & 0xFF);   // 0x 00 00 00 __
   
   SHARED_FM_WRITE_ENABLE();
   Output_low(Pin_A3);             //lower the CS PIN

   ///////////////////////////////////////////////////////////////////

   if( Sector_size == 4  ) spi_xfer(SHARED_FM,0x21);                    // 4KB Sector erase
   if( Sector_size == 32 ) spi_xfer(SHARED_FM,0x5C);                    // 32KB Sector erase
   if( Sector_size == 64 ) spi_xfer(SHARED_FM,0xDC);                    // 64KB Sector erase
   
   spi_xfer(SHARED_FM,address[0]);   
   spi_xfer(SHARED_FM,address[1]);    
   spi_xfer(SHARED_FM,address[2]);    
   spi_xfer(SHARED_FM,address[3]);
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_A3);           //take CS PIN higher back

   delay_ms(delay); 
   return;
}

SHARED_FM_BYTE_WRITE()

void SHARED_FM_BYTE_WRITE(unsigned int32 byte_address, unsigned int8 data)
{
   
   //Byte extraction
   address[0]  = (unsigned int8)((byte_address>>24) & 0xFF);   // 0x __ 00 00 00
   address[1]  = (unsigned int8)((byte_address>>16) & 0xFF);   // 0x 00 __ 00 00
   address[2]  = (unsigned int8)((byte_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   address[3]  = (unsigned int8)((byte_address)     & 0xFF);   // 0x 00 00 00 __
 
   SHARED_FM_WRITE_ENABLE();
   Output_low(Pin_A3);             //lower the CS PIN
   ///////////////////////////////////////////////////////////////////
   spi_xfer(SHARED_FM,0x12);         //Byte WRITE COMAND  (0x12)
   spi_xfer(SHARED_FM,address[0]);    
   spi_xfer(SHARED_FM,address[1]);    
   spi_xfer(SHARED_FM,address[2]);    
   spi_xfer(SHARED_FM,address[3]);

   spi_xfer(SHARED_FM,data); 
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_A3);           //take CS PIN higher back 
   
   return;
}

Configures SPI communication for shared flash memory:

  • SPI pins:
    • Clock = PIN_A0
    • Data In = PIN_A1
    • Data Out = PIN_A2
    • Chip Select = PIN_A3
  • Baud rate = 1 MHz.

SPI Operations: - SHARED_FM_WRITE_ENABLE: Enables writing to the flash memory. - SHARED_FM_SECTOR_ERASE: Erases a specified sector (4KB, 32KB, or 64KB). - SHARED_FM_BYTE_WRITE: Writes a byte to a specific address. - (Commented out) SHARED_FM_BYTE_READ: Reads a byte from a specified address.

CPLD Configuration Table

  • Defines the selection pin logic for different CPLD configurations (R1, R2, P1-P5)
  • Uses combinations of sel_0, sel_1, and sel_2 to select specific configurations.
/*
CPLD connection configuration
board   sel_0   sel_1   sel_2
#R1     0       0       1
#R2     0       1       0
#P1     0       1       1
#P2     1       0       0
#P3     1       0       1
#P4     1       1       0
#P5     1       1       1
*/

Key Observations:

  1. Interrupt Handling:
  • The SERIAL_ISR2 and SERIAL_ISR3 functions ensure efficient data collection from APRS and OBC without blocking execution.
  1. Buffer Management:
  • Both APRS and OBC implement circular-like buffer handling to store and retrieve data.
  1. Flexibility:
  • Uses #pin_select to dynamically assign UART pins, making it hardware-agnostic.
  1. SPI Commands:
  • The SPI functions use standard flash memory commands for sector erasure and byte writing, ensuring compatibility with various memory chips.

PIC18F67J94_registers.h

The use of #byte and #bit directives maps register addresses and specific bits to named variables. This abstraction makes it easier to interact with hardware registers directly in the code.

General Purpose I/O Registers

These registers control the configuration and state of the I/O pins of the microcontroller:

TRIS Registers (Data Direction)

#byte TRISG = 0xF98
#byte TRISF = 0xF97
#byte TRISE = 0xF96
#byte TRISD = 0xF95
#byte TRISC = 0xF94
#byte TRISB = 0xF93
#byte TRISA = 0xF92


#bit    TRISE0 = TRISE.0
#bit    TRISE1 = TRISE.1
#bit    TRISE2 = TRISE.2

#bit    TRISC4 = TRISC.4
#bit    TRISB3 = TRISB.3

#bit    TRISA0 = TRISA.0
#bit    TRISA1 = TRISA.1
#bit    TRISA2 = TRISA.2
#bit    TRISA5 = TRISA.5
  • TRIS registers (e.g., TRISA, TRISB, ..., TRISG) control whether the corresponding pins are inputs (1) or outputs (0).
    • TRISA = 0xF92 means the TRIS register for port A is located at address 0xF92.
    • #bit TRISA0 = TRISA.0 defines the first bit (bit 0) of the TRISA register.

LAT Registers (Latch)

#byte LATG = 0xF8F
#byte LATF = 0xF8E
#byte LATE = 0xF8D
#byte LATD = 0xF8C
#byte LATC = 0xF8B
#byte LATB = 0xF8A
#byte LATA = 0xF89

#bit    LATA0 = LATA.0
#bit    LATA1 = LATA.1
#bit    LATA2 = LATA.2
#bit    LATA3 = LATA.3
  • LAT registers (e.g., LATA, LATB, ..., LATG) hold the output data for output pins. Writing to these registers updates the state of the output pins.
    • LATB = 0xF8A means the latch register for port B is located at address 0xF8A.

PORT Registers (I/O State)

#byte PORTG = 0xF86
#bit    RG0 = PORTG.0
#bit    RG1 = PORTG.1
#bit    RG2 = PORTG.2
#bit    RG3 = PORTG.3
#bit    RG4 = PORTG.4
#bit    RG6 = PORTG.6
#bit    RG7 = PORTG.7
#byte PORTF = 0xF85
#bit    RF2 = PORTF.2
#bit    RF3 = PORTF.3
#bit    RF4 = PORTF.4
#bit    RF5 = PORTF.5
#bit    RF6 = PORTF.6
#bit    RF7 = PORTF.7

#byte PORTE = 0xF84
#bit    RE2 = PORTE.2

#byte PORTD = 0xF83
#byte PORTC = 0xF82
#bit    RC0 = PORTC.0
#bit    RC1 = PORTC.1
#bit    RC2 = PORTC.2
#bit    RC3 = PORTC.3
#bit    RC4 = PORTC.4
#bit    RC5 = PORTC.5
#bit    RC6 = PORTC.6
#bit    RC7 = PORTC.7

#byte PORTB = 0xF81
#bit    RB3 = PORTB.3

#byte PORTA = 0xF80
#bit    RA0 = PORTA.0
#bit    RA1 = PORTA.1
#bit    RA2 = PORTA.2
#bit    RA3 = PORTA.3
#bit    RA4 = PORTA.4
#bit    RA5 = PORTA.5
#bit    RA6 = PORTA.6
#bit    RA7 = PORTA.7
  • PORT registers (e.g., PORTA, PORTB, ..., PORTG) read the current state of the pins or write data directly to the pins (depending on TRIS configuration).
    • PORTC = 0xF82 defines the port register for port C at address 0xF82.
    • #bit RC0 = PORTC.0 accesses the first bit of the PORTC register.

Real-Time Clock (RTC) Registers

#byte RTCCON1 = 0xF5F
#bit    RTCPTR0 = RTCCON1.0
#bit    RTCPTR1 = RTCCON1.1
#bit    RTCOE = RTCCON1.2
#bit    HALFSEC = RTCCON1.3
#bit    RTCSYNC = RTCCON1.4
#bit    RTCWREN = RTCCON1.5
#bit    RTCEN = RTCCON1.7
#byte RTCCAL = 0xF5E
#byte RTCVALH = 0xF5D
#bit    WAITE0 = RTCVALH.0
#bit    WAITE1 = RTCVALH.1
#bit    WAITM0 = RTCVALH.2
#bit    WAITM1 = RTCVALH.3
#bit    WAITM2 = RTCVALH.4
#bit    WAITM3 = RTCVALH.5
#bit    WAITB0 = RTCVALH.6
#bit    WAITB1 = RTCVALH.7
//#byte RTCVALH = 0xF5D
#byte RTCVALL = 0xF5C
#byte ALRMCFG = 0xF5B
#bit    ALRMPTR0 = ALRMCFG.0
#bit    ALRMPTR1 = ALRMCFG.1
#bit    AMASK0 = ALRMCFG.2
#bit    AMASK1 = ALRMCFG.3
#bit    AMASK2 = ALRMCFG.4
#bit    AMASK3 = ALRMCFG.5
#bit    CHIME = ALRMCFG.6
#bit    ALRMEN = ALRMCFG.7
#byte ALRMRPT = 0xF5A
#byte ALRMVALH = 0xF59
#byte ALRMVALL = 0xF58
#byte RTCCON2 = 0xF57
#bit    RTCSECSEL0 = RTCCON2.0
#bit    RTCSECSEL1 = RTCCON2.1
#bit    RTCCLKSEL0 = RTCCON2.2
#bit    RTCCLKSEL1 = RTCCON2.3
#bit    PWCSPRE = RTCCON2.4
#bit    PWCCPRE = RTCCON2.5
#bit    PWCPOL = RTCCON2.6
#bit    PWCEN = RTCCON2.7

These registers control the real-time clock module, which tracks time and supports alarm functionality:

  • RTCCON1 (0xF5F): Controls the primary RTC operations:
    • RTCEN: Enable the RTC module (1 to enable).
    • RTCWREN: Allows writes to RTC registers (1 to allow writes).
    • HALFSEC: Indicates a half-second toggle bit.
  • RTCCON2 (0xF57): Additional configuration for RTC:
    • RTCCLKSEL: Selects the clock source for the RTC.
    • PWCEN: Power control enable.
  • ALRMCFG (0xF5B): Alarm configuration register:
    • ALRMEN: Enables the alarm feature.
    • CHIME: Enables the alarm to repeat (chime).
  • Other RTC-related registers, like RTCVALH, RTCVALL, and ALRMVALH, hold the time and alarm values.

Time Registers

T1CON (Timer1 Control)

#byte T1CON  = 0xFCD
#bit  T1CON7 = T1CON.7
#bit  T1CON6 = T1CON.6

#bit  SOSCEN1 = T1CON.3
#bit  TMR1ON = T1CON.0
  • Address: 0xFCD.
  • Controls Timer1 operation:
    • TMR1ON: Enables Timer1 (1 to enable).
    • SOSCEN1: Enables the secondary oscillator for Timer1.
#byte T3CON  = 0xFB1
#bit  SOSCEN3 = T3CON.3
  • Address: 0xFB1.
  • Contains similar settings as Timer1, including:
    • SOSCEN3: Enables the secondary oscillator for Timer3.

Interrupt Registers

These registers control and monitor interrupt settings

#byte PIE3 = 0xFA3
#bit  RC2IE = PIE3.5
  • Address: 0xFA3.
  • Enables specific peripheral interrupts:
    • RC2IE: Enables the interrupt for the UART2 receive module.

Miscellaneous Registers

#byte EECON2 = 0xF7E
#byte OSCCON = 0xFD3
  • EECON2 (0xF7E): Used for EEPROM read/write operations.
  • OSCCON (0xFD3): Controls the oscillator configuration for the microcontroller.

Flash_Memory.c

This setup allows the microcontroller to independently communicate with three SPI devices by selecting the appropriate SPI stream for each device.

//commands
#define READ_ID           0x9F
#define READ_STATUS_REG   0x05 

#use spi(MASTER, CLK=PIN_E1, DI=PIN_E0, DO=PIN_E6,  BAUD=1000000, BITS=8, STREAM=MAIN_FM, MODE=0)        //MAIN flash memory port
#use spi(MASTER, CLK=PIN_B2, DI=PIN_B5, DO=PIN_B4,  BAUD=1000000, BITS=8, STREAM=COM_FM, MODE=0)         //COM flash memory port
#use spi(MASTER, CLK=PIN_A3, DI=PIN_A0, DO=PIN_A1,  BAUD=1000000, BITS=8, STREAM=MSN_FM, MODE=0) 

READ_ID (0x9F): SPI command to read the device ID. Commonly used to retrieve manufacturer and memory type information from an SPI device. READ_STATUS_REG (0x05): SPI command to read the status register, often used to check the device's status, such as write enable latch or ready/busy state.

Syntax: MASTER: Configures the microcontroller as the SPI master. CLK (Clock): The pin used for the SPI clock line. DI (Data Input): The pin used for receiving data (MISO: Master In Slave Out). DO (Data Output): The pin used for sending data (MOSI: Master Out Slave In). BAUD: Communication speed in bits per second (1 MHz in this case). BITS: Number of bits per data frame (8-bit per frame here). STREAM: A named handle for the SPI interface to manage communication with multiple devices. MODE: SPI mode defines clock polarity and phase; Mode 0 (CPOL=0, CPHA=0) is selected here.

Streams: MAIN_FM: SPI interface for the main flash memory (pins on port E). COM_FM: SPI interface for the communication flash memory (pins on port B). MSN_FM: SPI interface for the mission flash memory (pins on port A).

The code below defines three functions, each enabling the "write" operation for a specific SPI flash memory device. The WRITE_ENABLE command (0x06) is a standard SPI flash memory command. Before any write, erase, or modification operation, the flash memory must be explicitly enabled to accept write operations. This mechanism helps prevent accidental data overwrites.

MAIN_FM_WRITE_ENABLE()

void MAIN_FM_WRITE_ENABLE()
{
  Output_low(Pin_E2);
  spi_xfer(MAIN_FM,0x06);                
  Output_high(Pin_E2);
  return;
}Controls the main flash memory.

Pulls the chip select (CS) line (Pin_E2) low to begin communication. Sends the WRITE_ENABLE command (0x06) over the MAIN_FM SPI stream. Sets the CS line (Pin_E2) high to end communication.

COM_FM_WRITE_ENABLE()

void COM_FM_WRITE_ENABLE()
{
  Output_low(Pin_B3);
  spi_xfer(COM_FM,0x06);                
  Output_high(Pin_B3);
  return;
}

Controls the communication flash memory. Operates similarly to MAIN_FM_WRITE_ENABLE() but uses Pin_B3 as the chip select line and the COM_FM SPI stream.

MSN_FM_WRITE_ENABLE()

void MSN_FM_WRITE_ENABLE()
{

  Output_low(Pin_A2);
  spi_xfer(MSN_FM,0x06);                
  Output_high(Pin_A2);
  return;
}

Controls the mission flash memory. Uses Pin_A2 as the chip select line and the MSN_FM SPI stream.

SPI Command Workflow Activate Chip Select: Pulling the CS line (Output_low) tells the SPI device to pay attention to incoming data. Send Command: The spi_xfer function transmits the command byte (0x06 for WRITE_ENABLE) to the flash memory. Deactivate Chip Select: Raising the CS line (Output_high) completes the transaction, ensuring the device latches the command.

unsigned int8 _data;

Initialisation of a data variable

Sector Erase Main FM (4KB,32KB,64KB )

void MAIN_FM_SECTOR_ERASE(unsigned int32 sector_address,char sector_size,  int16 delay = 1000)
{
   
   adsress[0]  = (unsigned int8)((sector_address>>24) & 0xFF);   // 0x __ 00 00 00
   adsress[1]  = (unsigned int8)((sector_address>>16) & 0xFF);   // 0x 00 __ 00 00
   adsress[2]  = (unsigned int8)((sector_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   adsress[3]  = (unsigned int8)((sector_address)     & 0xFF);   // 0x 00 00 00 __
   
   MAIN_FM_WRITE_ENABLE();
   Output_low(Pin_E2);             //lower the CS PIN

   ///////////////////////////////////////////////////////////////////

   if( Sector_size == 4  ) spi_xfer(MAIN_FM,0x21);                    // 4KB Sector erase
   if( Sector_size == 32 ) spi_xfer(MAIN_FM,0x5C);                    // 32KB Sector erase
   if( Sector_size == 64 ) spi_xfer(MAIN_FM,0xDC);                    // 64KB Sector erase
   
   spi_xfer(MAIN_FM,adsress[0]);   
   spi_xfer(MAIN_FM,adsress[1]);    
   spi_xfer(MAIN_FM,adsress[2]);    
   spi_xfer(MAIN_FM,adsress[3]);
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_E2);           //take CS PIN higher back

   delay_ms(delay); 
   return;
}

The MAIN_FM_SECTOR_ERASE function is intended to erase a specified sector of memory on a flash device using SPI communication. It does so by sending the appropriate erase command followed by the sector's address. The sector size is used to determine which erase command should be sent.

Function Breakdown

  • Function Parameters:

sector_address: The address of the sector to be erased (32-bit). sector_size: The size of the sector (e.g., 4KB, 32KB, 64KB). delay: A delay in milliseconds after the erase operation (default 1000 ms if not provided).

  • Address Formatting: The address is split into 4 bytes (since it's a 32-bit address) and stored in adsress[]. This is required for the SPI transmission.

  • SPI Write Enable: The function calls MAIN_FM_WRITE_ENABLE() to ensure that the flash memory is in a writable state.

  • Chip Select (CS): The Pin_E2 is used for the chip select signal. The Output_low(Pin_E2) brings the chip select low, indicating that the flash memory is being selected for communication.

  • Sector Erase Command: Depending on the size of the sector (Sector_size), the appropriate erase command is sent:

0x21 for 4KB sector erase 0x5C for 32KB sector erase 0xDC for 64KB sector erase

  • Address Transmission: After the erase command, the address bytes are transmitted via the SPI interface using spi_xfer().
  • CS Pin Reset: After the address is transmitted, Output_high(Pin_E2) sets the chip select pin high to deselect the flash memory.
  • Delay: After the operation, a delay (specified by delay) is introduced to allow the memory erase to complete.

Sector Erase COM FM (4KB,32KB,64KB )

Same as above but for COM flash memory

void COM_FM_SECTOR_ERASE(unsigned int32 sector_address, char sector_size, unsigned int16 delay = 1000)
{
   
   adsress[0]  = (unsigned int8)((sector_address>>24) & 0xFF);   // 0x __ 00 00 00
   adsress[1]  = (unsigned int8)((sector_address>>16) & 0xFF);   // 0x 00 __ 00 00
   adsress[2]  = (unsigned int8)((sector_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   adsress[3]  = (unsigned int8)((sector_address)     & 0xFF);   // 0x 00 00 00 __
   
   COM_FM_WRITE_ENABLE();
   Output_low(Pin_B3);             //lower the CS PIN
   ///////////////////////////////////////////////////////////////////
   if( Sector_size == 4  ) spi_xfer(COM_FM,0x21);                    // 4KB Sector erase
   if( Sector_size == 32 ) spi_xfer(COM_FM,0x5C);                    // 32KB Sector erase
   if( Sector_size == 64 ) spi_xfer(COM_FM,0xDC);                    // 64KB Sector erase
      
   spi_xfer(COM_FM,adsress[0]);   
   spi_xfer(COM_FM,adsress[1]);    
   spi_xfer(COM_FM,adsress[2]);    
   spi_xfer(COM_FM,adsress[3]);
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_B3);           //take CS PIN higher back
   delay_ms(delay); 
   return;
}

Sector Erase MSN FM (4KB,32KB,64KB )

Same as above but for MSN flash memory

void MSN_FM_SECTOR_ERASE(unsigned int32 sector_address,char sector_size, unsigned int16 delay = 1000)
{
   
   adsress[0]  = (unsigned int8)((sector_address>>24) & 0xFF);   // 0x __ 00 00 00
   adsress[1]  = (unsigned int8)((sector_address>>16) & 0xFF);   // 0x 00 __ 00 00
   adsress[2]  = (unsigned int8)((sector_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   adsress[3]  = (unsigned int8)((sector_address)     & 0xFF);   // 0x 00 00 00 __
   
   MSN_FM_WRITE_ENABLE();
   Output_low(Pin_A2);             //lower the CS PIN
   ///////////////////////////////////////////////////////////////////
   if( Sector_size == 4  ) spi_xfer(MSN_FM,0x21);                    // 4KB Sector erase
   if( Sector_size == 32 ) spi_xfer(MSN_FM,0x5C);                    // 32KB Sector erase
   if( Sector_size == 64 ) spi_xfer(MSN_FM,0xDC);                    // 64KB Sector erase
   
   spi_xfer(MSN_FM,adsress[0]);   
   spi_xfer(MSN_FM,adsress[1]);    
   spi_xfer(MSN_FM,adsress[2]);    
   spi_xfer(MSN_FM,adsress[3]);
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_A2);           //take CS PIN higher back
   delay_ms(delay); 
   return;
}

MAIN_FM_BYTE_WRITE(byte_address, data)

void MAIN_FM_BYTE_WRITE(unsigned int32 byte_address, unsigned int8 data)
{
   
   //Byte extraction
   adsress[0]  = (unsigned int8)((byte_address>>24) & 0xFF);   // 0x __ 00 00 00
   adsress[1]  = (unsigned int8)((byte_address>>16) & 0xFF);   // 0x 00 __ 00 00
   adsress[2]  = (unsigned int8)((byte_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   adsress[3]  = (unsigned int8)((byte_address)     & 0xFF);   // 0x 00 00 00 __
 
   MAIN_FM_WRITE_ENABLE();
   Output_low(Pin_E2);             //lower the CS PIN
   ///////////////////////////////////////////////////////////////////
   spi_xfer(MAIN_FM,0x12);         //Byte WRITE COMAND  (0x12)
   spi_xfer(MAIN_FM,adsress[0]);    
   spi_xfer(MAIN_FM,adsress[1]);    
   spi_xfer(MAIN_FM,adsress[2]);    
   spi_xfer(MAIN_FM,adsress[3]);

   spi_xfer(MAIN_FM,data); 
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_E2);           //take CS PIN higher back 
   
   return;
}

The MAIN_FM_BYTE_WRITE function is designed to write a byte of data to a specific address in flash memory via SPI. The function performs the following tasks:

Function Breakdown

  • Parameters:

byte_address: The 32-bit address where the byte data will be written. data: The byte data to be written to the specified address.

  • Address Extraction: The 32-bit byte_address is broken down into four 8-bit bytes and stored in the adsress[] array. This is done by shifting the address bits and applying a mask.

  • Write Enable: The function calls MAIN_FM_WRITE_ENABLE(), which is assumed to be another function that ensures the flash memory is in write-enabled mode.

  • Chip Select (CS): The function pulls the chip select pin (Pin_E2) low to select the flash memory for communication.

  • SPI Write Command: The function sends the Byte Write command (0x12) followed by the address bytes (adsress[0], adsress[1], adsress[2], and adsress[3]). Afterward, it sends the byte of data (data) to be written.

  • Chip Select Reset: After the data is sent, the chip select pin is pulled high to deselect the flash memory.

same as above for com

void COM_FM_BYTE_WRITE(unsigned int32 byte_address, unsigned int8 data)
{
   
   //Byte extraction
   adsress[0]  = (unsigned int8)((byte_address>>24) & 0xFF);   // 0x __ 00 00 00
   adsress[1]  = (unsigned int8)((byte_address>>16) & 0xFF);   // 0x 00 __ 00 00
   adsress[2]  = (unsigned int8)((byte_address>>8)  & 0xFF);   // 0x 00 00 __ 00
   adsress[3]  = (unsigned int8)((byte_address)     & 0xFF);   // 0x 00 00 00 __
 
   COM_FM_WRITE_ENABLE();
   Output_low(Pin_B3);             //lower the CS PIN
   ///////////////////////////////////////////////////////////////////
   spi_xfer(COM_FM,0x12);         //Byte WRITE COMAND  (0x12)
   spi_xfer(COM_FM,adsress[0]);    
   spi_xfer(COM_FM,adsress[1]);    
   spi_xfer(COM_FM,adsress[2]);    
   spi_xfer(COM_FM,adsress[3]);

   spi_xfer(COM_FM,data); 
   //////////////////////////////////////////////////////////////////
   Output_high(Pin_B3);           //take CS PIN higher back 
   
   return;
}

ComPic_Settings.c

Pin assignment definitions

#define TXMODE_PIN    PIN_F6     // PIN 42 in connector -> CP2NTRX_DIO0 
#define TXDONE_PIN    PIN_F5     // PIN 43 in connector -> CP2NTRX_DIO1 
#define TRXPWR_PIN    PIN_A3     // PIN 38 in connector -> CP2NTRX_DIO2  

#define RSTPIC_RESTART_PIN    PIN_F7

Constant definitions

   unsigned int16 MLC = 0;

mainloop counter

   unsigned int CW_DATA[16];
   unsigned int CW_DATA_TO_NEWTRX[55];

cw data array

   unsigned int8 CW_FLAG   = 0x00;
   
   unsigned int8 NEWTRX_CW     = 0xAA ;
   unsigned int8 OLDTRX_CW     = 0xBB ;
   unsigned int8 NEWTRX_1PCKT  = 0xCC ;
   unsigned int8 OLDTRX_1PCKT  = 0xDD ;
   
   unsigned int32 TLE_ADDRESS  = 0x07D00000;    // 2000 th sector start adress

CW transmission related variables

   unsigned int8 NEW_TRX_STATUS = 0x00;
   unsigned int8 ON_  = 0xAA;
   unsigned int8 OFF_ = 0xBB;

Transceiver operation

   unsigned int8 RPIC_TO_CPIC_ARRAY[55];  // only 20 bytes are used
   unsigned int8 CPIC_TO_RPIC_ARRAY[55];  // only 20 bytes are used

these arrays are used to comunicate with reset pic

   unsigned int8 NEWTRX_CMD_BUFFER[60];   
   unsigned int8 NEWTRX_TO_CPIC_ARRAY[60];
   unsigned int8 CPIC_TO_NEWTRX_ARRAY[110];

these arrays are used to comunicate with new tranceiver

   unsigned int8 OLDTRX_CMD_BUFFER[60];
   unsigned int8 OLDTRX_TO_CPIC_ARRAY[60];
   unsigned int8 CPIC_TO_OLDTRX_ARRAY[110];

these arrays are used to comunicate with old tranceiver

   unsigned int8 MPIC_TO_CPIC_ARRAY[55];  // only 32 bytes are used
   unsigned int8 CPIC_TO_MPIC_ARRAY[55];  // only 40 bytes are used

these arrays are used to comunicate with main pic

Call signs GS call sign

   unsigned int8 GS_callsign_leter_1 = 0x4A ;   // J
   unsigned int8 GS_callsign_leter_2 = 0x47 ;   // G
   unsigned int8 GS_callsign_leter_3 = 0x36 ;   // 6
   unsigned int8 GS_callsign_leter_4 = 0x59 ;   // Y
   unsigned int8 GS_callsign_leter_5 = 0x42 ;   // B
   unsigned int8 GS_callsign_leter_6 = 0x57 ;   // W

Satellite call sign

   unsigned int8 ST_callsign_leter_1 = 0x6A ;   // J
   unsigned int8 ST_callsign_leter_2 = 0x67 ;   // G
   unsigned int8 ST_callsign_leter_3 = 0x36 ;   // 6
   unsigned int8 ST_callsign_leter_4 = 0x79 ;   // Y
   unsigned int8 ST_callsign_leter_5 = 0x6F ;   // O
   unsigned int8 ST_callsign_leter_6 = 0x77 ;   // W

Tranciever beacons

   Void OLD_TRX_SEND_1PCKT_BEACON() ;
   Void NEW_TRX_SEND_1PCKT_BEACON() ;
   void COMUNICATION_WITH_MAIN_PIC_AND_WAIT_FOR_RESPONE(int16 numof_times, int16 time_delay, int16 wait_time = 200);  

debug software uart

#use rs232(baud=19200, parity=N, xmit=PIN_B7,  bits=8, stream = PC, errors, force_sw )  //Debug PGD line

UART port connected to main pic

This section handle the UART port connected to main pic

#define MP_BFR_SIZE 50
#pin_select TX2=PIN_G1  
#pin_select RX2=PIN_G2  
#use rs232(UART2, baud=38400, parity=N, bits=8, stream=MPic, errors) 

unsigned int8  MP_Buffer[MP_BFR_SIZE];
unsigned int16 MP_Byte_Counter = 0;
unsigned int8  MP_Overflow = 0;
unsigned int16 MP_Read_Byte_counter = 0;
unsigned int8  MP_Temp_byte = 0;

#INT_RDA2

SERIAL_ISR2()

Void SERIAL_ISR2()         // MAIN PIC uart interupt loop
{
   if( kbhit(MPic) )
   {
      if( MP_Byte_Counter < MP_BFR_SIZE )
      {
         MP_Buffer[MP_Byte_Counter] = fgetc(MPic);
         MP_Byte_Counter++;
      }
      else MP_Overflow = fgetc(MPic);
   }
}

MPic_Available()

unsigned int8 MPic_Available()
{
   if (MP_Byte_Counter > 0) return MP_Byte_Counter ;
   else return 0;
}

MPic_Read()

unsigned int8 MPic_Read()
{
   if (MP_Byte_Counter>0)
   {    
      MP_Temp_byte = MP_Buffer[MP_Read_Byte_counter];
      
      MP_Byte_Counter--;
      MP_Read_Byte_counter++;
      if(MP_Byte_Counter == 0) MP_Read_Byte_counter = 0;
      return MP_Temp_byte; 
   }
   
   if (MP_Byte_Counter == 0)
   { 
      MP_Read_Byte_counter = 0;
      MP_Temp_byte = 0x00;
      return MP_Temp_byte; 
   }
}

MPic_flush()

void MPic_flush()
{
   while( MPic_Available() ) MPic_Read() ;
}

UART port connected to RESET pic

pin select and constant definitions

SERIAL_ISR3()

MAIN PIC uart interupt loop

RPic_Available()

RPic_Read()

RPic_flush()

UART port connected to OLD TRANCEIVER definitons

SERIAL_ISR1()

OLDTRX_Available()

OLDTRX_Read()

OLDTRX_flush()

UART port connected to NEW TRANCEIVER

SERIAL_ISR4()

NEWTRX_Available()

NEWTRX_Read()

NEWTRX_flush()

printline()

just printing a line

CLEAR_DATA_ARRAY(array[], array_size)

using this function we can make any data array clear

crc calculating function

CALCULATE_CRC(data[], data_number)

CALCULATE_CRC_(data[], data_number)

flash memory functions

FM_WRITE_ENABLE()

FM_SECTOR_ERASE(sector_address, sector_size, delay = 1000)

FM_BYTE_WRITE(byte_address, data)

FM_BYTE_READ_(ADDRESS)

FM_BYTE_READ(ADDRESS)

Definitions.c

Initial constant definitons

#define ON    1
#define OFF   0
#define CLOSE 1
#define OPEN  0

values of on and off, close and open for 0 and 1

Flags

#define FAB_KILL_SWITCH_FLAG            5
#define OBC_KILL_SWITCH_FLAG            10

flags memory locations of kill switches in EEROM

Pin assignments

#define FAB_KILL_SWITCH_STATUS_PIN      PIN_D3
#define OBC_KILL_STATUS_STATUS_PIN      PIN_C1

kill switch status read pin

ADC Channel definitions

Temperature and vontage

unsigned int8 CNL_ADD = 0B1000001100110000 ;              // 0x8330

#define PLUS_X_TEMP         0       //  V0 channel TEMP1
#define MINUS_Y_TEMP        1       //  V1 channel TEMP2
#define MINUS_Z_TEMP        2       //  V2 channel TEMP3
#define PLUS_Y_TEMP         3       //  V3 channel TEMP4
#define PLUS_Z_TEMP         4       //  V4 channel TEMP5
#define BACKPLANE_TEMP      5       //  V5 channel TEMP6  // rename
#define MINUS_X_TEMP        6       //  V6 channel TEMP7
#define TEMP8               7       //  V7 channel TEMP8  // rename

#define PLUS_X_VOLTAGE      8       //  V_1 V8 channel voltage
#define PLUS_Y_VOLTAGE      9       //  V_2 V9 channel voltage
#define MINUS_Z_VOLTAGE    10       //  V_3 V10 channel voltage
#define MINUS_Y_VOLTAGE    11       //  V_4 V11 channel voltage
#define PLUS_Z_VOLTAGE     12       //  V_5 V12 channel voltage
#define MINUS_X_VOLTAGE    13       //  V_6 V13 channel voltage

Represents a 16-bit binary channel address (possibly for a specific hardware register or device). 0x8330: The hexadecimal representation of 1000001100110000.

definitions related external adc channel address, These constants map temperature sensors to specific channels in the telemetry system Maps voltage monitoring channels for solar panels or subsystems corresponding to satellite axes (+X, -Y, etc.).

Solar Panel Current & Power System Telemetry

#define MINUS_X_SP_CURRENT 5
#define MINUS_Y_SP_CURRENT 3
#define MINUS_Z_SP_CURRENT 2
#define PLUS_Y_SP_CURRENT  1
#define PLUS_Z_SP_CURRENT  4
#define PLUS_X_SP_CURRENT  0  //

#define RAW_VOLTAGE        9
#define RAW_CURRENT         6
#define BATTERY_VOLTAGE    10
#define BATTERY_CURRENT    12
#define BATTERY_TEMP       13
#define SP_BUS_VOLTAGE     8
#define SP_BUS_CURRENT     11

PIC ADC lines, Maps solar panel currents to channels for telemetry or control purposes, aligned with the satellite's axes. Represents telemetry values for monitoring the satellite's power system

#use spi(MASTER, CLK = PIN_C3, DI = PIN_C4, DO = PIN_C5,ENABLE = PIN_C2, BAUD = 100000, BITS = 16, STREAM = SPIPORT, MODE = 2 )

external adc spi port definition

  • MASTER: Configures the SPI module in Master mode. The microcontroller will generate the clock signal and control communication.
  • CLK = PIN_C3: Specifies the microcontroller pin used as the SPI clock (SCK).
  • DI = PIN_C4: Stands for Data In (MISO). This pin is used to receive data from the SPI slave device.
  • DO = PIN_C5: Stands for Data Out (MOSI). This pin is used to send data to the SPI slave device.
  • ENABLE = PIN_C2: This pin is used to control the chip select (CS) or slave select (SS) signal for the SPI slave device. Pulling this pin low enables communication with the slave device.
  • BAUD = 100000: Sets the SPI clock frequency to 100,000 bits per second (100 kHz).
  • BITS = 16: Specifies the bit width of each SPI data frame. In this case, each transfer is 16 bits wide.
  • STREAM = SPIPORT: Names this SPI configuration as SPIPORT. This allows referencing it in your code when using SPI functions, particularly useful when managing multiple SPI configurations.
  • MODE = 2: Specifies the SPI communication mode. Mode is defined by the clock polarity (CPOL) and clock phase (CPHA): Mode 2: Clock idle is high (CPOL = 1), and data is sampled on the falling edge of the clock (CPHA = 0).

SPI Modes Table

Mode CPOL CPHA Clock Idle State Data Sampling Edge
0 0 0 Low Rising edge
1 0 1 Low Falling edge
2 1 0 High Falling edge
3 1 1 High Rising edge

READ_EXT_ADC_CHANELS(channel)

using this function external ADC temperature channels can be read

unsigned int8 READ_EXT_ADC_CHANELS( unsigned int32 channel )
{
   unsigned int32 CNL_ADD = 0B1000001100110000 | (channel<<10) ;      // 0B1000001100110000 = 0x8330         
   spi_xfer(SPIPORT, CNL_ADD); 
   unsigned int16 ADC_READING = spi_xfer(SPIPORT) & 0x0FFF;           // reading ADC values and removing first address bits  
   
   return ADC_READING/16;                                             // converting to 8 bit
}

This function takes a 32-bit integer as the channel parameter. It returns an 8-bit value (unsigned int8) representing the ADC reading. The base command 0x8330 (binary: 1000001100110000) is shifted and OR-ed with the channel parameter shifted 10 bits to the left. This step selects the desired ADC channel for reading. For example:

If channel = 1: 0x8330 | (1 << 10) results in 0x8430. Sends the command CNL_ADD to the external ADC via the SPI interface. This tells the ADC which channel to read. The second SPI transfer reads the ADC result. & 0x0FFF masks out the upper 4 bits, leaving only the 12-bit ADC reading (common for many ADCs). Divides the 12-bit ADC reading by 16 to reduce it to 8 bits. This assumes the ADC's full range (12 bits, 0-4095) should be mapped to 8 bits (0-255).

READ_FAB_PIC_ADC(port_name)

designed to read an analog signal from a specified ADC channel on a PIC microcontroller.

unsigned int16 READ_FAB_PIC_ADC(int port_name)
{
   SET_ADC_CHANNEL(port_name); //  routing nth channel to adc
   delay_us(40);      

   return READ_ADC();       
}

Input: port_name: An integer representing the ADC channel to be read. Output: A 16-bit unsigned integer (unsigned int16) containing the ADC reading. Routes the specified channel (port_name) to the ADC module. This function configures the multiplexer in the ADC hardware to select the appropriate input pin. Introduces a delay of 40 microseconds to allow the input voltage on the ADC channel to stabilize after switching. The delay duration depends on the ADC specifications (settling time) and the input impedance. Triggers the ADC conversion and retrieves the result. Returns the ADC reading as an unsigned 16-bit value. However, the actual resolution depends on the PIC's ADC module (commonly 10-bit or 12-bit).

AVERAGE_PICADC_READING(channel, num_of_readings, timedelay)

The AVERAGE_PICADC_READING function calculates the average ADC reading over a specified number of readings from a given channel, adding a time delay between each reading. It then converts the final result to an 8-bit value.

unsigned int8 AVERAGE_PICADC_READING(int channel, int16 num_of_readings, int16 timedelay)
{
   unsigned int16 val = 0;
   
   for(int16 i=0; i<num_of_readings; i++)  
   {
      val = val + READ_FAB_PIC_ADC( channel ) ;
      delay_us(timedelay);
   }
   
   val = (val/num_of_readings);
   
   return (char)((val>>4)&0xFF);
}

Input Parameters: channel: The ADC channel to read from. num_of_readings: The number of ADC readings to take. timedelay: The delay (in microseconds) between successive readings. Return Value: Returns the averaged ADC value as an 8-bit unsigned integer. val accumulates the sum of ADC readings over the loop iterations. The ADC channel is read num_of_readings times. Each reading is added to val. A delay is introduced between readings to allow for stabilization. The total accumulated ADC value is divided by the number of readings. The 16-bit averaged value (val) is right-shifted by 4 bits (effectively dividing by 16). Only the lower 8 bits are returned.

AVERAGE_EXTADC_READING(channel, num_of_readings, timedelay)

The AVERAGE_EXTADC_READING function computes the average reading from an external ADC channel over a specified number of readings, with a delay between each reading. It then returns the averaged value as an 8-bit unsigned integer.

unsigned int8 AVERAGE_EXTADC_READING(int channel, int16 num_of_readings, int16 timedelay)
{
   unsigned int32 val = 0;
   
   for(int16 i=0; i<num_of_readings; i++)  
   {
      val = val + READ_EXT_ADC_CHANELS( channel ) ;
      delay_us(timedelay);
   }
   
   val = val /num_of_readings;
   
   return (char)val;
}

Input Parameters: channel: The external ADC channel to be read. num_of_readings: Number of readings to average. timedelay: Delay in microseconds between successive readings. Return Value: Returns the average reading as an 8-bit unsigned integer. val accumulates the sum of all readings. Each iteration reads the specified ADC channel and adds the result to val. A delay is introduced between readings to account for ADC stabilization. The total accumulated ADC value is divided by the number of readings to get the average. The final value is cast to an 8-bit type, which can result in truncation of higher bits if val exceeds 255.

AVERAGE_HECS_READING(channel, num_of_readings, timedelay)

The AVERAGE_HECS_READING function calculates the average ADC reading from a specified channel over a given number of readings with a delay between each. This function returns the averaged result as a 16-bit unsigned integer.

unsigned int16 AVERAGE_HECS_READING(int channel, int16 num_of_readings, int16 timedelay)
{
   unsigned int32 VAL = 0;
   
   for(int16 i=0; i<num_of_readings; i++)  
   {
      val = val + READ_FAB_PIC_ADC( channel ) ;
      delay_us(timedelay);
   }
   val = (val/num_of_readings); 
   
   return (unsigned int16)val;
}

Input Parameters: channel: ADC channel to read from. num_of_readings: Number of readings to average. timedelay: Time in microseconds between successive readings. Return Value: Returns the average reading as a 16-bit unsigned integer (unsigned int16). Accumulates the sum of all ADC readings. A 32-bit variable is used to prevent overflow during accumulation. Iterates num_of_readings times, reading from the ADC channel during each iteration and adding the result to VAL. Introduces a delay of timedelay microseconds between consecutive readings. Divides the accumulated total by the number of readings to compute the average. Casts the averaged value to unsigned int16 and returns it.

Debug.c

CHECK_UART_INCOMING_FROM_DEBUG_PORT()

this functions will check debug port incoming

PRINT_RECIVED_COMMAND_FROM_DEBUG_PORT()

this functions will print received comand prom bebug port

DEPLOY_ANTENNA_INSTANTLY()

sending command to reset pic wait for respone we will try 10 times then we start antena deployment

SET_UP_ANTENNA_DEPLOYMENT_VALUES()

GET_ANTENNA_DEPLOYMENT_VALUES()

UPDATE_RESET_PIC_TIME()

kill switch handling

CLOSE_FAB_KILL_SWITCH_DB()

CLOSE_OBC_KILL_SWITCH_DB()

OPEN_FAB_KILL_SWITCH_DB()

OPEN_OBC_KILL_SWITCH_DB()

POWER_LINE_CONTROL_USING_DEBUG_COMMAND_TO_RST_PIC()

RESET_SATELLITE_CMD()

SEND_COM_CMD_THROUGH_PC()

READ_FM_DATA_THROUGH_PC()

MPIC_CPIC.c

CHECK_UART_INCOMING_FROM_COM_PIC()

PRINT_RECIVED_COMMAND_FROM_COM_PIC()

COMUNICATION_WITH_COM_PIC_AND_WAIT_FOR_RESPONE(numof_times, time_delay, wait_time = 70)

GIVE_COMFM_ACCESS_TO_COMPIC_FOR_DATA_DOWNLOAD()

GIVE_CW_DATA_TO_COM_PIC()

CHECK_COMFM_ACCES_RELEASING()

DATA_COPY_FROM_MSN_FM_TO_CFM_USING_GS_CMD()

DATA_COPY_FROM_MAIN_FM_TO_CFM_USING_GS_CMD()

SECTOR_ERASE_USING_GS_COMMAND()

CONTROL_KILL_SWITCHES()

DEPLOY_ANTENNA_USING_GS_COMMAND()

SET_ANTENNA_VALUES_USING_GS_COMMAND()

LOAD_30DAY_COUNTER()

RESET_30DAY_CONTER()

MPIC_RPIC.c

RESERVE_fun.c

RTC_functions.c

4. COM PIC

File content

The files found in the COM PIC folder are detailed in Table 3 below.

File name pattern Scope Content

This program is designed to manage and control various subsystems of BIRDS-X using the PIC18F67J94 microcontroller. It includes functionalities such as antenna deployment, real-time clock (RTC) management, communication with other PICs (microcontrollers), and handling commands through UART interfaces.

The code is modular and uses external files to implement specific functionalities such as flash memory handling, RTC functions, and debugging.