問:玩轉(zhuǎn)STM32 - 使用 STM32 來控制 NeoPixels
目前,諸如Arduino和Feather等高級開發(fā)平臺已經(jīng)提供了出色的支持,可以通過易于使用的庫和普遍使用的示例代碼與NeoPixelLE?D、燈帶、矩陣等相連接。然而,更高級的平臺(例如STM32 開發(fā)板)通常缺乏相同水平的支持。因此,希望將NeoPixels整合到項(xiàng)目中的 開發(fā)人員需要全面了解NeoPixel通信協(xié)議以及如何克服它所帶來的挑戰(zhàn)。
NeoPixels
Adafruit推出的極受歡迎的可尋址全彩LED燈“NeoPixels”系列分為RGB和RGBW兩個種類。盡管二者都將紅、綠和藍(lán)色LED與驅(qū)動器芯片相集成,但RGBW組件還集成了第四個純白色的LED。可以使用類似的單線串行接口來控制這兩種類型的NeoPixel,其時間值和數(shù)據(jù)結(jié)構(gòu)僅存在微小的差異。
WS2812
RGB NeoPixels實(shí)際上是WS2812智能控制LED,包括數(shù)據(jù)信號輸入引腳(DIN)和數(shù)據(jù)信號輸出引腳(DOUT)。這允許多個LED級聯(lián)并且只用一個數(shù)據(jù)線進(jìn)行控制。鏈中的第一個LED負(fù)責(zé)處理從MCU接收到的前三個字節(jié)數(shù)據(jù),然后將后續(xù)的數(shù)據(jù)簡單地轉(zhuǎn)發(fā)給DOUT引腳,該引腳可以連接到另一個LED的DIN引腳。LED將以此方式繼續(xù)向下傳遞數(shù)據(jù),直到它們接收到復(fù)位信號為止(即,DIN線在一段時間內(nèi)持續(xù)保持低電平狀態(tài))。傳輸?shù)淖止?jié)按照圖1所示的協(xié)議進(jìn)行組織。第一個字節(jié)(G7-G0)表示綠色LED的8位PWM強(qiáng)度,其中0x00是完全關(guān)閉,0xFF是完全打開。類似地,第二個字節(jié)(R7-R0)用于控制紅色LED的強(qiáng)度,第三個字節(jié)(B7-B0)用于控制藍(lán)色LED的強(qiáng)度。

圖2:WS2812 LED的0和1位的計(jì)時圖
一種截然不同的組件,NeoPixels的RGBW種類實(shí)際上是SK6812智能控制LED,采用與WS2812 LED相同的運(yùn)作原理。然而,由于它們包含第四個LED,因此實(shí)施了圖3所示的4字節(jié)數(shù)據(jù)協(xié)議。與圖1相比,唯一的區(qū)別在于數(shù)據(jù)的串聯(lián)字節(jié)(W7-W0),該字節(jié)指定了白色LED的8位PWM強(qiáng)度。


步驟
由于NeoPixel的控制信號對計(jì)時要求非常嚴(yán)格,因此除非使用匯編語言,否則無法通過簡單的比特帶寬方法產(chǎn)生此信號。雖然還有許多其他方法可以利用各種MCU外設(shè)、外部硬件或其組合來生成該信號,但其中最直接的方法是配置MCU定時器來生成PWM輸出信號。這是因?yàn)椋缟弦徊糠种兴觯琋eoPixel控制信號只是一種固定頻率的PWM信號,采用不同的占空比表示0位和1位。為了以與傳輸協(xié)議相同的速率高效地在這兩個占空比之間進(jìn)行切換,還必須配置DMA流來管理更新。盡管這種方法可能是內(nèi)存效率最低的方式,但它易于理解、CPU高效并且易于實(shí)施(得益于STM32Cube環(huán)境)。以下應(yīng)用程式利用STM32CubeIDE(版本1.8.0)、NUCLEO-F401RE開發(fā)板和RGBW5x8 NeoPixel Shield實(shí)現(xiàn)上述的方法。不過,這些步驟可以輕松地推廣到任何STM32MCU/板和NeoPixel產(chǎn)品上。假定我們已經(jīng)創(chuàng)建了一個STM32CubeIDE項(xiàng)目。如需使用其他IDE,你可以改為使用獨(dú)立的STM32CubeMX代碼配置器工具,將項(xiàng)目導(dǎo)出到所需的開發(fā)平臺上。
1.配置PWMa. 先打開STM32CubeMX配置.ioc文件(如果還未打開的話)。隨后,STM32CubeIDE將切換到*器件配置工具(*Device Configuration Tool)視圖,供你配置MCU。
b. 將定時器通道備用功能分配給選定的GPIO引腳,以與NeoPixel進(jìn)行連接。所選定時器通道應(yīng)該能夠生成PWM輸出。圖5顯示了我的項(xiàng)目中的相關(guān)部分,我選擇了引腳PB10,并將它分配給定時器2、通道3(TIM2_CH3)功能。

d. 在定時器的*配置(*Configuration)面板中,驗(yàn)證“預(yù)分頻器”和“脈沖”值是否都設(shè)置為0。計(jì)數(shù)器周期,即自動重載寄存器(ARR),需要進(jìn)行設(shè)置以得到所需的PWM周期(如果使用RGB WS2812 LED,則為1.25μs;如果使用RGBW SK6812 LED,則為1.2μs)。這將取決于定時器外設(shè)輸入的速率。只需將所需的PWM周期除以時鐘周期,并減去1即可得到此值(減去1是因?yàn)槎〝?shù)器從0開始)。就我的器件而言,該公式得出的ARR值為99.8,我將其四舍五入為100(圖6)。請參見下文,了解有關(guān)計(jì)算理想ARR值的詳細(xì)說明。

假設(shè)定時器“預(yù)分頻器”值設(shè)為0,可以很容易的計(jì)算出ARR值


在我使用的MCU(STM32F401RE)規(guī)格書中,器件框圖中顯示我的定時器(TIM2)已連接到APB1(見圖8)。

圖9介紹了:通過切換到STM32CubeIDE中的*時鐘配置(*Clock Configuration)選項(xiàng)卡,我們可以發(fā)現(xiàn)TIM2的時鐘頻率為84MHz

因此,

a. 從組件列表中選擇DMA外設(shè)。
b. 在配置(Configuration)面板的DMA1選項(xiàng)卡下,點(diǎn)擊添加(Add)按鈕。在下拉菜單中,選擇你的定時器/通道組合。在我的項(xiàng)目中,我選擇了“TIM2_CH3/UP”。
c. 針對該新的DMA請求,將方向改為“內(nèi)存到外設(shè)”。
d. 同時,將優(yōu)先級改為“非常高”。
e. 驗(yàn)證默認(rèn)的DMA請求設(shè)置是否與圖10中顯示的相匹配。
f. 保存.ioc文件,以生成項(xiàng)目代碼。

3.編寫代碼
在main.c文件中,按從上到下的順序編寫,本部分展示了一個簡單的示例應(yīng)用,用于測試NeoPixel LED的全彩能力。此處提供了兩個版本的main()函數(shù),一個用于RGB WS2818 LED,另一個用于RGBW SK6812 LED。
a. 在main.c文件的私有typedef部分,你可以創(chuàng)建一個新的數(shù)據(jù)類型,以便輕松訪問單個LED顏色值以及整個NeoPixel數(shù)據(jù)結(jié)構(gòu)(如圖1和圖3所示)。列表1提供了RGB和RGBW NeoPixel組件的typedef。此代碼應(yīng)粘貼在/* USER CODE BEGIN PTD */和/* USER CODE END PTD */注釋之間。
列表1:為RGB WS2812和RGBW SK6812 LED自定義數(shù)據(jù)類型
typedef union
{
struct
{
uint8_t b;
uint8_t r;
uint8_t g;
} color;
uint32_t data;
} PixelRGB_t;
typedef union
{
struct
{
uint8_t w;
uint8_t b;
uint8_t r;
uint8_t g;
} color;
uint32_t data;
} PixelRGBW_t;
b. 更改“脈沖”寄存器(也稱為CCRx)的值,這樣可以改變PWM波形的占空比。因此,我們必須計(jì)算適當(dāng)?shù)腃CRx值,以實(shí)現(xiàn)使用的NeoPixels所需的代碼0和代碼1方波(無論是在圖2還是圖4中所示的那些)。對于RGBWS2812 LED,這些值計(jì)算如下:
ZERO=(ARR+1)(0.32)
ONE=(ARR+1)(0.64)
對于RGBW SK6812 LED,其計(jì)算過程稍有不同。
ZERO=(ARR+1)(0.25)
ONE=(ARR+1)(0.5)
當(dāng)然,這些計(jì)算出的值應(yīng)該四舍五入到最接近的整數(shù)。在main.c文件的私有定義部分,為每個值創(chuàng)建一個#define指令(請參見以下圖11中的示例)。
c. 除了CCRx值之外,還應(yīng)在私有定義部分中定義控制的NeoPixel LED數(shù)量和DMA緩沖區(qū)大小。如圖11所示,只需將LED的數(shù)量乘以相應(yīng)的NeoPixel數(shù)據(jù)結(jié)構(gòu)中的位數(shù)即可(回想圖1和圖3)。還必須分配一個額外的緩沖區(qū)元素,因?yàn)樽詈笠粋€CCRx值應(yīng)為零(復(fù)位信號)。

d. 將列表2中提供的DMA完成回調(diào)函數(shù)添加到/* USER CODE BEGIN 0/和/USER CODE END 0*/之間的私有用戶代碼部分。務(wù)必將TIM_CHANNEL_x更改為步驟1c中配置的通道。
列表2:HAL_TIM_PWM_PulseFinishedCallback()函數(shù)的實(shí)施
void HAL_TIM_PWM_PulseFinishedCallback(TIM_HandleTypeDef *htim)
{
HAL_TIM_PWM_Stop_DMA(htim, TIM_CHANNEL_x);
}
e. 最后,必須將應(yīng)用代碼添加到main()函數(shù)中。列表3提供了一個使用WS2812LED的示例main()函數(shù),而列表4提供了使用SK6812 LED的類似示例main()函數(shù)。請注意,HAL_TIM_PWM_Start_DMA()函數(shù)的TIM_CHANNEL_x參數(shù)必須再次進(jìn)行修改,以匹配步驟1c中配置的通道。
列表3:RGB WS2812 LED的示例main()函數(shù)
int main(void)
{
/* USER CODE BEGIN 1 */
PixelRGB_tpixel[NUM_PIXELS] = {0};
uint32_tdmaBuffer[DMA_BUFF_SIZE] = {0};
uint32_t *pBuff;
int i, j, k;
uint16_t stepSize;
/* USER CODE END 1 */
/* MCUConfiguration--------------------------------------------------------*/
/* Reset of allperipherals, Initializes the Flash interface and the Systick. */
HAL_Init();
/* USER CODE BEGIN Init*/
/* USER CODE END Init*/
/* Configure the systemclock */
SystemClock_Config();
/* USER CODE BEGINSysInit */
/* USER CODE ENDSysInit */
/* Initialize allconfigured peripherals */
MX_GPIO_Init();
MX_USART2_UART_Init();
MX_DMA_Init();
MX_TIM2_Init();
/* USER CODE BEGIN 2 */
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGINWHILE */
k = 0;
stepSize = 4;
while (1)
{
/* USER CODE ENDWHILE */
/* USER CODE BEGIN 3*/
for (i = (NUM_PIXELS- 1); i > 0; i--)
{
pixel[i].data =pixel[i-1].data;
}
if (k < 255)
{
pixel[0].color.g =254 - k; //[254, 0]
pixel[0].color.r= k + 1; //[1, 255]
pixel[0].color.b =0;
}
else if (k < 510)
{
pixel[0].color.g =0;
pixel[0].color.r =509 - k; //[254, 0]
pixel[0].color.b =k - 254; //[1, 255]
j++;
}
else if (k < 765)
{
pixel[0].color.g =k - 509; //[1, 255];
pixel[0].color.r =0;
pixel[0].color.b =764 - k; //[254, 0]
}
k = (k + stepSize) %765;
// not so bright
pixel[0].color.g>>= 2;
pixel[0].color.r>>= 2;
pixel[0].color.b>>= 2;
pBuff = dmaBuffer;
for (i = 0; i
{
for (j = 23; j>= 0; j--)
{
if((pixel[i].data >> j) & 0x01)
{
*pBuff =NEOPIXEL_ONE;
}
else
{
*pBuff =NEOPIXEL_ZERO;
}
pBuff++;
}
}
dmaBuffer[DMA_BUFF_SIZE - 1] = 0; // last element must be 0!
HAL_TIM_PWM_Start_DMA(&htim2, TIM_CHANNEL_x, dmaBuffer,DMA_BUFF_SIZE);
HAL_Delay(10);
}
/* USER CODE END 3 */
}
列表4:RGBW SK6812 LED的示例main()函數(shù)
int main(void)
{
/* USER CODE BEGIN 1 */
PixelRGBW_tpixel[NUM_PIXELS] = {0};
uint32_tdmaBuffer[DMA_BUFF_SIZE] = {0};
uint32_t *pBuff;
int i, j, k;
uint16_t stepSize;
/* USER CODE END 1 */
/* MCU Configuration--------------------------------------------------------*/
/* Reset of allperipherals, Initializes the Flash interface and the Systick. */
HAL_Init();
/* USER CODE BEGIN Init*/
/* USER CODE END Init*/
/* Configure the systemclock */
SystemClock_Config();
/* USER CODE BEGINSysInit */
/* USER CODE ENDSysInit */
/* Initialize allconfigured peripherals */
MX_GPIO_Init();
MX_USART2_UART_Init();
MX_DMA_Init();
MX_TIM2_Init();
/* USER CODE BEGIN 2 */
/* USER CODE END 2 */
/* Infinite loop */
/* USER CODE BEGINWHILE */
k = 0;
stepSize = 4;
while (1)
{
/* USER CODE ENDWHILE */
/* USER CODE BEGIN 3*/
for (i = (NUM_PIXELS- 1); i > 0; i--)
{
pixel[i].data =pixel[i-1].data;
}
if (k < 255)
{
pixel[0].color.g =254 - k; //[254, 0]
pixel[0].color.r= k + 1;//[1, 255]
pixel[0].color.b =0;
pixel[0].color.w =0;
}
else if (k < 510)
{
pixel[0].color.g =0;
pixel[0].color.r =509 - k; //[254, 0]
pixel[0].color.b =k - 254; //[1, 255]
pixel[0].color.w =0;
j++;
}
else if (k < 765)
{
pixel[0].color.g =0;
pixel[0].color.r =0;
pixel[0].color.b =764 - k; //[254, 0]
pixel[0].color.w =k - 509; //[1, 255]
}
else if (k < 1020)
{
pixel[0].color.g =k - 764; //[1, 255]
pixel[0].color.r =0;
pixel[0].color.b =0;
pixel[0].color.w =1019 - k; //[254, 0]
}
k = (k + stepSize) %1020;
// 50% brightness
pixel[0].color.g>>= 2;
pixel[0].color.r>>= 2;
pixel[0].color.b>>= 2;
pixel[0].color.w>>= 2;
pBuff = dmaBuffer;
for (i = 0; i
{
for (j = 31; j>= 0; j--)
{
if((pixel[i].data >> j) & 0x01)
{
*pBuff =NEOPIXEL_ONE;
}
else
{
*pBuff =NEOPIXEL_ZERO;
}
pBuff++;
}
}
dmaBuffer[DMA_BUFF_SIZE- 1] = 0; // last element must be 0!
HAL_TIM_PWM_Start_DMA(&htim2, TIM_CHANNEL_x, dmaBuffer,DMA_BUFF_SIZE);
HAL_Delay(10);
}
/* USER CODE END 3 */
}
該項(xiàng)目現(xiàn)在應(yīng)該能夠成功構(gòu)建,并支持你在器件上運(yùn)行代碼了。
結(jié)論
使用邏輯分析儀捕獲了上面提供的RGB和RGBW配置生成的控制信號。分別如圖12和圖13中所示。請注意,它們與圖2和圖4中指定的預(yù)期輸出相匹配。
圖12:生成的WS2812控制信號(正在發(fā)送0b0011……)
-
在STM32上輕松使用printf函數(shù)
-
在STM32上輕松使用scanf
-
輕松在 STM32 系列之間進(jìn)行遷移
-
利用 STM32CubeIDE 中構(gòu)建分析儀
- VL53L5CXToF傳感器使用入門

提示點(diǎn)擊菜單設(shè)計(jì)支持:工程師錦囊,獲取更多工程師小貼士
秘技知識學(xué)不停 專屬福利享不停
就等您加入!
點(diǎn)此登記
賺積分、換好禮
立即到「會員權(quán)益」查看您的禮遇! 如有任何問題,歡迎聯(lián)系得捷電子DigiKey的客服團(tuán)隊(duì)中國(人民幣)客服




中國(美金)/ 香港客服

400-882-4440



點(diǎn)擊下方“閱讀原文”查看更多
讓我知道你在看喲
原文標(biāo)題:按這個步驟 STM32即可完美控制 NeoPixels
文章出處:【微信公眾號:得捷電子DigiKey】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
-
得捷電子
+關(guān)注
關(guān)注
1文章
255瀏覽量
9772
原文標(biāo)題:按這個步驟 STM32即可完美控制 NeoPixels
文章出處:【微信號:得捷電子DigiKey,微信公眾號:得捷電子DigiKey】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
訓(xùn)練完模型后用cls_video.py在canmvIDE上運(yùn)行,按著步驟操作但是攝像頭沒有識別到是什么情況?
STM32與機(jī)智云連接實(shí)現(xiàn)步驟與技巧(下篇):機(jī)智云代碼移植與優(yōu)化

STM32與機(jī)智云連接實(shí)現(xiàn)步驟與技巧(上篇)

BNC座開孔標(biāo)準(zhǔn):確保完美適配的尺寸與步驟

STM32F405xx和STM32F407xx微控制器數(shù)據(jù)手冊
濕法刻蝕步驟有哪些
stm32 GPIO中斷配置教程
LMK05318手冊里的編寫EEPROM的步驟,這個掩碼什么意思?
STM32項(xiàng)目實(shí)戰(zhàn):基于STM32U5的智能燈光控制系統(tǒng)(LVGL),附項(xiàng)目教程/源碼

高低溫試驗(yàn)箱的操作步驟

STM32項(xiàng)目實(shí)戰(zhàn):基于STM32F4的智能燈光控制系統(tǒng)(LVGL),附項(xiàng)目教程/源碼

評論