aboutsummaryrefslogtreecommitdiff
path: root/apps/stepper2
diff options
context:
space:
mode:
authorTrygve Laugstøl <trygvis@inamo.no>2016-08-31 16:30:35 +0200
committerTrygve Laugstøl <trygvis@inamo.no>2016-08-31 16:30:35 +0200
commit07090d14e0dafc3b5b564a339cd615c2d9864125 (patch)
tree6d20c9168c3f059381981ff4cc8cb9774db4041d /apps/stepper2
parent7a3326e8687cd8c86750a2db8348ab34c261b426 (diff)
downloadstm32f103-playground-07090d14e0dafc3b5b564a339cd615c2d9864125.tar.gz
stm32f103-playground-07090d14e0dafc3b5b564a339cd615c2d9864125.tar.bz2
stm32f103-playground-07090d14e0dafc3b5b564a339cd615c2d9864125.tar.xz
stm32f103-playground-07090d14e0dafc3b5b564a339cd615c2d9864125.zip
o A couple of stepper motor driver applications.
Diffstat (limited to 'apps/stepper2')
-rw-r--r--apps/stepper2/CMakeLists.txt20
-rw-r--r--apps/stepper2/stepper2.cpp241
2 files changed, 261 insertions, 0 deletions
diff --git a/apps/stepper2/CMakeLists.txt b/apps/stepper2/CMakeLists.txt
new file mode 100644
index 0000000..67d76b0
--- /dev/null
+++ b/apps/stepper2/CMakeLists.txt
@@ -0,0 +1,20 @@
+add_executable(stepper2.elf stepper2.cpp
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/CMSIS/CM3/CoreSupport/core_cm3.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/CMSIS/CM3/DeviceSupport/ST/STM32F10x/system_stm32f10x.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/misc.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/stm32f10x_rcc.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/stm32f10x_spi.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/stm32f10x_gpio.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/stm32f10x_tim.c
+ ${STM32F10X_STDPERIPH_LIB}/Libraries/STM32F10x_StdPeriph_Driver/src/stm32f10x_dma.c
+ $<TARGET_OBJECTS:playground>
+ )
+
+target_include_directories(stepper2.elf PUBLIC
+ $<TARGET_PROPERTY:playground,INTERFACE_INCLUDE_DIRECTORIES>
+ )
+target_compile_definitions(stepper2.elf PUBLIC ${STM32F10X_STDPERIPH_DEFINES})
+target_link_libraries(stepper2.elf tinyprintf)
+
+set_target_properties(stepper2.elf PROPERTIES LINK_FLAGS "-nostartfiles -T${CMAKE_SOURCE_DIR}/cmake/stm32.ld")
+add_extra_commands(stepper2.elf)
diff --git a/apps/stepper2/stepper2.cpp b/apps/stepper2/stepper2.cpp
new file mode 100644
index 0000000..75603ac
--- /dev/null
+++ b/apps/stepper2/stepper2.cpp
@@ -0,0 +1,241 @@
+#include <stdint.h>
+#include <stm32f10x.h>
+#include <stm32f10x_rcc.h>
+#include <stm32f10x_gpio.h>
+#include <stm32f10x_dma.h>
+#include <stm32f10x_tim.h>
+#include <misc.h>
+
+#include "debug.h"
+#include "tinyprintf.h"
+#include "playground.h"
+
+extern "C"
+__attribute__((naked, used))
+void HardFault_Handler_C(uint32_t *hardfault_args) {
+ dbg_printf("r0 = 0x%08lx (%lu)\n", hardfault_args[0], hardfault_args[0]);
+ dbg_printf("r1 = 0x%08lx (%lu)\n", hardfault_args[1], hardfault_args[1]);
+ dbg_printf("r2 = 0x%08lx (%lu)\n", hardfault_args[2], hardfault_args[2]);
+ dbg_printf("r3 = 0x%08lx (%lu)\n", hardfault_args[3], hardfault_args[3]);
+ dbg_printf("r12 = 0x%08lx (%lu)\n", hardfault_args[4], hardfault_args[4]);
+ dbg_printf("lr = 0x%08lx (%lu)\n", hardfault_args[5], hardfault_args[5]);
+ dbg_printf("pc = 0x%08lx (%lu)\n", hardfault_args[6], hardfault_args[6]);
+ dbg_printf("psr = 0x%08lx (%lu)\n", hardfault_args[7], hardfault_args[7]);
+ dbg_printf("\n");
+
+ halt();
+}
+
+volatile bool dma_busy;
+
+struct motor_t {
+ GPIO_TypeDef *gpio_port;
+ const uint16_t ah;
+ const uint16_t al;
+};
+
+static struct motor_t motor_a = {
+ gpio_port: GPIOA,
+ ah : GPIO_Pin_0,
+ al : GPIO_Pin_1,
+};
+
+TIM_TypeDef *pwm_timer = TIM2;
+
+struct apb_bus_1 {
+};
+struct apb_bus_2 {
+};
+
+template<typename apb_bus>
+class apb_ctrl {
+public:
+ static void RCC_PeriphClockCmd(uint32_t peripherals, FunctionalState functionalState) {
+ RCC_PeriphClockCmd(apb_bus{}, peripherals, functionalState);
+ }
+
+private:
+ static void RCC_PeriphClockCmd(apb_bus_1, uint32_t peripherals, FunctionalState functionalState) {
+ RCC_APB1PeriphClockCmd(peripherals, functionalState);
+ }
+
+ static void RCC_PeriphClockCmd(apb_bus_2, uint32_t peripherals, FunctionalState functionalState) {
+ RCC_APB2PeriphClockCmd(functionalState, functionalState);
+ }
+
+public:
+ static void RCC_PeriphResetCmd(uint32_t peripherals, FunctionalState functionalState) {
+ RCC_PeriphResetCmd(apb_bus{}, peripherals, functionalState);
+ }
+
+private:
+ static void RCC_PeriphResetCmd(apb_bus_1, uint32_t peripherals, FunctionalState functionalState) {
+ RCC_APB1PeriphResetCmd(peripherals, functionalState);
+ }
+
+ static void RCC_PeriphResetCmd(apb_bus_2, uint32_t peripherals, FunctionalState functionalState) {
+ RCC_APB2PeriphResetCmd(functionalState, functionalState);
+ }
+};
+
+template<typename apb_bus, uint32_t peripheral>
+class apb_device {
+public:
+ static void reset() {
+ apb_ctrl<apb_bus>::RCC_PeriphResetCmd(peripheral, ENABLE);
+ apb_ctrl<apb_bus>::RCC_PeriphResetCmd(peripheral, DISABLE);
+ }
+
+ static void enable() {
+ apb_ctrl<apb_bus>::RCC_PeriphClockCmd(peripheral, ENABLE);
+ }
+
+ static void disable() {
+ apb_ctrl<apb_bus>::RCC_PeriphClockCmd(peripheral, DISABLE);
+ }
+};
+
+template<typename apb_bus, uint32_t peripheral>
+class gpio_port : public apb_device<apb_bus, peripheral> {
+public:
+};
+
+class gpio_port_a : public gpio_port<apb_bus_2, RCC_APB2Periph_GPIOA> {
+};
+
+class gpio_port_b : public gpio_port<apb_bus_2, RCC_APB2Periph_GPIOB> {
+};
+
+class gpio_port_c : public gpio_port<apb_bus_2, RCC_APB2Periph_GPIOC> {
+};
+
+class tim2 : public apb_device<apb_bus_1, RCC_APB1Periph_TIM2> {
+};
+
+void timer_init(TIM_TypeDef *tim, uint16_t period) {
+ RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
+// tim2::reset();
+// tim2::enable();
+
+ TIM_TimeBaseInitTypeDef timerInitStructure;
+ TIM_TimeBaseStructInit(&timerInitStructure);
+ timerInitStructure.TIM_Prescaler = 0;
+ timerInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
+ timerInitStructure.TIM_Period = period;
+ timerInitStructure.TIM_ClockDivision = TIM_CKD_DIV1;
+ timerInitStructure.TIM_RepetitionCounter = 0;
+ TIM_TimeBaseInit(tim, &timerInitStructure);
+}
+
+void pwm_control(TIM_TypeDef *tim, uint16_t pulse) {
+ TIM_OCInitTypeDef outputChannelInit;
+ outputChannelInit.TIM_OCMode = TIM_OCMode_PWM1;
+ outputChannelInit.TIM_Pulse = pulse;
+ outputChannelInit.TIM_OutputState = TIM_OutputState_Enable;
+ outputChannelInit.TIM_OCPolarity = TIM_OCPolarity_High;
+
+ TIM_OC1Init(tim, &outputChannelInit);
+ TIM_OC1PreloadConfig(tim, TIM_OCPreload_Enable);
+
+ TIM_OC2Init(tim, &outputChannelInit);
+ TIM_OC2PreloadConfig(tim, TIM_OCPreload_Enable);
+}
+
+void gpio_init(GPIO_TypeDef* gpio_port, uint16_t pins, bool use_af) {
+ GPIO_InitTypeDef init;
+ GPIO_StructInit(&init);
+ init.GPIO_Mode = use_af ? GPIO_Mode_AF_PP : GPIO_Mode_Out_PP;
+ init.GPIO_Pin = pins;
+ init.GPIO_Speed = GPIO_Speed_50MHz;
+ GPIO_Init(gpio_port, &init);
+}
+
+volatile bool run = true;
+
+bool use_gpio = false;
+
+int main() {
+ SystemInit();
+
+ init_printf(nullptr, dbg_putc);
+
+ dbg_printf("stepper2\n");
+
+ NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0);
+ NVIC_InitTypeDef NVIC_InitStruct = {
+ NVIC_IRQChannel: DMA1_Channel3_IRQn,
+ NVIC_IRQChannelPreemptionPriority: 0,
+ NVIC_IRQChannelSubPriority: 0,
+ NVIC_IRQChannelCmd: ENABLE,
+ };
+ NVIC_Init(&NVIC_InitStruct);
+ NVIC_EnableIRQ(DMA1_Channel3_IRQn);
+
+ RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO
+ | RCC_APB2Periph_GPIOA
+ | RCC_APB2Periph_GPIOB
+ | RCC_APB2Periph_GPIOC
+ | RCC_APB2Periph_ADC1
+ | RCC_APB2Periph_ADC2
+ | RCC_APB2Periph_ADC3
+ | RCC_APB2Periph_TIM1,
+ ENABLE);
+
+ const uint16_t period = 1000;
+
+ const int count_max = 10000;
+ int count = count_max - 1;
+
+ uint16_t pulse = 500;
+
+ gpio_init(motor_a.gpio_port, motor_a.ah | motor_a.al, !use_gpio);
+ timer_init(pwm_timer, period);
+ pwm_control(pwm_timer, pulse);
+
+ TIM_ARRPreloadConfig(pwm_timer, ENABLE);
+ TIM_Cmd(pwm_timer, ENABLE);
+
+ while (run) {
+ static bool on = false;
+
+ if (use_gpio) {
+ if (on) {
+ GPIO_SetBits(motor_a.gpio_port, motor_a.ah);
+ GPIO_SetBits(motor_a.gpio_port, motor_a.al);
+ } else {
+ GPIO_ResetBits(motor_a.gpio_port, motor_a.ah);
+ GPIO_ResetBits(motor_a.gpio_port, motor_a.al);
+ }
+ on = !on;
+ } else {
+ count++;
+
+ if (count == count_max) {
+ count = 0;
+
+ pulse++;
+
+ if (pulse > period) {
+ pulse = 0;
+ }
+
+ pwm_timer->CCR1 = pulse;
+ pwm_timer->CCR2 = period - pulse;
+ }
+ }
+ }
+
+ return 0;
+}
+
+extern "C"
+void DMA1_Channel3_IRQHandler() {
+ ITStatus tc = DMA_GetITStatus(DMA1_IT_TC3);
+ ITStatus te = DMA_GetITStatus(DMA1_IT_TE3);
+ ITStatus ht = DMA_GetITStatus(DMA1_IT_HT3);
+
+ if (tc || te || ht) {
+ DMA_ClearITPendingBit(DMA1_IT_GL3);
+ dma_busy = false;
+ }
+}