stm32 GPIO模拟SPI接口实现双机通信

这篇具有很好参考价值的文章主要介绍了stm32 GPIO模拟SPI接口实现双机通信。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、SPI协议简介
    一般主从方式工作,这种模式通常有一个主设备和一个或多个从设备,通常采用的是4根线,它们是MISO(主机输入从机输出)、MOSI(主机输出,针对主机来说)、SCLK(时钟,主机产生)、CS(片选,一般由主机发送或者直接使能,通常为低电平有效)

●SPI接口介绍

SCK:时钟信号,由主设备产生,所以主设备SCK信号为推挽输出模式,从设备的SCK信号为浮空输入模式。

CS:使能信号,由主设备控制从设备,,所以主设备CS信号为推挽输出模式,从设备的CS信号为浮空输入模式。

MOSI:主设备数据输出,从设备数据输入,所以主设备MOSI信号为推挽输出模式,从设备的MOSI信号为浮空输入模式。

MISO:主设备数据输入,从设备数据输出,所以主设备MISO信号为浮空输入模式,从设备的MISO信号为推挽输出模式。

二、四种模式(本次模拟采用的模式0)

模式0:CPOL=0,CPHA =0  SCK空闲为低电平,数据在SCK的上升沿被采样(提取数据)

模式1:CPOL=0,CPHA =1  SCK空闲为低电平,数据在SCK的下降沿被采样(提取数据)

模式2:CPOL=1,CPHA =0  SCK空闲为高电平,数据在SCK的下降沿被采样(提取数据)

模式3:CPOL=1,CPHA =1  SCK空闲为高电平,数据在SCK的上升沿被采样(提取数据)

三、双机通信实现

1、GPIO引脚

PC8为CS引脚        PC10为CLK引脚        PC11为MISO引脚        PC12为MOSI引脚

2、时序

首先主机将片选拉低(使能片选),在时钟为低电平时向从机发送数据,从机通过检测MOSI线上的高低电平实现数据的接收。在时钟为高电平时主机检测MISO线上的高低电平来实现数据的接收。

主机在时钟信号为低电平时发送单字节的最高位,然后将该字节左移一位,然后将时钟信号拉高,此时从机检测到时钟信号为高电平(时钟上升沿),从而检测MOSI线上的高低电平并将得到的高低电平放到变量中。同时从机向主机发送数据,即改变MISO线上的高低电平。此过程重复8次,即可完成发送和接收一个字节的数据。

四、代码如下

1、主设备GPIO的配置

void SPI_GPIO_Init(void){
	GPIO_InitTypeDef GPIO_InitStructure;			//定义结构体类型的变量
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);	 		//使能GPIOC的端口时种
	
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_8|GPIO_Pin_10|GPIO_Pin_12;	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;					//推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;					//IO口速度为50Mhz
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_11;	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;					//浮空输入
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	
	GPIO_SetBits(GPIOC,GPIO_Pin_8);	//CS拉高
	GPIO_ResetBits(GPIOC,GPIO_Pin_10);//CLK拉低
	GPIO_ResetBits(GPIOC,GPIO_Pin_11);	
}

2、主设备向从设备写以及读

/*
*********************************************************************************************************
*	函 数 名: SPI_WRByte
*	形    参: spi; wdat:写入的数据
*	返 回 值: spi读一个字节
*	功能说明: 主机spi同时读写一个字节的时序		MSB
*********************************************************************************************************
*/
u8 Master_SPI_WRByte(u8 wdat)
{
	u8 i=0,rdat=0;
	SPI_CS_LOW;
	for(i=0;i<8;i++)
	{
		if(wdat&0x80)
			SPI_MOSI_HIGH;
		else
			SPI_MOSI_LOW;
		wdat<<=1;
		delay_us(3);	
				
		SPI_CLK_HIGH;
				
		delay_us(2);
		rdat<<=1;
		if(SPI_MISO_READ)
			rdat |= 0x01;
		
		delay_us(1);
		
		SPI_CLK_LOW;
	}
	SPI_CS_HIGH;
	return rdat;	
}

3、从设备GPIO设置

void SPI_GPIO_Init(void){
	GPIO_InitTypeDef GPIO_InitStructure;			//定义结构体类型的变量
	
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);	 		//使能GPIOC的端口时种
	
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_11;	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;					//推挽输出
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;					//IO口速度为50Mhz
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	
	GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_8|GPIO_Pin_10|GPIO_Pin_12;	
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;					//浮空输入
	GPIO_Init(GPIOC,&GPIO_InitStructure);
	
	GPIO_ResetBits(GPIOC,GPIO_Pin_11);	
}

4、从设备读写函数

u8 Slave_SPI_RWByte(u8 wdat){
	u8 i=0,dat=0;
	if(!SPI_CS_READ){		//检测到片选拉低
		for(i=0;i<8;i++)
		{		
			while(!SPI_CLK_READ);		//检测到时钟上升沿的到来	
						
			if(wdat&0x80)SPI_MISO_HIGH;	//MSB
			else SPI_MISO_LOW;
			wdat<<=1;
			
			dat<<=1;
			if(SPI_MOSI_READ)dat |= 0x01;				
			while(SPI_CLK_READ);		
		}			
	}	
	return dat;	
}

5、头文件

#define SELECT    1		//0表示主机  其它表示从机

#if SELECT==0
//主机
//CS引脚
#define      SPI_CS_HIGH        PCout(8)=1
#define      SPI_CS_LOW         PCout(8)=0

//CLK引脚
#define      SPI_CLK_HIGH       PCout(10)=1
#define      SPI_CLK_LOW        PCout(10)=0


//MISO引脚
#define      SPI_MISO_READ      PCin(11)

//MOSI引脚
#define      SPI_MOSI_HIGH      PCout(12)=1
#define      SPI_MOSI_LOW       PCout(12)=0
void SPI_GPIO_Init(void);
u8 Master_SPI_WRByte(u8 wdat);


#else
//从机
//CS引脚
#define      SPI_CS_READ        PCin(8)

//CLK引脚
#define      SPI_CLK_READ       PCin(10)

//MISO引脚
#define      SPI_MISO_HIGH      PCout(11)=1
#define      SPI_MISO_LOW       PCout(11)=0

//MOSI引脚
#define      SPI_MOSI_READ      PCin(12)
void SPI_GPIO_Init(void);
u8 Slave_SPI_RWByte(u8 wdat);
#endif

6、主函数

 int main(void)
{	
	u8 key;
	delay_init();	    	 //延时函数初始化	  
	led_init();		  	//初始化与LED连接的硬件接口
	key_init();
	SPI_GPIO_Init();
	while(1)
	{
#if SELECT==0
	key=key_scan(0);
	switch(key){
		case key_up_value:{
			if(Master_SPI_WRByte(0x50)==0x60)LED1=!LED1;
			break;
		};
		case key_down_value:{
			
			break;
		};
		case key_left_value:{
			
			break;
		};
		case key_right_value:{
			
			break;
		};

	}		
		
#else
	Slave_SPI_RWByte(0x60);


	
#endif

五、实验说明

本实验通过主机向从机发送0x50命令来控制从机LED1亮灭。其他应用层的使用读者可自行完成。
 文章来源地址https://www.toymoban.com/news/detail-404014.html

到了这里,关于stm32 GPIO模拟SPI接口实现双机通信的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • STM32使用SPI协议主从通信

    目录 前言 一、理论部分      SPI简介   SPI特征 SPI物理层 SPI协议层 SPI配置过程 SPI数据发送与接收过程 二、代码部分 主机代码 从机代码 这是一篇学习笔记,记录自己学习SPI通信。方便之后运用的时候回顾。参考《STM32中文参考手册》          SPI 协议是由摩托罗拉公

    2024年02月13日
    浏览(31)
  • STM32 SPI通信协议1——协议基本原理及管脚定义

            SPI全称为 Serial Peripheral interface ,译为 串行外围设备接口 。SPI主要应用在EEPROM,FLASH,实时时钟,AD转换器,还有数字信号处理器和数字信号编码器之间。         SPI是一种高速,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,

    2024年01月25日
    浏览(32)
  • 【STM32学习】——SPI通信协议&SPI时序&W25Q64存储芯片&软件SPI读写

    目录 前言 一、SPI通信协议 1.概述​ 2.硬件电路  3.移位示意图 二、SPI时序 1.时序基本单元 2.完整时序波形 三、W25Q64存储芯片 1.芯片简介  2.硬件电路引脚定义  3.芯片框图 4.Flash操作注意事项 四、软件SPI读写W25Q64 五、SPI通信外设 总结 声明:学习笔记来自江科大自化协B站教

    2024年02月09日
    浏览(52)
  • STM32模拟SPI协议控制数字电位器MCP41010电阻值

    MCP41010是单路8位分辨率数字电位器,通过SPI接口可控制电位器阻值分配,相当于PW0端在PA0和PB0之间滑动。如下图所示: MCP41010是10K欧姆规格的数字电位器,即PA0和PB0之间的阻值恒定为10K,PW0与PA0或PW0与PB0之间的阻值互补可配。 另外有相关型号不同阻值规格和集成双路的MCP系列

    2024年02月16日
    浏览(31)
  • STM-32:SPI通信协议/W25Q64简介—软件SPI读写W25Q64

    SPI是串行外设接口(Serial Peripheral Interface)的缩写,是美国摩托罗拉公司(Motorola)最先推出的一种同步串行传输规范,也是一种单片机外设芯片串行扩展接口,是一种高速、全双工、同步通信总线,所以可以在同一时间发送和接收数据,SPI没有定义速度限制,通常能达到甚

    2024年02月16日
    浏览(36)
  • 【快速上手STM32】SPI通信协议&&1.8寸TFT-LCD(ST7735S)

    SPI,英文全称Serial Peripheral Interface,即串行外围设备接口,是一种高速、全双工、同步的串行通信总线。 我们之前说过I2C,那么我们就拿I2C和SPI做个对比。 SPI和I2C对比,优势在于SPI的传输速率比I2C快得多,劣势在于SPI需要用的通信线比较多。 I2C只需要两根线,而SPI至少需要

    2024年04月27日
    浏览(29)
  • STM32模拟SPI协议获取24位模数转换(24bit ADC)芯片AD7791电压采样数据

    STM32大部分芯片只有12位的ADC采样性能,如果要实现更高精度的模数转换如24位ADC采样,则需要连接外部ADC实现。AD7791是亚德诺(ADI)半导体一款用于低功耗、24位Σ-Δ型模数转换器(ADC) ,适合低频测量应用,提供50 Hz/60 Hz同步抑制。 这里介绍基于AD7791的24位ADC采样实现。 AD7791的管脚

    2024年02月09日
    浏览(49)
  • 【接口协议】FPGA实现SPI协议基于ADC128S022进行模拟信号采集

    使用vivado联合modelsim实现SPI协议基于ADC128S022进行模拟信号连续采集。 SPI是串行外设接口,是一种同步/全双工/主从式接口。通常由四根信号线构成: CS_N :片选信号,主从式接口,可以有多个从机,用片选信号进行从机选择; SCLK :串行时钟线,由主机提供给从机; MISO :主机

    2024年02月07日
    浏览(42)
  • 基于SPI实现stm32与fpga通信(一)

    SPI通信协议有以下4种模式: 模式0:时钟极性为0,时钟相位为0,数据在时钟下降沿捕获,数据在时钟上升沿改变。 模式1:时钟极性为0,时钟相位为1,数据在时钟上升沿捕获,数据在时钟下降沿改变。 模式2:时钟极性为1,时钟相位为0,数据在时钟上升沿捕获,数据在时钟

    2024年04月16日
    浏览(35)
  • 普冉PY32系列(九) GPIO模拟和硬件SPI方式驱动无线收发芯片XL2400

    普冉PY32系列(一) PY32F0系列32位Cortex M0+ MCU简介 普冉PY32系列(二) Ubuntu GCC Toolchain和VSCode开发环境 普冉PY32系列(三) PY32F002A资源实测 - 这个型号不简单 普冉PY32系列(四) PY32F002A/003/030的时钟设置 普冉PY32系列(五) 使用JLink RTT代替串口输出日志 普冉PY32系列(六) 通过I2C接口驱动PCF8574扩

    2024年02月08日
    浏览(56)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包