kobject.h 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  1. /*
  2. * Copyright (c) 2020 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_KOBJECT_H
  7. #define ZEPHYR_INCLUDE_SYS_KOBJECT_H
  8. #include <stdint.h>
  9. #include <stddef.h>
  10. #ifdef __cplusplus
  11. extern "C" {
  12. #endif
  13. struct k_thread;
  14. struct k_mutex;
  15. struct z_futex_data;
  16. /**
  17. * @brief Kernel Object Types
  18. *
  19. * This enumeration needs to be kept in sync with the lists of kernel objects
  20. * and subsystems in scripts/gen_kobject_list.py, as well as the otype_to_str()
  21. * function in kernel/userspace.c
  22. */
  23. enum k_objects {
  24. K_OBJ_ANY,
  25. /** @cond
  26. * Doxygen should ignore this build-time generated include file
  27. * when genrating API documentation. Enumeration values are
  28. * generated during build by gen_kobject_list.py. It includes
  29. * basic kernel objects (e.g. pipes and mutexes) and driver types.
  30. */
  31. #include <kobj-types-enum.h>
  32. /** @endcond
  33. */
  34. K_OBJ_LAST
  35. };
  36. /**
  37. * @defgroup usermode_apis User Mode APIs
  38. * @ingroup kernel_apis
  39. * @{
  40. */
  41. #ifdef CONFIG_USERSPACE
  42. #ifdef CONFIG_GEN_PRIV_STACKS
  43. /* Metadata struct for K_OBJ_THREAD_STACK_ELEMENT */
  44. struct z_stack_data {
  45. /* Size of the entire stack object, including reserved areas */
  46. size_t size;
  47. /* Stack buffer for privilege mode elevations */
  48. uint8_t *priv;
  49. };
  50. #endif /* CONFIG_GEN_PRIV_STACKS */
  51. /* Object extra data. Only some objects use this, determined by object type */
  52. union z_object_data {
  53. /* Backing mutex for K_OBJ_SYS_MUTEX */
  54. struct k_mutex *mutex;
  55. /* Numerical thread ID for K_OBJ_THREAD */
  56. unsigned int thread_id;
  57. #ifdef CONFIG_GEN_PRIV_STACKS
  58. /* Metadata for K_OBJ_THREAD_STACK_ELEMENT */
  59. const struct z_stack_data *stack_data;
  60. #else
  61. /* Stack buffer size for K_OBJ_THREAD_STACK_ELEMENT */
  62. size_t stack_size;
  63. #endif /* CONFIG_GEN_PRIV_STACKS */
  64. /* Futex wait queue and spinlock for K_OBJ_FUTEX */
  65. struct z_futex_data *futex_data;
  66. /* All other objects */
  67. int unused;
  68. };
  69. /* Table generated by gperf, these objects are retrieved via
  70. * z_object_find() */
  71. struct z_object {
  72. void *name;
  73. uint8_t perms[CONFIG_MAX_THREAD_BYTES];
  74. uint8_t type;
  75. uint8_t flags;
  76. union z_object_data data;
  77. } __packed __aligned(4);
  78. struct z_object_assignment {
  79. struct k_thread *thread;
  80. void * const *objects;
  81. };
  82. /**
  83. * @brief Grant a static thread access to a list of kernel objects
  84. *
  85. * For threads declared with K_THREAD_DEFINE(), grant the thread access to
  86. * a set of kernel objects. These objects do not need to be in an initialized
  87. * state. The permissions will be granted when the threads are initialized
  88. * in the early boot sequence.
  89. *
  90. * All arguments beyond the first must be pointers to kernel objects.
  91. *
  92. * @param name_ Name of the thread, as passed to K_THREAD_DEFINE()
  93. */
  94. #define K_THREAD_ACCESS_GRANT(name_, ...) \
  95. static void * const _CONCAT(_object_list_, name_)[] = \
  96. { __VA_ARGS__, NULL }; \
  97. static const STRUCT_SECTION_ITERABLE(z_object_assignment, \
  98. _CONCAT(_object_access_, name_)) = \
  99. { (&_k_thread_obj_ ## name_), \
  100. (_CONCAT(_object_list_, name_)) }
  101. /** Object initialized */
  102. #define K_OBJ_FLAG_INITIALIZED BIT(0)
  103. /** Object is Public */
  104. #define K_OBJ_FLAG_PUBLIC BIT(1)
  105. /** Object allocated */
  106. #define K_OBJ_FLAG_ALLOC BIT(2)
  107. /** Driver Object */
  108. #define K_OBJ_FLAG_DRIVER BIT(3)
  109. /**
  110. * Lookup a kernel object and init its metadata if it exists
  111. *
  112. * Calling this on an object will make it usable from userspace.
  113. * Intended to be called as the last statement in kernel object init
  114. * functions.
  115. *
  116. * @param obj Address of the kernel object
  117. */
  118. void z_object_init(const void *obj);
  119. #else
  120. /* LCOV_EXCL_START */
  121. #define K_THREAD_ACCESS_GRANT(thread, ...)
  122. /**
  123. * @internal
  124. */
  125. static inline void z_object_init(const void *obj)
  126. {
  127. ARG_UNUSED(obj);
  128. }
  129. /**
  130. * @internal
  131. */
  132. static inline void z_impl_k_object_access_grant(const void *object,
  133. struct k_thread *thread)
  134. {
  135. ARG_UNUSED(object);
  136. ARG_UNUSED(thread);
  137. }
  138. /**
  139. * @internal
  140. */
  141. static inline void k_object_access_revoke(const void *object,
  142. struct k_thread *thread)
  143. {
  144. ARG_UNUSED(object);
  145. ARG_UNUSED(thread);
  146. }
  147. /**
  148. * @internal
  149. */
  150. static inline void z_impl_k_object_release(const void *object)
  151. {
  152. ARG_UNUSED(object);
  153. }
  154. static inline void k_object_access_all_grant(const void *object)
  155. {
  156. ARG_UNUSED(object);
  157. }
  158. /* LCOV_EXCL_STOP */
  159. #endif /* !CONFIG_USERSPACE */
  160. /**
  161. * Grant a thread access to a kernel object
  162. *
  163. * The thread will be granted access to the object if the caller is from
  164. * supervisor mode, or the caller is from user mode AND has permissions
  165. * on both the object and the thread whose access is being granted.
  166. *
  167. * @param object Address of kernel object
  168. * @param thread Thread to grant access to the object
  169. */
  170. __syscall void k_object_access_grant(const void *object,
  171. struct k_thread *thread);
  172. /**
  173. * Revoke a thread's access to a kernel object
  174. *
  175. * The thread will lose access to the object if the caller is from
  176. * supervisor mode, or the caller is from user mode AND has permissions
  177. * on both the object and the thread whose access is being revoked.
  178. *
  179. * @param object Address of kernel object
  180. * @param thread Thread to remove access to the object
  181. */
  182. void k_object_access_revoke(const void *object, struct k_thread *thread);
  183. /**
  184. * @brief Release an object
  185. *
  186. * Allows user threads to drop their own permission on an object
  187. * Their permissions are automatically cleared when a thread terminates.
  188. *
  189. * @param object The object to be released
  190. *
  191. */
  192. __syscall void k_object_release(const void *object);
  193. /**
  194. * Grant all present and future threads access to an object
  195. *
  196. * If the caller is from supervisor mode, or the caller is from user mode and
  197. * have sufficient permissions on the object, then that object will have
  198. * permissions granted to it for *all* current and future threads running in
  199. * the system, effectively becoming a public kernel object.
  200. *
  201. * Use of this API should be avoided on systems that are running untrusted code
  202. * as it is possible for such code to derive the addresses of kernel objects
  203. * and perform unwanted operations on them.
  204. *
  205. * It is not possible to revoke permissions on public objects; once public,
  206. * any thread may use it.
  207. *
  208. * @param object Address of kernel object
  209. */
  210. void k_object_access_all_grant(const void *object);
  211. /**
  212. * Allocate a kernel object of a designated type
  213. *
  214. * This will instantiate at runtime a kernel object of the specified type,
  215. * returning a pointer to it. The object will be returned in an uninitialized
  216. * state, with the calling thread being granted permission on it. The memory
  217. * for the object will be allocated out of the calling thread's resource pool.
  218. *
  219. * Currently, allocation of thread stacks is not supported.
  220. *
  221. * @param otype Requested kernel object type
  222. * @return A pointer to the allocated kernel object, or NULL if memory wasn't
  223. * available
  224. */
  225. __syscall void *k_object_alloc(enum k_objects otype);
  226. #ifdef CONFIG_DYNAMIC_OBJECTS
  227. /**
  228. * Allocate memory and install as a generic kernel object
  229. *
  230. * This is a low-level function to allocate some memory, and register that
  231. * allocated memory in the kernel object lookup tables with type K_OBJ_ANY.
  232. * Initialization state and thread permissions will be cleared. The
  233. * returned z_object's data value will be uninitialized.
  234. *
  235. * Most users will want to use k_object_alloc() instead.
  236. *
  237. * Memory allocated will be drawn from the calling thread's reasource pool
  238. * and may be freed later by passing the actual object pointer (found
  239. * in the returned z_object's 'name' member) to k_object_free().
  240. *
  241. * @param align Required memory alignment for the allocated object
  242. * @param size Size of the allocated object
  243. * @return NULL on insufficient memory
  244. * @return A pointer to the associated z_object that is installed in the
  245. * kernel object tables
  246. */
  247. struct z_object *z_dynamic_object_aligned_create(size_t align, size_t size);
  248. /**
  249. * Allocate memory and install as a generic kernel object
  250. *
  251. * This is a low-level function to allocate some memory, and register that
  252. * allocated memory in the kernel object lookup tables with type K_OBJ_ANY.
  253. * Initialization state and thread permissions will be cleared. The
  254. * returned z_object's data value will be uninitialized.
  255. *
  256. * Most users will want to use k_object_alloc() instead.
  257. *
  258. * Memory allocated will be drawn from the calling thread's reasource pool
  259. * and may be freed later by passing the actual object pointer (found
  260. * in the returned z_object's 'name' member) to k_object_free().
  261. *
  262. * @param size Size of the allocated object
  263. * @return NULL on insufficient memory
  264. * @return A pointer to the associated z_object that is installed in the
  265. * kernel object tables
  266. */
  267. static inline struct z_object *z_dynamic_object_create(size_t size)
  268. {
  269. return z_dynamic_object_aligned_create(0, size);
  270. }
  271. /**
  272. * Free a kernel object previously allocated with k_object_alloc()
  273. *
  274. * This will return memory for a kernel object back to resource pool it was
  275. * allocated from. Care must be exercised that the object will not be used
  276. * during or after when this call is made.
  277. *
  278. * @param obj Pointer to the kernel object memory address.
  279. */
  280. void k_object_free(void *obj);
  281. #else
  282. /* LCOV_EXCL_START */
  283. static inline void *z_impl_k_object_alloc(enum k_objects otype)
  284. {
  285. ARG_UNUSED(otype);
  286. return NULL;
  287. }
  288. static inline struct z_object *z_dynamic_object_aligned_create(size_t align,
  289. size_t size)
  290. {
  291. ARG_UNUSED(align);
  292. ARG_UNUSED(size);
  293. return NULL;
  294. }
  295. static inline struct z_object *z_dynamic_object_create(size_t size)
  296. {
  297. ARG_UNUSED(size);
  298. return NULL;
  299. }
  300. /**
  301. * @brief Free an object
  302. *
  303. * @param obj
  304. */
  305. static inline void k_object_free(void *obj)
  306. {
  307. ARG_UNUSED(obj);
  308. }
  309. /* LCOV_EXCL_STOP */
  310. #endif /* CONFIG_DYNAMIC_OBJECTS */
  311. /** @} */
  312. #include <syscalls/kobject.h>
  313. #ifdef __cplusplus
  314. }
  315. #endif
  316. #endif