在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

用TIM的PWM輸出模式寫一個步進電機的Stepper庫函數

冬至子 ? 來源:Vulcan Matrix ? 作者:Vulcan ? 2023-07-24 14:36 ? 次閱讀

這是之前寫平衡小車時自己用TIM的PWM輸出模式寫了一個步進電機的Stepper庫函數。

1

調用順序

圖片

1.1

init函數

圖片

圖片

1.2

begin函數

圖片

1.3

setSpeed函數

圖片

圖片

2

Stepper類結構

圖片

3

TIM結構框圖

Stm32手冊中的結構框圖很重要,只要理解了外設的運行邏輯,按照邏輯一步一步給寄存器設值就可以讓外設按我們的要求運行。

圖片

圖片

#ifndef __STEPPER_H
#define __STEPPER_H


#include "peripheral.h"
#include "math.h"


#ifdef __cplusplus
extern "C"
{
#endif


    enum DIRCTION
    {
        POS,
        INV
    };


    class Stepper
    {
    private:
        /* data */
        uint16_t TIMx_prescaler = 0;
        uint32_t TIMx_freq = 0;
        TIM_TypeDef *TIMx;
        uint32_t Channel;
        float speed;


    public:
        Stepper(TIM_TypeDef *TIMx, uint32_t Channel);
        ~Stepper();
        void init();
        void gpio_init();
        void begin();
        void stop();


        void setDirection(DIRCTION dir);
        void setFreq(uint16_t freq);
        void setSpeed(float speed);


        float getSpeed();
    };


    extern Stepper Stepper_left;
    extern Stepper Stepper_right;


#ifdef __cplusplus
}
#endif


#endif
#include "Stepper.h"


Stepper Stepper_left(TIM1, LL_TIM_CHANNEL_CH1);
Stepper Stepper_right(TIM2, LL_TIM_CHANNEL_CH2);


Stepper::Stepper(TIM_TypeDef *TIMx, uint32_t Channel)
{
    this- >TIMx = TIMx;
    this- >Channel = Channel;
}


Stepper::~Stepper()
{
}


void Stepper::init()
{
    //開定時器外設時鐘
    if (TIMx == TIM1)
    {
        LL_APB2_GRP1_EnableClock(LL_APB2_GRP1_PERIPH_TIM1);
        //設置預分頻器
        LL_TIM_SetPrescaler(TIMx, 90);
        TIMx_freq = 90000000;
        TIMx_prescaler = 90;
    }


    if (TIMx == TIM2)
    {
        LL_APB1_GRP1_EnableClock(LL_APB1_GRP1_PERIPH_TIM2);
        //設置預分頻器
        LL_TIM_SetPrescaler(TIMx, 45);
        TIMx_freq = 45000000;
        TIMx_prescaler = 45;
    }


    //定時器選擇時鐘源
    LL_TIM_SetClockSource(TIMx, LL_TIM_CLOCKSOURCE_INTERNAL);


    //設置自動重載寄存器
    LL_TIM_SetAutoReload(TIMx, 2000 - 1);
    //設置計數方向
    LL_TIM_SetCounterMode(TIMx, LL_TIM_COUNTERMODE_CENTER_UP);
    //使能自動重載預裝載
    LL_TIM_EnableARRPreload(TIMx);


    if (Channel == LL_TIM_CHANNEL_CH1 || Channel == LL_TIM_CHANNEL_CH1N)
    {
        //設置比較值
        LL_TIM_OC_SetCompareCH1(TIMx, 1000 - 1);
        //設置成PWM模式
        LL_TIM_OC_SetMode(TIMx, LL_TIM_CHANNEL_CH1, LL_TIM_OCMODE_PWM1);


        //設置捕獲/比較寄存器值
        LL_TIM_OC_EnablePreload(TIMx, LL_TIM_CHANNEL_CH1);
    }


    if (Channel == LL_TIM_CHANNEL_CH2 || Channel == LL_TIM_CHANNEL_CH2N)
    {
        //設置比較值
        LL_TIM_OC_SetCompareCH2(TIMx, 1000 - 1);
        //設置成PWM模式
        LL_TIM_OC_SetMode(TIMx, LL_TIM_CHANNEL_CH2, LL_TIM_OCMODE_PWM1);


        //設置捕獲/比較寄存器值
        LL_TIM_OC_EnablePreload(TIMx, LL_TIM_CHANNEL_CH2);
    }

    //設置輸出極性
    LL_TIM_OC_SetPolarity(TIMx, Channel, LL_TIM_OCPOLARITY_HIGH);
    //使能輸出
    LL_TIM_EnableAllOutputs(TIMx);
    LL_TIM_CC_EnableChannel(TIMx, Channel);


    //GPIO初始化
    gpio_init();
}


void Stepper::gpio_init()
{
    if (TIMx == TIM1)
    {
        //開啟GPIO時鐘
        LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOA);


        //GPIO選為AF
        //M1-DIR
        LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
        GPIO_InitStruct.Mode = LL_GPIO_MODE_OUTPUT;
        GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
        GPIO_InitStruct.Pin = LL_GPIO_PIN_9;
        GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
        GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_MEDIUM;
        LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
        //M1-STEP
        GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
        GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
        GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
        GPIO_InitStruct.Pin = LL_GPIO_PIN_8;
        GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
        GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_MEDIUM;
        LL_GPIO_Init(GPIOA, &GPIO_InitStruct);
    }


    if (TIMx == TIM2)
    {
        //開啟GPIO時鐘
        LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOB);
        LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_GPIOC);


        //GPIO選為AF
        //M2-DIR
        LL_GPIO_InitTypeDef GPIO_InitStruct = {0};
        GPIO_InitStruct.Mode = LL_GPIO_MODE_OUTPUT;
        GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
        GPIO_InitStruct.Pin = LL_GPIO_PIN_7;
        GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
        GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_MEDIUM;
        LL_GPIO_Init(GPIOC, &GPIO_InitStruct);   
        //M2-STEP
        GPIO_InitStruct.Alternate = LL_GPIO_AF_1;
        GPIO_InitStruct.Mode = LL_GPIO_MODE_ALTERNATE;
        GPIO_InitStruct.OutputType = LL_GPIO_OUTPUT_PUSHPULL;
        GPIO_InitStruct.Pin = LL_GPIO_PIN_3;
        GPIO_InitStruct.Pull = LL_GPIO_PULL_NO;
        GPIO_InitStruct.Speed = LL_GPIO_SPEED_FREQ_MEDIUM;
        LL_GPIO_Init(GPIOB, &GPIO_InitStruct);
    }
}


void Stepper::begin()
{
    LL_TIM_EnableCounter(TIMx);
}


void Stepper::stop()
{
    LL_TIM_DisableCounter(TIMx);
}


void Stepper::setDirection(DIRCTION dir)
{
    if (dir == INV)
    {
        if (TIMx == TIM1)
        {
            LL_GPIO_WriteOutputPort(GPIOB, LL_GPIO_ReadOutputPort(GPIOB) & (~LL_GPIO_PIN_4));
        }
        if (TIMx == TIM2)
        {
            LL_GPIO_WriteOutputPort(GPIOB, LL_GPIO_ReadOutputPort(GPIOB) & (~LL_GPIO_PIN_10));
        }
    }
    if (dir == POS)
    {
        if (TIMx == TIM1)
        {
            LL_GPIO_WriteOutputPort(GPIOB, LL_GPIO_ReadOutputPort(GPIOB) | (LL_GPIO_PIN_4));
        }
        if (TIMx == TIM2)
        {
            LL_GPIO_WriteOutputPort(GPIOB, LL_GPIO_ReadOutputPort(GPIOB) | (LL_GPIO_PIN_10));
        }
    }
}


void Stepper::setFreq(uint16_t freq)
{
    if (freq == 0)
    {
        LL_TIM_DisableCounter(TIMx);
        return;
    }
    else if (!LL_TIM_IsEnabledCounter(TIMx))
    {
        LL_TIM_EnableCounter(TIMx);
    }


    uint32_t ARR_t = TIMx_freq / TIMx_prescaler / freq;


    if (ARR_t > 65535) //觸碰上限——頻率過低
    {
        TIMx_prescaler *= 10; //提高分頻比
    }


    if (ARR_t < 2) //觸碰下限——頻率過高
    {
        TIMx_prescaler /= 10; //降低分頻比
    }


    LL_TIM_SetPrescaler(TIMx, TIMx_prescaler);


    ARR_t = TIMx_freq / TIMx_prescaler / freq;


    LL_TIM_SetAutoReload(TIMx, ARR_t);


    uint16_t CCR_t = ARR_t * 0.5;


    if (Channel == LL_TIM_CHANNEL_CH1)
    {
        LL_TIM_OC_SetCompareCH1(TIMx, CCR_t);
    }
    if (Channel == LL_TIM_CHANNEL_CH2)
    {
        LL_TIM_OC_SetCompareCH2(TIMx, CCR_t);
    }
}

/**
 * @brief 設置轉速
 * 
 * @param speed 轉速-單位(度/秒)
 */
void Stepper::setSpeed(float speed)
{
    //判斷速度方向
    DIRCTION _dir = POS;
    if (speed != abs(speed))
    {
        _dir = INV;
        speed = abs(speed);
    }
    setDirection(_dir);


    //速度限幅
    if (speed > 5000)
        speed = 5000;
    if (speed < 10)
        speed = 10;


    this- >speed = speed;


    //將速度轉化為定時器頻率
    uint16_t _freq = speed / 1.8;
    setFreq(_freq);
}


float Stepper::getSpeed()
{
    return speed;
}
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 寄存器
    +關注

    關注

    31

    文章

    5359

    瀏覽量

    120833
  • STM32
    +關注

    關注

    2270

    文章

    10918

    瀏覽量

    356826
  • 步進電機
    +關注

    關注

    151

    文章

    3115

    瀏覽量

    147677
  • PWM波
    +關注

    關注

    0

    文章

    99

    瀏覽量

    16915
  • 預分頻器
    +關注

    關注

    0

    文章

    18

    瀏覽量

    8171
收藏 人收藏

    評論

    相關推薦

    stm32的官方庫函數步進電機的那個脈沖函數嗎?

    stm32的官方庫函數步進電機的那個脈沖函數嗎,還是說要自己編寫
    發表于 07-23 06:37

    stm32比較輸出模式及用于步進電機控制

    驅動器,PWM輸出模式只適用于電機直轉,不適合精準控制脈沖個數和精準角度以及做步進
    發表于 10-26 07:00

    Stepper電機驅動庫有哪些問題?

    Stepper庫,但是,結果是電機未轉動,以至于我認為購買的電機出了問題,自己還測試了下,最后發現電機是正確的,是
    發表于 07-08 08:01

    Stepper庫函數控制步進電機

    問題來源最近自學Arduino,在使用步進電機時開始沒能使步進電機轉起來,轉起來后感覺沒法調速,遂完成此篇筆記供自己后續查閱以及方便遇到相同問題的諸君尋找靈感。對于如何使
    發表于 09-07 09:15

    STM32F429如何使用TIM4和TIM14 的PWM波驅動步進電機

    STM32F429如何使用TIM4和TIM14 的PWM波驅動步進電機
    發表于 12-21 07:24

    PWM的兩比較模式是什么

    (TIMx_CCMR1)的作用PWM輸出極性是什么?PWM配置注意事項占空比怎么設置?庫函數配置步驟第步:使能GPIO/AFIO/
    發表于 02-16 07:14

    STM8的PWM輸出模式PWM1和PWM2的區別是什么

    STM8自學筆記:TIMPWM輸出模式中關于PWM1 和 PWM2的區別STM8的
    發表于 03-02 06:07

    TIM—高級定時器輸出PWM

    :TIM1->CH1CVR=30; CH1CVR表示通道1,不同通道數字不同,如CH2CVR表示通道23參數都不能超過16位,最大65535暫停PWM輸出,強制拉高拉低修改
    發表于 03-08 11:35

    步進電機只抖動不轉是什么問題?

    (\"ni\"); stepper.step(-1024); //4步模式下旋轉2048 步。 delay(500);}使用上述代碼 步進
    發表于 11-10 08:15

    開關手動控制步進電機的實現方法

    介紹了開關代替單片機對步進電機進行手動控制的原理電路,給出了步進
    發表于 04-29 13:52 ?23次下載

    MSP430 Stepper Motor Controller電機控制:步進電機

    MSP430 Stepper Motor Controller電機控制:步進電機
    發表于 10-12 14:50 ?12次下載
    MSP430 <b class='flag-5'>Stepper</b> Motor Controller<b class='flag-5'>電機</b>控制:<b class='flag-5'>步進</b><b class='flag-5'>電機</b>

    STM8TIM1配置

    基于STM8-TIM1的單通道PWM輸出,互補PWM輸出謹以本文記錄最近使用STM8進行電機驅動
    發表于 12-03 13:51 ?6次下載
    STM8<b class='flag-5'>TIM</b>1配置

    PWM輸出實驗詳細示例

    (TIMx_CCMR1)的作用PWM輸出極性是什么?PWM配置注意事項占空比怎么設置?庫函數配置步驟第步:使能GPIO/AFIO/
    發表于 12-20 18:58 ?2次下載
    <b class='flag-5'>PWM</b><b class='flag-5'>輸出</b>實驗詳細示例

    STM32驅動步進電機梯形算法庫函數

    關于梯形算法的原理查看:AVR446: Linear speed control of stepper motor 里面有原理和代碼(庫函數版F4)廢話不多說直接上鏈接: 梯形算法驅動步進電機
    發表于 03-23 10:20 ?0次下載
    STM32驅動<b class='flag-5'>步進</b><b class='flag-5'>電機</b>梯形算法<b class='flag-5'>庫函數</b>版

    PWM輸出控制電機

    PWM 輸出控制電機電機系列 PWM 輸出控制電機
    發表于 05-06 11:03 ?5次下載
    <b class='flag-5'>PWM</b><b class='flag-5'>輸出</b>控制<b class='flag-5'>電機</b>
    主站蜘蛛池模板: 日本亚洲天堂网| 看日本黄色大片| 成年女人在线观看| 黄色地址| www.五月婷婷| 理论片人人51| 精品欧美小视频在线观看| 国产精品一区二区三区免费视频 | 男啪女色黄无遮挡免费观看 | 亚洲黄色录像| 欧美黑粗硬| 奇米影视亚洲四色8888| 亚洲成在人天堂一区二区| av手机在线播放| 亚洲天堂bt| 中文字幕天天躁夜夜狠狠综合| 影音先锋ady69色资源网站 | 天天综合色天天综合| 免费午夜在线视频| 视频综合网| 国产亚洲片| 中文字幕二区三区| 在线观看免费国产| 色视频在线播放| 国产一级久久免费特黄| 天天干天天干| 日本人善交69xxx| 一区二区三区免费视频网站| 色老头永久免费网站| 国产亚洲精品自在久久77| 午夜国产福利在线观看| 亚洲精品视频在线| 国产午夜免费一区二区三区| 一本到卡二卡三卡四卡| 亚洲成在| 久久五月网| 天天做天天爰夜夜爽| 永久黄色免费网站| 在线视频毛片| 亚洲 自拍 欧美 综合| 久久鲁视频|