rsrc_pci.c 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. #include <linux/slab.h>
  2. #include <linux/module.h>
  3. #include <linux/kernel.h>
  4. #include <pcmcia/ss.h>
  5. #include <pcmcia/cistpl.h>
  6. #include "cs_internal.h"
  7. struct pcmcia_align_data {
  8. unsigned long mask;
  9. unsigned long offset;
  10. };
  11. static resource_size_t pcmcia_align(void *align_data,
  12. const struct resource *res,
  13. resource_size_t size, resource_size_t align)
  14. {
  15. struct pcmcia_align_data *data = align_data;
  16. resource_size_t start;
  17. start = (res->start & ~data->mask) + data->offset;
  18. if (start < res->start)
  19. start += data->mask + 1;
  20. return start;
  21. }
  22. static struct resource *find_io_region(struct pcmcia_socket *s,
  23. unsigned long base, int num,
  24. unsigned long align)
  25. {
  26. struct resource *res = pcmcia_make_resource(0, num, IORESOURCE_IO,
  27. dev_name(&s->dev));
  28. struct pcmcia_align_data data;
  29. int ret;
  30. data.mask = align - 1;
  31. data.offset = base & data.mask;
  32. ret = pci_bus_alloc_resource(s->cb_dev->bus, res, num, 1,
  33. base, 0, pcmcia_align, &data);
  34. if (ret != 0) {
  35. kfree(res);
  36. res = NULL;
  37. }
  38. return res;
  39. }
  40. static int res_pci_find_io(struct pcmcia_socket *s, unsigned int attr,
  41. unsigned int *base, unsigned int num,
  42. unsigned int align, struct resource **parent)
  43. {
  44. int i, ret = 0;
  45. /* Check for an already-allocated window that must conflict with
  46. * what was asked for. It is a hack because it does not catch all
  47. * potential conflicts, just the most obvious ones.
  48. */
  49. for (i = 0; i < MAX_IO_WIN; i++) {
  50. if (!s->io[i].res)
  51. continue;
  52. if (!*base)
  53. continue;
  54. if ((s->io[i].res->start & (align-1)) == *base)
  55. return -EBUSY;
  56. }
  57. for (i = 0; i < MAX_IO_WIN; i++) {
  58. struct resource *res = s->io[i].res;
  59. unsigned int try;
  60. if (res && (res->flags & IORESOURCE_BITS) !=
  61. (attr & IORESOURCE_BITS))
  62. continue;
  63. if (!res) {
  64. if (align == 0)
  65. align = 0x10000;
  66. res = s->io[i].res = find_io_region(s, *base, num,
  67. align);
  68. if (!res)
  69. return -EINVAL;
  70. *base = res->start;
  71. s->io[i].res->flags =
  72. ((res->flags & ~IORESOURCE_BITS) |
  73. (attr & IORESOURCE_BITS));
  74. s->io[i].InUse = num;
  75. *parent = res;
  76. return 0;
  77. }
  78. /* Try to extend top of window */
  79. try = res->end + 1;
  80. if ((*base == 0) || (*base == try)) {
  81. ret = adjust_resource(s->io[i].res, res->start,
  82. resource_size(res) + num);
  83. if (ret)
  84. continue;
  85. *base = try;
  86. s->io[i].InUse += num;
  87. *parent = res;
  88. return 0;
  89. }
  90. /* Try to extend bottom of window */
  91. try = res->start - num;
  92. if ((*base == 0) || (*base == try)) {
  93. ret = adjust_resource(s->io[i].res,
  94. res->start - num,
  95. resource_size(res) + num);
  96. if (ret)
  97. continue;
  98. *base = try;
  99. s->io[i].InUse += num;
  100. *parent = res;
  101. return 0;
  102. }
  103. }
  104. return -EINVAL;
  105. }
  106. static struct resource *res_pci_find_mem(u_long base, u_long num,
  107. u_long align, int low, struct pcmcia_socket *s)
  108. {
  109. struct resource *res = pcmcia_make_resource(0, num, IORESOURCE_MEM,
  110. dev_name(&s->dev));
  111. struct pcmcia_align_data data;
  112. unsigned long min;
  113. int ret;
  114. if (align < 0x20000)
  115. align = 0x20000;
  116. data.mask = align - 1;
  117. data.offset = base & data.mask;
  118. min = 0;
  119. if (!low)
  120. min = 0x100000UL;
  121. ret = pci_bus_alloc_resource(s->cb_dev->bus,
  122. res, num, 1, min, 0,
  123. pcmcia_align, &data);
  124. if (ret != 0) {
  125. kfree(res);
  126. res = NULL;
  127. }
  128. return res;
  129. }
  130. static int res_pci_init(struct pcmcia_socket *s)
  131. {
  132. if (!s->cb_dev || (!s->features & SS_CAP_PAGE_REGS)) {
  133. dev_err(&s->dev, "not supported by res_pci\n");
  134. return -EOPNOTSUPP;
  135. }
  136. return 0;
  137. }
  138. struct pccard_resource_ops pccard_nonstatic_ops = {
  139. .validate_mem = NULL,
  140. .find_io = res_pci_find_io,
  141. .find_mem = res_pci_find_mem,
  142. .init = res_pci_init,
  143. .exit = NULL,
  144. };
  145. EXPORT_SYMBOL(pccard_nonstatic_ops);