This document is prepared to share a thorough explaination of the execution of the BIRDS-RP satellite bus software.
The system block diagram of the BIRDS-RP 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-RP OBC board is detailed in figure 2 below.
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:
#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.
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.
#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.
ISR (SERIAL_ISR1()
): Handles incoming UART data efficiently, storing it in a buffer.
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.
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
).
RPic_Read()
Safely reads from the buffer, maintaining counters and avoiding overflow.
- 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
).
- If no data is available:
- Resets the read index and returns 0x00 as a default.
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. 1.Checks if data is available:
- Calls
Rpic_Available()
to see if the buffer has any bytes.
- 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.
- 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 theRPIC_TO_SPIC_ARRAY
array and exits the loop.
- Reads remaining data:
- Fills the rest of the
RPIC_TO_SPIC_ARRAY
array with the next 9 bytes from the buffer.
- 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.
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).
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 thePOWER_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).
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)
; andoutput_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.
for( int i = 0; i<10; i++ )
: Iterates through the 10 elements of theRPIC_TO_SPIC_ARRAY
.RPIC_TO_SPIC_ARRAY[i] = 0
: Sets each element of the array to 0, effectively clearing it.
This is the main code for the Start PIC MCU that manages power lines and checks for responses from another microcontroller (Reset PIC).
#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
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.
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
), incrementSEC_COUNT
every 100 ms. - Once
SEC_COUNT
reaches 36,000 (1 hour), setONEHOUR_FLAG
to0xAA
.
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. bytes
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 |
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:
Defines symbolic constants ON
and OFF
for readability, representing 1
(ON) and 0
(OFF) states.
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.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.
MP_CP_BuckBoost()
:
This function controls the Main PIC <-> Com PIC buck-boost converter.
- 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)
- 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)
- Similar to
MainPic_Power
, but controlsPIN_F6
and updates bit 6 ofPOWER_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
- Uses
PIN_D1
to control the buck-boost converter andPIN_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.
-
_3V3Power_Line2
- Uses
PIN_D2
to control the buck-boost converter andPIN_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.
- Uses
-
_5V0Power_Line
,- Uses
PIN_D3
to control the buck-boost converter andPIN_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.
- Uses
-
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.
- Uses
-
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.
- Uses
SYSTEM_RESET()
:
Performs a manual system reset of the satellite by command from PC or GS
- 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.
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. Each signal is mapped to a specific pin (or port) on the microcontroller, and the purpose of that pin is described:
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.
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.
A1 (AN1) = 3V3-1 current measure
- Analog channel 1 (
AN1
) measures the current flowing through the 3.3V#1 power rail (3V3-1
).
A0 (AN0) = 3V3-2 current measure
- Analog channel 0 (
AN0
) measures the current flowing through the 3.3V#2 power rail (3V3-2
).
A5 (AN4) = 5V0 current measure
- Analog channel 4 (
AN4
) measures the current on the 5V power line.
A4 (AN6) = UNREG-1 current measure
- Analog channel 6 (
AN6
) measures the current for an unregulated power line labeled UNREG-1.
C2 (AN9) = UNREG-2 current measure
- Analog channel 9 (
AN9
) measures the current for another unregulated power line labeled UNREG-2.
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
-
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.
- Configures the ADC to read from channel 1 (
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.
- Configures the ADC to read from channel 0 (
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.
- Configures the ADC to read from channel 4 (
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.
- Configures the ADC to read from channel 6 (
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.
- Configures the ADC to read from channel 9 (
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:
- Configure pin directions (TRIS registers).
- Control output states (LAT registers).
- Read or write pin states (PORT registers).
- Manage peripherals like RTCs, timers, and interrupts.
Here's a line-by-line explanation:
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.
- TRIS Registers are used to configure the direction of pins (input or output) for each port (
A
toG
). - Setting a TRIS bit to
1
makes the corresponding pin an input; setting it to0
makes it an output. - These definitions map the TRIS registers to their respective memory addresses (hex values starting at
0xF92
).
- 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.
- Individual bits within the TRIS registers are defined for ease of access.
- Example:
TRISE0
refers to bit 0 of theTRISE
register (controls direction for pinE0
).
- Defines specific bits in the LAT registers, corresponding to individual pins on port A.
- Example:
LATA0
controls the state of pinA0
.
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 definitions for individual pins on ports.
- Example:
RG0
refers to bit 0 of PORTG, representing the state of pinG0
.
The RTC (Real-Time Clock) section defines control registers and bits for managing the clock and alarms.
-
RTCCON1
is the control register for the RTC.RTCPTR0
andRTCPTR1
are specific bits within this register, for configuring the pointer to RTC data.RTCOE
HALFSEC
RTCSYNC
RTCWREN
RTCEN
-
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
-
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
-
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
- 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 ...
PIE3
is an interrupt enable register for specific peripherals.RC2IE
refers to bit 5 of PIE3, enabling an interrupt for the RC2 peripheral.
here we check main pic UART is available or not if available we save in MPIC_TO_RPIC array
this function will print recived command from com pic
this functions monitor the com pic. if no response within 10min reset pic will restart the com pic
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.
This function checks if data is available on the UART interface from the main PIC and saves it into an array.
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).
This function prints the command received from the main PIC to the PC.
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.
Monitors communication with the main PIC. If there’s no response for 10 minutes, it resets the main PIC.
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.
Responds to a 90-second communication command from the main PIC by sending housekeeping data.
- 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.
Updates the RTC (Real-Time Clock) based on a command from the main PIC.
- 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.
Controls power lines based on commands from the main PIC.
- 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.
Resets the satellite when a command is received.
- 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.
Controls the unregulated power line to deploy an antenna.
- 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.
This first section is commented out but nonetheless we will look at the use of it.
This function checks whether there is data available from the "START PIC" over UART and reads the data into an array if available.
- 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.
This function prints the contents of the received command from the "START PIC" to a PC console.
- 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).
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.
- 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.
- 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.
- 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.
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.
Initializes or updates the RTC with specific date and time values.
Updates the RTC values every second, accounting for leap years, month boundaries, and year rollover.
Prints the status of various power lines by extracting bit information from a POWER_LINE_STATUS
variable.
Prints the current RTC time and resets, with additional context like power line status and counters.
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.
#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
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.
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.
The main() function contains the core operational logic:
- initialises settings function
- RTC Management: Prints the current RTC time (Print_RTC()). Resets the system at midnight (SYSTEM_RESET_24H()).
- 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.
- 90-Second Communication Monitoring: Monitors communication with MAIN_PIC and COM_PIC. If no communication is detected for 90 seconds, resets are triggered.
- Command Execution: Handles specific commands received from MAIN_PIC and COM_PIC. Examples include updating the RTC, controlling power lines, and resetting the satellite.
- 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.
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.
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.
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.
#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.
The code processes incoming bytes, stores them in a buffer, and manages the overflow condition. 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.
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.
- 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_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).
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
using this function we can make any data array clear
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.
#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.
#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.
#PRIORITY INT_RDA4, INT_RDA2, INT_RDA3, INT_RDA
Configures interrupt priorities for UART communication on different ports.
- 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).
The main() function orchestrates all operations.
Initialization Delays for 1 second, initializes the system, and sends a debug message to the PC. System Initialization and Operations Loads settings for antenna deployment, reservation commands, and counters for monitoring system status. Infinite Loop
- 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
-
- Commands from Debug Port
-
- Handles specific command sequences received through UART, performs necessary tasks, and clears the command buffer.
- 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
- 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;
- 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)
- 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();
- 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();
-
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.
-
Antenna Deployment Logic: Includes deployment flags and counters to track and control antenna operations.
-
Mission Status Monitoring: Tracks multiple payloads and reference systems, allowing detailed monitoring of mission objectives.
-
Error Handling: Includes counters for communication failures and success rates, which can be used for diagnostics.
This function verifies and processes UART data received from the FAB PIC (Peripheral Interface Controller). Steps:
- Checks if there is data available from FABPic_available().
- Introduces a 100 ms delay (Delay_ms(100)) to allow data to accumulate in the buffer.
- 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.
- Reads the next 50 bytes from the FAB PIC into FAB_TO_MPIC_ARRAY.
This function prints the data received from the FAB PIC. Steps:
- Outputs a message indicating the data is being printed: "Received comand from FAB pic >> ".
- Iterates through the first 35 bytes of FAB_TO_MPIC_ARRAY and prints each in hexadecimal format.
- Adds two blank lines for clarity (printline();).
Handles communication with the FAB PIC, sends a command, and waits for the response.
- 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:
- Attempts communication numof_times times.
- Sends the MPIC_TO_FAB_ARRAY data via the FAB UART.
- Waits for wait_time milliseconds, then calls CHECK_UART_INCOMING_FROM_FAB_PIC() to receive the response.
- Prints the received data using PRINT_RECIVED_COMMAND_FROM_FAB_PIC().
- 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.
Closes the FAB kill switch. Steps:
- Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command sequence [0xFA, 0xF1, 0xFB].
- Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to communicate the command.
- Verifies the response matches [0xFA, 0xF1, 0xFB].
- If valid, prints a success message.
- Otherwise, prints a failure message.
Closes the OBC (Onboard Computer) kill switch. Steps:
- Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF2, 0xFB].
- Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
- Verifies the response.
- If valid, toggles the hardware pin PIN_A4 to close the switch and prints success.
- Otherwise, prints failure.
Opens the FAB kill switch. Steps:
- Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF3, 0xFB].
- Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
- Verifies the response.
- If valid, prints success.
- Otherwise, prints failure.
Opens the OBC kill switch. Steps:
- Clears FAB_TO_MPIC_ARRAY and prepares MPIC_TO_FAB_ARRAY with the command [0xFA, 0xF4, 0xFB].
- Calls COMUNICATION_WITH_FAB_PIC_AND_WAIT_FOR_RESPONE() to send the command.
- 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.
- 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.
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.
These arrays handle data exchange:
- MPIC_TO_FAB_ARRAY: Stores outgoing data.
- FAB_TO_MPIC_ARRAY: Stores incoming data from the FAB PIC.
- Configures the UART port for communication.
- Debug communication is established via the PC stream.
- 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:
- MPic_Available()
- Returns the number of bytes available in the buffer.
- MPic_Read()
- Reads a byte from the buffer, reducing the byte counter.
- Resets counters when the buffer is empty.
Clears any array, setting all elements to zero. Description Function usage example
Prints a blank line for debug readability. Description Function usage example
- 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. Description Function usage example
- 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. Description Function usage example
- 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. Description Function usage example
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. Description Function usage example
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).
- 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. Description Function usage example
- Purpose: Outputs all received data from MBOSS in a human-readable format for debugging. Description Function usage example
- Purpose: Sends an acknowledgment to the communication PIC (CPIC) with board and mission numbers. Description Function usage example
- Purpose: Decodes the board number from a command and ensures it is valid. Description Function usage example
- TURN_ON_MISSIONBOSS: Activates the Mission Boss system. Description Function usage example
- TURN_OFF_MISSIONBOSS: Deactivates the Mission Boss system. Description Function usage example
Description Function usage example
These functions control access to the Subsystem Function Module (SFM) for the Mission Boss. Description Function usage example
Resets all APRS board flags when the mission is not active (MISSION_STATUS == 0). Description Function usage example
- Sends an APRS command to Mission Boss.
- Grants SFM access if the command is a data transfer request. Description Function usage example
#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)
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
Description Function usage example
Description Function usage example
Description Function usage example
Description Function usage example
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.
Function usage example
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
Description Function usage example
Description Function usage example
Description Function usage example
Description Function usage example
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. Description Function usage example
#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;
Description Function usage example
Description Function usage example
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.
- 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:
- Interrupt Handling:
- The SERIAL_ISR2 and SERIAL_ISR3 functions ensure efficient data collection from APRS and OBC without blocking execution.
- Buffer Management:
- Both APRS and OBC implement circular-like buffer handling to store and retrieve data.
- Flexibility:
- Uses #pin_select to dynamically assign UART pins, making it hardware-agnostic.
- SPI Commands:
- The SPI functions use standard flash memory commands for sector erasure and byte writing, ensuring compatibility with various memory chips.
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.
These registers control the configuration and state of the I/O pins of the microcontroller:
#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.
#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.
#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.
#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.
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.
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.
#byte EECON2 = 0xF7E
#byte OSCCON = 0xFD3
- EECON2 (0xF7E): Used for EEPROM read/write operations.
- OSCCON (0xFD3): Controls the oscillator configuration for the microcontroller.
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.
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. Description Function usage example
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. Description Function usage example
Controls the mission flash memory. Uses Pin_A2 as the chip select line and the MSN_FM SPI stream. Description Function usage example
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
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.
Same as above but for COM flash memory
COM_FM_SECTOR_ERASE(sector_address, sector_size, delay = 1000)
Description
Function usage example
Same as above but for MSN flash memory
MSN_FM_SECTOR_ERASE(unsigned int32 sector_address,char sector_size, unsigned int16 delay = 1000)
Description
Function usage example
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. Description Function usage example
same as above for com
#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
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
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);
#use rs232(baud=19200, parity=N, xmit=PIN_B7, bits=8, stream = PC, errors, force_sw ) //Debug PGD line
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
pin select and constant definitions
MAIN PIC uart interupt loop
just printing a line
using this function we can make any data array clear
#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
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 |
using this function external ADC temperature channels can be read 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
).
designed to read an analog signal from a specified ADC channel on a PIC microcontroller. 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).
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.
- 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.
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.
- 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.
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.
- 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.
this functions will check debug port incoming
this functions will print received comand prom bebug port
sending command to reset pic wait for respone we will try 10 times then we start antena deployment
kill switch handling
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.