pci_auto_common.c 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * PCI auto-configuration library
  4. *
  5. * Author: Matt Porter <mporter@mvista.com>
  6. *
  7. * Copyright 2000 MontaVista Software Inc.
  8. *
  9. * Modifications for driver model:
  10. * Copyright 2015 Google, Inc
  11. * Written by Simon Glass <sjg@chromium.org>
  12. */
  13. #include <common.h>
  14. #include <dm.h>
  15. #include <errno.h>
  16. #include <pci.h>
  17. void pciauto_region_init(struct pci_region *res)
  18. {
  19. /*
  20. * Avoid allocating PCI resources from address 0 -- this is illegal
  21. * according to PCI 2.1 and moreover, this is known to cause Linux IDE
  22. * drivers to fail. Use a reasonable starting value of 0x1000 instead.
  23. */
  24. res->bus_lower = res->bus_start ? res->bus_start : 0x1000;
  25. }
  26. void pciauto_region_align(struct pci_region *res, pci_size_t size)
  27. {
  28. res->bus_lower = ((res->bus_lower - 1) | (size - 1)) + 1;
  29. }
  30. int pciauto_region_allocate(struct pci_region *res, pci_size_t size,
  31. pci_addr_t *bar, bool supports_64bit)
  32. {
  33. pci_addr_t addr;
  34. if (!res) {
  35. debug("No resource\n");
  36. goto error;
  37. }
  38. addr = ((res->bus_lower - 1) | (size - 1)) + 1;
  39. if (addr - res->bus_start + size > res->size) {
  40. debug("No room in resource");
  41. goto error;
  42. }
  43. if (upper_32_bits(addr) && !supports_64bit) {
  44. debug("Cannot assign 64-bit address to 32-bit-only resource\n");
  45. goto error;
  46. }
  47. res->bus_lower = addr + size;
  48. debug("address=0x%llx bus_lower=0x%llx\n", (unsigned long long)addr,
  49. (unsigned long long)res->bus_lower);
  50. *bar = addr;
  51. return 0;
  52. error:
  53. *bar = (pci_addr_t)-1;
  54. return -1;
  55. }
  56. static void pciauto_show_region(const char *name, struct pci_region *region)
  57. {
  58. pciauto_region_init(region);
  59. debug("PCI Autoconfig: Bus %s region: [%llx-%llx],\n"
  60. "\t\tPhysical Memory [%llx-%llxx]\n", name,
  61. (unsigned long long)region->bus_start,
  62. (unsigned long long)(region->bus_start + region->size - 1),
  63. (unsigned long long)region->phys_start,
  64. (unsigned long long)(region->phys_start + region->size - 1));
  65. }
  66. void pciauto_config_init(struct pci_controller *hose)
  67. {
  68. int i;
  69. hose->pci_io = NULL;
  70. hose->pci_mem = NULL;
  71. hose->pci_prefetch = NULL;
  72. for (i = 0; i < hose->region_count; i++) {
  73. switch (hose->regions[i].flags) {
  74. case PCI_REGION_IO:
  75. if (!hose->pci_io ||
  76. hose->pci_io->size < hose->regions[i].size)
  77. hose->pci_io = hose->regions + i;
  78. break;
  79. case PCI_REGION_MEM:
  80. if (!hose->pci_mem ||
  81. hose->pci_mem->size < hose->regions[i].size)
  82. hose->pci_mem = hose->regions + i;
  83. break;
  84. case (PCI_REGION_MEM | PCI_REGION_PREFETCH):
  85. if (!hose->pci_prefetch ||
  86. hose->pci_prefetch->size < hose->regions[i].size)
  87. hose->pci_prefetch = hose->regions + i;
  88. break;
  89. }
  90. }
  91. if (hose->pci_mem)
  92. pciauto_show_region("Memory", hose->pci_mem);
  93. if (hose->pci_prefetch)
  94. pciauto_show_region("Prefetchable Mem", hose->pci_prefetch);
  95. if (hose->pci_io)
  96. pciauto_show_region("I/O", hose->pci_io);
  97. }