linux SPI驱动——简单的gpio模拟SPI驱动测试 (二)
1: /*
2: * Add by xuyonghong for duotin car radio fm
3: * Copyright (C) 2016-5-24 xuyonghong@duotin.com
4: *
5: */
6: #include <linux/init.h>
7: #include <linux/miscdevice.h>
8: #include <linux/version.h>
9: #include <linux/kernel.h>
10: #include <linux/slab.h>
11: #include <linux/fs.h>
12: #include <linux/errno.h>
13: #include <linux/types.h>
14: #include <asm/io.h>
15: #include <linux/wait.h>
16: #include <linux/sched.h>
17: #include <linux/interrupt.h>
18: #include <linux/delay.h>
19: #include <linux/gpio.h>
20: #include <linux/i2c-gpio.h>
21:
22: #define NCS GPIO_PB(2) //定义SS所对应的GPIO接口编号
23: #define SCLK GPIO_PB(0) //定义SCLK所对应的GPIO接口编号
24: #define MOSI GPIO_PB(4) //定义SCLK所对应的GPIO接口编号
25: #define MISO GPIO_PB(01) //定义MISO所对应的GPIO接口编号
26: #define OUTP 1 //表示GPIO接口方向为输出
27: #define INP 0 //表示GPIO接口方向为输入
28:
29: //gpio_direction_output(unsigned gpio, int value)
30: //gpio_direction_input(unsigned gpio)
31: #define set_gpio_value gpio_set_value
32: #define get_gpio_value gpio_get_value
33:
34:
35: static int spi_request_gpio(void)
36: {
37:
38: if (gpio_request(NCS, "spi_ncs") < 0) {
39: return -1;
40: }
41: if (gpio_request(SCLK, "spi_sclk") < 0) {
42: return -1;
43: }
44: if (gpio_request(MOSI , "spi_mosi") < 0) {
45: return -1;
46: }
47: if (gpio_request(MISO, "spi_miso") < 0) {
48: return -1;
49: }
50: return 0;
51: }
52:
53: /* SPI端口初始化 */
54: static void spi_init(void)
55: {
56: gpio_direction_output(NCS, 1);
57: gpio_direction_output(SCLK, 1);
58: gpio_direction_output(MOSI, 0);
59: gpio_direction_input(MISO);
60: gpio_set_value(SCLK, 1);
61: gpio_set_value(MOSI, 0);
62: }
63:
64: /*
65: 从设备使能
66: enable:为1时,使能信号有效,SS低电平
67: 为0时,使能信号无效,SS高电平
68: */
69: void ss_enable(int enable)
70: {
71: if (enable)
72: set_gpio_value(NCS, 0); //SS低电平,从设备使能有效
73: else
74: set_gpio_value(NCS, 1); //SS高电平,从设备使能无效
75: }
76:
77:
78: /* SPI字节写 */
79: void spi_write_byte(unsigned char b)
80: {
81: int i;
82: for (i=7; i>=0; i--) {
83: set_gpio_value(SCLK, 0);
84: udelay(5);//延时
85: set_gpio_value(MOSI, b&(1<<i)); //从高位7到低位0进行串行写入
86: udelay(5);//延时
87: set_gpio_value(SCLK, 1); // CPHA=1,在时钟的第一个跳变沿采样
88: udelay(5);//延时
89: }
90: }
91:
92:
93: /* SPI字节读 */
94: unsigned char spi_read_byte(void)
95: {
96: int i;
97: unsigned char r = 0;
98: for (i=0; i<8; i++) {
99: set_gpio_value(SCLK, 0);
100: udelay(5);//延时 //延时
101: set_gpio_value(SCLK, 1); // CPHA=1,在时钟的第一个跳变沿采样
102: udelay(5);//延时
103: r = (r <<1) | get_gpio_value(MISO); //从高位7到低位0进行串行读出
104: udelay(5);//延时
105: }
106: return r;
107: }
108:
109: /*
110: SPI读操作
111: buf:写缓冲区
112: len:写入字节的长度
113: */
114: void spi_write (unsigned char reg_addr, unsigned char* buf, int len)
115: {
116: unsigned char reg_write_addr = reg_addr;
117: int i;
118:
119: /* SPI端口初始化 */
120: spi_init();
121: udelay(5);
122:
123: ss_enable(1); //从设备使能有效,通信开始
124: udelay(5); //延时
125:
126: spi_write_byte(reg_write_addr); //写寄存器写地址
127:
128: //写入数据
129: for (i=0; i<len; i++)
130: spi_write_byte(buf[i]);
131: udelay(5);//延时
132: ss_enable(0);
133: }
134:
135:
136: /*
137: SPI读操作
138: buf:读缓冲区
139: len:读入字节的长度
140: */
141: void spi_read(unsigned char reg_addr, unsigned char* buf, int len)
142: {
143: int i;
144: unsigned char reg_read_addr = 0x80 | reg_addr; //读的时候寄存地址最高位置1
145:
146: /* SPI端口初始化 */
147: spi_init();
148: udelay(5);
149:
150: ss_enable(1); //从设备使能有效,通信开始
151: udelay(5);//延时 //延时
152:
153: spi_write_byte(reg_read_addr); //写寄存器读地址
154:
155: //读入数据
156: for (i=0; i<len; i++)
157: buf[i] = spi_read_byte();
158: udelay(5);//延时
159: ss_enable(0); //从设备使能无效,通信结束
160: }
161:
162: static int __exit mpu9250_spi_init(void)
163: {
164: unsigned char data = 0x0;
165:
166: if (spi_request_gpio()) {
167: printk("mpu9250 spi initializer failure...\n");
168: return -1;
169: }
170:
171: spi_read(0x75, &data, 1);
172: printk("#################mpu9250 spi read id: 0x%x##############\n", data);
173:
174: return 0;
175: }
176:
177: static void __exit mpu9250_spi_exit(void)
178: {
179: printk("mpu9250 exit!\n");
180: }
181:
182: module_init(mpu9250_spi_init);
183: module_exit(mpu9250_spi_exit);
184: MODULE_AUTHOR("xuyonghong@duotin.com");
185: MODULE_LICENSE("GPL");
186:
测试结果:#################mpu9250 spi read id: 0x71##############
以上SPI设备工作在mode3模式
linux SPI驱动——简单的gpio模拟SPI驱动测试 (二)的更多相关文章
- linux SPI驱动——gpio模拟spi驱动(三)
一:首先在我的平台注册platform_device,保证能让spi-gpio.c能执行到probe函数. 1: struct spi_gpio_platform_data { 2: unsigned ...
- GPIO模拟SPI
上次用gpio模拟i2c理解i2c协议.相同的,我用gpio模拟spi来理解spi协议. 我用的是4线spi,四线各自是片选.时钟.命令/数据.数据. 数据在时钟上升沿传递,数据表示的是数据还是命令由 ...
- am335x gpio 模拟 spi 驱动添加
kernel 内 make menuconfig // make menuconfig Device Drivers ---> [*] SPI support ---> <*> ...
- gpio模拟i2c驱动
前段时间做项目,需要gpio模拟i2c通信,最后参考了一些资料,然后编写了一个程序.现在发出来,以免以后忘记,也为一些需要的朋友提供参考.不喜勿喷哈. /* 说明:该程序是基于atmel公司的sama ...
- linux enc28j60网卡驱动移植(硬件spi和模拟spi)
本来想移植DM9000网卡的驱动,无奈硬件出了点问题,通过杜邦线链接开发板和DM9000网卡模块,系统上电,还没加载网卡驱动就直接崩溃了,找不到原因...刚好手上有一个enc28j60的网卡模块,于是 ...
- STM32-24位AD7799驱动之手册代码详解,支持模拟SPI和硬件SPI
1.AD7799介绍 AD7799结构图如下所示: 其中REFIN参考电压建议为2.5V, REFIN电压低于0.1V时,则差分输入ad值就无法检测了,如下图所示: 注意: 如果REG_CONFIG的 ...
- LPC4370使用学习:GPIO的引脚功能使用,和12864OLED模拟I2C驱动
一: 手中有块LPC4370的开发板,因为便宜,所以引脚引出的不多,而且只有基本的底板资源驱动代码和例程. 看着手册和例程看了老半天,写程序写了半天,结果GPIO老是驱动不起来,因为引脚配置寄存器中有 ...
- OLED的波形曲线、进度条、图片显示(STM32 HAL库 模拟SPI通信 5线OLED屏幕)详细篇
少废话,先上效果图 屏幕显示效果 全家福 一.基础认识及引脚介绍 屏幕参数: 尺寸:0.96英寸 分辨率:128*64 驱动芯片:SSD1306 驱动接口协议:SPI 引脚说明: 二. ...
- S5PV210之GPIO模拟I2c时序之pcf8591与at24xx linux3.0.8驱动
目录:一. 说明 二. 驱动程序说明及问题 三. 案例一 四. 案例二 一. 说明 mini210开发板上带了at24c08, 看了linux内核自带的at24.c的驱动程序,编译下载到看 ...
随机推荐
- 给某个li标签家样式
HTML: <div class="tabs clearfix"> <ul id="der"> <li ><a hre ...
- Java面试题之ArrayList和LinkedList的区别
先看下类图: 相同点: 都实现了List接口和Collection: 不同点: 1.ArrayList是基于数组实现的:LinkedList是基于链表实现的: 2.ArrayList随机查询速度快:L ...
- java面试题之死锁产生的条件,以及如何避免死锁,银行家算法,产生死锁后如何解决(阿里面试题)
死锁产生的四个必要条件: 互斥:一个资源每次只能被一个进程使用(资源独立) 请求与保持:一个进程因请求资源而阻塞时,对已获得的资源保持不放(不释放锁) 不剥夺:进程已获得的资源,在未使用之前,不能强行 ...
- Keepalived中Master和Backup角色选举策略
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://ixdba.blog.51cto.com/2895551/1544858 在Kee ...
- net7:Web用户控件ascx的使用及其动态加载
原文发布时间为:2008-07-30 -- 来源于本人的百度文章 [由搬家工具导入] Web用户控件test.ascx的源代码: using System;using System;using Sys ...
- JVM指令助记符
以下只是JVM指令助记符,关于JVM指令的详细内容请阅读<JVM指令详解> 变量到操作数栈:iload,iload_,lload,lload_,fload,fload_,dload,dlo ...
- 【MFC】禁用鼠标拖拽标题栏移动窗口
解决方案:重载WM_NCLBUTTONDOWN消息 (1) .h 文件 afx_msg void OnNcLButtonDown(UINT nHitTest, CPoint point); (2) . ...
- inline关键词的使用(转载)
(一)inline函数(摘自C++ Primer的第三版) 在函数声明或定义中函数返回类型前加上关键字inline即把min()指定为内联. inline int min(int first, int ...
- Emmet插件的快捷键
Emmet插件的快捷键 html:5+tab键,可以生成html标签.!+tab键,也可以生成html标签.============================================== ...
- 浅谈APP的分享功能,有时候社交裂变形式比内容更“重要”
回顾2018年的移动互联网,“社交裂变”“下沉”等成为年度关键词.一方面我们可以看到社交裂变助推用户增长,另一方面我们也看到了以拼多多.趣头条为代表的互联网企业对于社交裂变模式表现出的空前关注度.作为 ...