io-pgtable.c 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Generic page table allocator for IOMMUs.
  4. *
  5. * Copyright (C) 2014 ARM Limited
  6. *
  7. * Author: Will Deacon <will.deacon@arm.com>
  8. */
  9. #include <linux/bug.h>
  10. #include <linux/io-pgtable.h>
  11. #include <linux/kernel.h>
  12. #include <linux/types.h>
  13. static const struct io_pgtable_init_fns *
  14. io_pgtable_init_table[IO_PGTABLE_NUM_FMTS] = {
  15. #ifdef CONFIG_IOMMU_IO_PGTABLE_LPAE
  16. [ARM_32_LPAE_S1] = &io_pgtable_arm_32_lpae_s1_init_fns,
  17. [ARM_32_LPAE_S2] = &io_pgtable_arm_32_lpae_s2_init_fns,
  18. [ARM_64_LPAE_S1] = &io_pgtable_arm_64_lpae_s1_init_fns,
  19. [ARM_64_LPAE_S2] = &io_pgtable_arm_64_lpae_s2_init_fns,
  20. [ARM_MALI_LPAE] = &io_pgtable_arm_mali_lpae_init_fns,
  21. #endif
  22. #ifdef CONFIG_IOMMU_IO_PGTABLE_DART
  23. [APPLE_DART] = &io_pgtable_apple_dart_init_fns,
  24. [APPLE_DART2] = &io_pgtable_apple_dart_init_fns,
  25. #endif
  26. #ifdef CONFIG_IOMMU_IO_PGTABLE_ARMV7S
  27. [ARM_V7S] = &io_pgtable_arm_v7s_init_fns,
  28. #endif
  29. #ifdef CONFIG_AMD_IOMMU
  30. [AMD_IOMMU_V1] = &io_pgtable_amd_iommu_v1_init_fns,
  31. [AMD_IOMMU_V2] = &io_pgtable_amd_iommu_v2_init_fns,
  32. #endif
  33. };
  34. static int check_custom_allocator(enum io_pgtable_fmt fmt,
  35. struct io_pgtable_cfg *cfg)
  36. {
  37. /* No custom allocator, no need to check the format. */
  38. if (!cfg->alloc && !cfg->free)
  39. return 0;
  40. /* When passing a custom allocator, both the alloc and free
  41. * functions should be provided.
  42. */
  43. if (!cfg->alloc || !cfg->free)
  44. return -EINVAL;
  45. /* Make sure the format supports custom allocators. */
  46. if (io_pgtable_init_table[fmt]->caps & IO_PGTABLE_CAP_CUSTOM_ALLOCATOR)
  47. return 0;
  48. return -EINVAL;
  49. }
  50. struct io_pgtable_ops *alloc_io_pgtable_ops(enum io_pgtable_fmt fmt,
  51. struct io_pgtable_cfg *cfg,
  52. void *cookie)
  53. {
  54. struct io_pgtable *iop;
  55. const struct io_pgtable_init_fns *fns;
  56. if (fmt >= IO_PGTABLE_NUM_FMTS)
  57. return NULL;
  58. if (check_custom_allocator(fmt, cfg))
  59. return NULL;
  60. fns = io_pgtable_init_table[fmt];
  61. if (!fns)
  62. return NULL;
  63. iop = fns->alloc(cfg, cookie);
  64. if (!iop)
  65. return NULL;
  66. iop->fmt = fmt;
  67. iop->cookie = cookie;
  68. iop->cfg = *cfg;
  69. return &iop->ops;
  70. }
  71. EXPORT_SYMBOL_GPL(alloc_io_pgtable_ops);
  72. /*
  73. * It is the IOMMU driver's responsibility to ensure that the page table
  74. * is no longer accessible to the walker by this point.
  75. */
  76. void free_io_pgtable_ops(struct io_pgtable_ops *ops)
  77. {
  78. struct io_pgtable *iop;
  79. if (!ops)
  80. return;
  81. iop = io_pgtable_ops_to_pgtable(ops);
  82. io_pgtable_tlb_flush_all(iop);
  83. io_pgtable_init_table[iop->fmt]->free(iop);
  84. }
  85. EXPORT_SYMBOL_GPL(free_io_pgtable_ops);