mem_domain.h 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. /*
  2. * Copyright (c) 2017 Linaro Limited
  3. * Copyright (c) 2018-2020 Intel Corporation
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef INCLUDE_APP_MEMPORY_MEM_DOMAIN_H
  8. #define INCLUDE_APP_MEMPORY_MEM_DOMAIN_H
  9. #include <stdint.h>
  10. #include <stddef.h>
  11. #include <sys/dlist.h>
  12. #include <toolchain.h>
  13. #include <kernel/thread.h>
  14. #ifdef __cplusplus
  15. extern "C" {
  16. #endif
  17. /**
  18. * @defgroup mem_domain_apis Memory domain APIs
  19. * @ingroup kernel_apis
  20. * @{
  21. */
  22. #ifdef CONFIG_USERSPACE
  23. /**
  24. * @def K_MEM_PARTITION_DEFINE
  25. *
  26. * @brief Statically declare a memory partition
  27. */
  28. #ifdef _ARCH_MEM_PARTITION_ALIGN_CHECK
  29. #define K_MEM_PARTITION_DEFINE(name, start, size, attr) \
  30. _ARCH_MEM_PARTITION_ALIGN_CHECK(start, size); \
  31. struct k_mem_partition name =\
  32. { (uintptr_t)start, size, attr}
  33. #else
  34. #define K_MEM_PARTITION_DEFINE(name, start, size, attr) \
  35. struct k_mem_partition name =\
  36. { (uintptr_t)start, size, attr}
  37. #endif /* _ARCH_MEM_PARTITION_ALIGN_CHECK */
  38. /**
  39. * @brief Memory Partition
  40. *
  41. * A memory partition is a region of memory in the linear address space
  42. * with a specific access policy.
  43. *
  44. * The alignment of the starting address, and the alignment of the size
  45. * value may have varying requirements based on the capabilities of the
  46. * underlying memory management hardware; arbitrary values are unlikely
  47. * to work.
  48. */
  49. struct k_mem_partition {
  50. /** start address of memory partition */
  51. uintptr_t start;
  52. /** size of memory partition */
  53. size_t size;
  54. /** attribute of memory partition */
  55. k_mem_partition_attr_t attr;
  56. };
  57. /**
  58. * @brief Memory Domain
  59. *
  60. * A memory domain is a collection of memory partitions, used to represent
  61. * a user thread's access policy for the linear addresss space. A thread
  62. * may be a member of only one memory domain, but any memory domain may
  63. * have multiple threads that are members.
  64. *
  65. * Supervisor threads may also be a member of a memory domain; this has
  66. * no implications on their memory access but can be useful as any child
  67. * threads inherit the memory domain membership of the parent.
  68. *
  69. * A user thread belonging to a memory domain with no active partitions
  70. * will have guaranteed access to its own stack buffer, program text,
  71. * and read-only data.
  72. */
  73. struct k_mem_domain {
  74. #ifdef CONFIG_ARCH_MEM_DOMAIN_DATA
  75. struct arch_mem_domain arch;
  76. #endif /* CONFIG_ARCH_MEM_DOMAIN_DATA */
  77. /** partitions in the domain */
  78. struct k_mem_partition partitions[CONFIG_MAX_DOMAIN_PARTITIONS];
  79. /** Doubly linked list of member threads */
  80. sys_dlist_t mem_domain_q;
  81. /** number of active partitions in the domain */
  82. uint8_t num_partitions;
  83. };
  84. /**
  85. * Default memory domain
  86. *
  87. * All threads are a member of some memory domain, even if running in
  88. * supervisor mode. Threads belong to this default memory domain if they
  89. * haven't been added to or inherited membership from some other domain.
  90. *
  91. * This memory domain has the z_libc_partition partition for the C library
  92. * added to it if exists.
  93. */
  94. extern struct k_mem_domain k_mem_domain_default;
  95. #else
  96. /* To support use of IS_ENABLED for the APIs below */
  97. struct k_mem_domain;
  98. struct k_mem_partition;
  99. #endif /* CONFIG_USERSPACE */
  100. /**
  101. * @brief Initialize a memory domain.
  102. *
  103. * Initialize a memory domain with given name and memory partitions.
  104. *
  105. * See documentation for k_mem_domain_add_partition() for details about
  106. * partition constraints.
  107. *
  108. * Do not call k_mem_domain_init() on the same memory domain more than once,
  109. * doing so is undefined behavior.
  110. *
  111. * @param domain The memory domain to be initialized.
  112. * @param num_parts The number of array items of "parts" parameter.
  113. * @param parts An array of pointers to the memory partitions. Can be NULL
  114. * if num_parts is zero.
  115. */
  116. extern void k_mem_domain_init(struct k_mem_domain *domain, uint8_t num_parts,
  117. struct k_mem_partition *parts[]);
  118. /**
  119. * @brief Add a memory partition into a memory domain.
  120. *
  121. * Add a memory partition into a memory domain. Partitions must conform to
  122. * the following constraints:
  123. *
  124. * - Partitions in the same memory domain may not overlap each other.
  125. * - Partitions must not be defined which expose private kernel
  126. * data structures or kernel objects.
  127. * - The starting address alignment, and the partition size must conform to
  128. * the constraints of the underlying memory management hardware, which
  129. * varies per architecture.
  130. * - Memory domain partitions are only intended to control access to memory
  131. * from user mode threads.
  132. * - If CONFIG_EXECUTE_XOR_WRITE is enabled, the partition must not allow
  133. * both writes and execution.
  134. *
  135. * Violating these constraints may lead to CPU exceptions or undefined
  136. * behavior.
  137. *
  138. * @param domain The memory domain to be added a memory partition.
  139. * @param part The memory partition to be added
  140. */
  141. extern void k_mem_domain_add_partition(struct k_mem_domain *domain,
  142. struct k_mem_partition *part);
  143. /**
  144. * @brief Remove a memory partition from a memory domain.
  145. *
  146. * Remove a memory partition from a memory domain.
  147. *
  148. * @param domain The memory domain to be removed a memory partition.
  149. * @param part The memory partition to be removed
  150. */
  151. extern void k_mem_domain_remove_partition(struct k_mem_domain *domain,
  152. struct k_mem_partition *part);
  153. /**
  154. * @brief Add a thread into a memory domain.
  155. *
  156. * Add a thread into a memory domain. It will be removed from whatever
  157. * memory domain it previously belonged to.
  158. *
  159. * @param domain The memory domain that the thread is going to be added into.
  160. * @param thread ID of thread going to be added into the memory domain.
  161. *
  162. */
  163. extern void k_mem_domain_add_thread(struct k_mem_domain *domain,
  164. k_tid_t thread);
  165. #ifdef __cplusplus
  166. }
  167. #endif
  168. /** @} */
  169. #endif /* INCLUDE_APP_MEMORY_MEM_DOMAIN_H */