ioremap.c 2.1 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. /*
  2. * Re-map IO memory to kernel address space so that we can access it.
  3. * This is needed for high PCI addresses that aren't mapped in the
  4. * 640k-1MB IO memory area on PC's
  5. *
  6. * (C) Copyright 1995 1996 Linus Torvalds
  7. */
  8. #include <linux/vmalloc.h>
  9. #include <linux/mm.h>
  10. #include <linux/sched.h>
  11. #include <linux/io.h>
  12. #include <asm/cacheflush.h>
  13. #include <asm/pgtable.h>
  14. static int ioremap_pte_range(pmd_t *pmd, unsigned long addr,
  15. unsigned long end, unsigned long phys_addr, pgprot_t prot)
  16. {
  17. pte_t *pte;
  18. unsigned long pfn;
  19. pfn = phys_addr >> PAGE_SHIFT;
  20. pte = pte_alloc_kernel(pmd, addr);
  21. if (!pte)
  22. return -ENOMEM;
  23. do {
  24. BUG_ON(!pte_none(*pte));
  25. set_pte_at(&init_mm, addr, pte, pfn_pte(pfn, prot));
  26. pfn++;
  27. } while (pte++, addr += PAGE_SIZE, addr != end);
  28. return 0;
  29. }
  30. static inline int ioremap_pmd_range(pud_t *pud, unsigned long addr,
  31. unsigned long end, unsigned long phys_addr, pgprot_t prot)
  32. {
  33. pmd_t *pmd;
  34. unsigned long next;
  35. phys_addr -= addr;
  36. pmd = pmd_alloc(&init_mm, pud, addr);
  37. if (!pmd)
  38. return -ENOMEM;
  39. do {
  40. next = pmd_addr_end(addr, end);
  41. if (ioremap_pte_range(pmd, addr, next, phys_addr + addr, prot))
  42. return -ENOMEM;
  43. } while (pmd++, addr = next, addr != end);
  44. return 0;
  45. }
  46. static inline int ioremap_pud_range(pgd_t *pgd, unsigned long addr,
  47. unsigned long end, unsigned long phys_addr, pgprot_t prot)
  48. {
  49. pud_t *pud;
  50. unsigned long next;
  51. phys_addr -= addr;
  52. pud = pud_alloc(&init_mm, pgd, addr);
  53. if (!pud)
  54. return -ENOMEM;
  55. do {
  56. next = pud_addr_end(addr, end);
  57. if (ioremap_pmd_range(pud, addr, next, phys_addr + addr, prot))
  58. return -ENOMEM;
  59. } while (pud++, addr = next, addr != end);
  60. return 0;
  61. }
  62. int ioremap_page_range(unsigned long addr,
  63. unsigned long end, unsigned long phys_addr, pgprot_t prot)
  64. {
  65. pgd_t *pgd;
  66. unsigned long start;
  67. unsigned long next;
  68. int err;
  69. BUG_ON(addr >= end);
  70. start = addr;
  71. phys_addr -= addr;
  72. pgd = pgd_offset_k(addr);
  73. do {
  74. next = pgd_addr_end(addr, end);
  75. err = ioremap_pud_range(pgd, addr, next, phys_addr+addr, prot);
  76. if (err)
  77. break;
  78. } while (pgd++, addr = next, addr != end);
  79. flush_cache_vmap(start, end);
  80. return err;
  81. }