[ CH32V003]  CH32V003 のSPI SPI -- ESP8266 SPI slav

 CH32V003 でのSPIは結構難しいという話で、あまり良い解を紹介しているものではないのであしからず。とりあえず最低限の事をやるといった状態なので。

前回、ESP8266をSPIのスレーブにして、Arduino Nanoをマスターをして通信することをテストした。

https://funasover.blogspot.com/2025/05/esp8266-spi-slaveesp8266spi.html

Arduino Nanoは、互換品でも最近は、パンデミック以降半導体不足以降結構なお値段。そこでATmega328P単体をを購入して使うようにしているが、購入時に、以前からCH32シリーズ気になっていたのでポチリと一緒に注文していた10個かっても400~500円ですから。

ATTinyの愛用者ですが、それよりはるかに安く、容量、クロックともに優れている。

Lチカとかの基本動作は、他のサイトに任せるとし、SPI中心に。

【開発環境】

CH32V003には主に開発環境として3つ

①Arduino IDE

    ・さっと使う事を考えると一番簡単

 ・開発中の模様で、いろいろ動かないものがあるらしい。

②MounRiver

 ・公式IDE。

 ・STM32を意識して作っているので、STM32を使った事があれば理解はすぐ

 ・コマンド詳細を知りたい時は、大抵STM32のコマンドで調べても問題ない。

 ・WCH-LinkE という専用の書き込み機を使って書き込み。線1本。シリアルでの書き込みもできるが、最初にLinkEで設定する必要があるので、いずれにせよ必要。

 ・Pallav Aggarwalさんというありがたい方が、丁寧なチュートリアルを用意してくれているのでここ読めば大抵の事はできる。

https://pallavaggarwal.in/2023/10/01/ch32v003-low-cost-mcu-programming/

 ・ただし、やはり情報量は少ないので、困る事も多い・・・

③CH32V003fun

一番盛んそうだが、未着手。

使い手の方がなかなか初めて触るので、謎が多い。①~③のうち、①、②を試してみた。

まず

①Aruino IDE 

WCHからCH32シリーズのEVTボード用のArduinoIDE向けの開発環境が出ているのでそれを取り込んで使ってみたが、まだまだ開発中の模様。

undefined reference to `PinMap_SPI_MOSI'

というリンカーでエラーが出る・・・サクッといかなさそう。中身なしで#include <SPI.h>しただけで出ていたと思う・・・
もう一つ、UIAPさんのPrpMicroボード用として公開されているので、そちらも使ってみた。
こちらは同じ所でエラーは吐かなくなった。ソース側の問題ではなかった模様。
簡単に済むかと思ったのですが、使い慣れていない事もあり、現時点で不明な事が残る。・
まあかなり惜しい所までできて、ちゃんとSPIで通信するが、1秒ごとに読みとるようにしても、最初の2つぐらいで止まってしまって・・・理由が切り分けできず。
(推測するに、バッファーが空になるまで待機とかそうい所で無限ループに入ってもどってこれらなくなっていたりするのが要因では?とは。まったくの推測で根拠はない。経験上。)
相手が自作のESPのスレーブなので、ちゃんとしたICだと動くかも。

次に②MounRiver 
動かせたは動かせたが、問題が残り、この時点では完全に理解できていない。


255,255,113,10,その後0が続く所まではOK。しかし後半16バイト分が、毎回同じだがパルスが出てくる・・・。ん・・・スレーブに自作のESPを使っているので、そちらが悪いのかさえも切り分けが難しい。arudino nano相手では正しく通信できていたので・・・。

スレーブがESPなので定常状態ではLOWにしている。自作ESP相手でない時は定常状態HIGHで問題ない。

//CH32V003
#include "debug.h"
#include "string.h"

/* SPI Mode Definition */
#define HOST_MODE   0
#define SLAVE_MODE   1

/* SPI Communication Mode Selection */
//#define SPI_MODE   HOST_MODE
#define SPI_MODE   HOST_MODE

/* Global define */
#define Size 32


/* Global Variable */

//u8 TxData[Size] = {0x03, 0x00}
u8 RxData[Size];

/*********************************************************************
 * @fn      SPI_FullDuplex_Init
 *
 * @brief   Configuring the SPI for full-duplex communication.
 *
 * @return  none
 */
void SPI_FullDuplex_Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure={0};
    SPI_InitTypeDef SPI_InitStructure={0};

    RCC_APB2PeriphClockCmd( RCC_APB2Periph_GPIOC | RCC_APB2Periph_SPI1, ENABLE );

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_4;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

#if (SPI_MODE == HOST_MODE)
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_30MHz;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_30MHz;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

#elif (SPI_MODE == SLAVE_MODE)
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_7;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_30MHz;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init( GPIOC, &GPIO_InitStructure );

#endif

    SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;//(SPI_DIRECTION_2LINES_RXONLY //SPI_Direction_2Lines_FullDuplex

#if (SPI_MODE == HOST_MODE)
    SPI_InitStructure.SPI_Mode = SPI_Mode_Master;

#elif (SPI_MODE == SLAVE_MODE)
    SPI_InitStructure.SPI_Mode = SPI_Mode_Slave;

#endif

    SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;
    SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;// SPI_CPOL_Low SPI_CPOL_High
    SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;
    SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;
    SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_8;
    SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;//SPI_FirstBit_MSB
    SPI_InitStructure.SPI_CRCPolynomial = 7;
    SPI_Init( SPI1, &SPI_InitStructure );

    SPI_Cmd( SPI1, ENABLE );
}

void SPISendReceiveBytes(uint8_t *sendData, uint8_t *getData, uint32_t length)
{
    for(int i = 0; i  < length; i ++)
    {
        //Send SPI Byte
        while( SPI_I2S_GetFlagStatus( SPI1, SPI_I2S_FLAG_TXE ) == RESET ); // wait while flag is zero or TX buffer not empty
        SPI_I2S_SendData( SPI1, sendData[i] );

        //Receive SPI Byte
        while(SPI_I2S_GetFlagStatus( SPI1, SPI_I2S_FLAG_RXNE ) == RESET ); // wait while flag is zero or RX buffer is empty
        getData[i] = SPI_I2S_ReceiveData( SPI1 );

    }
}

/*********************************************************************
 * @fn      main
 *
 * @brief   Main program.
 *
 * @return  none
 */
int main(void)
{
    SystemCoreClockUpdate();
    Delay_Init();

    USART_Printf_Init(115200);//460800
    printf("SystemClk:%d\r\n",SystemCoreClock);
    printf( "ChipID:%08x\r\n", DBGMCU_GetCHIPID() );

    GPIO_WriteBit(GPIOC, GPIO_Pin_4, Bit_RESET);
    GPIO_ResetBits(GPIOC, GPIO_Pin_4); // CS LOW
#if (SPI_MODE == SLAVE_MODE)
    printf("Slave Mode\r\n");
    Delay_Ms(1000);

#endif

    SPI_FullDuplex_Init();

#if (SPI_MODE == HOST_MODE)
    printf("Host Mode\r\n");
    Delay_Ms(2000);

#endif

    uint8_t txBuf[34];
    uint8_t rxBuf[34];
    for(int i = 0; i < 34; i++){
        txBuf[i] = 0x0;
        rxBuf[i] = 0x0;
    }
    txBuf[0] = 0x03;

    while(1)
    {
        GPIO_SetBits(GPIOC, GPIO_Pin_4); // CS HIGH
        Delay_Us(5);
        GPIO_ResetBits(GPIOC, GPIO_Pin_4); // CS LOW
        SPISendReceiveBytes(txBuf,rxBuf,34);
        GPIO_SetBits(GPIOC, GPIO_Pin_4); // CS HIGH
        Delay_Us(5);
        GPIO_ResetBits(GPIOC, GPIO_Pin_4); // CS LOW

        for(int i = 0; i < 34; i++){
            printf("%d ",rxBuf[i]);
        }
        printf("\r\n");
        Delay_Ms(1000);

    }
}






コメント

このブログの人気の投稿

Attiny85とAQM0802A(LCD)のI2C接続

CH9329で日本語キーボード109で正しく表示する方法