stm32 hal库驱动ULN2003+28BYJ-48步进电机程序

一、特点

ULN2003是高耐压、大电流达林顿陈列,由七个硅NPN达林顿管组成。该电路的特点如下:ULN2003的每一对达林顿都串联一个2.7K的基极电阻,在5V的工作电压下它能与TTL和CMOS电路直接相连,可以直接处理原先需要标准逻辑缓冲器来处理的数据。 ULN2003工作电压高,工作电流大,灌电流可达500mA,并且能够在关态时承受50V的电压,输出还可以在高负载电流并行运行。ULN2003采用DIP—16或SOP—16塑料封装。

2b5c22810de047ba941cb4181943efc0.jpg

7ac93c880cf84d448b1cea1cfcdc503e.jpg

d9f7e26ef205458d972fb2b989d8592b.jpg

二、原理

a2ac1aed483e44a99c3686afed1b5169.jpg 28BYJ-48步进电机参数

74c51b262c2b435ca759b693a6876a04.jpg

A~D为相线

12268711b43a40629a9d95f452997db7.jpg

根据电机拆解结果可知,一共被分成了32份,平均相间11.25°(即360°/32。而32是通过8*4得到,因为一共ABCD 4组,一组8个齿)

6736710669b1480e9f588587bfed4002.jpg

 减速比:

58dfa1da8d8f44d08b8f6ba853d998b4.jpg

stm32驱动代码

时钟72MHZ

ec7ce5288fc74604a87a8c2e6fca2925.jpg

 28BYJ-48初始化

edcb6794fa834b8db25024f14322cddb.jpg

 Main.c

378bc496065f4a3f9159f0a6012d6430.jpg

 

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
电机驱动的实现可以采用28BYJ-48步电机驱动板,驱动板上自带 ULN2003 驱动芯片,可以方便地使用基于 STM32 的控制器行控制。在使用 STM32 HAL 驱动电机时,可以按照以下步骤行: 1. 初始化 GPIO 端口和定时器。在 HAL 中,可以使用 HAL_GPIO_Init 和 HAL_TIM_Base_Init 函数行初始化。 2. 配置 GPIO 端口的输出模式,设置为推挽输出模式。可以使用 HAL_GPIO_WritePin 函数将 GPIO 端口输出设置为高电平或低电平,从而控制步电机的转向。 3. 配置定时器的计数值和分频值,以控制步电机的速度。可以使用 HAL_TIM_Base_Start_IT 函数启动定时器,以产生中断并控制步电机的转动。 4. 在定时器中断服务函数中,根据步电机的转向和步长控制 GPIO 端口的输出状态,从而实现步电机的正转、反转和速度控制。 下面是一个示例代码,可以实现控制28BYJ-48步电机的正转、反转和速度控制功能: ```c #include "stm32f1xx_hal.h" #define STEPPER_PORT GPIOA #define STEPPER_IN1_PIN GPIO_PIN_0 #define STEPPER_IN2_PIN GPIO_PIN_1 #define STEPPER_IN3_PIN GPIO_PIN_2 #define STEPPER_IN4_PIN GPIO_PIN_3 #define STEPPER_SPEED 1000 TIM_HandleTypeDef htim2; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_TIM2_Init(void); void HAL_TIM_MspPostInit(TIM_HandleTypeDef *htim); void stepper_forward(void); void stepper_reverse(void); void stepper_stop(void); int main(void) { HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_TIM2_Init(); HAL_TIM_Base_Start_IT(&htim2); while (1) { stepper_forward(); HAL_Delay(2000); stepper_reverse(); HAL_Delay(2000); } } void stepper_forward(void) { HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN1_PIN, GPIO_PIN_SET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_SET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN4_PIN, GPIO_PIN_RESET); } void stepper_reverse(void) { HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN1_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_SET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN4_PIN, GPIO_PIN_SET); } void stepper_stop(void) { HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN1_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN4_PIN, GPIO_PIN_RESET); } void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim) { if (htim->Instance == TIM2) { static int step = 0; static int dir = 1; if (dir == 1) { switch (step) { case 0: stepper_forward(); break; case 1: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN1_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_SET); break; case 2: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN4_PIN, GPIO_PIN_SET); break; case 3: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_SET); break; default: break; } } else { switch (step) { case 0: stepper_reverse(); break; case 1: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_SET); break; case 2: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN1_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN2_PIN, GPIO_PIN_SET); break; case 3: HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN3_PIN, GPIO_PIN_RESET); HAL_GPIO_WritePin(STEPPER_PORT, STEPPER_IN4_PIN, GPIO_PIN_SET); break; default: break; } } step += dir; if (step >= 4) { step = 0; } else if (step < 0) { step = 3; } TIM2->ARR = STEPPER_SPEED; } } void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct; RCC_ClkInitTypeDef RCC_ClkInitStruct; __HAL_RCC_PWR_CLK_ENABLE(); __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1); RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.HSEState = RCC_HSE_ON; RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) { Error_Handler(); } HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq() / 1000); HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK); HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0); } static void MX_TIM2_Init(void) { TIM_MasterConfigTypeDef sMasterConfig; htim2.Instance = TIM2; htim2.Init.Prescaler = 7200 - 1; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = STEPPER_SPEED; htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; if (HAL_TIM_Base_Init(&htim2) != HAL_OK) { Error_Handler(); } sMasterConfig.MasterOutputTrigger = TIM_TRGO_RESET; sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK) { Error_Handler(); } HAL_TIM_MspPostInit(&htim2); } static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct; __HAL_RCC_GPIOA_CLK_ENABLE(); GPIO_InitStruct.Pin = STEPPER_IN1_PIN | STEPPER_IN2_PIN | STEPPER_IN3_PIN | STEPPER_IN4_PIN; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(STEPPER_PORT, &GPIO_InitStruct); } void HAL_TIM_MspPostInit(TIM_HandleTypeDef *htim) { GPIO_InitTypeDef GPIO_InitStruct; if (htim->Instance == TIM2) { __HAL_RCC_GPIOA_CLK_ENABLE(); /**TIM2 GPIO Configuration PA1 ------> TIM2_CH2 */ GPIO_InitStruct.Pin = GPIO_PIN_1; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; GPIO_InitStruct.Alternate = GPIO_AF1_TIM2; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } } void Error_Handler(void) { while (1) { } } #ifdef USE_FULL_ASSERT void assert_failed(uint8_t *file, uint32_t line) { } #endif ``` 需要注意的是,步电机的控制需要按照一定的步序行,因此在定时器中断服务函数中需要控制每一步的输出状态,从而实现步电机的正转、反转和速度控制。在上述代码中,步电机的控制采用了半步控制方式,可以根据需要行调整。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

嵌入式冰箱

你的鼓励是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值