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

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

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

3天內不再提示

【XR806開發板試用】使用編碼器進行調光

冬至配餃子 ? 來源:阿珣 ? 作者:阿珣 ? 2023-10-08 11:08 ? 次閱讀

之前做過一個LED調光的項目,這次想拿XR806來實現,后續打算加入遠程控制的功能。這個項目使用旋轉編碼器來調節LED的亮度,基本原理是MCU識別編碼器的旋轉方向和步數,調節PWM輸出占空比,從而實現亮度調節。

識別其旋轉方向和步數,要考慮消除抖動,否則會出現識別錯誤,導致系統不穩定,這和按鍵是類似的。

Github上有對應的Arduino庫,注意它的開源協議是GNU GPL V3!本人曾經移植到STM32,現已移植到XR806,效果良好。旋轉編碼器相關代碼如下:

頭文件re.h源碼:

/*
 * Rotary encoder library for Arduino.
 * Port to XR806 by Zixun Chen.
 */

#ifndef _ROTARY_ENCODER_H_
#define _ROTARY_ENCODER_H_

#ifdef __cplusplus
extern "C" {
#endif

#include "main.h"

// 根據編碼器的輸出類型來選擇是否定義RE_HALF_STEP
#define RE_HALF_STEP

// 旋轉編碼器通常外接上拉電阻,對應空閑電平是00B。如果外接電阻是下拉的,需要定義RE_PINS_PULL_DOWN
// #define RE_PINS_PULL_DOWN

#define DIR_NONE 0    // 尚無完整有效的步進
#define DIR_CW   0x10 // 順時針步進
#define DIR_CCW  0x20 // 逆時針步進

typedef struct {
    // 定義編碼器A端所連的GPIO引腳
    GPIO_Port GPIO_A;
    GPIO_Pin PIN_A;
    // 定義編碼器B端所連的GPIO引腳
    GPIO_Port GPIO_B;
    GPIO_Pin PIN_B;
    uint8_t RetVal; // 保存返回值
    uint8_t State; // 內部變量,保存狀態機狀態
} REHandle_t;

void RotaryEncoderInit(REHandle_t *REVal); // 初始化
void RotaryEncoderProcess(REHandle_t *REVal); // 讀取步進

#ifdef __cplusplus
}
#endif

#endif // _ROTARY_ENCODER_H_

源文件re.c:

/* Rotary encoder handler for arduino.
 *
 * Copyright 2011 Ben Buxton. Licenced under the GNU GPL Version 3.
 * Contact: bb@cactii.net
 *
 * Port to XR806 by Zixun Chen.
 */

#include "re.h"

/*
 * The below state table has, for each state (row), the new state
 * to set based on the next encoder output. From left to right in,
 * the table, the encoder outputs are 00, 01, 10, 11, and the value
 * in that position is the new state to set.
 */

#define R_START 0x0

#ifdef RE_HALF_STEP
// Use the half-step state table (emits a code at 00 and 11)
#define R_CCW_BEGIN 0x1
#define R_CW_BEGIN 0x2
#define R_START_M 0x3
#define R_CW_BEGIN_M 0x4
#define R_CCW_BEGIN_M 0x5
const unsigned char ttable[6][4] = {
    // R_START (00)
    {R_START_M,           R_CW_BEGIN,    R_CCW_BEGIN,  R_START},
    // R_CCW_BEGIN
    {R_START_M | DIR_CCW, R_START,       R_CCW_BEGIN,  R_START},
    // R_CW_BEGIN
    {R_START_M | DIR_CW,  R_CW_BEGIN,    R_START,      R_START},
    // R_START_M (11)
    {R_START_M,           R_CCW_BEGIN_M, R_CW_BEGIN_M, R_START},
    // R_CW_BEGIN_M
    {R_START_M,           R_START_M,     R_CW_BEGIN_M, R_START | DIR_CW},
    // R_CCW_BEGIN_M
    {R_START_M,           R_CCW_BEGIN_M, R_START_M,    R_START | DIR_CCW},
};
#else
// Use the full-step state table (emits a code at 00 only)
#define R_CW_FINAL 0x1
#define R_CW_BEGIN 0x2
#define R_CW_NEXT 0x3
#define R_CCW_BEGIN 0x4
#define R_CCW_FINAL 0x5
#define R_CCW_NEXT 0x6

const unsigned char ttable[7][4] = {
  // R_START
  {R_START,    R_CW_BEGIN,  R_CCW_BEGIN, R_START},
  // R_CW_FINAL
  {R_CW_NEXT,  R_START,     R_CW_FINAL,  R_START | DIR_CW},
  // R_CW_BEGIN
  {R_CW_NEXT,  R_CW_BEGIN,  R_START,     R_START},
  // R_CW_NEXT
  {R_CW_NEXT,  R_CW_BEGIN,  R_CW_FINAL,  R_START},
  // R_CCW_BEGIN
  {R_CCW_NEXT, R_START,     R_CCW_BEGIN, R_START},
  // R_CCW_FINAL
  {R_CCW_NEXT, R_CCW_FINAL, R_START,     R_START | DIR_CCW},
  // R_CCW_NEXT
  {R_CCW_NEXT, R_CCW_FINAL, R_CCW_BEGIN, R_START},
};
#endif

static uint8_t ReadPinLevel(GPIO_Port GPIOx, GPIO_Pin PINy); // 內部函數,讀取引腳電平

void RotaryEncoderInit(REHandle_t *REVal) {
    GPIO_InitParam GPIO_InitVal={0};
    // 初始化GPIO引腳
    GPIO_InitVal.driving=GPIO_DRIVING_LEVEL_1;
    GPIO_InitVal.mode=GPIOx_Pn_F0_INPUT;
    GPIO_InitVal.pull=GPIO_PULL_NONE;
    HAL_GPIO_Init(REVal- >GPIO_A, REVal- >PIN_A, &GPIO_InitVal);
    HAL_GPIO_Init(REVal- >GPIO_B, REVal- >PIN_B, &GPIO_InitVal);
    // 初始化狀態機
    REVal- >State=R_START;
}

void RotaryEncoderProcess(REHandle_t *REVal) {
    uint8_t pinstate;
    // 讀取AB端電平
    pinstate=(ReadPinLevel(REVal- >GPIO_B, REVal- >PIN_B)< 1) | 
              ReadPinLevel(REVal- >GPIO_A, REVal- >PIN_A);
    // 狀態機操作
    REVal- >State=ttable[REVal- >State & 0xf][pinstate];
    // 返回編碼器步進信息
    REVal- >RetVal=REVal- >State & 0x30;
}

static uint8_t ReadPinLevel(GPIO_Port GPIOx, GPIO_Pin PINy)
{
    GPIO_PinState RDPin;
    RDPin=HAL_GPIO_ReadPin(GPIOx, PINy);
    #ifdef RE_PINS_PULL_DOWN
    // 如果定義RE_PINS_PULL_DOWN,需要反轉引腳電平
    if(GPIO_PIN_HIGH==RDPin) {
        return 0;
    } else {
        return 1;
    }
    #else // RE_PINS_PULL_DOWN
    if(GPIO_PIN_HIGH==RDPin) {
        return 1;
    } else {
        return 0;
    }
    #endif // RE_PINS_PULL_DOWN
}

編碼器A端和B端分別連接PA12和PA13,使用板載LED即可,引腳是PA21,對應PWM_CH2。開發環境基于FreeRTOS,XR806 SDK在 ~/tools/目錄下。在 ~/tools/xr806_sdk/project/demo/ 目錄下新建 tryre 文件夾,并在其中添加源代碼,makefile等文件,然后按照教程編譯鏈接下載即可。主要代碼如下:
頭文件main.h:

#ifndef __MAIN_H
#define __MAIN_H

#ifdef __cplusplus
extern "C" {
#endif

// 需要包含的頭文件
#include < stdio.h >
#include "driver/chip/hal_gpio.h"
#include "driver/chip/hal_pwm.h"
#include "re.h"
#include "FreeRTOS.h"
#include "task.h"

#ifdef __cplusplus
}
#endif

#endif /* __MAIN_H */

main.c:

#include "main.h"

// 定義編碼器占用的GPIO引腳
static REHandle_t REVal={
    .GPIO_A=GPIO_PORT_A,
    .PIN_A=GPIO_PIN_12,
    .GPIO_B=GPIO_PORT_A,
    .PIN_B=GPIO_PIN_13
};
static const uint8_t STEPMAX=10, STEPMIN=0;
static uint8_t step=0; // 控制LED亮度等級
static void RotaryScan(void); // 編碼器識別與處理
// PWM輸出初始化
#define PWM_CHANNEL   PWM_GROUP1_CH2
#define PWM_MODE      PWM_CYCLE_MODE
#define PWM_GPIO_PORT GPIO_PORT_A
#define PWM_GPIO_PIN  GPIO_PIN_21
#define PWM_GPIO_MODE GPIOA_P21_F4_PWM2_ECT2
static int max_duty_ratio; // PWM計數上限
static void PWMCycleModeSet(void); // PWM重復輸出模式初始化
static HAL_Status PWMDutyRatioSet(int val); // 設置PWM輸出占空比
// FreeRTOS配置
#define TASK_RE_PRIO     1
#define TASK_RE_STK_SIZE 200
static TaskHandle_t TaskRE_Handler=NULL;
static void TaskCreation(void); // 創建任務
static void TaskRE(void *pvParameters); // 編碼器識別任務

int main(void)
{
    printf("Rotary encoder & PWM demo.rn"); // 串口輸出相關信息
    RotaryEncoderInit(&REVal); // 初始化編碼器
    PWMCycleModeSet(); // 初始化PWM
    PWMDutyRatioSet(max_duty_ratio*step/STEPMAX); // 設置PWM輸出占空比
    TaskCreation(); // 創建任務
    // 任務調度不需要用戶指定
    return 0;
}

static void TaskCreation(void)
{
    BaseType_t xRet = NULL;
    taskENTER_CRITICAL();
    xRet = xTaskCreate((TaskFunction_t )TaskRE, (const char *)"TaskRE", (uint16_t)TASK_RE_STK_SIZE, 
                       (void *)NULL, (UBaseType_t)TASK_RE_PRIO, (TaskHandle_t *)&TaskRE_Handler);
    if(pdPASS == xRet) {
        printf("TaskRE created!rn"); // 任務創建成功
    }
    taskEXIT_CRITICAL();
}

static void TaskRE(void *pvParameters)
{
    while (1) {
        RotaryScan(); // 識別編碼器步進
        vTaskDelay(10); // 延遲10(ms)
    }
}

static void RotaryScan(void)
{
    RotaryEncoderProcess(&REVal); // 識別編碼器步進
    if(DIR_CW == REVal.RetVal) { // 順時針步進
        if(step< STEPMAX) {
            step++; // 增大亮度,上限是STEPMAX
            PWMDutyRatioSet(max_duty_ratio*step/STEPMAX);
            printf("%d  ", step);
        }
    } else if(DIR_CCW == REVal.RetVal) {
        if(step >STEPMIN) {
            step--; // 減小亮度,下限是STEPMIN
            PWMDutyRatioSet(max_duty_ratio*step/STEPMAX);
            printf("%d  ", step);
        }
    }
}

static void PWMCycleModeSet(void)
{
    // 初始化硬件所需變量聲明
    GPIO_InitParam io_param = {0};
    HAL_Status status = HAL_ERROR;
    PWM_ClkParam clk_param = {0};
    PWM_ChInitParam ch_param = {0};
    // 配置GPIO復用,官方例程里面缺了這一部分
    io_param.driving = GPIO_DRIVING_LEVEL_1;
    io_param.mode = PWM_GPIO_MODE;
    io_param.pull = GPIO_PULL_NONE;
    HAL_GPIO_Init(PWM_GPIO_PORT, PWM_GPIO_PIN, &io_param);
    // 配置PWM時鐘
    clk_param.clk = PWM_CLK_HOSC;
    clk_param.div = PWM_SRC_CLK_DIV_1;
    status = HAL_PWM_GroupClkCfg(PWM_CHANNEL / 2, &clk_param);
    if (status != HAL_OK) {
        printf("%s(): %d, PWM group clk config errorn", __func__, __LINE__);
    }
    // 配置PWM模式,頻率和極性
    ch_param.hz = 1000;
    ch_param.mode = PWM_MODE;
    ch_param.polarity = PWM_HIGHLEVE;
    max_duty_ratio = HAL_PWM_ChInit(PWM_CHANNEL, &ch_param);
    if (max_duty_ratio == -1) {
        printf("%s(): %d, PWM ch init errorn", __func__, __LINE__);
    }
    // 設置占空比
    status = HAL_PWM_ChSetDutyRatio(PWM_CHANNEL, max_duty_ratio / 2);
    if (status != HAL_OK) {
        printf("%s(): %d, PWM set duty ratio errorn", __func__, __LINE__);
    }
    // 使能通道
    status = HAL_PWM_EnableCh(PWM_CHANNEL, PWM_MODE, 1);
    if (status != HAL_OK) {
        printf("%s(): %d, PWM ch enable errorn", __func__, __LINE__);
    }
}

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

    關注

    5

    文章

    159

    瀏覽量

    26025
  • LED調光
    +關注

    關注

    0

    文章

    61

    瀏覽量

    16372
  • GNU
    GNU
    +關注

    關注

    0

    文章

    143

    瀏覽量

    17542
  • MCU控制
    +關注

    關注

    0

    文章

    48

    瀏覽量

    6790
  • PWM輸出
    +關注

    關注

    1

    文章

    66

    瀏覽量

    5219
收藏 人收藏

    評論

    相關推薦

    XR806開發板適用】基于多源信息融合的糧食霉變檢測系統設計

    XR806開發板具有強大的數據處理和傳輸能力,廣泛應用于物聯網和嵌入式系統開發
    的頭像 發表于 10-27 17:06 ?991次閱讀
    【<b class='flag-5'>XR806</b><b class='flag-5'>開發板</b>適用】基于多源信息融合的糧食霉變檢測系統設計

    基于全志XR806開發板的家用環境監測傳感設計

    本文來自全志XR806開發板評測活動文章,作者使用XR806開發板制作一個簡單的家用環境監測傳感,可以獲取當前的溫度、濕度、氣壓、海拔這四
    的頭像 發表于 12-03 10:20 ?719次閱讀
    基于全志<b class='flag-5'>XR806</b><b class='flag-5'>開發板</b>的家用環境監測傳感<b class='flag-5'>器</b>設計

    如何使用XR806開發板來設計一個糧食霉變檢測系統?

    XR806開發板具有強大的數據處理和傳輸能力,廣泛應用于物聯網和嵌入式系統開發
    的頭像 發表于 01-05 17:30 ?879次閱讀
    如何使用<b class='flag-5'>XR806</b><b class='flag-5'>開發板</b>來設計一個糧食霉變檢測系統?

    OpenHarmony南向嵌入式:【XR806開發板指導文檔】

    XR806_OpenHarmony開發板是基于XR806芯片設計開發的參考評估,身集成了XR806
    的頭像 發表于 04-14 09:25 ?810次閱讀
    OpenHarmony南向嵌入式:【<b class='flag-5'>XR806</b><b class='flag-5'>開發板</b>指導文檔】

    XR806芯片、XR806開源鴻蒙開發板簡介

    。可廣泛滿足 智能家居、智慧樓宇、工業互聯、兒童玩具、電子競賽、極客DIY 等領域的無線連接需求。芯片框圖芯片參數XR806開源鴻蒙開發板介紹XR806開發板是基于
    發表于 11-15 17:08

    XR806芯片、XR806開源鴻蒙開發板簡介

    。可廣泛滿足 智能家居、智慧樓宇、工業互聯、兒童玩具、電子競賽、極客DIY 等領域的無線連接需求。芯片框圖芯片參數XR806開源鴻蒙開發板介紹XR806開發板是基于
    發表于 11-15 18:32

    如何對XR806開發板的環境進行配置呢

    如何對XR806開發板的環境進行配置呢?有哪些配置步驟?
    發表于 12-28 07:51

    請問大佬XR806鴻蒙開發板怎么樣?

    請問大佬XR806鴻蒙開發板怎么樣?
    發表于 12-29 07:06

    簡述星辰處理的全志XR806鴻蒙開發板上手體驗

    本文轉載于極術社區極術專欄:STAR CPU(星辰處理)作者:PingYang XR806鴻蒙開發板是全志科技新出的一款支持WiFi和BLE的高集成度無線MCU芯片,支持鴻蒙L0系統。CPU采用
    的頭像 發表于 11-03 15:00 ?8645次閱讀
    簡述星辰處理<b class='flag-5'>器</b>的全志<b class='flag-5'>XR806</b>鴻蒙<b class='flag-5'>開發板</b>上手體驗

    簡述XR806開發板與STAR-MC1處理

    全志XR806開發板基于XR806芯片設計,搭載了安謀中國的星辰系列STAR-MC1處理。該開發板體積較小,具備WiFi、低功耗藍牙技術等
    的頭像 發表于 04-11 18:08 ?3264次閱讀

    XR806開發板試用】留言功能開發

    XR806開源鴻蒙開發板是一款基于XR806芯片設計,高度集成WiFi/BLE/常用外設,可供開發進行方案評估、DIY或小規模產品研發,可
    的頭像 發表于 10-08 10:58 ?738次閱讀

    XR806開發板試用】FreeRTOS創建任務測試

    這篇來學習下,XR806開發板在FreeRTOS系統下創建兩個任務測試,由于沒有找到學習的文檔,試著參考例程來測試。
    的頭像 發表于 10-10 11:47 ?678次閱讀
    【<b class='flag-5'>XR806</b><b class='flag-5'>開發板</b><b class='flag-5'>試用</b>】FreeRTOS創建任務測試

    使用XR806芯片驅動一下DHT11溫濕度傳感

    參與全志在線與極術社區聯合舉辦的開發板試用活動,收到寄來的全志XR806開發板,之前用過很多全志的SOC芯片,但是像這種無線芯片還是第一次用。這次打算使用
    發表于 10-13 17:28 ?693次閱讀
    使用<b class='flag-5'>XR806</b>芯片驅動一下DHT11溫濕度傳感<b class='flag-5'>器</b>

    XR806串口驅動CM32M對小廚寶的控制實驗

    非常感謝基于安謀科技STAR-MC1的全志XR806 Wi-Fi+BLE開源鴻蒙開發板試用活動,并獲得開發板試用
    的頭像 發表于 10-23 11:39 ?627次閱讀
    <b class='flag-5'>XR806</b>串口驅動CM32M對小廚寶的控制實驗

    全志XR806開發板原理圖

    全志XR806開發板原理圖
    發表于 10-19 15:11 ?12次下載
    主站蜘蛛池模板: 一久久| 国产精品臀控福利在线观看 | 国产精品永久免费自在线观看 | 午夜啪啪免费视频 | 乱好看的的激情伦小说 | 天天插综合 | 久久在线精品 | 亚色在线 | 最近最新视频中文字幕4 | 天堂资源地址在线 | 精品一区二区三区免费毛片爱 | 激情理论 | 四虎影永久地址www 四虎影永久在线观看精品 四虎影永久在线观看网址 四虎影院.com | 亚洲精品美女视频 | 狠狠色丁香久久综合婷婷 | 亚洲欧美色一区二区三区 | 操女人免费视频 | 五月天婷婷社区 | 日本一卡二卡3卡四卡网站精品 | 久草视频资源在线 | 亚洲欧美天堂网 | 福利视频一区二区三区 | 色爱区综合激月婷婷激情五月 | 国内精品久久久久影 | 亚洲αv久久久噜噜噜噜噜 亚洲аv电影天堂网 | 亚洲一区在线播放 | 亚洲专区一区 | 成人黄色免费 | 天天爽天天干 | 日韩一级欧美一级一级国产 | 国产精品入口免费视频 | 国产一级特黄在线播放 | 天天天天操 | 麻豆国产一区二区在线观看 | 色噜噜亚洲男人的天堂 | 五月情视频在线观看 | 中文字幕亚洲天堂 | 亚洲欧美日韩动漫 | 男男失禁play 把尿bl | 亚洲经典一区二区三区 | 老师下面很湿很爽很紧 |