spi-sh-sci.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. /*
  2. * SH SCI SPI interface
  3. *
  4. * Copyright (c) 2008 Magnus Damm
  5. *
  6. * Based on S3C24XX GPIO based SPI driver, which is:
  7. * Copyright (c) 2006 Ben Dooks
  8. * Copyright (c) 2006 Simtec Electronics
  9. *
  10. * This program is free software; you can redistribute it and/or modify
  11. * it under the terms of the GNU General Public License version 2 as
  12. * published by the Free Software Foundation.
  13. *
  14. */
  15. #include <linux/kernel.h>
  16. #include <linux/init.h>
  17. #include <linux/delay.h>
  18. #include <linux/spinlock.h>
  19. #include <linux/workqueue.h>
  20. #include <linux/platform_device.h>
  21. #include <linux/spi/spi.h>
  22. #include <linux/spi/spi_bitbang.h>
  23. #include <linux/module.h>
  24. #include <asm/spi.h>
  25. #include <asm/io.h>
  26. struct sh_sci_spi {
  27. struct spi_bitbang bitbang;
  28. void __iomem *membase;
  29. unsigned char val;
  30. struct sh_spi_info *info;
  31. struct platform_device *dev;
  32. };
  33. #define SCSPTR(sp) (sp->membase + 0x1c)
  34. #define PIN_SCK (1 << 2)
  35. #define PIN_TXD (1 << 0)
  36. #define PIN_RXD PIN_TXD
  37. #define PIN_INIT ((1 << 1) | (1 << 3) | PIN_SCK | PIN_TXD)
  38. static inline void setbits(struct sh_sci_spi *sp, int bits, int on)
  39. {
  40. /*
  41. * We are the only user of SCSPTR so no locking is required.
  42. * Reading bit 2 and 0 in SCSPTR gives pin state as input.
  43. * Writing the same bits sets the output value.
  44. * This makes regular read-modify-write difficult so we
  45. * use sp->val to keep track of the latest register value.
  46. */
  47. if (on)
  48. sp->val |= bits;
  49. else
  50. sp->val &= ~bits;
  51. iowrite8(sp->val, SCSPTR(sp));
  52. }
  53. static inline void setsck(struct spi_device *dev, int on)
  54. {
  55. setbits(spi_master_get_devdata(dev->master), PIN_SCK, on);
  56. }
  57. static inline void setmosi(struct spi_device *dev, int on)
  58. {
  59. setbits(spi_master_get_devdata(dev->master), PIN_TXD, on);
  60. }
  61. static inline u32 getmiso(struct spi_device *dev)
  62. {
  63. struct sh_sci_spi *sp = spi_master_get_devdata(dev->master);
  64. return (ioread8(SCSPTR(sp)) & PIN_RXD) ? 1 : 0;
  65. }
  66. #define spidelay(x) ndelay(x)
  67. #include "spi-bitbang-txrx.h"
  68. static u32 sh_sci_spi_txrx_mode0(struct spi_device *spi,
  69. unsigned nsecs, u32 word, u8 bits)
  70. {
  71. return bitbang_txrx_be_cpha0(spi, nsecs, 0, 0, word, bits);
  72. }
  73. static u32 sh_sci_spi_txrx_mode1(struct spi_device *spi,
  74. unsigned nsecs, u32 word, u8 bits)
  75. {
  76. return bitbang_txrx_be_cpha1(spi, nsecs, 0, 0, word, bits);
  77. }
  78. static u32 sh_sci_spi_txrx_mode2(struct spi_device *spi,
  79. unsigned nsecs, u32 word, u8 bits)
  80. {
  81. return bitbang_txrx_be_cpha0(spi, nsecs, 1, 0, word, bits);
  82. }
  83. static u32 sh_sci_spi_txrx_mode3(struct spi_device *spi,
  84. unsigned nsecs, u32 word, u8 bits)
  85. {
  86. return bitbang_txrx_be_cpha1(spi, nsecs, 1, 0, word, bits);
  87. }
  88. static void sh_sci_spi_chipselect(struct spi_device *dev, int value)
  89. {
  90. struct sh_sci_spi *sp = spi_master_get_devdata(dev->master);
  91. if (sp->info && sp->info->chip_select)
  92. (sp->info->chip_select)(sp->info, dev->chip_select, value);
  93. }
  94. static int sh_sci_spi_probe(struct platform_device *dev)
  95. {
  96. struct resource *r;
  97. struct spi_master *master;
  98. struct sh_sci_spi *sp;
  99. int ret;
  100. master = spi_alloc_master(&dev->dev, sizeof(struct sh_sci_spi));
  101. if (master == NULL) {
  102. dev_err(&dev->dev, "failed to allocate spi master\n");
  103. ret = -ENOMEM;
  104. goto err0;
  105. }
  106. sp = spi_master_get_devdata(master);
  107. platform_set_drvdata(dev, sp);
  108. sp->info = dev->dev.platform_data;
  109. /* setup spi bitbang adaptor */
  110. sp->bitbang.master = spi_master_get(master);
  111. sp->bitbang.master->bus_num = sp->info->bus_num;
  112. sp->bitbang.master->num_chipselect = sp->info->num_chipselect;
  113. sp->bitbang.chipselect = sh_sci_spi_chipselect;
  114. sp->bitbang.txrx_word[SPI_MODE_0] = sh_sci_spi_txrx_mode0;
  115. sp->bitbang.txrx_word[SPI_MODE_1] = sh_sci_spi_txrx_mode1;
  116. sp->bitbang.txrx_word[SPI_MODE_2] = sh_sci_spi_txrx_mode2;
  117. sp->bitbang.txrx_word[SPI_MODE_3] = sh_sci_spi_txrx_mode3;
  118. r = platform_get_resource(dev, IORESOURCE_MEM, 0);
  119. if (r == NULL) {
  120. ret = -ENOENT;
  121. goto err1;
  122. }
  123. sp->membase = ioremap(r->start, resource_size(r));
  124. if (!sp->membase) {
  125. ret = -ENXIO;
  126. goto err1;
  127. }
  128. sp->val = ioread8(SCSPTR(sp));
  129. setbits(sp, PIN_INIT, 1);
  130. ret = spi_bitbang_start(&sp->bitbang);
  131. if (!ret)
  132. return 0;
  133. setbits(sp, PIN_INIT, 0);
  134. iounmap(sp->membase);
  135. err1:
  136. spi_master_put(sp->bitbang.master);
  137. err0:
  138. return ret;
  139. }
  140. static int sh_sci_spi_remove(struct platform_device *dev)
  141. {
  142. struct sh_sci_spi *sp = platform_get_drvdata(dev);
  143. iounmap(sp->membase);
  144. setbits(sp, PIN_INIT, 0);
  145. spi_bitbang_stop(&sp->bitbang);
  146. spi_master_put(sp->bitbang.master);
  147. return 0;
  148. }
  149. static struct platform_driver sh_sci_spi_drv = {
  150. .probe = sh_sci_spi_probe,
  151. .remove = sh_sci_spi_remove,
  152. .driver = {
  153. .name = "spi_sh_sci",
  154. .owner = THIS_MODULE,
  155. },
  156. };
  157. module_platform_driver(sh_sci_spi_drv);
  158. MODULE_DESCRIPTION("SH SCI SPI Driver");
  159. MODULE_AUTHOR("Magnus Damm <damm@opensource.se>");
  160. MODULE_LICENSE("GPL");
  161. MODULE_ALIAS("platform:spi_sh_sci");