始创于2000年 股票代码:831685
咨询热线:0371-60135900 注册有礼 登录
  • 挂牌上市企业
  • 60秒人工响应
  • 99.99%连通率
  • 7*24h人工
  • 故障100倍补偿
您的位置: 网站首页 > 帮助中心>文章内容

Linux spi子操作系统驱动分析

发布时间:  2012/8/24 13:58:57
  2.6.18内核下已经添加了完整的spi子系统了,参考mtd的分析,将从下到上层,再从上到下层的对其进行分析。

 

  以下先从下到上的进行分析:

  driver/spi下有两个底层相关的spi驱动程序:

  spi_s3c24xx.c和spi_s3c24xx_gpio.c

  其中spi_s3c24xx.c是基于s3c24xx下相应的spi接口的驱动程序,spi_s3c24xx_gpio.c允许用户指定3个gpio口,分别充当spi_clk、spi_mosi和spi_miso接口,模拟标准的spi总线。

  s3c2410自带了两个spi接口(spi0和spi1),在此我只研究基于s3c2410下spi接口的驱动程序spi_s3c24xx.c。

  首先从spi驱动的检测函数进行分析:

  static int s3c24xx_spi_probe(struct platform_device *pdev)

  {

  struct s3c24xx_spi *hw;

  struct spi_master *master;

  struct spi_board_info *bi;

  struct resource *res;

  int err = 0;

  int i;

  /* pi_alloc_master函数申请了struct spi_master+struct s3c24xx_spi大小的数据,

  * spi_master_get_devdata和pi_master_get分别取出struct s3c24xx_spi和struct spi_master结构指针

  */

  master = spi_alloc_master(&pdev->dev, sizeof(struct s3c24xx_spi));

  if (master == NULL) {

  dev_err(&pdev->dev, "No memory for spi_master\n");

  err = -ENOMEM;

  goto err_nomem;

  }

  /* 填充struct spi_master结构 */

  hw = spi_master_get_devdata(master);

  memset(hw, 0, sizeof(struct s3c24xx_spi));

  hw->master = spi_master_get(master);

  hw->pdata = pdev->dev.platform_data;

  hw->dev = &pdev->dev;

  if (hw->pdata == NULL) {

  dev_err(&pdev->dev, "No platform data supplied\n");

  err = -ENOENT;

  goto err_no_pdata;

  }

  platform_set_drvdata(pdev, hw);//dev_set_drvdata(&pdev->dev, hw)

  init_completion(&hw->done);

  /* setup the state for the bitbang driver */

  /* 填充hw->bitbang结构(hw->bitbang结构充当一个中间层,相当与input system的input_handle struct) */

  hw->bitbang.master = hw->master;

  hw->bitbang.setup_transfer = s3c24xx_spi_setupxfer;

  hw->bitbang.chipselect = s3c24xx_spi_chipsel;

  hw->bitbang.txrx_bufs = s3c24xx_spi_txrx;

  hw->bitbang.master->setup = s3c24xx_spi_setup;

  dev_dbg(hw->dev, "bitbang at %p\n", &hw->bitbang);

  /* find and map our resources */

  /* 申请spi所用到的资源:io、irq、时钟等 */

  res = platform_get_resource(pdev, IORESOURCE_MEM, 0);

  if (res == NULL) {

  dev_err(&pdev->dev, "Cannot get IORESOURCE_MEM\n");

  err = -ENOENT;

  goto err_no_iores;

  }

  hw->ioarea = request_mem_region(res->start, (res->end - res->start)+1,

  pdev->name);

  if (hw->ioarea == NULL) {

  dev_err(&pdev->dev, "Cannot reserve region\n");

  err = -ENXIO;

  goto err_no_iores;

  }

  hw->regs = ioremap(res->start, (res->end - res->start)+1);

  if (hw->regs == NULL) {

  dev_err(&pdev->dev, "Cannot map IO\n");

  err = -ENXIO;

  goto err_no_iomap;

  }

  hw->irq = platform_get_irq(pdev, 0);

  if (hw->irq < 0) {

  dev_err(&pdev->dev, "No IRQ specified\n");

  err = -ENOENT;

  goto err_no_irq;

  }

  err = request_irq(hw->irq, s3c24xx_spi_irq, 0, pdev->name, hw);

  if (err) {

  dev_err(&pdev->dev, "Cannot claim IRQ\n");

  goto err_no_irq;

  }

  hw->clk = clk_get(&pdev->dev, "spi");

  if (IS_ERR(hw->clk)) {

  dev_err(&pdev->dev, "No clock for device\n");

  err = PTR_ERR(hw->clk);

  goto err_no_clk;

  }

  /* for the moment, permanently enable the clock */

  clk_enable(hw->clk);

  /* program defaults into the registers */

  /* 初始化spi相关的寄存器 */

  writeb(0xff, hw->regs + S3C2410_SPPRE);

  writeb(SPPIN_DEFAULT, hw->regs + S3C2410_SPPIN);

  writeb(SPCON_DEFAULT, hw->regs + S3C2410_SPCON);

  /* add by lfc */

  s3c2410_gpio_setpin(S3C2410_GPE13, 0);

  s3c2410_gpio_setpin(S3C2410_GPE12, 0);

  s3c2410_gpio_cfgpin(S3C2410_GPE13, S3C2410_GPE13_SPICLK0);

  s3c2410_gpio_cfgpin(S3C2410_GPE12, S3C2410_GPE12_SPIMOSI0);

  s3c2410_gpio_cfgpin(S3C2410_GPE11, S3C2410_GPE11_SPIMISO0);

  /* end add */

  /* setup any gpio we can */

  /* 片选 */

  if (!hw->pdata->set_cs) {

  s3c2410_gpio_setpin(hw->pdata->pin_cs, 1);

  s3c2410_gpio_cfgpin(hw->pdata->pin_cs, S3C2410_GPIO_OUTPUT);

  }

  /* register our spi controller */

  /* 最终通过调用spi_register_master来注册spi控制器(驱动) */

  err = spi_bitbang_start(&hw->bitbang);

  if (err) {

  dev_err(&pdev->dev, "Failed to register SPI master\n");

  goto err_register;

  }

  dev_dbg(hw->dev, "shutdown=%d\n", hw->bitbang.shutdown);

  /* register all the devices associated */

  /* 注册所用使用本spi驱动的设备 */

  bi = &hw->pdata->board_info[0];

  for (i = 0; i < hw->pdata->board_size; i++, bi++) {

  dev_info(hw->dev, "registering %s\n", bi->modalias);

  bi->controller_data = hw;

  spi_new_device(master, bi);

  }

  return 0;

  err_register:

  clk_disable(hw->clk);

  clk_put(hw->clk);

  err_no_clk:

  free_irq(hw->irq, hw);

  err_no_irq:

  iounmap(hw->regs);

  err_no_iomap:

  release_resource(hw->ioarea);

  kfree(hw->ioarea);

  err_no_iores:

  err_no_pdata:

  spi_master_put(hw->master);;

  err_nomem:

  return err;

  }


本文出自:亿恩科技【www.enkj.com】

服务器租用/服务器托管中国五强!虚拟主机域名注册顶级提供商!15年品质保障!--亿恩科技[ENKJ.COM]

  • 您可能在找
  • 亿恩北京公司:
  • 经营性ICP/ISP证:京B2-20150015
  • 亿恩郑州公司:
  • 经营性ICP/ISP/IDC证:豫B1.B2-20060070
  • 亿恩南昌公司:
  • 经营性ICP/ISP证:赣B2-20080012
  • 服务器/云主机 24小时售后服务电话:0371-60135900
  • 虚拟主机/智能建站 24小时售后服务电话:0371-60135900
  • 专注服务器托管17年
    扫扫关注-微信公众号
    0371-60135900
    Copyright© 1999-2019 ENKJ All Rights Reserved 亿恩科技 版权所有  地址:郑州市高新区翠竹街1号总部企业基地亿恩大厦  法律顾问:河南亚太人律师事务所郝建锋、杜慧月律师   京公网安备41019702002023号
      1
     
     
     
     

    0371-60135900
    7*24小时客服服务热线