IGKBoard(imx6ull)-SPI接口编程-回环测试


对于SIP不了解的可以参考这篇文章:SPI协议介绍

1- 使能imx6ull开发板SPI驱动

想要使能40pin扩展口的SPI1的话,需要修改开发板上的DTOverlay配置文件,添加该管脚对SPI1的支持,具体修改具体方法为修改 eMMC 启动介质的 boot 分区下的 config.txt 文件,将dtoverlay_spi1的选项修改为yes,然后重启应用就可以了。

root@igkboard:~# vi /run/media/mmcblk1p1/config.txt
# Enable SPI overlay, SPI1 conflict with UART8(NB-IoT/4G module)
dtoverlay_spi1=yes

系统启动时将会自动加载 SPI 协议驱动。查看/dev下是否存在spi设备节点,已验证spi驱动是否加载。

root@igkboard:~# ls -l /dev/spidev0.0
crw------- 1 root root 153, 0 Mar  4 05:56 /dev/spidev0.0

2- 回环测试imx6ull开发板物理连接

我们可以看见开发板上的40pin扩展口上:
GPIO03_IO27 -----> ECSPI1_MOSI
GPIO03_IO28 -----> ECSPI1_MISO

在这里插入图片描述
回环测试,找到IGKBoard的SPI1的MISO和MOSI管脚,使用杜邦线或跳线帽短接即可,如下图所

在这里插入图片描述


3- 编程SPI回环测试

源码:

#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <linux/spi/spidev.h>

typedef struct spi_ctx_s
{
    int       fd;
    char      dev[64];
    uint8_t   bits;
    uint16_t  delay;
    uint32_t  mode;
    uint32_t  speed;
}spi_ctx_t;

static int spi_init(spi_ctx_t *spi_ctx);/*设备初始化函数*/
static int spi_transmit_receive(spi_ctx_t *spi_ctx, uint8_t const *tx, uint8_t const *rx,size_t len);

int main(int argc, char* argv[])
{
    char          *input_tx = "Hello.I am WangDengtao.";//默认发送的数据
    char          *spi_dev = "/dev/spidev0.0";//默认设备的位置
    uint32_t      spi_speed = 500000;//默认速率500k
    spi_ctx_t     spi_ctx;
    char          rx_buf[60];

    if(2 != argc)
    {
        printf("This is the procedure of spi loopback test.\n");
        printf("Please input %s /dev/spidev***.\n", argv[0]);
        return 0;
    }

    memset(&spi_ctx, 0, sizeof(spi_ctx));
    strncpy(spi_ctx.dev, spi_dev, sizeof(spi_ctx.dev));//将设备地址拷贝到结构体中,方便初始化调用

    spi_ctx.bits = 8;//数据长度
    spi_ctx.delay = 100;//两个spi_ioc_transfer之间的延时,微秒
    /*
#define SPI_MODE_0  (0|0)               //模式0
#define SPI_MODE_1  (0|SPI_CPHA)        //模式1
#define SPI_MODE_2  (SPI_CPOL|0)        //模式2
#define SPI_MODE_3  (SPI_CPOL|SPI_CPHA) //模式3
     */
    spi_ctx.mode = SPI_MODE_2;//设置spi模式
    spi_ctx.speed = spi_speed;///通讯速率

    if(spi_init(&spi_ctx) < 0)
    {
        printf("spi_init error\n");
        return -1;
    }
    printf("SPI %s [fd=%d] init successfully\n",spi_ctx.dev, spi_ctx.fd);

    if(spi_transmit_receive(&spi_ctx, input_tx, rx_buf, strlen(input_tx)) < 0)
    {
        printf("spi_transmit_receive error\n");
        return -2;
    }

    /*打印 tx_buf 和 rx_buf*/
    printf("tx_buf: | %s |\n", input_tx);
    printf("rx_buf: | %s |\n", rx_buf);

    return 0;
}

/*数据收发函数*/
static int spi_transmit_receive(spi_ctx_t *spi_ctx, uint8_t const *tx, uint8_t const *rx,size_t len)
{
    /*应用程序空间需要从spi设备传输数据时候每组数据元素就是 struct spi_ioc_transfer 结构体类型*/
    struct spi_ioc_transfer tr = {
        .tx_buf = (unsigned long)tx,
        .rx_buf = (unsigned long)rx,
        .len = len,
        .delay_usecs = spi_ctx->delay,
        .speed_hz = spi_ctx->speed,
        .bits_per_word = spi_ctx->bits,
    };

    if(ioctl(spi_ctx->fd, SPI_IOC_MESSAGE(1), &tr) < 0)
    {
        printf("SPI transmit and receive error: %s\n", strerror(errno));
        return -1;
    }

    return 0;
}

/*设备初始化函数*/
int spi_init(spi_ctx_t *spi_ctx)
{
    int ret;
    
    spi_ctx->fd = open(spi_ctx->dev, O_RDWR);
    if(spi_ctx->fd < 0)
    {
        printf("Open %s error:%s\n", spi_ctx -> dev, strerror(errno));
        return -1;
    }
    /*设置spi接收和发送的工作模式*/
    ret = ioctl(spi_ctx->fd, SPI_IOC_RD_MODE, &spi_ctx->mode);
    if(ret < 0)
    {
        printf("SPI set SPI_IOC_RD_MODE [0x%x] error:%s\n", spi_ctx->mode, strerror(errno));
        goto CleanUp;
    }
    ret = ioctl(spi_ctx->fd, SPI_IOC_WR_MODE, &spi_ctx->mode);
    if(ret < 0)
    {
        printf("SPI set SPI_IOC_WR_MODE [0x%x] error:%s\n", spi_ctx->mode, strerror(errno));
        goto CleanUp;
    }
    /*设置spi通信接收和发送的字长*/
    ret = ioctl(spi_ctx->fd, SPI_IOC_RD_BITS_PER_WORD, &spi_ctx->bits);
    if(ret < 0)
    {
        printf("SPI set SPI_IOC_RD_BITS_PER_WORD [%d] error:%s\n",spi_ctx->bits, strerror(errno));
        goto CleanUp;
    }
    ret = ioctl(spi_ctx->fd, SPI_IOC_WR_BITS_PER_WORD, &spi_ctx->bits);
    if(ret < 0)
    {
        printf("SPI set SPI_IOC_WR_BITS_PER_WORD [%d] error:%s\n",spi_ctx->bits, strerror(errno));
        goto CleanUp;
    }
    /*设置最高工作频率*/
    ret = ioctl(spi_ctx->fd, SPI_IOC_WR_MAX_SPEED_HZ, &spi_ctx->speed);
    if(ret == -1)
    {
        printf("SPI set SPI_IOC_WR_MAX_SPEED_HZ [%d] error:%s\n", spi_ctx->speed, strerror(errno));
        goto CleanUp;
    }
    ret = ioctl(spi_ctx->fd, SPI_IOC_RD_MAX_SPEED_HZ, &spi_ctx->speed);
    if(ret == -1)
    {
        printf("SPI set SPI_IOC_RD_MAX_SPEED_HZ [%d] error:%s\n", spi_ctx->speed, strerror(errno));
        goto CleanUp;
    }
    printf("spi mode: 0x%x\n", spi_ctx->mode);
    printf("bits per word: %d\n", spi_ctx->bits);
    printf("max speed: %d KHz\n", spi_ctx->speed /1000);

    return spi_ctx->fd;

CleanUp:
    close(spi_ctx->fd);
    return -1;
}

makefile:

CC=arm-linux-gnueabihf-gcc
APP_NAME=spi_test

all:clean
	@${CC} ${APP_NAME}.c -o ${APP_NAME}

clean:
	@rm -f ${APP_NAME}

tftp不了解的小伙伴可以参考这篇文章:wpa_supplicant无线网络配置imx6ull以及搭建tftp服务器
tftp服务器下载到开发板运行:

root@igkboard:~# tftp -gr spi_test 192.168.10.168
root@igkboard:~# chmod a+x spi_test
root@igkboard:~# ./spi_test /dev/spidev0.0 
spi mode: 0x4
bits per word: 8
max speed: 500 KHz
SPI /dev/spidev0.0 [fd=3] init successfully
tx_buf: | Hello.I am WangDengtao. |
rx_buf: | Hello.I am WangDengtao. |

4- 代码重难点分析

(1)spi_device结构体

虽然用户空间不需要直接用到spi_device结构体,但是这个结构体和用户空间的程序有密切的关系,理解它的成员有助于理解SPI设备节点的IOCTL命令,所以首先来介绍它。
在我们的代码中,我们就使用到了:

spi_ctx.mode = SPI_MODE_2;//设置spi模式
//设置读写的工作模式
ioctl(spi_ctx->fd, SPI_IOC_RD_MODE, &spi_ctx->mode);
ioctl(spi_ctx->fd, SPI_IOC_WR_MODE, &spi_ctx->mode);

SPI_MODE_2是设置spi模式为第二种模式,在开头推荐的第一篇文章中有讲到。

在内核中,每个spi_device代表一个物理的SPI设备:

struct spi_device {
         structdevice              dev;
         structspi_master         *master;
         u32                       max_speed_hz;        /* 通信时钟最大频率 */
         u8                        chip_select;         /* 片选号 */
         u8                        mode;                /*SPI设备的模式,下面的宏是它各bit的含义  */
#define       SPI_CPHA        0x01                      /* 采样的时钟相位                            */
#define       SPI_CPOL        0x02                      /* 时钟信号起始相位:高或者是低电平*/
#define       SPI_MODE_0     (0|0)                    
#define       SPI_MODE_1     (0|SPI_CPHA)
#define       SPI_MODE_2     (SPI_CPOL|0)
#define       SPI_MODE_3     (SPI_CPOL|SPI_CPHA)
#define       SPI_CS_HIGH     0x04                      /* 为1时片选的有效信号是高电平*/
#define       SPI_LSB_FIRST   0x08                      /* 发送时低比特在前  */
#define       SPI_3WIRE       0x10                      /* 输入输出信号使用同一根信号线 */
#define       SPI_LOOP        0x20                      /* 回环模式 */
         u8                        bits_per_word;       /* 每个通信字的字长(比特数) */
         int                       irq;                 /*使用到的中断 */
         void                     *controller_state;
         void                     *controller_data;
         char                      modalias[32];        /* 设备驱动的名字*/
};

(2)spi_ioc_transfer结构体

应用程序空间需要从spi设备传输数据时候,每组数据元素就是 struct spi_ioc_transfer 结构体类型,该结构体定义如下:

struct spi_ioc_transfer {
	__u64 tx_buf; //发送数据缓存
	__u64 rx_buf; //接收数据缓存
	
	__u32 len; //数据长度
	__u32 speed_hz; //通讯速率
	
	__u16 delay_usecs; //两个spi_ioc_transfer之间的延时,微秒
	__u8 bits_per_word; //数据长度
	__u8 cs_change; //取消选中片选
	__u8 tx_nbits; //单次数据宽度(多数据线模式)
	__u8 rx_nbits; //单次数据宽度(多数据线模式)
	__u8 word_delay_usecs;
	__u8 pad;
	/* If the contents of 'struct spi_ioc_transfer' ever change
	* incompatibly, then the ioctl number (currently 0) must change;
	* ioctls with constant size fields get a bit more in the way of
	* error checking than ones (like this) where that field varies.
	*
	* NOTE: struct layout is the same in 64bit and 32bit userspace.
	*/
};

(3)ioctl函数

在编写应用程序时还需要使用ioctl函数设置spi相关配置,其函数原型如下:

#include <sys/ioctl.h>
int ioctl(int fd, unsigned long request, ...);

其中对于SPI设备request的值常用的有以下几种:
在这里插入图片描述
当然,如果要使用上述的参数的话,需要头文件#include <linux/spi/spidev.h>


  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值