spi驱动--IO模拟

该spi驱动,应用于stm32f103 访问外部的flash(23K256).

  1. 23K256.c
#include "includes.h"
/*
1.  spi根据配置可以使用为半双工或者全双工,全双工下总共有四个引脚,MISO,MOSI,SCK,CS
    MISO: 主入从出
    MOSI: 主出从入
    SCK: 时钟线
    CS: 片选
2. spi可以有四种工作模式,有CPOL和CPHA决定:
    - CPOL: 时钟极性,决定是SCK的空闲态是低电平还是高电平,CPOL=0:空闲态是低电平;CPOL=1:空闲态是高电平。
    - CPHA:时钟相位,决定采样的点,是第一个时钟沿还是第二个,CPHA=0,采样在第一个时钟延,输出在第二个时钟沿;CPHA=1,输出在第一个时钟沿,采样在第二个时钟沿。
                   CPOL           CPHA
    Mode 0 :        0               0  
    Mode 1 :        0               1 
    Mode 2 :        1               0
    Mode 3 :        1               1
一般工作模式对于特定芯片是固定的,可以看时序图确定。
3. spi数据传输,先高位后低位。
/*
此处,spi协议工作在mode0,低电平的idle状态,采样在第一个边沿(即下降沿)。
接口:
    1. spi协议写一个字节
        主机写一个字节,就是输出,所以写MOSI数据(先高位后低位),然后时钟线产生一个下降沿,将数据输出。
    2. spi协议读一个字节
        主机读一个数据,就是采样,所以时钟产生一个上升沿,开始采样MISO线数据(先高位后低位)
-------------------上部分严格和spi协议相关---------------------
-----------------下部分需要参照对应的spi器件手册----------------
    3. spi发送一个字节
        - 片选拉低,选中SPI从机
        - 发送写指令
        - 发送地址高位
        - 发送地址低位
        - 发送数据(此处可以单字节或多字节发送)
        - 片选拉高,取消SPI从机
    4. spi接收一个字节
        - 片选拉低,选中SPI从机
        - 发送读指令
        - 发送地址高位
        - 发送地址低位
        - 读取数据(此处单字节或多字节接收)
        - 片选拉高,取消SPI从机
*/
*/
//1.spi gpio_init 
void spi_gpio_config(void)
{
    //CS: GPB12  SCK: GPB13  MISO:PB14  MOSI:PB15
    GPIO_InitTypeDef GPIO_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
    //NSS
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
    //SCK
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_13;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
    //MISO
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_14;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;    

    GPIO_Init(GPIOB, &GPIO_InitStructure);
    //MOSI
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_15;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOB, &GPIO_InitStructure);
}
//2. 主机向从机写数据  上升沿读取写入数据
void spi_write_byte(uint8_t data)
{
    uint8_t i;
    SPI_SCK(0);
    delay(10);
    for(i = 0;i<8;i++)
    {
        if(data&0x80)
            SPI_MOSI(1);
        else
            SPI_MOSI(0);
        data<<=1;

        delay(10);
        SPI_SCK(1);
        delay(10);

        SPI_SCK(0);  //out
    delay(10);
/*      
        delay(10);
        SPI_SCK(1);
        delay(10);

        SPI_SCK(0);
        //delay(10);
*/      
    }
}
//3. 从机向主机传送一个数据
uint8_t spi_read_byte(void)
{
    uint8_t i;
    uint8_t temp;
    SPI_SCK(0);
//  delay(10);
    temp = 0;
    for(i = 0;i<8;i++)
    {       
        SPI_SCK(1); 
        delay(10);
        temp<<=1;
        if(SPI_MISO())
            temp |= 0x01;
        else
            temp &= ~0x01; 
        SPI_SCK(0);
        delay(10);
    }       
    return temp;
}
//4. write one byte
void spi_send_byte(uint16_t addr,uint8_t dat)
{
    SPI_NSS(0); 
    spi_write_byte(SPI_WRITE_CMD);
    spi_write_byte(AddrHigh(addr));
    spi_write_byte(AddrLow(addr));
    spi_write_byte(dat);

    SPI_NSS(1); 
}
//5. read one byte
uint8_t spi_recv_byte(uint16_t addr)
{
    uint8_t temp = 0;

    SPI_NSS(0); 
    spi_write_byte(SPI_READ_CMD);
    spi_write_byte(AddrHigh(addr));
    spi_write_byte(AddrLow(addr));
    temp = spi_read_byte();
    SPI_NSS(1);

    return temp;
}
//6. spi_init
void spi_init(void)
{
    spi_gpio_config();
    SPI_SCK(0);
    SPI_NSS(1);

}
//7. spi_buff_write
void spi_write_buff(uint8_t *pBuff,uint16_t WriteAddress,uint16_t BufferSize)
{
    uint8_t i;
    SPI_NSS(0); 
    //发送写指令
    spi_write_byte(SPI_WRITE_CMD);
    //写寄存器地址
    spi_write_byte(AddrHigh(WriteAddress));
    spi_write_byte(AddrLow(WriteAddress));
    //写数据
    for(i = 0;i<BufferSize;i++){
        spi_write_byte(pBuff[i]);
    }
    SPI_NSS(1);  
}

//8. spi_buff_read
void spi_read_buff(uint8_t *pBuff,uint16_t ReadAddress,uint16_t BufferSize)
{
    uint8_t i;
    SPI_NSS(0); 
    delay(20);
    //发送读指令
    spi_write_byte(SPI_READ_CMD);
    //发送寄存器地址
    spi_write_byte(AddrHigh(ReadAddress));
    spi_write_byte(AddrLow(ReadAddress));
    //读取数据
    for(i = 0;i<BufferSize;i++)
    {
        pBuff[i] = spi_read_byte();
    }
    SPI_NSS(1); 
    delay(20);
}
//9. spi read status
uint8_t spi_read_status(void)
{
    uint8_t spi_status = 0;

    SPI_NSS(0); 

    spi_write_byte(SPI_RDSR_CMD);   
    spi_status = spi_read_byte();

    SPI_NSS(1); 
    return spi_status;

}
//10. spi write status
void spi_write_status(uint8_t stas)
{
    SPI_NSS(0); 

    spi_write_byte(SPI_WRSP_CMD);
    spi_write_byte(stas);

    SPI_NSS(1); 
}
//////////////////////////以下为测试使用/////////////////////////////////
// spi chip test 0: success -1: failed, bytes num: 32768
int sram_23k256_test(void)
{
      int i,j;
        uint8_t temp;
        int DatAddr = 0;

        printf("spi testing %d....\r\n",systick_ms_counter);

        for(i = 0;i<128;i++)
        {
            for(j = 0;j<256;j++)  //写数据
            {
                spi_send_byte(DatAddr,j);
                DatAddr++;
            }
            for(j = 0;j<256;j++)  //读数据
            {
                DatAddr--;
                temp = spi_recv_byte(DatAddr);
                if(temp != (DatAddr%256)){
                    return -1;
                }
            }
            DatAddr += 256;
        }
        printf("spi test complete %d!\r\n",systick_ms_counter);
        return 0;
}
// spi sck 时序测试
void spi_sck_test(void)
{
      SPI_SCK(0);
        delay(10);
        SPI_SCK(1);
        delay(10);
}

  1. 23K256.h
#ifndef __23k256_H
#define __23k256_H

#include "stm32f10x.h"

#define SPI_READ_CMD  0X03
#define SPI_WRITE_CMD 0x02
#define SPI_RDSR_CMD  0x05
#define SPI_WRSP_CMD  0x01

#define SPI_PAGES_MAX  (1024)      //页大小
#define SPI_BYTES_MAX  (32768)     //总字节个数

#define  SPI_MOSI(bitval)  GPIO_WriteBit(GPIOB,GPIO_Pin_15,(BitAction)bitval)                                                   
#define  SPI_SCK(bitval)   GPIO_WriteBit(GPIOB,GPIO_Pin_13,(BitAction)bitval)
#define  SPI_NSS(bitval)   GPIO_WriteBit(GPIOB,GPIO_Pin_12,(BitAction)bitval)
#define  SPI_MISO()          (GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_14)==RESET ? 0:1)


void spi_init(void);
uint8_t spi_recv_byte(uint16_t addr);
void spi_send_byte(uint16_t addr,uint8_t dat);

uint8_t spi_read_status(void);
void spi_write_status(uint8_t stas);
void spi_write_buff(uint8_t *pBuff,uint16_t WriteAddress,uint16_t BufferSize);
void spi_read_buff(uint8_t *pBuff,uint16_t ReadAddress,uint16_t BufferSize);


int sram_23k256_test(void);
void spi_sck_test(void);






#endif
  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: BL0939是一款数字温湿度传感器,通常使用SPI接口与主控芯片通信。下面是一个使用C语言模拟BL0939的SPI驱动的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <stdint.h> #define SPI_MODE 0 /* SPI模式,0表示模式0,CPOL=0,CPHA=0 */ /* 模拟SPI总线的IO口 */ #define MOSI_PIN 1 /* MOSI口 */ #define MISO_PIN 2 /* MISO口 */ #define SCK_PIN 3 /* 时钟口 */ #define CS_PIN 4 /* 片选口 */ /* 定义BL0939寄存器地址 */ #define BL0939_TEMPERATURE_REG 0x00 #define BL0939_HUMIDITY_REG 0x01 /* 模拟SPI总线传输一个字节 */ uint8_t spi_transfer(uint8_t data) { uint8_t recv_data = 0; int i; for (i = 7; i >= 0; i--) { /* 先将时钟置低 */ digitalWrite(SCK_PIN, 0); /* 将数据写入MOSI口 */ digitalWrite(MOSI_PIN, (data >> i) & 1); /* 模拟SPI传输延迟 */ delay(1); /* 拉高时钟,读取MISO口的数据 */ recv_data |= digitalRead(MISO_PIN) << i; digitalWrite(SCK_PIN, 1); /* 模拟SPI传输延迟 */ delay(1); } return recv_data; } /* 从BL0939读取温度数据 */ float bl0939_read_temperature() { uint8_t tx_data[3] = {0x03, BL0939_TEMPERATURE_REG, 0x00}; uint8_t rx_data[3]; /* 拉低片选口 */ digitalWrite(CS_PIN, 0); /* 发送读取温度命令和寄存器地址 */ int i; for (i = 0; i < 3; i++) { spi_transfer(tx_data[i]); } /* 接收温度数据 */ for (i = 0; i < 3; i++) { rx_data[i] = spi_transfer(0); } /* 拉高片选口 */ digitalWrite(CS_PIN, 1); /* 计算温度值 */ int16_t temp_raw = (rx_data[1] << 8) | rx_data[2]; return (float)temp_raw / 100.0; } /* 从BL0939读取湿度数据 */ float bl0939_read_humidity() { uint8_t tx_data[3] = {0x03, BL0939_HUMIDITY_REG, 0x00}; uint8_t rx_data[3]; /* 拉低片选口 */ digitalWrite(CS_PIN, 0); /* 发送读取湿度命令和寄存器地址 */ int i; for (i = 0; i < 3; i++) { spi_transfer(tx_data[i]); } /* 接收湿度数据 */ for (i = 0 ### 回答2: 在C语言中实现BL0939的模拟IO口的SPI驱动,可以采用以下步骤: 1.定义必要的引脚和寄存器的宏: 首先,我们需要定义与BL0939连接的引脚和相关寄存器的宏。这些宏将帮助我们在代码中方便地访问这些引脚和寄存器。 2.编写初始化函数: 初始化函数应包含引脚和寄存器的初始化,以及SPI的设置。这些包括设置引脚为输出或输入,配置SPI控制器的参数(如数据位数、时钟速度等)。 3.编写发送数据函数: 发送数据函数用于向BL0939写入数据。首先,我们将写入片选信号使BL0939响应。然后,我们可以使用SPI控制器发送数据。 4.编写接收数据函数: 接收数据函数用于从BL0939读取数据。首先,我们将写入片选信号使BL0939响应。然后,我们可以使用SPI控制器接收数据。 5.编写其他必要的功能函数: 根据BL0939的需求,可能还需要编写其他的功能函数。这些函数可以包括设置寄存器、读取状态等。 6.编写主函数: 在主函数中,我们可以使用上述编写的函数进行测试。我们可以编写代码以模拟与BL0939的交互,例如设置寄存器的值并读取其状态。 需要注意的是,由于无法直接访问硬件,我们在编写完整代码后,可以使用模拟器或硬件评估板进行测试。此外,还应仔细阅读BL0939的数据手册以了解其具体的IO口和SPI驱动要求。 ### 回答3: BL0939是一种模拟IO口的SPI(Serial Peripheral Interface)驱动芯片。要用C语言实现BL0939的模拟IO口的SPI驱动,首先需要了解BL0939的通信协议和功能。 BL0939通信协议基于SPI的主从模式,其中主设备发送的数据帧由8位控制字和8位数据字组成,从设备通过IO口进行通信。下面是一个简单的C语言示例代码,展示了如何实现BL0939的模拟IO口的SPI驱动。 ```c #include <stdio.h> #include <wiringPi.h> // 定义BL0939的IO口 #define BL0939_CS 10 #define BL0939_CLK 11 #define BL0939_IN 12 #define BL0939_OUT 13 // 初始化BL0939的IO口 void bl0939_init() { wiringPiSetup(); pinMode(BL0939_CS, OUTPUT); pinMode(BL0939_CLK, OUTPUT); pinMode(BL0939_IN, INPUT); pinMode(BL0939_OUT, OUTPUT); } // 发送数据帧到BL0939 void bl0939_send_frame(unsigned char control, unsigned char data) { digitalWrite(BL0939_CS, LOW); // 使能BL0939 // 发送控制字 for (int i = 7; i >= 0; i--) { digitalWrite(BL0939_CLK, LOW); // 下降沿时发送位 digitalWrite(BL0939_OUT, (control >> i) & 0x01); digitalWrite(BL0939_CLK, HIGH); // 上升沿时采样位 } // 发送数据字 for (int i = 7; i >= 0; i--) { digitalWrite(BL0939_CLK, LOW); // 下降沿时发送位 digitalWrite(BL0939_OUT, (data >> i) & 0x01); digitalWrite(BL0939_CLK, HIGH); // 上升沿时采样位 } digitalWrite(BL0939_CS, HIGH); // 禁止BL0939 } int main() { bl0939_init(); // 初始化BL0939的IO口 unsigned char control = 0x01; // 控制字 unsigned char data = 0x55; // 数据字 bl0939_send_frame(control, data); // 发送数据帧到BL0939 return 0; } ``` 在上述示例代码中,首先定义了BL0939的IO口引脚,并在`bl0939_init`函数中进行了初始化。然后在`bl0939_send_frame`函数中对控制字和数据字进行SPI通信,按位发送位和采样位。 在`main`函数中,定义了控制字和数据字,并调用`bl0939_send_frame`函数发送数据帧到BL0939。 需要注意的是,以上代码仅为示例,实际使用时还需要根据具体的硬件平台及接线进行适配和调整。同时还需要根据BL0939的功能文档,结合具体的需求进行更详细的代码编写。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值