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驱动测试 (二)的更多相关文章

  1. linux SPI驱动——gpio模拟spi驱动(三)

    一:首先在我的平台注册platform_device,保证能让spi-gpio.c能执行到probe函数. 1: struct spi_gpio_platform_data { 2: unsigned ...

  2. GPIO模拟SPI

    上次用gpio模拟i2c理解i2c协议.相同的,我用gpio模拟spi来理解spi协议. 我用的是4线spi,四线各自是片选.时钟.命令/数据.数据. 数据在时钟上升沿传递,数据表示的是数据还是命令由 ...

  3. am335x gpio 模拟 spi 驱动添加

    kernel 内 make menuconfig // make menuconfig Device Drivers ---> [*] SPI support ---> <*> ...

  4. gpio模拟i2c驱动

    前段时间做项目,需要gpio模拟i2c通信,最后参考了一些资料,然后编写了一个程序.现在发出来,以免以后忘记,也为一些需要的朋友提供参考.不喜勿喷哈. /* 说明:该程序是基于atmel公司的sama ...

  5. linux enc28j60网卡驱动移植(硬件spi和模拟spi)

    本来想移植DM9000网卡的驱动,无奈硬件出了点问题,通过杜邦线链接开发板和DM9000网卡模块,系统上电,还没加载网卡驱动就直接崩溃了,找不到原因...刚好手上有一个enc28j60的网卡模块,于是 ...

  6. STM32-24位AD7799驱动之手册代码详解,支持模拟SPI和硬件SPI

    1.AD7799介绍 AD7799结构图如下所示: 其中REFIN参考电压建议为2.5V, REFIN电压低于0.1V时,则差分输入ad值就无法检测了,如下图所示: 注意: 如果REG_CONFIG的 ...

  7. LPC4370使用学习:GPIO的引脚功能使用,和12864OLED模拟I2C驱动

    一: 手中有块LPC4370的开发板,因为便宜,所以引脚引出的不多,而且只有基本的底板资源驱动代码和例程. 看着手册和例程看了老半天,写程序写了半天,结果GPIO老是驱动不起来,因为引脚配置寄存器中有 ...

  8. OLED的波形曲线、进度条、图片显示(STM32 HAL库 模拟SPI通信 5线OLED屏幕)详细篇

    少废话,先上效果图 屏幕显示效果         全家福 一.基础认识及引脚介绍 屏幕参数: 尺寸:0.96英寸 分辨率:128*64 驱动芯片:SSD1306 驱动接口协议:SPI 引脚说明: 二. ...

  9. S5PV210之GPIO模拟I2c时序之pcf8591与at24xx linux3.0.8驱动

    目录:一. 说明 二. 驱动程序说明及问题 三. 案例一       四. 案例二 一. 说明 mini210开发板上带了at24c08, 看了linux内核自带的at24.c的驱动程序,编译下载到看 ...

随机推荐

  1. BZOJ4816 [Sdoi2017]数字表格 【莫比乌斯反演】

    题目 Doris刚刚学习了fibonacci数列.用f[i]表示数列的第i项,那么 f[0]=0 f[1]=1 f[n]=f[n-1]+f[n-2],n>=2 Doris用老师的超级计算机生成了 ...

  2. P4513 小白逛公园 (线段树)

    题目链接 Solution 线段树是一门比较刁钻的手艺... 此题我们需要维护 \(4\) 个变量: \(amx\) 代表当前节点的最大值. \(lmx\) 代表当前节点以左端点为起点的区间最大值. ...

  3. Element 'plugin' cannot have character [children], because the type's content type is element-only

    原因是你复制的时候,带了一些特殊符号. 解决方案: 将那一串代码复制到notpad++ 或者文本上面,再复制到你的编译器里面,就可以解决问题了

  4. 挑战程序设计2 KD树

    Range Query - Range Search (kD Tree) Time Limit : 1 sec, Memory Limit : 262144 KB Japanese version i ...

  5. best corder MG loves gold

    MG loves gold  Accepts: 451  Submissions: 1382  Time Limit: 3000/1500 MS (Java/Others)  Memory Limit ...

  6. Docker 组件如何协作?

    还记得我们运行的第一个容器吗?现在通过它来体会一下 Docker 各个组件是如何协作的. 容器启动过程如下: Docker 客户端执行 docker run 命令. Docker daemon 发现本 ...

  7. python算法与数据结构-顺序表(37)

    1.顺序表介绍 顺序表是最简单的一种线性结构,逻辑上相邻的数据在计算机内的存储位置也是相邻的,可以快速定位第几个元素,中间不允许有空,所以插入.删除时需要移动大量元素.顺序表可以分配一段连续的存储空间 ...

  8. 哈工大CSAPP大作业

    第1章 概述 1.1 Hello简介 hello的源码hello.c文件,要生成可执行文件,首先要进行预处理,其次要进行编译生成汇编代码,接着进行汇编处理生成目标文件,目标文件通过链接器形成一个可执行 ...

  9. Linux 之 Memcached

    Memcached的安装使用 参考教程:[千峰教育] 环境:CentOS 6.8 一.简介: memcached作为高速运行的分布式缓存服务器,具有以下的特点. · 协议简单 · 基于libevent ...

  10. Xcode文件名后的字母含义

    Xcode文件名后的字母含义   在Xcode中,左侧的dock区域显示项目结构.很多时候,文件名后方会出现一个字母提示,如M.A之类的.这实际是一种提示符号.如果项目中使用SVN.Git等版本控制工 ...