lwp_pmutex.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2021/01/02 bernard the first version
  9. * 2022/12/18 bernard fix the _m_lock to tid in user land.
  10. */
  11. #include "lwp_internal.h"
  12. #include <rtthread.h>
  13. #ifdef ARCH_MM_MMU
  14. #include <lwp_user_mm.h>
  15. #endif
  16. #include <sys/time.h>
  17. #include <syscall_generic.h>
  18. #define PMUTEX_NORMAL 0 /* Unable to recursion */
  19. #define PMUTEX_RECURSIVE 1 /* Can be recursion */
  20. #define PMUTEX_ERRORCHECK 2 /* This type of mutex provides error checking */
  21. struct rt_pmutex
  22. {
  23. union
  24. {
  25. rt_mutex_t kmutex;
  26. rt_sem_t ksem; /* use sem to emulate the mutex without recursive */
  27. } lock;
  28. struct lwp_avl_struct node;
  29. struct rt_object *custom_obj;
  30. rt_uint8_t type; /* pmutex type */
  31. };
  32. /*
  33. * userspace mutex definitions in musl
  34. */
  35. struct rt_umutex
  36. {
  37. union
  38. {
  39. int __i[6];
  40. volatile int __vi[6];
  41. volatile void *volatile __p[6];
  42. } __u;
  43. };
  44. #define _m_type __u.__i[0]
  45. #define _m_lock __u.__vi[1]
  46. #define _m_waiters __u.__vi[2]
  47. #define _m_prev __u.__p[3]
  48. #define _m_next __u.__p[4]
  49. #define _m_count __u.__i[5]
  50. static struct rt_mutex _pmutex_lock;
  51. static int pmutex_system_init(void)
  52. {
  53. rt_mutex_init(&_pmutex_lock, "pmtxLock", RT_IPC_FLAG_FIFO);
  54. return 0;
  55. }
  56. INIT_PREV_EXPORT(pmutex_system_init);
  57. static rt_err_t pmutex_destory(void *data)
  58. {
  59. rt_err_t ret = -1;
  60. struct rt_pmutex *pmutex = (struct rt_pmutex *)data;
  61. if (pmutex)
  62. {
  63. lwp_mutex_take_safe(&_pmutex_lock, RT_WAITING_FOREVER, 0);
  64. /* remove pmutex from pmutext avl */
  65. lwp_avl_remove(&pmutex->node, (struct lwp_avl_struct **)pmutex->node.data);
  66. lwp_mutex_release_safe(&_pmutex_lock);
  67. if (pmutex->type == PMUTEX_NORMAL)
  68. {
  69. rt_sem_delete(pmutex->lock.ksem);
  70. }
  71. else
  72. {
  73. rt_mutex_delete(pmutex->lock.kmutex);
  74. }
  75. /* release object */
  76. rt_free(pmutex);
  77. ret = 0;
  78. }
  79. return ret;
  80. }
  81. static struct rt_pmutex* pmutex_create(void *umutex, struct rt_lwp *lwp)
  82. {
  83. struct rt_pmutex *pmutex = RT_NULL;
  84. struct rt_object *obj = RT_NULL;
  85. rt_ubase_t type;
  86. if (!lwp)
  87. {
  88. return RT_NULL;
  89. }
  90. long *p = (long *)umutex;
  91. /* umutex[0] bit[0-1] saved mutex type */
  92. type = *p & 3;
  93. if (type != PMUTEX_NORMAL && type != PMUTEX_RECURSIVE && type != PMUTEX_ERRORCHECK)
  94. {
  95. return RT_NULL;
  96. }
  97. pmutex = (struct rt_pmutex *)rt_malloc(sizeof(struct rt_pmutex));
  98. if (!pmutex)
  99. {
  100. return RT_NULL;
  101. }
  102. if (type == PMUTEX_NORMAL)
  103. {
  104. pmutex->lock.ksem = rt_sem_create("pmutex", 1, RT_IPC_FLAG_PRIO);
  105. if (!pmutex->lock.ksem)
  106. {
  107. rt_free(pmutex);
  108. return RT_NULL;
  109. }
  110. }
  111. else
  112. {
  113. pmutex->lock.kmutex = rt_mutex_create("pmutex", RT_IPC_FLAG_PRIO);
  114. if (!pmutex->lock.kmutex)
  115. {
  116. rt_free(pmutex);
  117. return RT_NULL;
  118. }
  119. }
  120. obj = rt_custom_object_create("pmutex", (void *)pmutex, pmutex_destory);
  121. if (!obj)
  122. {
  123. if (pmutex->type == PMUTEX_NORMAL)
  124. {
  125. rt_sem_delete(pmutex->lock.ksem);
  126. }
  127. else
  128. {
  129. rt_mutex_delete(pmutex->lock.kmutex);
  130. }
  131. rt_free(pmutex);
  132. return RT_NULL;
  133. }
  134. pmutex->node.avl_key = (avl_key_t)umutex;
  135. pmutex->node.data = &lwp->address_search_head;
  136. pmutex->custom_obj = obj;
  137. pmutex->type = type;
  138. /* insert into pmutex head */
  139. lwp_avl_insert(&pmutex->node, &lwp->address_search_head);
  140. return pmutex;
  141. }
  142. static struct rt_pmutex* pmutex_get(void *umutex, struct rt_lwp *lwp)
  143. {
  144. struct rt_pmutex *pmutex = RT_NULL;
  145. struct lwp_avl_struct *node = RT_NULL;
  146. node = lwp_avl_find((avl_key_t)umutex, lwp->address_search_head);
  147. if (!node)
  148. {
  149. return RT_NULL;
  150. }
  151. pmutex = rt_container_of(node, struct rt_pmutex, node);
  152. return pmutex;
  153. }
  154. static int _pthread_mutex_init(void *umutex)
  155. {
  156. struct rt_lwp *lwp = RT_NULL;
  157. struct rt_pmutex *pmutex = RT_NULL;
  158. rt_err_t lock_ret = 0;
  159. /* umutex union is 6 x (void *) */
  160. if (!lwp_user_accessable(umutex, sizeof(void *) * 6))
  161. {
  162. rt_set_errno(EINVAL);
  163. return -EINVAL;
  164. }
  165. lock_ret = rt_mutex_take_interruptible(&_pmutex_lock, RT_WAITING_FOREVER);
  166. if (lock_ret != RT_EOK)
  167. {
  168. rt_set_errno(EAGAIN);
  169. return -EAGAIN;
  170. }
  171. lwp = lwp_self();
  172. pmutex = pmutex_get(umutex, lwp);
  173. if (pmutex == RT_NULL)
  174. {
  175. /* create a pmutex according to this umutex */
  176. pmutex = pmutex_create(umutex, lwp);
  177. if (pmutex == RT_NULL)
  178. {
  179. rt_mutex_release(&_pmutex_lock);
  180. rt_set_errno(ENOMEM);
  181. return -ENOMEM;
  182. }
  183. if (lwp_user_object_add(lwp, pmutex->custom_obj) != 0)
  184. {
  185. rt_custom_object_destroy(pmutex->custom_obj);
  186. rt_set_errno(ENOMEM);
  187. return -ENOMEM;
  188. }
  189. }
  190. else
  191. {
  192. lwp_mutex_take_safe(&_pmutex_lock, RT_WAITING_FOREVER, 1);
  193. if (pmutex->type == PMUTEX_NORMAL)
  194. {
  195. pmutex->lock.ksem->value = 1;
  196. }
  197. else
  198. {
  199. pmutex->lock.kmutex->owner = RT_NULL;
  200. pmutex->lock.kmutex->priority = 0xFF;
  201. pmutex->lock.kmutex->hold = 0;
  202. pmutex->lock.kmutex->ceiling_priority = 0xFF;
  203. }
  204. lwp_mutex_release_safe(&_pmutex_lock);
  205. }
  206. rt_mutex_release(&_pmutex_lock);
  207. return 0;
  208. }
  209. static int _pthread_mutex_lock_timeout(void *umutex, struct timespec *timeout)
  210. {
  211. struct rt_lwp *lwp = RT_NULL;
  212. struct rt_pmutex *pmutex = RT_NULL;
  213. struct rt_umutex *umutex_p = (struct rt_umutex*)umutex;
  214. rt_err_t lock_ret = 0;
  215. rt_int32_t time = RT_WAITING_FOREVER;
  216. if (!lwp_user_accessable((void *)umutex, sizeof(struct rt_umutex)))
  217. {
  218. rt_set_errno(EINVAL);
  219. return -EINVAL;
  220. }
  221. if (timeout)
  222. {
  223. if (!lwp_user_accessable((void *)timeout, sizeof(struct timespec)))
  224. {
  225. rt_set_errno(EINVAL);
  226. return -EINVAL;
  227. }
  228. time = rt_timespec_to_tick(timeout);
  229. }
  230. lock_ret = rt_mutex_take_interruptible(&_pmutex_lock, RT_WAITING_FOREVER);
  231. if (lock_ret != RT_EOK)
  232. {
  233. rt_set_errno(EINTR);
  234. return -EINTR;
  235. }
  236. lwp = lwp_self();
  237. pmutex = pmutex_get(umutex, lwp);
  238. if (pmutex == RT_NULL)
  239. {
  240. rt_mutex_release(&_pmutex_lock);
  241. rt_set_errno(EINVAL);
  242. return -ENOMEM; /* umutex not recored in kernel */
  243. }
  244. rt_mutex_release(&_pmutex_lock);
  245. switch (pmutex->type)
  246. {
  247. case PMUTEX_NORMAL:
  248. lock_ret = rt_sem_take_interruptible(pmutex->lock.ksem, time);
  249. break;
  250. case PMUTEX_RECURSIVE:
  251. lock_ret = rt_mutex_take_interruptible(pmutex->lock.kmutex, time);
  252. if (lock_ret == RT_EOK)
  253. {
  254. umutex_p->_m_lock = rt_thread_self()->tid;
  255. }
  256. break;
  257. case PMUTEX_ERRORCHECK:
  258. lock_ret = lwp_mutex_take_safe(&_pmutex_lock, RT_WAITING_FOREVER, 1);
  259. if (lock_ret != RT_EOK)
  260. {
  261. return -EINTR;
  262. }
  263. if (pmutex->lock.kmutex->owner == rt_thread_self())
  264. {
  265. lwp_mutex_release_safe(&_pmutex_lock);
  266. return -EDEADLK;
  267. }
  268. lock_ret = rt_mutex_take_interruptible(pmutex->lock.kmutex, time);
  269. if (lock_ret == RT_EOK)
  270. {
  271. umutex_p->_m_lock = rt_thread_self()->tid;
  272. }
  273. lwp_mutex_release_safe(&_pmutex_lock);
  274. break;
  275. default: /* unknown type */
  276. return -EINVAL;
  277. }
  278. if (lock_ret != RT_EOK)
  279. {
  280. if (lock_ret == -RT_ETIMEOUT)
  281. {
  282. if (time == 0) /* timeout is 0, means try lock failed */
  283. {
  284. rt_set_errno(EBUSY);
  285. return -EBUSY;
  286. }
  287. else
  288. {
  289. rt_set_errno(ETIMEDOUT);
  290. return -ETIMEDOUT;
  291. }
  292. }
  293. else if (lock_ret == -RT_EINTR)
  294. {
  295. rt_set_errno(EINTR);
  296. return -EINTR;
  297. }
  298. else
  299. {
  300. rt_set_errno(EAGAIN);
  301. return -EAGAIN;
  302. }
  303. }
  304. return 0;
  305. }
  306. static int _pthread_mutex_unlock(void *umutex)
  307. {
  308. rt_err_t lock_ret = 0;
  309. struct rt_lwp *lwp = RT_NULL;
  310. struct rt_pmutex *pmutex = RT_NULL;
  311. struct rt_umutex *umutex_p = (struct rt_umutex*)umutex;
  312. lock_ret = rt_mutex_take_interruptible(&_pmutex_lock, RT_WAITING_FOREVER);
  313. if (lock_ret != RT_EOK)
  314. {
  315. rt_set_errno(EAGAIN);
  316. return -EAGAIN;
  317. }
  318. lwp = lwp_self();
  319. pmutex = pmutex_get(umutex, lwp);
  320. if (pmutex == RT_NULL)
  321. {
  322. rt_mutex_release(&_pmutex_lock);
  323. rt_set_errno(EPERM);
  324. return -EPERM;//unlock static mutex of unlock state
  325. }
  326. rt_mutex_release(&_pmutex_lock);
  327. switch (pmutex->type)
  328. {
  329. case PMUTEX_NORMAL:
  330. if(pmutex->lock.ksem->value >=1)
  331. {
  332. rt_set_errno(EPERM);
  333. return -EPERM;//unlock dynamic mutex of unlock state
  334. }
  335. else
  336. {
  337. lock_ret = rt_sem_release(pmutex->lock.ksem);
  338. }
  339. break;
  340. case PMUTEX_RECURSIVE:
  341. case PMUTEX_ERRORCHECK:
  342. lock_ret = rt_mutex_release(pmutex->lock.kmutex);
  343. if ((lock_ret == RT_EOK) && pmutex->lock.kmutex->owner == NULL)
  344. {
  345. umutex_p->_m_lock = 0;
  346. }
  347. break;
  348. default: /* unknown type */
  349. return -EINVAL;
  350. }
  351. if (lock_ret != RT_EOK)
  352. {
  353. rt_set_errno(EPERM);
  354. return -EPERM;
  355. }
  356. return 0;
  357. }
  358. static int _pthread_mutex_destroy(void *umutex)
  359. {
  360. struct rt_lwp *lwp = RT_NULL;
  361. struct rt_pmutex *pmutex = RT_NULL;
  362. rt_err_t lock_ret = 0;
  363. lock_ret = rt_mutex_take_interruptible(&_pmutex_lock, RT_WAITING_FOREVER);
  364. if (lock_ret != RT_EOK)
  365. {
  366. rt_set_errno(EAGAIN);
  367. return -EAGAIN;
  368. }
  369. lwp = lwp_self();
  370. pmutex = pmutex_get(umutex, lwp);
  371. if (pmutex == RT_NULL)
  372. {
  373. rt_mutex_release(&_pmutex_lock);
  374. rt_set_errno(EINVAL);
  375. return -EINVAL;
  376. }
  377. lwp_user_object_delete(lwp, pmutex->custom_obj);
  378. rt_mutex_release(&_pmutex_lock);
  379. return 0;
  380. }
  381. sysret_t sys_pmutex(void *umutex, int op, void *arg)
  382. {
  383. int ret = -EINVAL;
  384. switch (op)
  385. {
  386. case PMUTEX_INIT:
  387. ret = _pthread_mutex_init(umutex);
  388. break;
  389. case PMUTEX_LOCK:
  390. ret = _pthread_mutex_lock_timeout(umutex, (struct timespec*)arg);
  391. if (ret == -ENOMEM)
  392. {
  393. /* lock not init, try init it and lock again. */
  394. ret = _pthread_mutex_init(umutex);
  395. if (ret == 0)
  396. {
  397. ret = _pthread_mutex_lock_timeout(umutex, (struct timespec*)arg);
  398. }
  399. }
  400. break;
  401. case PMUTEX_UNLOCK:
  402. ret = _pthread_mutex_unlock(umutex);
  403. break;
  404. case PMUTEX_DESTROY:
  405. ret = _pthread_mutex_destroy(umutex);
  406. break;
  407. default:
  408. rt_set_errno(EINVAL);
  409. break;
  410. }
  411. return ret;
  412. }