0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

SPI硬件基础知识科普

xCb1_yikoulinux 来源:一口Linux 作者:Jasonangel 2022-05-16 16:12 次阅读

SPI硬件基础

1、SPI hardware

SPI:Serial Perripheral Interface,串行外围设备接口,由 Motorola 公司提出,是一种高速、全双工、同步通信总线。SPI 以主从方式工作,通常是有一个主设备和一个或多个从设备,无应答机制。

本文我们讲解标准的 4 线 SPI,四根线如下:

①、CS/SS,Slave Select/Chip Select,片选信号线,用于选择需要进行通信的从设备。

②、SCK,Serial Clock,串行时钟,和 I2C 的 SCL 一样,为 SPI 通信提供时钟。

③、MOSI/SDO,Master Out Slave In/Serial Data Output,主输出从输入。

④、MISO/SDI,Master In Slave Out/Serial Data Input,主输入从输出。

c7489992-d3f7-11ec-bce3-dac502259ad0.png

2、SPI 四种工作模式

SPI 有四种工作模式,通过时钟极性(CPOL)和时钟相位(CPHA)的搭配来得到四种工作模式:

①、CPOL=0,串行时钟空闲状态为低电平。
②、CPOL=1,串行时钟空闲状态为高电平。
③、CPHA=0,串行时钟的第一个跳变沿(上升沿或下降沿)采集数据。
④、CPHA=1,串行时钟的第二个跳变沿(上升沿或下降沿)采集数据。
c77588d0-d3f7-11ec-bce3-dac502259ad0.png

示例波形图如下:

c7a1857a-d3f7-11ec-bce3-dac502259ad0.png

SPI 是全双工的,所以读写时序可以一起完成。

3、SPI 传输机制

c7c79b84-d3f7-11ec-bce3-dac502259ad0.png

从图可以看出,主机和从机都有一个串行移位寄存器,主机通过向它的 SPI 串行寄存器写入一个字节来发起一次传输。寄存器通过 MOSI 信号线将字节传送给从机,从机也将自己的移位寄存器中的内容通过 MISO 信号线返回给主机。这样,两个移位寄存器中的内容就被交换。

外设的写操作和读操作是同步完成的。如果只进行写操作,主机只需忽略接收到的字节;反之,若主机要读取从机的一个字节,就必须发送一个空字节来引发从机的传输。

c7eb934a-d3f7-11ec-bce3-dac502259ad0.gifc826bc4a-d3f7-11ec-bce3-dac502259ad0.gifc846a6f4-d3f7-11ec-bce3-dac502259ad0.gif

虽然 SPI 四线制支持读写同时进行,但实际上我们很多时候并不需要又读又写,见以下两种情况(参考 BMA223 数据手册):

注意:如下三幅图示均为 CPOL=1,CPHA=1

1、主机向从机写数据

c867a4a8-d3f7-11ec-bce3-dac502259ad0.png

主机发送先发送 8 bits,第一个 bit 为 0 代表这次主机是想写数据到从机,AD6~AD0 表示要写的寄存器地址。然后,主机就会一直写下去。在这期间 SDO 一直没用,一直是高阻态,算是一直读到1。

2、主机从从机读数据

c8824ef2-d3f7-11ec-bce3-dac502259ad0.png

这种情况下,主机先发送 8 bits,第一位为 1 代表这次是读,然后 AD6 ~ AD0 是想要读的寄存器地址,然后 SDO 开始返回数据。

4、SPI timing diagram

c8a207d8-d3f7-11ec-bce3-dac502259ad0.png
Tcsb_setup:建立时间
Tcsb_hold:保持时间
tsckl:低电平时间
tsckh:高电平时间
SCK period :Tsckl + tsckh
一般情况下Tsckl=tsckh

注意:真实的波形图如上,高低电平并不是到达最高点才算,0.3Vdd 以下为低电平,0.7Vdd 以上为高电平,计算信号时间长度的时候需要注意这个微小的时间,硬件设计必须注意信号质量风险,软件开发人员也要会看波形图。

这里的参数,一般 spi 驱动不需要设置,但是半导体厂商提供的 spi 控制器驱动中,可以修改这些参数。我们写 SPI 驱动时候,可以根据从设备的要求来修改这些参数

5、DMA 与 FIFO

不同平台对于 SPI FIFO 和 DMA 的 buffer size 设置不同:

c8bc0e8a-d3f7-11ec-bce3-dac502259ad0.png

传输 32bytes 以下使用 FIFO,传输 32bytes 以上使用 DMA。

DMA 可以自动发起多次传输,一次最大 256K 。

6、I2C 与 SPI 对比

功能 I2C SPI
线数 2(SDA,SCL) 4(MOSI,MISO,SCLK,CS)
主机数量 >=1 ==1
类型 半双工 全双工
回应机制 yes no
速度 <=3.4Mbps high
应用 重要数据 大量数据
流控 yes no
设备地址 yes no
常规用途 命令 数据

I2C 和 SPI 的速率如下:

I2C模式 速度
标准 100KHz
快速 400KHz
快速+ 1MHz
高速 3.4MHz

SPI 速率:几十 MHz 甚至上百 MHz,速度取决于 CPU 的 SPI 控制器和时钟 clock

STM32F103 的 SPI 最高支持 18MHz,imx6ull 的 SPI 最高支持 52MHz,其他芯片一般用不到更高的,因为速度越快波形质量越不好,越容易出问题。

具体采用多大速率还和外设有关,比如 EEPROM 的 W25Q128 的 SPI 最高支持 80MHz,ICM20608 传感器的 SPI 最高支持8MHz。一般用在 flash 上的速度会较快。

7、扩展

SPI 协议其实是包括:Standard SPI、Dual SPI 和 Queued SPI 三种协议接口。

Dual SPI 还是四线制,只是传输线可以变为同方向,速度是 Standard SPI 的两倍。

Queued SPI 是六线制,多了两根数据线,传输速度是 Standard SPI 的四倍。

SPILinux驱动

1、SPI 驱动源文件目录

Linux common spi driver

kernel-4.14/drivers/spi/spi.cLinux提供的通用接口封装层驱动
kernel-4.14/drivers/spi/spidev.clinux提供的SPI通用设备驱动程序
kernel-4.14/include/linux/spi/spi.hlinux提供的包含SPI的主要数据结构和函数

spi 控制器驱动,IC 厂商提供,不同厂商命名不同

kernel-4.14/drivers/spi/spi-mt65xx.cMTKSPI控制器驱动
kernel-4.14/drivers/spi/spi-mt65xx-dev.c
kernel-4.14/include/linux/platform_data/spi-mt65xx.h

dts

kernel-4.14/arch/arm/boot/dts/...
kernel-4.14/arch/arm64/boot/dts/...

以上文件对应如下 SPI 驱动软件架构:

c911c0d2-d3f7-11ec-bce3-dac502259ad0.png
SPI 控制器驱动程序

SPI 控制器不用关心设备的具体功能,它只负责把上层协议驱动准备好的数据按 SPI 总线的时序要求发送给 SPI 设备,同时把从设备收到的数据返回给上层的协议驱动,因此,内核把 SPI 控制器的驱动程序独立出来。

SPI 控制器驱动负责控制具体的控制器硬件,诸如 DMA 和中断操作等等,因为多个上层的协议驱动可能会通过控制器请求数据传输操作,所以,SPI 控制器驱动同时也要负责对这些请求进行队列管理,保证先进先出的原则。

SPI 通用接口封装层

为了简化 SPI 驱动程序的编程工作,同时也为了降低【协议驱动程序】和【控制器驱动程序】的耦合程度,内核把控制器驱动和协议驱动的一些通用操作封装成标准的接口,加上一些通用的逻辑处理操作,组成了 SPI 通用接口封装层。

这样的好处是,对于控制器驱动程序,只要实现标准的接口回调 API,并把它注册到通用接口层即可,无需直接和协议层驱动程序进行交互。而对于协议层驱动来说,只需通过通用接口层提供的 API 即可完成设备和驱动的注册,并通过通用接口层的 API 完成数据的传输,无需关注 SPI 控制器驱动的实现细节。

SPI 协议驱动程序

SPI 设备的具体功能是由 SPI 协议驱动程序完成的,SPI 协议驱动程序了解设备的功能和通信数据的协议格式。向下,协议驱动通过通用接口层和控制器交换数据,向上,协议驱动通常会根据设备具体的功能和内核的其它子系统进行交互。

例如,和 MTD 层交互以便把 SPI 接口的存储设备实现为某个文件系统,和 TTY 子系统交互把 SPI 设备实现为一个 TTY 设备,和网络子系统交互以便把一个 SPI 设备实现为一个网络设备。如果是一个专有的 SPI 设备,我们也可以按设备的协议要求,实现自己的专有协议驱动。

SPI 通用设备驱动程序

考虑到连接在 SPI 控制器上的设备的可变性,在内核没有配备相应的协议驱动程序,对于这种情况,内核为我们准备了通用的 SPI 设备驱动程序,该通用设备驱动程序向用户空间提供了控制 SPI 控制的控制接口,具体的协议控制和数据传输工作交由用户空间根据具体的设备来完成,在这种方式中,只能采用同步的方式和 SPI 设备进行通信,所以通常用于一些数据量较少的简单 SPI 设备。

2、SPI 通用接口层

  1. SPI 通用接口层把具体的 SPI 设备的协议驱动和 SPI 控制器驱动连接在一起。
  2. 负责 SPI 系统与 Linux 设备模型相关的初始化工作。
  3. 为协议驱动和控制器驱动提供一系列的标准接口 API 及其数据结构。
  4. SPI 设备、SPI 协议驱动、SPI 控制器的数据抽象
  5. 协助数据传输而定义的数据结构

kernel-4.14/drivers/spi/spi.c

staticint__initspi_init(void)
{
intstatus;

buf=kmalloc(SPI_BUFSIZ,GFP_KERNEL);
if(!buf){
status=-ENOMEM;
gotoerr0;
}

//创建/sys/bus/spi节点
status=bus_register(&spi_bus_type);
if(status< 0)
gotoerr1;

//创建/sys/class/spi_master节点
status=class_register(&spi_master_class);
if(status< 0)
gotoerr2;

if(IS_ENABLED(CONFIG_SPI_SLAVE)){
status=class_register(&spi_slave_class);
if(status< 0)
gotoerr3;
}
......
}

在这里创建了 SPI 总线,创建 /sys/bus/spi 节点和 /sys/class/spi_master 节点。

重要数据结构:

spi_device
spi_driver
spi_board_info
spi_controller/spi_master
spi_transfer
spi_message

重要 API

spi_message_init
spi_message_add_tail
spi_sync
spi_async
spi_write
spi_read

接下来详细解析结构体和API,只讲解重点部分,完整解析请参考官方文档

https://www.kernel.org/doc/html/v4.14//driver-api/spi.html

只有熟悉每个结构体存储的是什么东西,才能真正搞懂 SPI 模块。

spi_master/spi_controller:描述一个 spi 主机设备

structspi_master{
//Linux驱动模型中的设备
structdevicedev;

//此spi_master设备在全局spi_master链表中的节点
structlist_headlist;

//此spi_master编号
s16bus_num;

//此spi_master支持的片选信号数量
u16num_chipselect;

//dma地址对齐
u16dma_alignment;

//此spi_master支持传输的mode
u16mode_bits;
u32bits_per_word_mask;
/*limitsontransferspeed*/
u32min_speed_hz;
u32max_speed_hz;

/*otherconstraintsrelevanttothisdriver*/
u16flags;

/*lockandmutexforSPIbuslocking*/
spinlock_tbus_lock_spinlock;//总线自旋锁
structmutexbus_lock_mutex;//总线互斥锁

//总线是否处于lock状态
boolbus_lock_flag;

//准备传输,设置传输的参数
int(*setup)(structspi_device*spi);

//传输数据
int(*transfer)(structspi_device*spi,
structspi_message*mesg);
//设备release时的清除工作
void(*cleanup)(structspi_device*spi);

bool(*can_dma)(structspi_master*master,
structspi_device*spi,
structspi_transfer*xfer);

boolqueued;//是否采用系统的序列化传输
structkthread_workerkworker;//序列化传输时的线程worker
structtask_struct*kworker_task;//序列化传输的线程
structkthread_workpump_messages;//序列化传输时的处理函数
spinlock_tqueue_lock;//序列化传输时的queue_lock
structlist_headqueue;//序列化传输时的msg队列头
structspi_message*cur_msg;//序列化传输时当前的msg
boolidling;
boolbusy;//序列化传输时线程是否处于busy状态
boolrunning;//序列化传输时线程是否在运行
boolrt;//是否实时传输
......

int(*prepare_transfer_hardware)(structspi_master*master);

//一个msg的传输实现
int(*transfer_one_message)(structspi_master*master,
structspi_message*mesg);
......

/*gpiochipselect*/
int*cs_gpios;
......
};

spi_device:描述一个 spi 从机设备

structspi_device{
//Linux驱动模型中的设备
structdevicedev;
structspi_master*master;//设备所连接的spi主机设备
u32max_speed_hz;//该设备最大传输速率
u8chip_select;//CS片选信号编号
u8bits_per_word;//每次传输长度
u16mode;//传输模式
......
intirq;//软件中断号
void*controller_state;//控制器状态
void*controller_data;//控制参数
charmodalias[SPI_NAME_SIZE];//设备名称
//CS片选信号对应的GPIOnumber
intcs_gpio;/*chipselectgpio*/

/*thestatistics*/
structspi_statisticsstatistics;
};

spi_driver:描述一个 spi 设备驱动

structspi_driver{
//此driver所支持的spi设备list
conststructspi_device_id*id_table;
int(*probe)(structspi_device*spi);
int(*remove)(structspi_device*spi);
//系统shutdown时的回调函数
void(*shutdown)(structspi_device*spi);
structdevice_driverdriver;
};

spi_board_info:描述一个 spi 从机设备板级信息,无设备树时使用

structspi_board_info{
//设备名称
charmodalias[SPI_NAME_SIZE];
constvoid*platform_data;//设备的平台数据
void*controller_data;//设备的控制器数据
intirq;//设备的中断号
u32max_speed_hz;//设备支持的最大速率
u16bus_num;//设备连接的spi总线编号
u16chip_select;//设备连接的CS信号编号
u16mode;//设备使用的传输mode
};

spi_transfer:描述 spi 传输的具体数据

structspi_transfer{

constvoid*tx_buf;//spi_transfer的发送buf
void*rx_buf;//spi_transfer的接收buf
unsignedlen;//spi_transfer发送和接收的长度

dma_addr_ttx_dma;//tx_buf对应的dma地址
dma_addr_trx_dma;//rx_buf对应的dma地址
structsg_tabletx_sg;
structsg_tablerx_sg;

//spi_transfer传输完成后是否要改变CS片选信号
unsignedcs_change:1;
unsignedtx_nbits:3;
unsignedrx_nbits:3;
......
u8bits_per_word;//spi_transfer中一个word占的bits
u16delay_usecs;//两个spi_transfer直接的等待延迟
u32speed_hz;//spi_transfer的传输速率

structlist_headtransfer_list;//spi_transfer挂载到的message节点
};

spi_message:描述一次 spi 传输的信息

structspi_message{
//挂载在此msg上的transfer链表头
structlist_headtransfers;
//此msg需要通信的spi从机设备
structspi_device*spi;
//所使用的地址是否是dma地址
unsignedis_dma_mapped:1;

//msg发送完成后的处理函数
void(*complete)(void*context);
void*context;//complete函数的参数
unsignedframe_length;
unsignedactual_length;//此msg实际成功发送的字节数
intstatus;//此 msg 的发送状态,0:成功,负数,失败

structlist_headqueue;//此msg在所有msg中的链表节点
void*state;//此msg的私有数据
};

队列化

SPI 数据传输可以有两种方式:同步方式和异步方式

同步方式:数据传输的发起者必须等待本次传输的结束,期间不能做其它事情,用代码来解释就是,调用传输的函数后,直到数据传输完成,函数才会返回。

异步方式:数据传输的发起者无需等待传输的结束,数据传输期间还可以做其它事情,用代码来解释就是,调用传输的函数后,函数会立刻返回而不用等待数据传输完成,我们只需设置一个回调函数,传输完成后,该回调函数会被调用以通知发起者数据传送已经完成。

同步方式简单易用,很适合处理那些少量数据的单次传输。但是对于数据量大、次数多的传输来说,异步方式就显得更加合适。

对于 SPI 控制器来说,要支持异步方式必须要考虑以下两种状况:

  1. 对于同一个数据传输的发起者,既然异步方式无需等待数据传输完成即可返回,返回后,该发起者可以立刻又发起一个 message,而这时上一个message还没有处理完。
  2. 对于另外一个不同的发起者来说,也有可能同时发起一次message传输请求。

队列化正是为了为了解决以上的问题,所谓队列化,是指把等待传输的 message 放入一个等待队列中,发起一个传输操作,其实就是把对应的 message 按先后顺序放入一个等待队列中,系统会在不断检测队列中是否有等待传输的 message,如果有就不停地调度数据传输内核线程,逐个取出队列中的 message 进行处理,直到队列变空为止。SPI 通用接口层为我们实现了队列化的基本框架。

c934340a-d3f7-11ec-bce3-dac502259ad0.png

spi_message 就是一次 SPI 数据交换的原子操作,不可打断。

3、SPI 控制器驱动层

SPI 控制器驱动层负责最底层的数据收发,主要有以下功能:

  1. 申请必要的硬件资源,比如中断、DMA 通道、DMA 内存缓冲区等等
  2. 配置 SPI 控制器的工作模式和参数,使之可以和相应的设备进行正确的数据交换
  3. 向通用接口层提供接口,使得上层的协议驱动可以通过通用接口层访问控制器驱动
  4. 配合通用接口层,完成数据消息队列的排队和处理,直到消息队列变空为止

SPI 主机驱动就是 SOC 的 SPI 控制器驱动。Linux 内核使用 spi_master/spi_controller 表示 SPI 主机驱动,spi_master 是个结构体,定义在 include/linux/spi/spi.h 文件中。

SPI 主机驱动的核心就是申请 spi_master,然后初始化 spi_master,最后向 Linux 内核注册 spi_master。

API 如下:

spi_alloc_master 函数:申请 spi_master。
spi_master_put 函数:释放 spi_master。

spi_register_master函数:注册 spi_master。
spi_unregister_master 函数:注销 spi_master。

spi_bitbang_start函数:注册 spi_master。
spi_bitbang_stop 函数:注销 spi_master。

SPI 主机驱动的加载

以 MTK 为例,源码来自于小米开源项目

https://github.com/MiCode/Xiaomi_Kernel_OpenSource

小米每做一个项目,都会把 kernel 部分开源,因为需要遵循 Linux GPL 开源协议。

【设备】声明在设备树中

kernel-4.14/arch/arm64/boot/dts/mediatek/mt6885.dts
c9657024-d3f7-11ec-bce3-dac502259ad0.png

【驱动】

kernel-4.14/drivers/spi/spi-mt65xx.c

c97a6ea2-d3f7-11ec-bce3-dac502259ad0.pngc9a77c3a-d3f7-11ec-bce3-dac502259ad0.png

匹配以后,probe 函数执行,申请 spi_master,初始化 spi_master,最后向 Linux 内核注册 spi_master。

c9c3b3f0-d3f7-11ec-bce3-dac502259ad0.pngca1e7c68-d3f7-11ec-bce3-dac502259ad0.png

4、软件流程

ca391b9a-d3f7-11ec-bce3-dac502259ad0.png

看懂该图,对 SPI 驱动框架就有完整的了解了。

1、2、3 按顺执行,首先有 spi 总线的注册,然后是 spi 控制器驱动加载,然后是设备驱动加载。

区别在于,spi 控制器驱动加载时,是靠 platform 总线匹配设备(控制器)与驱动。spi 设备驱动加载时,是靠 spi 总线匹配设备(外设IC)与驱动。

init flow

ca6d8b28-d3f7-11ec-bce3-dac502259ad0.png

spi_register_master 的调用序列图

ca83c424-d3f7-11ec-bce3-dac502259ad0.png

队列化的工作机制及过程

cae4f5c8-d3f7-11ec-bce3-dac502259ad0.pngcb03331c-d3f7-11ec-bce3-dac502259ad0.png

当协议驱动程序通过 spi_async 发起一个 message 请求时,队列化和工作线程被激活,触发一些列的操作,最终完成 message 的传输操作。

spi_sync 与 spi_async 类似,只是有一个等待过程。

5、SPI 设备驱动

【设备】声明在设备树中

cb42e03e-d3f7-11ec-bce3-dac502259ad0.pngcb55e15c-d3f7-11ec-bce3-dac502259ad0.png

注意:设备的声明,slave device node 应该包含在你所要挂载的 &spi node 下,将 device 绑定在 master 上。然后通过 pinctrl 方式指定 GPIO,并在驱动中操作 pinctrl 句柄。

【驱动】demo

Linux 内核使用 spi_driver 结构体来表示 spi 设备驱动,我们在编写 SPI 设备驱动的时候需要实现 spi_driver。spi_driver 结构体定义在 include/linux/spi/spi.h 文件中。

spi_register_driver:注册 spi_driver
spi_unregister_driver:销掉 spi_driver
/*probe函数*/
staticintxxx_probe(structspi_device*spi)
{

/*具体函数内容*/
return0;
}

/*remove函数*/
staticintxxx_remove(structspi_device*spi)
{

/*具体函数内容*/
return0;
}

/*传统匹配方式ID列表*/
staticconststructspi_device_idxxx_id[]={

{"xxx",0},
{}
};

/*设备树匹配列表*/
staticconststructof_device_idxxx_of_match[]={

{.compatible="xxx"},
{/*Sentinel*/}
};

/*SPI驱动结构体*/
staticstructspi_driverxxx_driver={

.probe=xxx_probe,
.remove=xxx_remove,
.driver={
.owner=THIS_MODULE,
.name="xxx",
.of_match_table=xxx_of_match,
},
.id_table=xxx_id,
};

/*驱动入口函数*/
staticint__initxxx_init(void)
{

returnspi_register_driver(&xxx_driver);
}

/*驱动出口函数*/
staticvoid__exitxxx_exit(void)
{

spi_unregister_driver(&xxx_driver);
}

module_init(xxx_init);
module_exit(xxx_exit);

在驱动入口函数中调用 spi_register_driver 来注册 spi_driver。

在驱动出口函数中调用 spi_unregister_driver 来注销 spi_driver。

spi 读写数据demo

/*SPI多字节发送*/
staticintspi_send(structspi_device*spi,u8*buf,intlen)
{
intret;
structspi_messagem;

structspi_transfert={
.tx_buf=buf,
.len=len,
};

spi_message_init(&m);/*初始化spi_message*/
spi_message_add_tail(t,&m);/*将spi_transfer添加到spi_message队列*/
ret=spi_sync(spi,&m);/*同步传输*/
returnret;
}
/*SPI多字节接收*/
staticintspi_receive(structspi_device*spi,u8*buf,intlen)
{
intret;
structspi_messagem;

structspi_transfert={
.rx_buf=buf,
.len=len,
};

spi_message_init(&m);/*初始化spi_message*/
spi_message_add_tail(t,&m);/*将spi_transfer添加到spi_message队列*/
ret=spi_sync(spi,&m);/*同步传输*/
returnret;
}

除了 init、exit、probe、remove、read、write 函数外,其他的函数看需求实现,这几个是最基本的。

6、总结

Linux 是 总线、设备、驱动 的框架,理解了这个框架,就能理解所有的模块驱动框架。

SPI 驱动比 I2C 驱动还是简单很多的。

end

原文标题:SPI 硬件+Linux驱动详解

文章出处:【微信公众号:一口Linux】欢迎添加关注!文章转载请注明出处。

审核编辑:汤梓红

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    11292

    浏览量

    209332
  • SPI硬件
    +关注

    关注

    0

    文章

    2

    浏览量

    800
  • 传输机制
    +关注

    关注

    0

    文章

    2

    浏览量

    1148

原文标题:SPI 硬件+Linux驱动详解

文章出处:【微信号:yikoulinux,微信公众号:一口Linux】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    C语言基础知识科普

    C语言是单片机开发中的必备基础知识,本文列举了部分STM32学习中比较常见的一些C语言基础知识
    发表于 07-21 10:58 1890次阅读

    科普一下CAN总线的基础知识

    CAN总线是一种常用的总线,对于刚开始接触CAN总线的,面对着各式各样的资料,可能不知道从何看起,今天科普一下CAN总线的基础知识。CAN2.0协议分为A版本和B版本,A版本协议为11位标识符(标准帧),B版本在兼容11位ID标识符的同时,向上扩展到29位ID标识符。
    发表于 05-16 09:49 3253次阅读
    <b class='flag-5'>科普</b>一下CAN总线的<b class='flag-5'>基础知识</b>

    手机硬件设计基础知识

    从别处得来的,基础知识讲解,纯分享{:4_95:}
    发表于 01-17 15:38

    硬件设计基础知识

    硬件设计基础知识
    发表于 02-25 21:28

    ecs700硬件基础知识

    ecs700硬件基础知识,目录1、电磁干扰1、电磁干扰
    发表于 07-14 06:35

    示波器基础知识

    第1章 示波器基础知识本章的内容整理自网络,主要讲解示波器的基础知识。如果初学的话非常有必要对这部分知识有一个了解。因为示波器是硬件调试必不可少的设备。1.1 什么是示波器1.2 示波
    发表于 08-09 07:21

    五分钟读懂WiFi基础知识

    家1、嵌入式技术常识科普【物联网】WiFi基础知识五分钟读懂TCP/IP;协议STM32开发 -- Keil基本使用如何看懂时序图(以SPI/I2C为例)ESP8266配网思路(不使用...
    发表于 12-01 06:36

    SPI通信协议的基础知识解析

    SPI通信协议详解写在最前: 本文讲述了SPI通信协议的基本内容包括如下SPI基础知识SPI的读写时序本文重点参考 英文维基百科 中文维基
    发表于 12-13 08:05

    记录一下SPI基础知识与软件开发环境

    的,毕竟速率够高,数据量能传的比较大。1. 准备1.1 SPI基础知识1.2 开发环境1.2.1 软件开发环境1.2.2 硬件环境SPI2. Demo...
    发表于 02-17 06:29

    科普】卷积神经网络基础知识

    本文的主要目的,是简单介绍时下流行的深度学习算法的基础知识,本人也看过许多其他教程,感觉其中大部分讲的还是太过深奥,于是便有了写一篇科普文的想法。博主也是现学现卖,文中如有不当之处,请各位指出
    发表于 11-10 14:49 1718次阅读
    【<b class='flag-5'>科普</b>】卷积神经网络<b class='flag-5'>基础知识</b>

    电源管理基础知识电源管理基础知识电源管理基础知识

    电源管理基础知识电源管理基础知识电源管理基础知识
    发表于 09-15 14:36 76次下载
    电源管理<b class='flag-5'>基础知识</b>电源管理<b class='flag-5'>基础知识</b>电源管理<b class='flag-5'>基础知识</b>

    硬件工程师必备要了解哪些基础知识

    硬件工程师必备基础知识 目的:基于实际经验与实际项目详细理解并掌握成为合格的硬件工程师的最基本知识
    发表于 10-30 08:00 0次下载

    单片机基础知识学习笔记

    单片机基础知识学习笔记有关总线1.IIC总线2.SPI总线
    发表于 11-14 16:51 26次下载
    单片机<b class='flag-5'>基础知识</b>学习笔记

    SPI协议基础知识

    电子发烧友网站提供《SPI协议基础知识.pdf》资料免费下载
    发表于 11-16 10:32 1次下载
    <b class='flag-5'>SPI</b>协议<b class='flag-5'>基础知识</b>

    硬件工程师需要掌握的硬件基础知识

    作为一个资深硬件工程师,我们需要掌握一些硬件基础知识,今天总结一下哪些算是基础知识。给学电子方面想从事硬件工作的同学们一点提示。给未走出大学
    的头像 发表于 12-02 09:22 248次阅读
    <b class='flag-5'>硬件</b>工程师需要掌握的<b class='flag-5'>硬件</b><b class='flag-5'>基础知识</b>