精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

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

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

3天內不再提示

無刷電機小車開發記錄03——PWM信號輸入捕獲驅動

RTThread物聯網操作系統 ? 來源:未知 ? 2023-10-20 02:00 ? 次閱讀

前言


??好吧,轉眼一看,距離上一篇文章已經過去了兩個月了。只能說工作和家庭的各種事情確實太多了,人到中年屬于自己的時間確實越來越少了,再不過來調一下這個小車估計都要生銹了。所以今天過來接著做下一步工作。之前是完成了BSP的移植和導入,接下來就要嘗試移植FOC算法了,開源的FOC算法也比較多,我這里打算利用SimpleFOC進行移植。本身的SimpleFOC是基于C++的,這里要移植成C代碼。另外,SimpleFOC的SDK其中已經適配了很多種類的傳感器驅動器以及無刷電機。如果硬件使用的是它已經適配的方案,則只需要簡單配置一下就可以驅動了。而我這里是要在RTThread下移植FOC,更傾向于使用RTThread的框架,所以各種傳感器和驅動器的適配計劃加到RTThread的驅動這邊來做。FOC那邊只移植SimpleFOC的核心算法即可。所以在正式移植FOC算法之前,還需要先搭建用到的底層驅動。今天就先整理一下讀取磁編碼器PWM信號的輸入捕獲驅動的移植記錄。其實某些適配更好的BSP內的RTThread驅動庫里面已經有了輸入捕獲驅動,但只是捕獲了輸入脈寬的時間,而我這里需要的是捕獲PWM信號的占空比,也就對應了磁編碼器探測到的電機位置。但大體功能類似,所以隨便找一個類似的底層驅動進行一下修改和移植即可。

磁編碼器簡介


??我這里用的是賽卓電子的國產磁編碼器芯片SC60228,詳情請看其數據手冊,主要特性如下:

wKgaomUxb86Ad0frAADRKt9ULqs978.jpg

移植RTT驅動


??這個比較簡單,因為RTT驅動庫內已經有了“rt_inputcapture.c”的驅動文件,在SDK的“rt-thread/components/drivers/misc”目錄下。只不過大多數的BSP沒有做對應的適配而已。那先不管BSP那邊的適配問題,先把這個C文件和對應的頭文件拷貝一份,比如我重命名為“PWM_input_capture.c”和“PWM_input_capture.h”。

wKgaomUxb86AJ7ovAABMOSbJPxk665.png

然后代碼內容改動不大,主要改的是返回的數據除了脈寬時間還有一個周期時間,這樣就可以計算輸入PWM信號的占空比了。另外,原有的驅動上使用的是ringbuffer做了一個數據緩存,這樣數據處理可以異步話,什么時候需要什么時候把緩存內的數據全部讀走即可。但各人考慮,我應用的場合是用這個信號來驅動無刷電機,這個PWM信號的輸入頻率也才1Khz,市面上大多數的無刷電機驅動,底層控制頻率基本都達到了10Khz以上。所以我這里肯定不需要異步處理的,會直接用這個信號觸發底層控制。而且控制效果還需要測試,如果轉速上不去或者抖動厲害的話,可能還需要想辦法插值細化或者改用SPI讀取編碼器數據(這也是為什么硬件上做了兩種接口的原因,就是想去測試探索一些好玩的東西)。所以我這里是直接去掉了ringbuffer,加入了信號量。到時候上層開一個線程去等待這個信號量去跑FOC算法。頭文件修改如下:


		

1structpwm_inputcapture_data 2{ 3rt_uint32_tpulsewidth_us;//脈寬 4rt_uint32_tpulsecycle_us;//周期 5}; 6structpwm_inputcapture_device 7{ 8structrt_deviceparent; 9conststructpwm_inputcapture_ops*ops; 10rt_sem_t*sem; 11structpwm_inputcapture_datapulse_param; 12}; 13/** 14*captureoperators 15*/ 16structpwm_inputcapture_ops 17{ 18rt_err_t(*init)(structpwm_inputcapture_device*inputcapture); 19rt_err_t(*open)(structpwm_inputcapture_device*inputcapture); 20rt_err_t(*close)(structpwm_inputcapture_device*inputcapture); 21}; 22voidpwm_hw_inputcapture_isr(structpwm_inputcapture_device*inputcapture); 23rt_err_trt_device_pwm_inputcapture_register(structpwm_inputcapture_device*inputcapture, 24constchar*name, 25void*data);

C文件主要修改的是回調函數,把之前的數據加入ringbuffer的操作改成了釋放信號量,其它地方的修改都是一些簡單的適配,由于C代碼較多,我這里就不都貼出來了,相信大家肯定會自己完成適配,甚至比我的還要適配的好。而我的代碼,等我第一期的功能開發完了,會整體開源出來。C代碼主要修改的回調函數如下:


		

1voidrt_hw_pwm_inputcapture_isr(structpwm_inputcapture_device*inputcapture) 2{ 3rt_sem_release(inputcapture->sem); 4if(inputcapture->parent.rx_indicate!=RT_NULL) 5inputcapture->parent.rx_indicate(&inputcapture->parent,1); 6}

適配BSP驅動


??BSP驅動的適配稍微麻煩一點,如果大家能找到其它類似BSP內的相似驅動可以進行移植,那我這里簡單找了一下并沒有找到,所以仿照RTT的驅動適配方式,自己適配了一下。主要實現要點就是開啟每個Timer的CH0和CH1雙通道對CI0或者CI1輸入的PWM信號進行采樣,一個捕獲脈寬,一個捕獲周期,從而得到占空比。剩下的就是一些向下調用GD32的驅動庫API,向上適配RTT的驅動接口。同樣,下面只給出主要的初始化代碼和中斷處理代碼,其它的可自行實現或者關注我后續開源的代碼。


		

1rt_err_tpwm_inputcap_init(structpwm_inputcapture_device*pwm_incap) 2{ 3uint32_tsys_clk_freq; 4uint32_ttimer_prescaler=1; 5uint32_ttrigger_ch; 6timer_parameter_structTimerConfig; 7timer_ic_parameter_structTimerICConfig; 8structgd32_pwm_inputcapture_device*pwm_incap_device; 9pwm_incap_device=(structgd32_pwm_inputcapture_device*)pwm_incap; 10rcu_periph_clock_enable(pwm_incap_device->timer_rcu); 11rcu_periph_clock_enable(pwm_incap_device->GPIO_rcu); 12gpio_init(pwm_incap_device->GPIOx,GPIO_MODE_IN_FLOATING,GPIO_OSPEED_50MHZ,pwm_incap_device->PINx); 13sys_clk_freq=rcu_clock_freq_get(CK_SYS); 14LOG_I("systemclockfrequency:%d",sys_clk_freq); 15TimerConfig.alignedmode=TIMER_COUNTER_EDGE; 16TimerConfig.clockdivision=TIMER_CKDIV_DIV1; 17TimerConfig.counterdirection=TIMER_COUNTER_UP; 18TimerConfig.period=65535U; 19do{ 20if(sys_clk_freq/timer_prescaler/TimerConfig.periodinput_freq_min) 21break; 22if(timer_prescaler==65536) 23{ 24rt_kprintf("cannotconfiguretheprescalerforinputsignalfrequency:%dhz ",pwm_incap_device->input_freq_min); 25returnRT_ERROR; 26} 27timer_prescaler++; 28}while(1); 29TimerConfig.prescaler=timer_prescaler-1; 30TimerConfig.repetitioncounter=0; 31timer_init(pwm_incap_device->timerx,&TimerConfig); 32LOG_I("%stimerprescaler:%d",pwm_incap_device->name,timer_prescaler); 33TimerICConfig.icfilter=10; 34TimerICConfig.icpolarity=TIMER_IC_POLARITY_RISING; 35TimerICConfig.icprescaler=TIMER_IC_PSC_DIV1; 36TimerICConfig.icselection=TIMER_IC_SELECTION_DIRECTTI; 37timer_input_pwm_capture_config(pwm_incap_device->timerx,pwm_incap_device->input_ch,&TimerICConfig); 38timer_interrupt_flag_clear(pwm_incap_device->timerx,TIMER_INT_FLAG_UP); 39timer_interrupt_enable(pwm_incap_device->timerx,TIMER_INT_UP); 40trigger_ch=((pwm_incap_device->input_ch==TIMER_CH_0)?TIMER_SMCFG_TRGSEL_CI0FE0:TIMER_SMCFG_TRGSEL_CI1FE1); 41timer_input_trigger_source_select(pwm_incap_device->timerx,trigger_ch); 42timer_slave_mode_select(pwm_incap_device->timerx,TIMER_SLAVE_MODE_RESTART); 43timer_external_trigger_config(pwm_incap_device->timerx,TIMER_EXT_TRI_PSC_OFF,TIMER_ETP_RISING,10); 44NVIC_SetPriority(pwm_incap_device->timerx_irqn,3); 45NVIC_EnableIRQ(pwm_incap_device->timerx_irqn); 46timer_enable(pwm_incap_device->timerx); 47returnRT_EOK; 48} 49voidpwm_inputcapture_update_isr(structgd32_pwm_inputcapture_device*device) 50{ 51uint32_twidth_ch; 52/*TIMUpdateevent*/ 53if(timer_interrupt_flag_get(device->timerx,TIMER_INT_FLAG_UP)!=RESET) 54{ 55timer_interrupt_flag_clear(device->timerx,TIMER_INT_FLAG_UP); 56device->pwm_inputcap.pulse_param.pulsecycle_us=timer_channel_capture_value_register_read(device->timerx,device->input_ch); 57width_ch=((device->input_ch==TIMER_CH_0)?(TIMER_CH_1):(TIMER_CH_0)); 58device->pwm_inputcap.pulse_param.pulsewidth_us=timer_channel_capture_value_register_read(device->timerx,width_ch); 59rt_hw_pwm_inputcapture_isr(device); 60} 61}

修改工程構建文件


修改相關SConscript文件

??在“libraries/gd32_drivers/SConscript”文件內的適當位置加入如下代碼:


		

1ifGetDepend(['RT_USING_PWM_INPUT_CAPTURE']): 2src+=['drv_pwm_inputcapture.c']

??在“rt-thread/components/drivers/misc/SConscript”文件內的適當位置加入如下代碼:


		

1ifGetDepend(['RT_USING_INPUT_CAPTURE']): 2src=src+['rt_inputcapture.c']

??意思很簡單,就是當rtconfig.h內定義了”RT_USING_PWM_INPUT_CAPTURE”宏,則把“drv_pwm_inputcapture.c”和“rt_inputcapture.c”驅動文件加入工程,更準確的是加入編譯。

修改相關Kconfig文件

??在“rt-thread/components/drivers/Kconfig”文件內的適當位置加入如下代碼:


		

1configRT_USING_INPUT_CAPTURE 2bool"UsingINPUTCAPTUREdevicedrivers" 3defaultn

??管理BSP驅動代碼的Kconfig文件不再librares目錄下,而是在board目錄下。于是在“board/Kconfig”文件內的適當位置,仿照其它驅動加入如下代碼:


		

1menuconfigBSP_USING_PWM_INPUTCAPTURE 2bool"Enablepwminputcapture" 3defaultn 4selectRT_USING_PWM_INPUT_CAPTURE 5ifBSP_USING_PWM_INPUTCAPTURE 6configBSP_USING_PWM_INPUTCAPTURE1 7bool"Enablepwminputcapture1" 8defaultn 9configBSP_USING_PWM_INPUTCAPTURE2 10bool"Enablepwminputcapture2" 11defaultn 12configBSP_USING_PWM_INPUTCAPTURE3 13bool"Enablepwminputcapture3" 14defaultn 15configBSP_USING_PWM_INPUTCAPTURE4 16bool"Enablepwminputcapture4" 17defaultn 18configBSP_USING_PWM_INPUTCAPTURE5 19bool"Enablepwminputcapture5" 20defaultn 21configBSP_USING_PWM_INPUTCAPTURE6 22bool"Enablepwminputcapture6" 23defaultn 24endif

??意思也比較簡單,我這里適配了6個PWM的輸入捕獲驅動,并且利用“select”語句,在BSP的驅動管理里面自動開啟了RTT驅動里面的“RT_USING_PWM_INPUT_CAPTURE”選項。修改完上述代碼后,就可以用menuconfig命令或者RTThreadIDE的RT-Thread Settings圖形配置界面內進行配置了:

wKgaomUxb86AcVJtAAC__4hZpLo688.png

測試


??驅動有了,再在頂層邏輯內創建并實現兩個測試線程:


		

1intmain(void) 2{ 3... 4rt_thread_tMotorL_encoder_thread; 5MotorL_encoder_thread=rt_thread_create("MotorLEncoder",MotorLEncoder_thread_entry,RT_NULL,1024,4,20); 6rt_thread_startup(MotorL_encoder_thread); 7rt_thread_tMotorR_encoder_thread; 8MotorR_encoder_thread=rt_thread_create("MotorREncoder",MotorREncoder_thread_entry,RT_NULL,1024,4,20); 9rt_thread_startup(MotorR_encoder_thread); 10... 11} 12voidMotorLEncoder_thread_entry(void*parameter) 13{ 14rt_device_tLpwm_input_dev; 15rt_uint16_twait_i; 16structpwm_inputcapture_device*inputcap_dev; 17Lpwm_input_dev=rt_device_find("pwm_inputcap1"); 18if(Lpwm_input_dev==RT_NULL) 19return; 20inputcap_dev=(structpwm_inputcapture_device*)Lpwm_input_dev; 21rt_device_open(Lpwm_input_dev,RT_DEVICE_OFLAG_RDONLY); 22while(1) 23{ 24if(inputcap_dev->sem!=RT_NULL) 25{ 26rt_sem_take(inputcap_dev->sem,RT_WAITING_FOREVER); 27if(wait_i++>=1000) 28{ 29rt_kprintf("MotorLencoder: %d ",inputcap_dev->pulse_param.pulsewidth_us*10000/inputcap_dev->pulse_param.pulsecycle_us); 30wait_i=0; 31} 32} 33} 34} 35voidMotorREncoder_thread_entry(void*parameter) 36{ 37rt_device_tRpwm_input_dev; 38rt_uint16_twait_i; 39structpwm_inputcapture_device*inputcap_dev; 40Rpwm_input_dev=rt_device_find("pwm_inputcap3"); 41if(Rpwm_input_dev==RT_NULL) 42return; 43inputcap_dev=(structpwm_inputcapture_device*)Rpwm_input_dev; 44rt_device_open(Rpwm_input_dev,RT_DEVICE_OFLAG_RDONLY); 45while(1) 46{ 47if(inputcap_dev->sem!=RT_NULL) 48{ 49rt_sem_take(inputcap_dev->sem,RT_WAITING_FOREVER); 50if(wait_i++>=1000) 51{ 52rt_kprintf("MotorRencoder:%d ",inputcap_dev->pulse_param.pulsewidth_us*10000/inputcap_dev->pulse_param.pulsecycle_us); 53wait_i=0; 54} 55} 56} 57}

??目前只是實現了大概1S鐘打印一次編碼器位置,一圈的機械角度范圍擴大到了0~10000(我用的是12位磁編碼器,分辨率是4096,我這里統一歸一化到了10000),終端輸出如下:

wKgaomUxb86AWvABAADp-6S53R4467.png

??可以看到,慢慢向一個方向推動小車,兩個編碼器的變化規律是相反的,和實際的兩個電機對向安裝相匹配,實際使用的時候按照其中一個為基準,把另外一個編碼器數據反向即可。

??只看其中一個輪子,輸出頻率改為原有的1Khz,輸出值轉換為浮點的角度值,可得到如下的測試曲線:

wKgaomUxb8-ASU3AAABtDl44Qdk079.jpg

靜止不動,暫時也沒有驅動電機,也就沒有電機的電磁干擾,在次條件下測得的靜態數據如下,靜態穩定度在0.2度左右,比12位的最小測量精度0.088度大了二倍多一點:

wKgaomUxb8-ASXSrAACu0yvNQTo492.jpg

相關鏈接


本系列首篇文章鏈接:

https://club.rt-thread.org/ask/article/5c0c4ba7eb4ab1ad.html

———————End——————

wKgaomUxb9CAVirTAHiX-BnG6Ho919.gif

點擊閱讀原文進入官網

?


原文標題:無刷電機小車開發記錄03——PWM信號輸入捕獲驅動

文章出處:【微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。


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

    關注

    31

    文章

    1274

    瀏覽量

    39944

原文標題:無刷電機小車開發記錄03——PWM信號輸入捕獲驅動

文章出處:【微信號:RTThread,微信公眾號:RTThread物聯網操作系統】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    淺談無刷電機的工作流程

    上一期的芝識課堂,我們跟大家一起分析了無刷電機的四個功能單元,并詳細分析了PWM和逆變器單元的工作情況,今天我們繼續來熟悉無刷電機工作流程中另外兩個重要的部分——轉子位置檢測和波形驅動
    的頭像 發表于 11-12 13:46 ?362次閱讀
    淺談<b class='flag-5'>無刷電機</b>的工作流程

    無刷電機驅動方案的應用

    : 有刷電機:有刷電機通過碳刷和換向器(或稱為整流子)來傳遞電流。 無刷電機無刷電機沒有碳刷和換向器,而是通過電子換向器(通常是霍爾傳感器和電子
    發表于 10-21 16:00

    其利天下技術·無刷電機中開環控制和閉環控制的區別·無刷電機驅動方案

    風扇驅動方案開環控制的原理開環控制是指系統的輸出不影響輸入控制信號的策略。在無刷電機中,控制信號(如PW
    的頭像 發表于 10-16 10:30 ?356次閱讀
    其利天下技術·<b class='flag-5'>無刷電機</b>中開環控制和閉環控制的區別·<b class='flag-5'>無刷電機</b><b class='flag-5'>驅動</b>方案

    其利天下技術·無刷電機壽命有多長?軟件能改善嗎?無刷電機驅動

    作為無刷電機驅動方案開發商,經常跟客戶都會討論到一個問題——無刷電機的壽命受什么因素影響?現在無刷電機的制作工藝對比以前已經有了極大的提升,
    的頭像 發表于 09-23 16:01 ?269次閱讀
    其利天下技術·<b class='flag-5'>無刷電機</b>壽命有多長?軟件能改善嗎?<b class='flag-5'>無刷電機</b><b class='flag-5'>驅動</b>

    其利天下技術·低速無刷電機如何更好的控制?無刷電機驅動方案開發

    需要低速運轉的無刷電機經常運用在電動車、機器人關節、醫療設備等場景。最近有客戶找到我們開發無刷電機驅動方案,他的需求是低轉速,而且需要精準控制電流。經常
    的頭像 發表于 09-09 11:50 ?413次閱讀
    其利天下技術·低速<b class='flag-5'>無刷電機</b>如何更好的控制?<b class='flag-5'>無刷電機</b><b class='flag-5'>驅動</b>方案<b class='flag-5'>開發</b>

    其利天下技術開發·無刷電機常見的5種啟動方式·無刷電機驅動方案

    作為經常跟無刷電機打交道的驅動方案商,能夠根據不同的無刷電機選擇匹配的啟動方式,是基本的技術要求。今天分享一下,無刷電機常見的啟動方式有哪些?分別又有什么優缺點呢?
    的頭像 發表于 08-21 10:59 ?491次閱讀
    其利天下技術<b class='flag-5'>開發</b>·<b class='flag-5'>無刷電機</b>常見的5種啟動方式·<b class='flag-5'>無刷電機</b><b class='flag-5'>驅動</b>方案

    PWM驅動電機發熱的原因

    PWM技術被廣泛應用于電機的調速和控制。 1.2 PWM驅動電機的工作原理 PWM
    的頭像 發表于 08-08 15:25 ?804次閱讀

    其利天下技術·無刷電機常見的5種啟動方式及優缺點·無刷電機驅動方案設計開發

    作為經常跟無刷電機打交道的驅動方案商,能夠根據不同的無刷電機選擇匹配的啟動方式,是基本的技術要求。今天分享一下,無刷電機常見的啟動方式有哪些?分別又有什么優缺點呢? ? 其利天下·
    的頭像 發表于 08-07 10:26 ?642次閱讀
    其利天下技術·<b class='flag-5'>無刷電機</b>常見的5種啟動方式及優缺點·<b class='flag-5'>無刷電機</b><b class='flag-5'>驅動</b>方案設計<b class='flag-5'>開發</b>

    無刷電機驅動原理 直流有刷電機無刷電機的差異

    在上期芝識課堂中,我們一起認識了東芝無刷電機的基本知識,對于生活中無處不在的無刷電機,了解它的更多知識才能上手開發哦~今天芝子就帶你一起來學習無刷電機
    的頭像 發表于 07-19 14:29 ?1289次閱讀
    <b class='flag-5'>無刷電機</b>的<b class='flag-5'>驅動</b>原理 直流有刷<b class='flag-5'>電機</b>和<b class='flag-5'>無刷電機</b>的差異

    無刷電機驅動器的工作原理和結構組成

    無刷電機驅動器,也被稱為無刷電調或無刷電機電子調速器,是一種能有效控制電機轉速、扭矩和位置的新型電機驅動
    的頭像 發表于 06-12 16:54 ?2236次閱讀

    CS32F03X生成多路PWM控制無刷電機換相

    本技術文檔旨在幫助用戶快速配置 MCU CS32F03X 輸出多路經過調制的 PWM 波,用于控制直流無刷電機(BLDC)換相。HPWM-LON 是直流無刷電機較為常用的
    發表于 05-16 10:41

    無刷電機是什么意思 無刷電機是交流還是直流

    無刷電機(Brushless Motor)是一種將電能轉換為機械能的驅動裝置,也被稱為無刷直流電機(BLDC Motor)。與傳統的有刷電機相比,
    的頭像 發表于 02-01 13:57 ?1.2w次閱讀

    一個通道如何捕獲PWM的頻率和占空比?

    捕獲。雖然也實現了捕獲PWM的功能,但是代價也太大了,且很難同時捕獲多個PWM,那有沒有更好的方法呢?本文介紹了另一種
    的頭像 發表于 12-30 08:00 ?2794次閱讀
    一個通道如何<b class='flag-5'>捕獲</b><b class='flag-5'>PWM</b>的頻率和占空比?

    無刷電機驅動板怎么接線

    驅動板的接線方法,包括電源接線、電機接線和傳感器接線等方面,以幫助讀者更好地理解和應用無刷電機驅動板。 電源接線 電源輸入
    的頭像 發表于 12-14 11:22 ?4847次閱讀

    pwm是怎么驅動電機的?

    pwm是怎么驅動電機的? 本文將詳盡探討脈寬調制(PWM)技術及其在電機驅動中的應用。首先,我們
    的頭像 發表于 12-07 15:48 ?1778次閱讀