STM32F407 SPI配置并读取磁角度传感器AS5048a笔记

STM32的SPI模块是一种标准的全双工数据传输模块,很多传感器芯片都将其作为标准的数据传输协议。一般来说,STM32F4XX系列都自带3路SPI。

初始化

初始化主要分为两步,IO口初始化和SPI内设初始化。
为使SPI使用起来比较通用,故定义了一些标号。这样更换SPI会非常方便。

//#define SPI1_OPEN
//#define SPI2_OPEN
#define SPI3_OPEN

#ifdef              SPI1_OPEN
    #define Open_SPIx_NSS_PIN                   GPIO_Pin_0
    #define Open_SPIx_NSS_PORT                GPIOA
    #define Open_SPIx_NSS_GPIO_CLK                RCC_AHB1Periph_GPIOA
    
    //#define Open_RCC_APB2Periph_SPIx              RCC_APB2Periph_SPI1

    #define Open_SPIx                           SPI1
    #define Open_SPIx_CLK                       RCC_APB2Periph_SPI1
    #define Open_SPIx_CLK_INIT                  RCC_APB2PeriphClockCmd
    #define Open_SPIx_IRQn                      SPI1_IRQn
    #define Open_SPIx_IRQHANDLER                SPI1_IRQHandler

    #define Open_SPIx_SCK_PIN                   GPIO_Pin_3
    #define Open_SPIx_SCK_GPIO_PORT             GPIOB
    #define Open_SPIx_SCK_GPIO_CLK              RCC_AHB1Periph_GPIOB
    #define Open_SPIx_SCK_SOURCE                GPIO_PinSource3
    #define Open_SPIx_SCK_AF                    GPIO_AF_SPI1

    #define Open_SPIx_MISO_PIN                  GPIO_Pin_4
    #define Open_SPIx_MISO_GPIO_PORT            GPIOB
    #define Open_SPIx_MISO_GPIO_CLK             RCC_AHB1Periph_GPIOB
    #define Open_SPIx_MISO_SOURCE               GPIO_PinSource4
    #define Open_SPIx_MISO_AF                   GPIO_AF_SPI1

    #define Open_SPIx_MOSI_PIN                  GPIO_Pin_5
    #define Open_SPIx_MOSI_GPIO_PORT            GPIOB
    #define Open_SPIx_MOSI_GPIO_CLK             RCC_AHB1Periph_GPIOB
    #define Open_SPIx_MOSI_SOURCE               GPIO_PinSource5
    #define Open_SPIx_MOSI_AF                   GPIO_AF_SPI1

#elif defined       SPI2_OPEN
    #define Open_SPIx_NSS_PIN                   GPIO_Pin_0
    #define Open_SPIx_NSS_PORT                  GPIOA
    #define Open_SPIx_NSS_GPIO_CLK              RCC_AHB1Periph_GPIOA
    //#define Open_SPIx_NSS_SOURCE                GPIO_PinSource10
    
    #define Open_SPIx                           SPI2
    #define Open_SPIx_CLK                       RCC_APB1Periph_SPI2
    #define Open_SPIx_CLK_INIT                  RCC_APB1PeriphClockCmd
    #define Open_SPIx_IRQn                      SPI2_IRQn
    #define Open_SPIx_IRQHANDLER                SPI2_IRQHandler

    #define Open_SPIx_SCK_PIN                   GPIO_Pin_13
    #define Open_SPIx_SCK_GPIO_PORT             GPIOB
    #define Open_SPIx_SCK_GPIO_CLK              RCC_AHB1Periph_GPIOB
    #define Open_SPIx_SCK_SOURCE                GPIO_PinSource13
    #define Open_SPIx_SCK_AF                    GPIO_AF_SPI2

    #define Open_SPIx_MISO_PIN                  GPIO_Pin_14
    #define Open_SPIx_MISO_GPIO_PORT            GPIOB
    #define Open_SPIx_MISO_GPIO_CLK             RCC_AHB1Periph_GPIOB
    #define Open_SPIx_MISO_SOURCE               GPIO_PinSource14
    #define Open_SPIx_MISO_AF                   GPIO_AF_SPI2

    #define Open_SPIx_MOSI_PIN                  GPIO_Pin_15
    #define Open_SPIx_MOSI_GPIO_PORT            GPIOB
    #define Open_SPIx_MOSI_GPIO_CLK             RCC_AHB1Periph_GPIOB
    #define Open_SPIx_MOSI_SOURCE               GPIO_PinSource15
    #define Open_SPIx_MOSI_AF                   GPIO_AF_SPI2
    
#elif defined       SPI3_OPEN
    #define Open_SPIx_NSS_PIN                   GPIO_Pin_0
    #define Open_SPIx_NSS_PORT                  GPIOA
    #define Open_SPIx_NSS_GPIO_CLK              RCC_AHB1Periph_GPIOA
    
    #define Open_SPIx                           SPI3
    #define Open_SPIx_CLK                       RCC_APB1Periph_SPI3
    #define Open_SPIx_CLK_INIT                  RCC_APB1PeriphClockCmd
    #define Open_SPIx_IRQn                      SPI3_IRQn
    #define Open_SPIx_IRQHANDLER                SPI3_IRQHandler

    #define Open_SPIx_SCK_PIN                   GPIO_Pin_10
    #define Open_SPIx_SCK_GPIO_PORT             GPIOC
    #define Open_SPIx_SCK_GPIO_CLK              RCC_AHB1Periph_GPIOC
    #define Open_SPIx_SCK_SOURCE                GPIO_PinSource10
    #define Open_SPIx_SCK_AF                    GPIO_AF_SPI3

    #define Open_SPIx_MISO_PIN                  GPIO_Pin_11
    #define Open_SPIx_MISO_GPIO_PORT            GPIOC
    #define Open_SPIx_MISO_GPIO_CLK             RCC_AHB1Periph_GPIOC
    #define Open_SPIx_MISO_SOURCE               GPIO_PinSource11
    #define Open_SPIx_MISO_AF                   GPIO_AF_SPI3

    #define Open_SPIx_MOSI_PIN                  GPIO_Pin_12
    #define Open_SPIx_MOSI_GPIO_PORT            GPIOC
    #define Open_SPIx_MOSI_GPIO_CLK             RCC_AHB1Periph_GPIOC
    #define Open_SPIx_MOSI_SOURCE               GPIO_PinSource12
    #define Open_SPIx_MOSI_AF                   GPIO_AF_SPI3

#else
    #error "Please select The COM to be used (in spi.h)"
#endif  

初始化的源代码。

void SPI_Configuration(void)
{
    SPI_InitTypeDef SPI_InitStruct;  
    GPIO_InitTypeDef GPIO_InitStructure;
    
    RCC_AHB1PeriphClockCmd(Open_SPIx_SCK_GPIO_CLK | Open_SPIx_MISO_GPIO_CLK | Open_SPIx_MOSI_GPIO_CLK,ENABLE); // 将SPI用到的IO管脚挂在到时钟上
    Open_SPIx_CLK_INIT(Open_SPIx_CLK,ENABLE);   //初始化SPI的时钟,注意,SPI2和SPI3用函数RCC_APB1PeriphClockCmd,SPI1用函数RCC_APB2PeriphClockCmd
    //注意,以上两步实现的功能是不一样的。SPI作为单片机的内设,需要时钟进行工作。IO是不是SPI的一部分,只是可以复用成某一路SPI,IO口也有自己的时钟。
    
    //端口复用
    GPIO_PinAFConfig(Open_SPIx_SCK_GPIO_PORT, Open_SPIx_SCK_SOURCE,  Open_SPIx_MOSI_AF);
    GPIO_PinAFConfig(Open_SPIx_MISO_GPIO_PORT, Open_SPIx_MISO_SOURCE, Open_SPIx_MOSI_AF);
    GPIO_PinAFConfig(Open_SPIx_MOSI_GPIO_PORT, Open_SPIx_MOSI_SOURCE, Open_SPIx_MOSI_AF);
    // 端口复用设置,一般来说对SPI的使用影响并不是很大。
    GPIO_InitStructure.GPIO_Pin = Open_SPIx_SCK_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
    GPIO_InitStructure.GPIO_PuPd  = GPIO_PuPd_UP;  
    GPIO_Init(Open_SPIx_SCK_GPIO_PORT, &GPIO_InitStructure);
    GPIO_InitStructure.GPIO_Pin = Open_SPIx_MISO_PIN;
    GPIO_Init(Open_SPIx_MISO_GPIO_PORT, &GPIO_InitStructure);
    GPIO_InitStructure.GPIO_Pin = Open_SPIx_MOSI_PIN;
    GPIO_Init(Open_SPIx_MOSI_GPIO_PORT, &GPIO_InitStructure);
    
        // SPI的具体配置,每一步都很关键。具体需要查询所操作芯片的datasheet
    SPI_I2S_DeInit(Open_SPIx);
    SPI_InitStruct.SPI_Direction= SPI_Direction_2Lines_FullDuplex;     //
    SPI_InitStruct.SPI_DataSize = SPI_DataSize_16b;                          //一定要确定SPI的位数,这很重要
    SPI_InitStruct.SPI_Mode = SPI_Mode_Master;
    SPI_InitStruct.SPI_CPOL = SPI_CPOL_Low;           //CPOL和下面的CPHA决定了SPI工作的MODE
    SPI_InitStruct.SPI_CPHA = SPI_CPHA_2Edge;
    SPI_InitStruct.SPI_NSS = SPI_NSS_Soft ;
    SPI_InitStruct.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_16;    //分频因子,跟SPI的工作频率有关
    SPI_InitStruct.SPI_FirstBit = SPI_FirstBit_MSB;      //一般来说都是高位在前,具体可以参考传感器的芯片手册
    SPI_InitStruct.SPI_CRCPolynomial = 15;                // 校验,设置7位和设置15位好像没有区别
    SPI_Init(Open_SPIx, &SPI_InitStruct);
        // 使能SPI
    SPI_Cmd(Open_SPIx, ENABLE);
    
    
    /* 初始化NSS/Csn 管脚,初始化片选管脚*/
    RCC_AHB1PeriphClockCmd(Open_SPIx_NSS_GPIO_CLK, ENABLE);
    GPIO_InitStructure.GPIO_Pin = Open_SPIx_NSS_PIN;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_OUT;//
    GPIO_InitStructure.GPIO_PuPd =  GPIO_PuPd_UP;
    GPIO_Init(Open_SPIx_NSS_PORT, &GPIO_InitStructure);//

    GPIO_SetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
}

通过查阅芯片手册可以知悉,AS5048的SPI工作在mode2的模式。并且SPI的位数为16bits。

image.png

SPI发送接收函数

SPI是一种全双工的通信协议,意思是master每发向slave发送一个bit,slave同时也向master发送一个bit。
具体到as5048,SCK的上升沿是master发送slave接收的时刻,下降沿是salve发送master接收的时刻。明白这些时序,对用模拟IO口读取as5048有重要帮助,也对理解SPI的时序有重要帮助。

u16 SPI_ReadWriteData(u16 data)
{
    while(SPI_I2S_GetFlagStatus(Open_SPIx, SPI_I2S_FLAG_TXE)==RESET);
    SPI_I2S_SendData(Open_SPIx,data);

    while(SPI_I2S_GetFlagStatus(Open_SPIx, SPI_I2S_FLAG_RXNE)==RESET);
    return SPI_I2S_ReceiveData(Open_SPIx);
}

此处操作的是STM32F407的库函数,具体到每个单片机,写法不尽相同,是代码移植时替换的核心代码。

as5048读取简介

首先看一下as5048的读取方式

image.png

读取一个寄存器内的数据需要进行两次通信。第一次是发送带有寄存器地址的read command(16位),以表明我们想要采集的是哪个数据;第二次按理说是随便一个command,但是一般来说,可以给出下一步要操作的寄存器,这样节省读取的次数。
一个command package由14位地址、1位r/w标志位和1位校验位共16位组成。

image.png

读回来的数据由14位数据位、1位错误标志位和1位校验位共16位组成。

image.png

as5048清除错误标志

当SPI的工作方式不对、频率过高、两次读取的间隔太短、信号线之间互相干扰、电源不稳等等情况出现时,读取的数据可能会出错,错位标志位会被置高。此时需要清除错误标志位。

image.png

清除错误标志位的方式,即为读取错误标志寄存器。
值得注意的是,如果返回的数据持续有错,在检查接线、供电都没问题的情况下,那很有可能是SPI工作的方式选取的不对或者信号出现了干扰。

as5048读取源程序

struct as5048_data 
{
    uint8_t iserror;
    uint16_t value;
    uint16_t mag;
    uint8_t agc;
    double angle;
    
};

uint16_t SPI2_Read5048Data(uint16_t TxData);
u16 ClearAndNop(void);
struct as5048_data CollectData(void);
// 定义好附加偶校验位的各个寄存器读取指令
#define CMD_ANGLE            0xffff
#define CMD_AGC              0x7ffd
#define CMD_MAG              0x7ffe
#define CMD_CLAER            0x4001
#define CMD_NOP              0xc000


// as5048读取接收函数
uint16_t SPI2_Read5048Data(uint16_t TxData)
{
    uint16_t data;
    GPIO_ResetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
    data=SPI_ReadWriteData(TxData);
    GPIO_SetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
    return data;
}

struct as5048_data CollectData()
{
    uint16_t anglereg = 0, magreg = 0, agcreg = 0;
    uint16_t mag = 0, value = 0;
    double angle = 0.0;
    uint8_t agc = 0;
    struct as5048_data Temp = {1,0,0,0,0.0};

    SPI2_Read5048Data(CMD_ANGLE);SPI2_Read5048Data(CMD_ANGLE);
    anglereg = SPI2_Read5048Data(CMD_MAG); value = anglereg & 0x3fff;
    magreg = SPI2_Read5048Data(CMD_AGC);   mag = magreg & 0x3fff;
    agcreg = SPI2_Read5048Data(CMD_NOP);   agc = (uint8_t)agcreg & 0x00ff;
    angle = (value * 360.0)/16384.0;
    
    if ((anglereg & 0x4000) | (magreg & 0x4000) | (agcreg & 0x4000))
    {
        ClearAndNop();
        //rt_kprintf("There is and error!\n");
        Temp.iserror = 1;
    }
    else
    {
        Temp.iserror = 0;
        Temp.angle = angle;
        Temp.mag = mag;
        Temp.agc = agc;
        Temp.value = value;
    }
    return Temp;
}

// as5048清除错误标志位
u16 ClearAndNop(void)
{
    GPIO_ResetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
    SPI2_Read5048Data(0x4001);              // 附加偶校验的错误标志位清除命令
    GPIO_SetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
    delay_us(10);              // 两次命令之间有350ns的间隔,源自官方datasheet
    GPIO_ResetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
    SPI2_Read5048Data(0xc000);               // 附加偶校验的错误标志位清除命令
    GPIO_SetBits(Open_SPIx_NSS_PORT, Open_SPIx_NSS_PIN);
}

附1 as5048的时序图

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,701评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,649评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,037评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,994评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,018评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,796评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,481评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,370评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,868评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,014评论 3 338
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,153评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,832评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,494评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,039评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,156评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,437评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,131评论 2 356

推荐阅读更多精彩内容