ipc.h 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. /*
  2. * Copyright (c) 2022 Agustina Arzille.
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 3 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. *
  17. * Definitions for inter-process communication.
  18. */
  19. #ifndef KERN_IPC_H
  20. #define KERN_IPC_H
  21. #include <iovec.h>
  22. #include <stdbool.h>
  23. #include <stdint.h>
  24. #include <kern/types.h>
  25. struct ipc_msg_vme
  26. {
  27. uintptr_t addr;
  28. size_t size;
  29. int prot;
  30. int max_prot;
  31. };
  32. struct ipc_msg_cap
  33. {
  34. int cap;
  35. uint32_t flags;
  36. };
  37. #define IPC_IOV_ITER_CACHE_SIZE 8
  38. struct ipc_iov_iter
  39. {
  40. struct iovec cache[IPC_IOV_ITER_CACHE_SIZE];
  41. uint32_t cache_idx;
  42. struct iovec head;
  43. struct iovec *begin;
  44. uint32_t cur;
  45. uint32_t end;
  46. };
  47. struct ipc_cap_iter
  48. {
  49. struct ipc_msg_cap *begin;
  50. uint32_t cur;
  51. uint32_t end;
  52. };
  53. struct ipc_vme_iter
  54. {
  55. struct ipc_msg_vme *begin;
  56. uint32_t cur;
  57. uint32_t end;
  58. };
  59. struct ipc_msg
  60. {
  61. uint32_t size;
  62. struct iovec *iovs;
  63. uint32_t iov_cnt;
  64. struct ipc_msg_vme *vmes;
  65. uint32_t vme_cnt;
  66. struct ipc_msg_cap *caps;
  67. uint32_t cap_cnt;
  68. };
  69. // Bits for the 'flags' member of a IPC message metadata.
  70. #define IPC_MSG_TRUNC 0x01 // Reply was truncated.
  71. #define IPC_MSG_ERROR 0x02 // There was an error during IPC.
  72. #define IPC_MSG_KERNEL 0x04 // Message was sent on behalf of the kernel.
  73. struct ipc_msg_data
  74. {
  75. uint32_t size;
  76. int task_id;
  77. int thread_id;
  78. uint32_t flags;
  79. uintptr_t tag;
  80. ssize_t bytes_sent;
  81. uint32_t vmes_sent;
  82. uint32_t caps_sent;
  83. ssize_t bytes_recv;
  84. uint32_t vmes_recv;
  85. uint32_t caps_recv;
  86. };
  87. struct task;
  88. // Flags for the ipc copy interfaces.
  89. #define IPC_COPY_FROM (1 << 0) // Copy from the remote task.
  90. #define IPC_COPY_TO (1 << 1) // Copy to the remote task.
  91. #define IPC_CHECK_LOCAL (1 << 2)
  92. #define IPC_CHECK_REMOTE (1 << 3)
  93. #define IPC_CHECK_BOTH (IPC_CHECK_LOCAL | IPC_CHECK_REMOTE)
  94. /*
  95. * IPC iterator functions.
  96. *
  97. * These come in 3 flavors: iovec, capabilities and VMEs. The former works
  98. * by simply copying data; the other 2 by transfering objects across tasks.
  99. */
  100. static inline void
  101. ipc_cap_iter_init (struct ipc_cap_iter *it,
  102. struct ipc_msg_cap *msg, uint32_t nr_msgs)
  103. {
  104. it->begin = msg;
  105. it->cur = 0, it->end = nr_msgs;
  106. }
  107. static inline int
  108. ipc_cap_iter_size (const struct ipc_cap_iter *it)
  109. {
  110. return ((int)(it->end - it->cur));
  111. }
  112. static inline void
  113. ipc_vme_iter_init (struct ipc_vme_iter *it,
  114. struct ipc_msg_vme *msg, uint32_t nr_msgs)
  115. {
  116. it->begin = msg;
  117. it->cur = 0, it->end = nr_msgs;
  118. }
  119. static inline int
  120. ipc_vme_iter_size (const struct ipc_vme_iter *it)
  121. {
  122. return ((int)(it->end - it->cur));
  123. }
  124. static inline void
  125. ipc_iov_iter_init_buf (struct ipc_iov_iter *it, void *buf, size_t size)
  126. {
  127. it->head = IOVEC (buf, size);
  128. it->cache_idx = IPC_IOV_ITER_CACHE_SIZE;
  129. it->begin = &it->head;
  130. it->cur = it->end = 1;
  131. }
  132. static inline bool
  133. ipc_iov_iter_empty (const struct ipc_iov_iter *it)
  134. {
  135. return (it->cur >= it->end && !it->head.iov_len &&
  136. it->cache_idx >= IPC_IOV_ITER_CACHE_SIZE);
  137. }
  138. static inline void
  139. ipc_iov_iter_init (struct ipc_iov_iter *it, struct iovec *vecs, uint32_t cnt)
  140. {
  141. it->head.iov_len = 0;
  142. it->begin = vecs;
  143. it->cur = 0, it->end = cnt;
  144. it->cache_idx = IPC_IOV_ITER_CACHE_SIZE;
  145. }
  146. // Advance an iovec iterator with pointers from userspace.
  147. struct iovec* ipc_iov_iter_usrnext (struct ipc_iov_iter *it, ssize_t *errp);
  148. // Copy bytes between a local and a remote task.
  149. ssize_t ipc_bcopy (struct task *r_task, void *r_ptr, size_t r_size,
  150. void *l_ptr, size_t l_size, uint32_t flags);
  151. // Copy bytes in iterators between a local and a remote task.
  152. ssize_t ipc_iov_iter_copy (struct task *r_task, struct ipc_iov_iter *r_it,
  153. struct ipc_iov_iter *l_it, uint32_t direction);
  154. // Transfer capabilities in iterators between a local and a remote task.
  155. int ipc_cap_iter_copy (struct task *r_task, struct ipc_cap_iter *r_it,
  156. struct ipc_cap_iter *l_it, uint32_t flags);
  157. // Transfer VMEs in iterators between a local and a remote task.
  158. int ipc_vme_iter_copy (struct task *r_task, struct ipc_vme_iter *r_it,
  159. struct ipc_vme_iter *l_it, uint32_t flags);
  160. // Copy bytes in iovecs between a local and a remote task.
  161. static inline ssize_t
  162. ipc_bcopyv (struct task *r_task, struct iovec *r_iov, uint32_t r_niov,
  163. struct iovec *l_iov, uint32_t l_niov, uint32_t flags)
  164. {
  165. struct ipc_iov_iter r_it, l_it;
  166. ipc_iov_iter_init (&l_it, l_iov, l_niov);
  167. ipc_iov_iter_init (&r_it, r_iov, r_niov);
  168. return (ipc_iov_iter_copy (r_task, &r_it, &l_it, flags));
  169. }
  170. // Transfer VMEs between a remote and a local task.
  171. static inline int
  172. ipc_copy_vmes (struct task *r_task, struct ipc_msg_vme *r_vmes,
  173. uint32_t r_nvmes, struct ipc_msg_vme *l_vmes,
  174. uint32_t l_nvmes, uint32_t flags)
  175. {
  176. struct ipc_vme_iter r_it, l_it;
  177. ipc_vme_iter_init (&r_it, r_vmes, r_nvmes);
  178. ipc_vme_iter_init (&l_it, l_vmes, l_nvmes);
  179. return (ipc_vme_iter_copy (r_task, &r_it, &l_it, flags));
  180. }
  181. // Transfer capabilities between a remote and a local task.
  182. static inline int
  183. ipc_copy_caps (struct task *r_task, struct ipc_msg_cap *r_caps,
  184. uint32_t r_ncaps, struct ipc_msg_cap *l_caps,
  185. uint32_t l_ncaps, uint32_t flags)
  186. {
  187. struct ipc_cap_iter r_it, l_it;
  188. ipc_cap_iter_init (&r_it, r_caps, r_ncaps);
  189. ipc_cap_iter_init (&l_it, l_caps, l_ncaps);
  190. return (ipc_cap_iter_copy (r_task, &r_it, &l_it, flags));
  191. }
  192. #endif