国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

這篇具有很好參考價值的文章主要介紹了普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

目錄

  • 普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU簡介
  • 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode開發(fā)環(huán)境
  • 普冉PY32系列(三) PY32F002A資源實測 - 這個型號不簡單
  • 普冉PY32系列(四) PY32F002A/003/030的時鐘設(shè)置
  • 普冉PY32系列(五) 使用JLink RTT代替串口輸出日志
  • 普冉PY32系列(六) 通過I2C接口驅(qū)動PCF8574擴展的1602LCD
  • 普冉PY32系列(七) SOP8,SOP10,SOP16封裝的PY32F002A/PY32F003管腳復(fù)用
  • 普冉PY32系列(八) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XN297LBW
  • 普冉PY32系列(九) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XL2400
  • 普冉PY32系列(十) 基于PY32F002A的6+1通道遙控小車I - 綜述篇
  • 普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇
  • 普冉PY32系列(十二) 基于PY32F002A的6+1通道遙控小車III - 驅(qū)動篇

基于PY32F002A的6+1通道遙控小車II - 控制篇

這篇繼續(xù)介紹6+1通道遙控小車的控制端, 關(guān)于遙控手柄的硬件和軟件設(shè)計的說明

PCB實物

正面

在嘉立創(chuàng)下單了PCB, 收到的是這個樣子的.

  • PCB中二極管的位置稍微偏上, 存在與螺絲短接的風險, 在新的PCB設(shè)計中已經(jīng)將其下移.
  • 無線模塊的天線沒有覆漆, 在LCEDA中不知道怎么修改. PCB做出來是焊盤的效果(上錫了), 不影響使用.

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

背面

因為空間限制, PY32F002A和74HC595/165都放到了背面

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

分割后的各個模塊

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

遙控面板成品

遙控面板的焊接過程運氣不錯, 從貼片到接插件都是一次成功, 沒有返工.

正面

  • 空間限制, 只比一張名片稍微大點, 布局比較局促.
  • LCD因為是裸片沒有托板, 和背光板一起是用熱熔膠直接固定在PCB上的.

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

LCD試車, 顯示沒問題

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

背面

  • 正面基本上全是接插件, 如果PY32F002A放到這面, 將來萬一燒壞更換非常麻煩, 所以貼片元件都放到了背面
  • 電源接口用的是XH2.54
  • LCD背光擔心電流過大, 補焊串了一顆1KR的電阻

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

LCD控制界面

這是最終的LCD控制界面

  • 上面兩道橫桿代表旋鈕的模擬量
  • 中間和下方的四道橫桿代表搖桿的模擬量
  • 兩邊的6個數(shù)字代表了模擬量的數(shù)值, 都是8bit, 從0 - 255
  • 下方的8個方格代表了8個開關(guān)量, 高亮(黑)代表按鍵按下(低電壓), 正常(白)代表按鍵松開(高電壓)

普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇

軟件設(shè)計

整體結(jié)構(gòu)

因為只考慮發(fā)送, 所以控制端的流程較為簡單, 做一個大循環(huán)肯定可行, 采集數(shù)據(jù) -> 發(fā)送數(shù)據(jù) -> 采集數(shù)據(jù) -> 發(fā)送數(shù)據(jù). 如果要提升大循環(huán)的效率, 因為LCD顯示和無線發(fā)送共用SPI, 需要保留在大循環(huán), ADC可以用定時器觸發(fā)做成DMA, 節(jié)省出ADC的時間.

最終使用的執(zhí)行流程是

  • 使用一個uint8_t pad_state[8]存儲6+1通道的數(shù)據(jù)
  • ADC使用定時器觸發(fā), 通過DMA存儲轉(zhuǎn)換結(jié)果到6個雙字節(jié)內(nèi)存地址, ADC DMA轉(zhuǎn)換完成后
    • 將結(jié)果轉(zhuǎn)為8bit, 存入 pad_state,
    • 收集74HC165的按鍵狀態(tài), 合成一個byte 也存入pad_state
    • 計算CRC并存至 pad_state 最后一個字節(jié)
  • 外層大循環(huán)讀取 pad_state
    • 更新LCD顯示
    • 通過無線發(fā)送數(shù)據(jù)

主循環(huán)

int main(void)
{
  // ...

  /* Infinite loop */
  while(1)
  {
    // 更新LCD顯示
    DRV_Display_Update(pad_state);
    // 發(fā)送
    wireless_tx++;
    if (XL2400_Tx(pad_state, XL2400_PLOAD_WIDTH) == 0x20)
    {
      wireless_tx_succ++;
    }
    // 每 255 次發(fā)送, 打印一次成功次數(shù), 用于標識成功率
    if (wireless_tx == 0xFF)
    {
      wireless_state[10] = wireless_tx_succ;
      DEBUG_PRINTF("TX_SUCC: %02X\r\n", wireless_tx_succ);
      wireless_tx = 0;
      wireless_tx_succ = 0;
    }
    // 延遲可以調(diào)節(jié)
    LL_mDelay(20);
  }
}

DMA中斷

void DMA1_Channel1_IRQHandler(void)
{
  uint8_t crc = 0;
  if (LL_DMA_IsActiveFlag_TC1(DMA1) == 1)
  {
    LL_DMA_ClearFlag_TC1(DMA1);
    // 轉(zhuǎn)換DMA讀數(shù)為uint8_t并存入pad_state
    for (uint8_t i = 0; i < 6; i++)
    {
      pad_state[i] = (uint8_t)(*(adc_dma_data + i) >> 4);
      crc += pad_state[i];
    }
    // 從 74HC165 讀取按鍵狀態(tài)
    pad_state[6] = HC165_Read();
    // 存入CRC結(jié)果
    pad_state[7] = crc + pad_state[6];
  }
}

無線通訊

無線部分使用的是硬件SPI驅(qū)動的 XL2400, 代碼可以參考
https://github.com/IOsetting/py32f0-template/tree/main/Examples/PY32F0xx/LL/SPI/XL2400_Wireless

傳輸?shù)臄?shù)據(jù)格式為固定長度8字節(jié)

#define XL2400_PLOAD_WIDTH       8   // Payload width

其中字節(jié)[0, 5]為6個ADC采集的數(shù)值結(jié)果, 字節(jié)[6]為74HC165采集的按鍵結(jié)果, 字節(jié)[7]為CRC校驗.

收發(fā)的地址是固定的(將來需要改進)

const uint8_t TX_ADDRESS[5] = {0x11,0x33,0x33,0x33,0x11};
const uint8_t RX_ADDRESS[5] = {0x33,0x55,0x33,0x44,0x33};

輸入采集

ADC采集

DMA初始化

void MSP_DMA_Config(void)
{
  LL_AHB1_GRP1_EnableClock(LL_AHB1_GRP1_PERIPH_DMA1);
  LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_SYSCFG);

  // Remap ADC to LL_DMA_CHANNEL_1
  LL_SYSCFG_SetDMARemap_CH1(LL_SYSCFG_DMA_MAP_ADC);
  // Transfer from peripheral to memory
  LL_DMA_SetDataTransferDirection(DMA1, LL_DMA_CHANNEL_1, LL_DMA_DIRECTION_PERIPH_TO_MEMORY);
  // Set priority
  LL_DMA_SetChannelPriorityLevel(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PRIORITY_HIGH);
  // Circular mode
  LL_DMA_SetMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MODE_CIRCULAR);
  // Peripheral address no increment
  LL_DMA_SetPeriphIncMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PERIPH_NOINCREMENT);
  // Memory address increment
  LL_DMA_SetMemoryIncMode(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MEMORY_INCREMENT);
  // Peripheral data alignment : 16bit
  LL_DMA_SetPeriphSize(DMA1, LL_DMA_CHANNEL_1, LL_DMA_PDATAALIGN_HALFWORD);
  // Memory data alignment : 16bit
  LL_DMA_SetMemorySize(DMA1, LL_DMA_CHANNEL_1, LL_DMA_MDATAALIGN_HALFWORD);
  // Data length
  LL_DMA_SetDataLength(DMA1, LL_DMA_CHANNEL_1, 6);
  // Sorce and target address
  LL_DMA_ConfigAddresses(DMA1, LL_DMA_CHANNEL_1, (uint32_t)&ADC1->DR, (uint32_t)adc_dma_data, LL_DMA_GetDataTransferDirection(DMA1, LL_DMA_CHANNEL_1));
  // Enable DMA channel 1
  LL_DMA_EnableChannel(DMA1, LL_DMA_CHANNEL_1);
  // Enable transfer-complete interrupt
  LL_DMA_EnableIT_TC(DMA1, LL_DMA_CHANNEL_1);

  NVIC_SetPriority(DMA1_Channel1_IRQn, 0);
  NVIC_EnableIRQ(DMA1_Channel1_IRQn);
}

ADC初始化

void MSP_ADC_Init(void)
{
  __IO uint32_t backup_setting_adc_dma_transfer = 0;

  LL_APB1_GRP2_EnableClock(LL_APB1_GRP2_PERIPH_ADC1);

  LL_ADC_Reset(ADC1);
  // Calibrate start
  if (LL_ADC_IsEnabled(ADC1) == 0)
  {
    /* Backup current settings */
    backup_setting_adc_dma_transfer = LL_ADC_REG_GetDMATransfer(ADC1);
    /* Turn off DMA when calibrating */
    LL_ADC_REG_SetDMATransfer(ADC1, LL_ADC_REG_DMA_TRANSFER_NONE);
    LL_ADC_StartCalibration(ADC1);

    while (LL_ADC_IsCalibrationOnGoing(ADC1) != 0);

    /* Delay 1ms(>= 4 ADC clocks) before re-enable ADC */
    LL_mDelay(1);
    /* Apply saved settings */
    LL_ADC_REG_SetDMATransfer(ADC1, backup_setting_adc_dma_transfer);
  }
  // Calibrate end

  /* PA0 ~ PA5 as ADC input */
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_0, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_1, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_2, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_3, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_4, LL_GPIO_MODE_ANALOG);
  LL_GPIO_SetPinMode(GPIOA, LL_GPIO_PIN_5, LL_GPIO_MODE_ANALOG);
  /* Set ADC channel and clock source when ADEN=0, set other configurations when ADSTART=0 */
  LL_ADC_SetCommonPathInternalCh(__LL_ADC_COMMON_INSTANCE(ADC1), LL_ADC_PATH_INTERNAL_NONE);

  LL_ADC_SetClock(ADC1, LL_ADC_CLOCK_SYNC_PCLK_DIV2);
  LL_ADC_SetResolution(ADC1, LL_ADC_RESOLUTION_12B);
  LL_ADC_SetDataAlignment(ADC1, LL_ADC_DATA_ALIGN_RIGHT);
  LL_ADC_SetLowPowerMode(ADC1, LL_ADC_LP_MODE_NONE);
  LL_ADC_SetSamplingTimeCommonChannels(ADC1, LL_ADC_SAMPLINGTIME_41CYCLES_5);

  /* Set TIM1 as trigger source */
  LL_ADC_REG_SetTriggerSource(ADC1, LL_ADC_REG_TRIG_EXT_TIM1_TRGO);
  LL_ADC_REG_SetTriggerEdge(ADC1, LL_ADC_REG_TRIG_EXT_RISING);
  /* Single conversion mode (CONT = 0, DISCEN = 0), performs a single sequence of conversions, converting all the channels once */
  LL_ADC_REG_SetContinuousMode(ADC1, LL_ADC_REG_CONV_SINGLE);

  LL_ADC_REG_SetDMATransfer(ADC1, LL_ADC_REG_DMA_TRANSFER_UNLIMITED);
  LL_ADC_REG_SetOverrun(ADC1, LL_ADC_REG_OVR_DATA_OVERWRITTEN);
  /* Enable: each conversions in the sequence need to be triggerred separately */
  LL_ADC_REG_SetSequencerDiscont(ADC1, LL_ADC_REG_SEQ_DISCONT_DISABLE);
  /* Set channel 0/1/2/3/4/5 */
  LL_ADC_REG_SetSequencerChannels(ADC1, LL_ADC_CHANNEL_0 | LL_ADC_CHANNEL_1 | LL_ADC_CHANNEL_2 | LL_ADC_CHANNEL_3 | LL_ADC_CHANNEL_4 | LL_ADC_CHANNEL_5);

  LL_ADC_Enable(ADC1);

  // Start ADC regular conversion
  LL_ADC_REG_StartConversion(ADC1);
}

用于觸發(fā)ADC的TIM1定時器初始化

void MSP_TIM1_Init(void)
{
  LL_TIM_InitTypeDef TIM1CountInit = {0};

  // RCC_APBENR2_TIM1EN == LL_APB1_GRP2_PERIPH_TIM1 
  LL_APB1_GRP2_EnableClock(RCC_APBENR2_TIM1EN);
  
  TIM1CountInit.ClockDivision       = LL_TIM_CLOCKDIVISION_DIV1;
  TIM1CountInit.CounterMode         = LL_TIM_COUNTERMODE_UP;
  // 系統(tǒng)時鐘48MHz, 預(yù)分頻8K, 預(yù)分頻后定時器時鐘為6KHz
  TIM1CountInit.Prescaler           = (SystemCoreClock / 6000) - 1;
  // 每600次計數(shù)一個周期, 每秒10個周期, 可以減小數(shù)值提高頻率
  TIM1CountInit.Autoreload          = 600 - 1;
  TIM1CountInit.RepetitionCounter   = 0;
  LL_TIM_Init(TIM1, &TIM1CountInit);
  /* Triggered by update */
  LL_TIM_SetTriggerOutput(TIM1, LL_TIM_TRGO_UPDATE);
  LL_TIM_EnableCounter(TIM1);
}

開關(guān)量采集

74HC165的狀態(tài)讀取

uint8_t HC165_Read(void)
{
    uint8_t i, data = 0;

    HC165_LD_LOW;  // Pull down LD to load parallel inputs
    HC165_LD_HIGH; // Pull up to inhibit parallel loading

    for (i = 0; i < 8; i++)
    {
        data = data << 1;
        HC165_SCK_LOW;
        HC165_NOP; // NOP to ensure reading correct value
        if (HC165_DATA_READ)
        {
            data |= 0x01;
        }
        HC165_SCK_HIGH;
    }
    return data;
}

74HC165的示例代碼, 可以參考 https://github.com/IOsetting/py32f0-template/tree/main/Examples/PY32F0xx/LL/GPIO/74HC165_8bit_Parallel_In_Serial_Out

LCD顯示

PY32F002A驅(qū)動ST7567的示例代碼可以參考 Examples/PY32F0xx/LL/SPI/ST7567_128x64LCD, 但是這個示例, 包括GitHub上可以搜到的其它示例, 都是使用 128 x 8 的內(nèi)存作為顯示緩存, 通過讀寫這塊緩存再將緩存內(nèi)容寫入 ST7567 實現(xiàn)的顯示內(nèi)容更新. 這種方式可以實現(xiàn)非常靈活的顯示, 缺點就是需要占用1KB的內(nèi)存. 對于STM32F103這類有16KB或20KB內(nèi)存的控制器, 1KB內(nèi)存不算什么, 但是 PY32F002A 只有4KB內(nèi)存, 1KB就值得考慮一下了. 因為遙控部分的數(shù)顯, 顯示格式相對固定, page之間可以相互獨立, 沒有相互交疊的部分, 啟動后只需要顯示滑動條和讀數(shù), 因此完全可以采用直接輸出的方式.

換成直接輸出后就變成這樣的顯示函數(shù)了, 定制LCD顯示是比較費時費事的一步.

移動光標到坐標

void ST7567_SetCursor(uint8_t page, uint8_t column)
{
    ST7567_WriteCommand(ST7567_SET_PAGE_ADDRESS | (page & ST7567_SET_PAGE_ADDRESS_MASK));
    ST7567_WriteCommand(ST7567_SET_COLUMN_ADDRESS_MSB | ((column + ST7567_X_OFFSET) >> 4));
    ST7567_WriteCommand(ST7567_SET_COLUMN_ADDRESS_LSB | ((column + ST7567_X_OFFSET) & 0x0F));
}

指定寬度和偏移量, 填入固定內(nèi)容

static void DRV_DrawRepeat(uint8_t symbol, uint8_t width, uint8_t offset, uint8_t colorInvert)
{
  symbol = symbol << offset;
  symbol = colorInvert? ~symbol : symbol;
  ST7567_TransmitRepeat(symbol, width);
}

畫出橫條

static void DRV_DrawHorizBar(uint8_t page, uint8_t column, uint8_t size)
{
  ST7567_SetCursor(page, column);
  DRV_DrawRepeat(0x7E, 1, 0, 0);
  DRV_DrawRepeat(0x42, size, 0, 0);
  DRV_DrawRepeat(0x7E, 1, 0, 0);
}

在橫條中畫出高亮滑塊

static void DRV_DrawHorizBarCursor(uint8_t page, uint8_t column, uint8_t value, uint8_t barWidth, uint8_t cursorWidth, uint8_t direction)
{
  value = direction? value : 255 - value;
  ST7567_SetCursor(page, column + 1);
  DRV_DrawRepeat(0x42, barWidth, 0, 0);
  ST7567_SetCursor(page, column + 1 + (value * (barWidth - cursorWidth) / 255));
  DRV_DrawRepeat(0x7E, cursorWidth, 0, 0);
}

畫出豎條和豎條光標的方法更復(fù)雜, 這里就不貼代碼了.

在main函數(shù)的while循環(huán)中, 每次會更新LCD顯示

void DRV_Display_Update(uint8_t *state)
{
  // 更新按鍵顯示
  DRV_DrawKeyState(*(state + 6));
  // 更新4個橫條的顯示
  DRV_DrawHorizBarCursor(0, 10, *(state + 4), 50, 4, 0);
  DRV_DrawHorizBarCursor(0, 65, *(state + 5), 50, 4, 1);
  DRV_DrawHorizBarCursor(7,  0, *(state + 1), 60, 4, 0);
  DRV_DrawHorizBarCursor(7, 65, *(state + 2), 60, 4, 1);
  // 更新2個豎條顯示, 因為豎條處于多個page, 每次更新顯示都需要全部重繪
  DRV_DrawVertiBar(0, 1, 52);
  DRV_DrawVertiBarCursor(0, 1, 52, *(state + 0), 4, 0);
  DRV_DrawVertiBar(121, 1, 52);
  DRV_DrawVertiBarCursor(121, 1, 52, *(state + 3), 4, 1);
  // 輸出6個模擬通道的數(shù)值(0 ~ 255)
  DRV_DrawNumber(1, 10, *(state + 4));
  DRV_DrawNumber(1, 100, *(state + 5));

  DRV_DrawNumber(4, 10, *(state + 0));
  DRV_DrawNumber(4, 100, *(state + 3));

  DRV_DrawNumber(5, 10, *(state + 1));
  DRV_DrawNumber(5, 100, *(state + 2));
}

用直接寫入的方式, 在不開JLink RTT 的情況下, 整機內(nèi)存只需要不到400個字節(jié), 資源節(jié)約效果明顯.文章來源地址http://www.zghlxwxcb.cn/news/detail-746814.html

到了這里,關(guān)于普冉PY32系列(十一) 基于PY32F002A的6+1通道遙控小車II - 控制篇的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔相關(guān)法律責任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • 32 位 ARM? Cortex?-M0+ 單片機,PY32F002B 系列微控制器

    32 位 ARM? Cortex?-M0+ 單片機,PY32F002B 系列微控制器

    PY32F002B 系列微控制器采用高性能的 32 位 ARM? Cortex?-M0+內(nèi)核,寬電壓工作范圍的 MCU。嵌入24Kbytes Flash 和 3Kbytes SRAM 存儲器,最高工作頻率 24MHz。包含多種不同封裝類型多款產(chǎn)品。 芯片集成I2C、SPI、USART 等通訊外設(shè),1 路 12bit ADC,2 個 16bit 定時器,以及 2 路比較器。PY32F002B 系

    2024年02月05日
    瀏覽(24)
  • 普冉PY32系列(十三) SPI驅(qū)動WS2812全彩LED

    普冉PY32系列(十三) SPI驅(qū)動WS2812全彩LED

    普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU簡介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode開發(fā)環(huán)境 普冉PY32系列(三) PY32F002A資源實測 - 這個型號不簡單 普冉PY32系列(四) PY32F002A/003/030的時鐘設(shè)置 普冉PY32系列(五) 使用JLink RTT代替串口輸出日志 普冉PY32系列(六) 通過I2C接口驅(qū)動PCF8574擴

    2024年02月05日
    瀏覽(134)
  • 普冉PY32系列(十四) 從XL2400遷移到XL2400P

    普冉PY32系列(十四) 從XL2400遷移到XL2400P

    普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU簡介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode開發(fā)環(huán)境 普冉PY32系列(三) PY32F002A資源實測 - 這個型號不簡單 普冉PY32系列(四) PY32F002A/003/030的時鐘設(shè)置 普冉PY32系列(五) 使用JLink RTT代替串口輸出日志 普冉PY32系列(六) 通過I2C接口驅(qū)動PCF8574擴

    2024年02月04日
    瀏覽(17)
  • 普冉PY32系列(九) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XL2400

    普冉PY32系列(九) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XL2400

    普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU簡介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode開發(fā)環(huán)境 普冉PY32系列(三) PY32F002A資源實測 - 這個型號不簡單 普冉PY32系列(四) PY32F002A/003/030的時鐘設(shè)置 普冉PY32系列(五) 使用JLink RTT代替串口輸出日志 普冉PY32系列(六) 通過I2C接口驅(qū)動PCF8574擴

    2024年02月08日
    瀏覽(21)
  • 普冉PY32系列(八) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XN297LBW

    普冉PY32系列(八) GPIO模擬和硬件SPI方式驅(qū)動無線收發(fā)芯片XN297LBW

    普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU簡介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode開發(fā)環(huán)境 普冉PY32系列(三) PY32F002A資源實測 - 這個型號不簡單 普冉PY32系列(四) PY32F002A/003/030的時鐘設(shè)置 普冉PY32系列(五) 使用JLink RTT代替串口輸出日志 普冉PY32系列(六) 通過I2C接口驅(qū)動PCF8574擴

    2024年02月08日
    瀏覽(52)
  • 普冉PY32L020單片機簡介,主頻最高48MHZ

    普冉PY32L020單片機簡介,主頻最高48MHZ

    PY32L020單片機是一顆32 位 ARM? Cortex?-M0+內(nèi)核,寬電壓工作范圍的 MCU。這顆MCU的價格跟八位單片機相差不大,性價比可以說是非常的高了。來看看PY32L020的配置吧。 ? PY32L020單片機產(chǎn)品特性: 內(nèi)核: — 32 位 ARM? Cortex? - M0+ — 最高 48MHz 工作頻率 存儲器: — 24Kbytes Flash 存儲器

    2024年02月05日
    瀏覽(17)
  • 普冉32位單片機 PY32C642,M0+內(nèi)核,1.7 V ~ 5.5 V寬工作電壓

    普冉32位單片機 PY32C642,M0+內(nèi)核,1.7 V ~ 5.5 V寬工作電壓

    PY32C642 單片機采用高性能的 32 位 ARM? Cortex?-M0+內(nèi)核,寬電壓工作范圍。嵌入 24Kbytes Flash 和 3 Kbytes SRAM 存儲器,最高工作頻率 24 MHz。包含多種不同封裝類型產(chǎn)品。工作溫度范圍為-40°C ~ 85°C,工作電壓范圍 1.7 V ~ 5.5 V。1 路 12 位ADC,2 個 16 位定時器,以及 2 路比較器。 PY32C6

    2024年01月23日
    瀏覽(23)
  • 新版PY系列離線燒錄器,支持PY002A/002B/003/030/071等MCU各封裝,不同 FLASH 大小型號

    新版PY系列離線燒錄器,支持PY002A/002B/003/030/071等MCU各封裝,不同 FLASH 大小型號

    PY系列離線燒錄器,目前支持PY32F002A/002B/002/003/030/071/072/040/403/303 各封裝、不同 FLASH 大小型號。PY離線燒錄器需要搭配上位機軟件使用,上位機軟件可以在芯嶺技術(shù)官網(wǎng)上下載,還包括了離線燒錄器的使用說明。PY離線燒錄器使用MINI USB線供電,燒錄器僅提供 3.3V 的 SWD 燒錄。需

    2024年01月16日
    瀏覽(16)
  • 基于STM32F103C8T6的高速DMA傳輸多通道ADC數(shù)據(jù)

    ADC在STM32系列單片機的使用中占用著很大的比例,常見的案例是通過ADC單次轉(zhuǎn)換電壓值,這種方式的缺陷在于轉(zhuǎn)換效率不高。一般的單片機帶有ADC1和ADC2兩個ADC轉(zhuǎn)換,單次轉(zhuǎn)換需要執(zhí)行一定的程序,想得到結(jié)果需要耗費一些時間在賦值,調(diào)用中斷上面。在此基礎(chǔ)上,為了提高轉(zhuǎn)

    2024年02月11日
    瀏覽(29)
  • PY32F003F18點燈

    PY32F003F18點燈

    延時函數(shù)學習完之后,可以學習PY32F003F18的GPIO輸出功能。 1、Debug引腳默認被置于復(fù)用功能上拉或下拉模式: PA14默認為SWCLK : 置于下拉模式 PA13默認為SWDIO : 置于上拉模式 PF4默認為Boot :Boot引腳默認置于輸入下拉模式 2、GPIO輸出狀態(tài): 1)、push-pull,推挽輸出 2)、open drain,開漏

    2024年02月10日
    瀏覽(15)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包