STM32:GPIO模拟SPI驱动ADS8361

ADS8361是TI公司开发的一款模拟量输入芯片。ADS8361有四种工作模式,本文主要针对模式三进行通信驱动。官方方案使用两路SPI来通信,一路SPI Master,一路SPI Slave。我在使用STM32主控芯片的两路SPI进行通信的时候,发现只有SPI Master这路可以正常工作。Slave一直无法接收到数据。估计是两路SPI同时通信会有问题。所以使用了GPIO模拟SPI来驱动ADS8361。

硬件系统接线示意图:

模式三通信时序图:

 

 下面是驱动程序:

/*Global variables*/
float Ain1, Ain2 , Ain3;

static float Calculate_Volt_By_SPI_Data(uint8_t* data)
{
    uint32_t temp1, temp2, temp3,  temp4;

    temp1 = data[0] & 0X1F;
    temp2 = data[1];
    temp3 = data[2] >> 5;
    temp4 = temp1 * 2048 + temp2 * 8 + temp3;

    if(temp4 >= 0x8000)
    {
        temp4 -= 0x8000;
    }
    else
    {
        temp4 += 0x8000;
    }

	return 10.0 - (float)temp4/65535*20;
}

static void Monitor_AIN_Level(void)
{
    uint8_t SPI1_RD[3] = {0x00, 0x00, 0x00};
    uint8_t SPI2_RD[3] = {0x00, 0x00, 0x00};
    uint8_t Invalid_ARR[3] = {0x00, 0x00, 0x00};

    for(int i = 0; i < 20; i++)
    {
        HAL_GPIO_WritePin(SPI1_SCK_GPIO_Port, SPI1_SCK_Pin, GPIO_PIN_SET);

        if(i == 0)
        {
            HAL_GPIO_WritePin(SPI1_MOSI_GPIO_Port, SPI1_MOSI_Pin, GPIO_PIN_SET);
        }
        else if(i == 1)
        {
            HAL_GPIO_WritePin(SPI1_MOSI_GPIO_Port, SPI1_MOSI_Pin, GPIO_PIN_RESET);
        }

        delay_us(1);

        HAL_GPIO_WritePin(SPI1_SCK_GPIO_Port, SPI1_SCK_Pin, GPIO_PIN_RESET);

        if(HAL_GPIO_ReadPin(SPI1_MISO_GPIO_Port, SPI1_MISO_Pin) == GPIO_PIN_SET)
        {
            SPI1_RD[i/8] += (1 << (7 - (i % 8)));
        }

        if(HAL_GPIO_ReadPin(SPI2_MOSI_GPIO_Port, SPI2_MOSI_Pin) == GPIO_PIN_SET)
        {
            SPI2_RD[i/8] += (1 << (7 - (i % 8)));
        }

        delay_us(1);
    }

    HAL_GPIO_WritePin(SPI1_SCK_GPIO_Port, SPI1_SCK_Pin, GPIO_PIN_RESET);
 
    if(memcmp(SPI1_RD, Invalid_ARR, sizeof(SPI1_RD)) == 0 ||  memcmp(SPI2_RD, Invalid_ARR, sizeof(SPI2_RD)) == 0) 
    {
        return;
    }

    if(SPI1_RD[0] >= 0x60 || SPI2_RD[0] >= 0x60)
    {
        return;
    }

    if((SPI1_RD[0] & 0x40) > 0) Ain2 = Calculate_Volt_By_SPI_Data(SPI1_RD);
    else Ain1 = Calculate_Volt_By_SPI_Data(SPI1_RD);
    
    if((SPI2_RD[0] & 0x40) == 0) Ain3 = Calculate_Volt_By_SPI_Data(SPI2_RD);
    
    //Usart_Printf("Ain1: %f\n", Ain1);
    //Usart_Printf("Ain2: %f\n", Ain2);
    //Usart_Printf("Ain3: %f\n", Ain3);
}

 三路模拟量输入通道的测试结果:

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
ADS1256是一款高精度、低功耗的24位模数转换器(ADC),常用于电子测量、仪器仪表、传感器接口等领域。STM32F103是一款基于ARM Cortex-M3内核的微控制器,拥有丰富的外设资源,其中包括SPI接口。 为了驱动ADS1256,我们需要通过SPI接口与其进行通信。以下是基于STM32F103的ADS1256驱动程序示例: 首先,我们需要定义一些常量和变量: ``` #define ADS1256_SPI SPI1 #define ADS1256_CS GPIO_Pin_4 #define ADS1256_START 0x08 #define ADS1256_WAKEUP 0x00 #define ADS1256_RDATA 0x01 #define ADS1256_RDATAC 0x03 #define ADS1256_SDATAC 0x0F #define ADS1256_RREG 0x10 #define ADS1256_WREG 0x50 #define ADS1256_SELFCAL 0xF0 #define ADS1256_SELFOCAL 0xF1 #define ADS1256_SELFGCAL 0xF2 #define ADS1256_SYSOCAL 0xF3 #define ADS1256_SYSGCAL 0xF4 #define ADS1256_SYNC 0xFC #define ADS1256_STANDBY 0xFD #define ADS1256_RESET 0xFE uint8_t ADS1256_Buffer[3]; ``` 这里定义了SPI接口、ADS1256的CS引脚、ADS1256的指令常量以及一个用于缓存数据的数组。 接着,我们需要实现SPI的初始化函数: ``` void ADS1256_SPI_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; SPI_InitTypeDef SPI_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5 | GPIO_Pin_6 | GPIO_Pin_7; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); GPIO_InitStructure.GPIO_Pin = ADS1256_CS; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex; SPI_InitStructure.SPI_Mode = SPI_Mode_Master; SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b; SPI_InitStructure.SPI_CPOL = SPI_CPOL_Low; SPI_InitStructure.SPI_CPHA = SPI_CPHA_1Edge; SPI_InitStructure.SPI_NSS = SPI_NSS_Soft; SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64; SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB; SPI_Init(ADS1256_SPI, &SPI_InitStructure); SPI_Cmd(ADS1256_SPI, ENABLE); ADS1256_CS_HIGH(); } ``` 该函数首先初始化GPIOSPI外设,并将ADS1256的CS引脚设置为输出模式。然后设置SPI的工作模式、传输速率等参数,并使能SPI外设。 接下来,我们可以编写一个函数用于向ADS1256发送指令: ``` void ADS1256_SendCommand(uint8_t command) { ADS1256_CS_LOW(); SPI_I2S_SendData(ADS1256_SPI, command); while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_BSY) == SET); ADS1256_CS_HIGH(); } ``` 该函数将ADS1256的CS引脚拉低,发送指令,等待传输完成,然后将CS引脚拉高。 接着,我们可以编写一个函数用于向ADS1256写寄存器: ``` void ADS1256_WriteReg(uint8_t reg, uint8_t data) { ADS1256_CS_LOW(); ADS1256_Buffer[0] = ADS1256_WREG | reg; ADS1256_Buffer[1] = 0x00; ADS1256_Buffer[2] = data; SPI_I2S_SendData(ADS1256_SPI, ADS1256_Buffer[0]); while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_BSY) == SET); SPI_I2S_SendData(ADS1256_SPI, ADS1256_Buffer[1]); while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_BSY) == SET); SPI_I2S_SendData(ADS1256_SPI, ADS1256_Buffer[2]); while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_BSY) == SET); ADS1256_CS_HIGH(); } ``` 该函数首先将ADS1256的CS引脚拉低,然后发送写寄存器指令和寄存器地址。接着,发送两个字节的数据,等待传输完成,最后将CS引脚拉高。 最后,我们可以编写一个函数用于从ADS1256读取数据: ``` int32_t ADS1256_ReadData(void) { uint32_t i; uint32_t timeout = 100000; ADS1256_CS_LOW(); ADS1256_Buffer[0] = ADS1256_RDATA; SPI_I2S_SendData(ADS1256_SPI, ADS1256_Buffer[0]); while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_BSY) == SET); for (i = 0; i < 3; i++) { timeout = 100000; while (SPI_I2S_GetFlagStatus(ADS1256_SPI, SPI_I2S_FLAG_RXNE) == RESET) { if (--timeout == 0) { ADS1256_CS_HIGH(); return 0; } } ADS1256_Buffer[i] = SPI_I2S_ReceiveData(ADS1256_SPI); } ADS1256_CS_HIGH(); return ((int32_t)ADS1256_Buffer[0] << 16) | ((int32_t)ADS1256_Buffer[1] << 8) | ((int32_t)ADS1256_Buffer[2]); } ``` 该函数首先将ADS1256的CS引脚拉低,然后发送读数据指令。接着,循环读取三个字节的数据,等待传输完成,最后将CS引脚拉高。返回值为读取到的数据的24位有符号整数值。 以上就是基于STM32F103的ADS1256驱动程序示例。需要注意的是,该示例仅提供了基本的SPI通信和指令发送功能,具体的应用需要根据实际情况进行修改和完善。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AllenSun-1990

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值