Merge branch 'main' into dev

This commit is contained in:
Ondrej Hladuvka 2025-05-13 17:43:45 +03:00
commit 48fd2e6aed
9 changed files with 393 additions and 6 deletions

View File

@ -19,4 +19,34 @@ uint32_t __attribute__((naked)) CPU_cpsie(void)
// naked attribute).
//
return(ret);
}
}
uint32_t __attribute__((naked)) CPU_cpsid(void)
{
uint32_t ret;
//
// Read PRIMASK and disable interrupts.
//
__asm(" mrs r0, PRIMASK\n"
" cpsid i\n"
" bx lr\n"
: "=r" (ret));
//
// The return is handled in the inline assembly, but the compiler will
// still complain if there is not an explicit return here (despite the fact
// that this does not result in any code being produced because of the
// naked attribute).
//
return(ret);
}
void __attribute__((naked)) CPU_wfi(void)
{
//
// Wait for the next interrupt.
//
__asm(" wfi\n"
" bx lr\n");
}

View File

@ -4,5 +4,7 @@
#include <stdint.h>
uint32_t CPU_cpsie(void);
uint32_t CPU_cpsid(void);
void CPU_wfi(void);
#endif /* CPU_H */

56
common/input_capture.c Normal file
View File

@ -0,0 +1,56 @@
#include "input_capture.h"
#include "inc/msp432p401r.h"
void (*capture_task)(uint16_t time);
// Initialize Timer A0 in edge time mode to request interrupts on
// the rising edge of P7.3 (TA0CCP0). The interrupt service routine
// acknowledges the interrupt and calls a user function.
void TimerA0CaptureInit(void (*task)(uint16_t time)) {
capture_task = task;
P7->SEL0 |= 0x08;
P7->SEL1 &= ~0x08; // configure P7.3 as TA0CCP0
P7->DIR &= ~0x08;
TIMER_A0->CTL &= ~0x30; // halt Timer A0
TIMER_A0->CTL = 0x200; // SMCLK
TIMER_A0->CCTL[0] = 0x4910; // capture on rising edge, synchronous capture, capture mode, interrupt enabled
TIMER_A0->EX0 &= ~0x7; // input clock divider /1
NVIC->IP[2] = (NVIC->IP[2] & 0xFFFFFF00) | 0x00000002; // priority 2
// interrupts enabled in the main program after all devices initialized
NVIC->ISER[0] = 0x100; // enable interrupt 8 in NVIC
TIMER_A0->CTL |= 0x24; // reset and start Timer A0 in continuous up mode
}
void TA0_0_IRQHandler(void) {
TIMER_A0->CCTL[0] &= ~0x1; // acknowledge capture/compare interrupt
capture_task(TIMER_A0->CCR[0]); // execute user task
}
// Left Encoder A connected to P10.5 (J5)
// Right Encoder A connected to P10.4 (J5)
// user function, time is up-counting timer value when edge occurred in units of 0.083 usec
// called when P10.4/.5 (TA3CCP0/1) edge occurs
void (*capture_task0)(uint16_t time);
void (*capture_task1)(uint16_t time);
// Initialize Timer A3 in edge time mode to request interrupts on
// the rising edges of P10.4 (TA3CCP0) and P10.5 (TA3CCP1).
// Interrupt service routines acknowledge the interrupt and call a user function.
// Assumes: low-speed subsystem master clock is 12 MHz
void TimerA3CaptureInit(void (*task0)(uint16_t time),
void (*task1)(uint16_t time)) {
// write this code
}
// The rising edge of P10.4 will cause an interrupt
void TA3_0_IRQHandler(void) {
// write this code
}
// The rising edge of P10.5 will cause an interrupt
void TA3_N_IRQHandler(void) {
// write this code
}

9
common/input_capture.h Normal file
View File

@ -0,0 +1,9 @@
#ifndef INPUT_CAPTURE_H
#define INPUT_CAPTURE_H
#include <stdint.h>
void TimerA3CaptureInit(void (*task0)(uint16_t time),
void (*task1)(uint16_t time));
#endif /* INPUT_CAPTURE */

View File

@ -66,8 +66,8 @@ void MotorInit(void) {
// Stop the motors, power down the drivers, and
// set the PWM speed control to 0% duty cycle.
void MotorStop(void) {
P5->OUT &= ~(LEFT_DIR_PIN | RIGHT_DIR_PIN); // set direction - PH low
P3->OUT |= (LEFT_SLEEP_PIN | RIGHT_SLEEP_PIN); // wake up motors
P3->OUT &= ~(LEFT_SLEEP_PIN | RIGHT_SLEEP_PIN); // SLEEP pins low
P2->OUT &= ~(LEFT_SLEEP_PIN | RIGHT_SLEEP_PIN); // EN pins low
SetPWMDutyLeftMotor(0);
SetPWMDutyRightMotor(0);
@ -78,6 +78,7 @@ void MotorStop(void) {
void MotorForward(uint16_t left_duty, uint16_t right_duty) {
P5->OUT &= ~(LEFT_DIR_PIN | RIGHT_DIR_PIN); // set direction - PH low
P3->OUT |= (LEFT_SLEEP_PIN | RIGHT_SLEEP_PIN); // wake up motors
P2->OUT |= (LEFT_PWM_PIN | RIGHT_PWM_PIN);
SetPWMDutyRightMotor(right_duty);
SetPWMDutyLeftMotor(left_duty);
@ -86,17 +87,33 @@ void MotorForward(uint16_t left_duty, uint16_t right_duty) {
// Left and right wheels backward with the given duty cycles.
// Duty cycle is 0 to 14,998.
void MotorBackward(uint16_t left_duty, uint16_t right_duty) {
// write this code
P5->OUT |= (LEFT_DIR_PIN | RIGHT_DIR_PIN); // set direction - PH high
P3->OUT |= (LEFT_SLEEP_PIN | RIGHT_SLEEP_PIN); // wake up motors
P2->OUT |= (LEFT_PWM_PIN | RIGHT_PWM_PIN);
SetPWMDutyRightMotor(right_duty);
SetPWMDutyLeftMotor(left_duty);
}
// Left wheel backward, right wheel forward with the given duty cycles.
// Duty cycle is 0 to 14,998.
void MotorLeft(uint16_t left_duty, uint16_t right_duty) {
// write this code
P5->OUT &= ~(LEFT_DIR_PIN); // set direction
P3->OUT |= (LEFT_SLEEP_PIN); // wake up motor
P2->OUT |= (LEFT_PWM_PIN | RIGHT_PWM_PIN);
SetPWMDutyRightMotor(right_duty);
SetPWMDutyLeftMotor(left_duty);
}
// Left wheel forward, right wheel backward with the given duty cycles.
// Duty cycle is 0 to 14,998.
void MotorRight(uint16_t left_duty, uint16_t right_duty) {
// write this code
P5->OUT &= ~(RIGHT_DIR_PIN); // set direction - PH high
P3->OUT |= (RIGHT_SLEEP_PIN); // wake up motors
P2->OUT |= (LEFT_PWM_PIN | RIGHT_PWM_PIN);
SetPWMDutyRightMotor(right_duty);
SetPWMDutyLeftMotor(left_duty);
}

41
common/tachometer.c Normal file
View File

@ -0,0 +1,41 @@
#include "tachometer.h"
#include "input_capture.h"
#include "inc/msp432p401r.h"
#define P5_0_IN (*((volatile uint8_t *)(0x42098800)))
#define P5_2_IN (*((volatile uint8_t *)(0x42098808)))
uint16_t tach_r_time1, tach_r_time2, tach_l_time1, tach_l_time2;
// incremented with every step forward, decremented with every step backward
int tach_r_steps, tach_l_steps;
TachDirection_t tach_r_dir, tach_l_dir;
// Right Encoder B connected to P5.0 (J2.13)
// When high, increase r_steps and set dir forward
// When low, decrease r_steps and set dir reverse
void TachometerRightInt(uint16_t current_time) {
tach_r_time1 = tach_r_time2;
tach_r_time2 = current_time;
// write this code
}
// Left Encoder B connected to P5.2 (J2.12)
// When high, increase l_steps and set dir forward
// When low, decrease l_steps and set dir reverse
void TachometerLeftInt(uint16_t current_time) {
tach_l_time1 = tach_l_time2;
tach_l_time2 = current_time;
// write this code
}
// Initialize P5.0 and P5.2 and make them GPIO inputs,
// which will be used to determine the direction of rotation.
// Initialize the input capture interface, which
// will be used to measure the speed of rotation.
void TachometerInit(void) {
// write this code
}

15
common/tachometer.h Normal file
View File

@ -0,0 +1,15 @@
#ifndef TACHOMETER_H
#define TACHOMETER_H
#include <stdint.h>
// Specifies the direction of the motor rotation, relative to the front of the robot
typedef enum TachDirection {
Reverse = -1,
Stopped,
Forward,
} TachDirection_t;
void TachometerInit(void);
#endif /* TACHOMETER_H */

77
lab6/Makefile Normal file
View File

@ -0,0 +1,77 @@
CC = arm-none-eabi-gcc
# The location of the C compiler
# ARMGCC_ROOT is used by some makefiles that need to know where the compiler
# is installed.
ARMGCC_BIN := ${shell which ${CC}}
ifeq (${shell test -h ${ARMGCC_BIN} && echo "true" || echo "false"}, true)
ARMGCC_ROOT := ${shell dirname ${shell readlink ${ARMGCC_BIN}}}/..
else
ARMGCC_ROOT := ${shell dirname ${ARMGCC_BIN}}/..
endif
# Search for source files that are not in this directory
VPATH = ../common/
OBJECTS = main.o timer_a1.o motor.o pwm.o input_capture.o tachometer.o cpu.o clock.o system.o startup.o syscalls.o
NAME = lab
CFLAGS = -I.. -I../inc -I../common \
-D__MSP432P401R__ \
-DDeviceFamily_MSP432P401x \
-mcpu=cortex-m4 \
-march=armv7e-m \
-mthumb \
-std=c99 \
-mfloat-abi=hard \
-mfpu=fpv4-sp-d16 \
-ffunction-sections \
-fdata-sections \
-gstrict-dwarf \
-Wall \
-I$(ARMGCC_ROOT)/arm-none-eabi/include/newlib-nano \
-I$(ARMGCC_ROOT)/arm-none-eabi/include
LFLAGS = -Wl,-T,../config.lds \
-Wl,-Map,$(NAME).map \
-march=armv7e-m \
-mthumb \
-mfloat-abi=hard \
-mfpu=fpv4-sp-d16 \
-static \
-Wl,--gc-sections \
-lgcc \
-lc \
-lm \
-lnosys \
--specs=nano.specs
all: $(NAME).out
@echo "SUCCESS: Compilation completed successfully."
%.o: %.c
@$(CC) $(CFLAGS) -g $< -c -o $@
cpu.o: ../common/cpu.c
@$(CC) $(CFLAGS) $< -c -o $@
clock.o: ../common/clock.c
@$(CC) $(CFLAGS) $< -c -o $@
system.o: ../system.c
@$(CC) $(CFLAGS) $< -c -o $@
startup.o: ../startup.c
@$(CC) $(CFLAGS) $< -c -o $@
syscalls.o: ../syscalls.c
@$(CC) $(CFLAGS) $< -c -o $@
$(NAME).out: $(OBJECTS)
@$(CC) $(OBJECTS) $(LFLAGS) -o $(NAME).out
clean: # Redirecting both the stderr and stdout to /dev/null
@rm -f $(OBJECTS) > /dev/null 2>&1
@rm -f $(NAME).out > /dev/null 2>&1
@rm -f $(NAME).map > /dev/null 2>&1

140
lab6/main.c Normal file
View File

@ -0,0 +1,140 @@
#include <stdint.h>
#include <stdbool.h>
#include "cpu.h"
#include "motor.h"
#include "pwm.h"
#include "clock.h"
#include "input_capture.h"
#include "timer_a1.h"
#include "tachometer.h"
#include "inc/msp432p401r.h"
void RedLEDInit(void) {
P1->SEL0 &= ~0x01;
P1->SEL1 &= ~0x01; // 1) configure P1.0 as GPIO
P1->DIR |= 0x01; // 2) make P1.0 out
}
// bit-banded address
#define REDLED (*((volatile uint8_t *)(0x42098040)))
void ColorLEDInit(void) {
P2->SEL0 &= ~0x07;
P2->SEL1 &= ~0x07; // 1) configure P2.2-P2.0 as GPIO
P2->DIR |= 0x07; // 2) make P2.2-P2.0 out
P2->DS |= 0x07; // 3) activate increased drive strength
P2->OUT &= ~0x07; // all LEDs off
}
// bit-banded addresses
#define BLUEOUT (*((volatile uint8_t *)(0x42098068)))
#define GREENOUT (*((volatile uint8_t *)(0x42098064)))
#define REDOUT (*((volatile uint8_t *)(0x42098060)))
uint16_t period0, period1; // (1/SMCLK) units = 83.3 ns units
uint16_t first0, first1; // Timer A3 first edge, P10.4/.5
bool done0, done1; // set each rising
// max period is (2^16-1)*83.3 ns = 5.4612 ms
// min period determined by time to run ISR, which is about 1 us
void PeriodMeasure0(uint16_t time) {
REDOUT ^= 1;
period0 = time - first0; // 83.3 ns resolution
first0 = time; // setup for next
done0 = true;
}
// max period is (2^16-1)*83.3 ns = 5.4612 ms
// min period determined by time to run ISR, which is about 1 us
void PeriodMeasure1(uint16_t time) {
REDLED ^= 1;
period1 = time - first1; // 83.3 ns resolution
first1 = time; // setup for next
done1 = true;
}
int main_period_measure(void) {
CPU_cpsid(); // Disable interrupts
ClockInit48MHz(); // 48 MHz clock; 12 MHz Timer A clock
RedLEDInit();
ColorLEDInit();
PWMInitMotor(?);
MotorInit();
TimerA3CaptureInit(&PeriodMeasure0, &PeriodMeasure1);
MotorForward(3751, 3751); // 50%
CPU_cpsie(); // Enable interrupts
CPU_wfi(); // Wait for interrupt
return 0;
}
uint16_t speed_buf[500]; // RPM, radius 0.035m wheels
uint32_t period_buf[500]; // 1/12MHz = 0.083 usec
uint32_t duty; // 0 to 15000
uint32_t time; // in 0.01 sec
void Collect(void) {
GREENOUT ^= 1;
if (!done0) period0 = 0xFFFF - 1; // stopped
if (!done1) period1 = 0xFFFF - 1;
done0 = done1 = false; // set on subsequent
switch (time) {
case 0:
duty = 1875; // 25%
MotorForward(duty, duty);
break;
case 100:
duty = 3751; // 50%
MotorForward(duty, duty);
break;
case 200:
duty = 5626; // 75%
MotorForward(duty, duty);
break;
case 300:
duty = 3751; // 50%
MotorForward(duty, duty);
break;
case 400:
duty = 1875; // 25%
MotorForward(duty, duty);
break;
case 500:
MotorStop();
TIMER_A1->CTL &= ~0x30; // Halt timer A1
break;
default:
speed_buf[time] = 2000000 / period0;
period_buf[time] = period0;
break;
}
++time;
}
int main_collect(void) {
CPU_cpsid(); // Disable interrupts
ClockInit48MHz(); // 48 MHz clock; 12 MHz Timer A clock
RedLEDInit();
ColorLEDInit();
PWMInitMotor(?);
MotorInit();
TimerA3CaptureInit(&PeriodMeasure0, &PeriodMeasure1);
TimerA1Init(&Collect, 5000); // 100 Hz
CPU_cpsie(); // Enable interrupts
CPU_wfi(); // Wait for interrupt
return 0;
}
int main_tach(void) {
CPU_cpsid(); // Disable interrupts
ClockInit48MHz();
PWMInitMotor(?);
MotorInit();
TachometerInit();
MotorForward(2000, 2000); // confirm if rotating forward
//MotorBackward(2000, 2000); // confirm if rotating backward
CPU_cpsie(); // Enable interrupts
CPU_wfi(); // Wait for interrupt
return 0;
}