sa1100_simpad.c 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. /*
  2. * drivers/pcmcia/sa1100_simpad.c
  3. *
  4. * PCMCIA implementation routines for simpad
  5. *
  6. */
  7. #include <linux/module.h>
  8. #include <linux/kernel.h>
  9. #include <linux/sched.h>
  10. #include <linux/device.h>
  11. #include <linux/init.h>
  12. #include <asm/hardware.h>
  13. #include <asm/mach-types.h>
  14. #include <asm/irq.h>
  15. #include <asm/arch/simpad.h>
  16. #include "sa1100_generic.h"
  17. extern long get_cs3_shadow(void);
  18. extern void set_cs3_bit(int value);
  19. extern void clear_cs3_bit(int value);
  20. static struct pcmcia_irqs irqs[] = {
  21. { 1, IRQ_GPIO_CF_CD, "CF_CD" },
  22. };
  23. static int simpad_pcmcia_hw_init(struct soc_pcmcia_socket *skt)
  24. {
  25. clear_cs3_bit(VCC_3V_EN|VCC_5V_EN|EN0|EN1);
  26. skt->irq = IRQ_GPIO_CF_IRQ;
  27. return soc_pcmcia_request_irqs(skt, irqs, ARRAY_SIZE(irqs));
  28. }
  29. static void simpad_pcmcia_hw_shutdown(struct soc_pcmcia_socket *skt)
  30. {
  31. soc_pcmcia_free_irqs(skt, irqs, ARRAY_SIZE(irqs));
  32. /* Disable CF bus: */
  33. //set_cs3_bit(PCMCIA_BUFF_DIS);
  34. clear_cs3_bit(PCMCIA_RESET);
  35. }
  36. static void
  37. simpad_pcmcia_socket_state(struct soc_pcmcia_socket *skt,
  38. struct pcmcia_state *state)
  39. {
  40. unsigned long levels = GPLR;
  41. long cs3reg = get_cs3_shadow();
  42. state->detect=((levels & GPIO_CF_CD)==0)?1:0;
  43. state->ready=(levels & GPIO_CF_IRQ)?1:0;
  44. state->bvd1=1; /* Not available on Simpad. */
  45. state->bvd2=1; /* Not available on Simpad. */
  46. state->wrprot=0; /* Not available on Simpad. */
  47. if((cs3reg & 0x0c) == 0x0c) {
  48. state->vs_3v=0;
  49. state->vs_Xv=0;
  50. } else {
  51. state->vs_3v=1;
  52. state->vs_Xv=0;
  53. }
  54. }
  55. static int
  56. simpad_pcmcia_configure_socket(struct soc_pcmcia_socket *skt,
  57. const socket_state_t *state)
  58. {
  59. unsigned long flags;
  60. local_irq_save(flags);
  61. /* Murphy: see table of MIC2562a-1 */
  62. switch (state->Vcc) {
  63. case 0:
  64. clear_cs3_bit(VCC_3V_EN|VCC_5V_EN|EN0|EN1);
  65. break;
  66. case 33:
  67. clear_cs3_bit(VCC_3V_EN|EN1);
  68. set_cs3_bit(VCC_5V_EN|EN0);
  69. break;
  70. case 50:
  71. clear_cs3_bit(VCC_5V_EN|EN1);
  72. set_cs3_bit(VCC_3V_EN|EN0);
  73. break;
  74. default:
  75. printk(KERN_ERR "%s(): unrecognized Vcc %u\n",
  76. __FUNCTION__, state->Vcc);
  77. clear_cs3_bit(VCC_3V_EN|VCC_5V_EN|EN0|EN1);
  78. local_irq_restore(flags);
  79. return -1;
  80. }
  81. local_irq_restore(flags);
  82. return 0;
  83. }
  84. static void simpad_pcmcia_socket_init(struct soc_pcmcia_socket *skt)
  85. {
  86. soc_pcmcia_enable_irqs(skt, irqs, ARRAY_SIZE(irqs));
  87. }
  88. static void simpad_pcmcia_socket_suspend(struct soc_pcmcia_socket *skt)
  89. {
  90. soc_pcmcia_disable_irqs(skt, irqs, ARRAY_SIZE(irqs));
  91. set_cs3_bit(PCMCIA_RESET);
  92. }
  93. static struct pcmcia_low_level simpad_pcmcia_ops = {
  94. .owner = THIS_MODULE,
  95. .hw_init = simpad_pcmcia_hw_init,
  96. .hw_shutdown = simpad_pcmcia_hw_shutdown,
  97. .socket_state = simpad_pcmcia_socket_state,
  98. .configure_socket = simpad_pcmcia_configure_socket,
  99. .socket_init = simpad_pcmcia_socket_init,
  100. .socket_suspend = simpad_pcmcia_socket_suspend,
  101. };
  102. int __init pcmcia_simpad_init(struct device *dev)
  103. {
  104. int ret = -ENODEV;
  105. if (machine_is_simpad())
  106. ret = sa11xx_drv_pcmcia_probe(dev, &simpad_pcmcia_ops, 1, 1);
  107. return ret;
  108. }