msg_q.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. /* Copyright (c) 2011-2012, 2014, 2017 The Linux Foundation. All rights reserved.
  2. *
  3. * Redistribution and use in source and binary forms, with or without
  4. * modification, are permitted provided that the following conditions are
  5. * met:
  6. * * Redistributions of source code must retain the above copyright
  7. * notice, this list of conditions and the following disclaimer.
  8. * * Redistributions in binary form must reproduce the above
  9. * copyright notice, this list of conditions and the following
  10. * disclaimer in the documentation and/or other materials provided
  11. * with the distribution.
  12. * * Neither the name of The Linux Foundation, nor the names of its
  13. * contributors may be used to endorse or promote products derived
  14. * from this software without specific prior written permission.
  15. *
  16. * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
  17. * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
  18. * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
  19. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
  20. * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  21. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  22. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
  23. * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
  24. * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
  25. * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
  26. * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  27. */
  28. // Uncomment to log verbose logs
  29. #define LOG_NDEBUG 1
  30. #define LOG_TAG "LocSvc_utils_q"
  31. #include <stdio.h>
  32. #include <stdlib.h>
  33. #include <pthread.h>
  34. #include <loc_pla.h>
  35. #include <log_util.h>
  36. #include "linked_list.h"
  37. #include "msg_q.h"
  38. typedef struct msg_q {
  39. void* msg_list; /* Linked list to store information */
  40. pthread_cond_t list_cond; /* Condition variable for waiting on msg queue */
  41. pthread_mutex_t list_mutex; /* Mutex for exclusive access to message queue */
  42. int unblocked; /* Has this message queue been unblocked? */
  43. } msg_q;
  44. /*===========================================================================
  45. FUNCTION convert_linked_list_err_type
  46. DESCRIPTION
  47. Converts from one set of enum values to another.
  48. linked_list_val: Value to convert to msg_q_enum_type
  49. DEPENDENCIES
  50. N/A
  51. RETURN VALUE
  52. Corresponding linked_list_enum_type in msg_q_enum_type
  53. SIDE EFFECTS
  54. N/A
  55. ===========================================================================*/
  56. static msq_q_err_type convert_linked_list_err_type(linked_list_err_type linked_list_val)
  57. {
  58. switch( linked_list_val )
  59. {
  60. case eLINKED_LIST_SUCCESS:
  61. return eMSG_Q_SUCCESS;
  62. case eLINKED_LIST_INVALID_PARAMETER:
  63. return eMSG_Q_INVALID_PARAMETER;
  64. case eLINKED_LIST_INVALID_HANDLE:
  65. return eMSG_Q_INVALID_HANDLE;
  66. case eLINKED_LIST_UNAVAILABLE_RESOURCE:
  67. return eMSG_Q_UNAVAILABLE_RESOURCE;
  68. case eLINKED_LIST_INSUFFICIENT_BUFFER:
  69. return eMSG_Q_INSUFFICIENT_BUFFER;
  70. case eLINKED_LIST_FAILURE_GENERAL:
  71. default:
  72. return eMSG_Q_FAILURE_GENERAL;
  73. }
  74. }
  75. /* ----------------------- END INTERNAL FUNCTIONS ---------------------------------------- */
  76. /*===========================================================================
  77. FUNCTION: msg_q_init
  78. ===========================================================================*/
  79. msq_q_err_type msg_q_init(void** msg_q_data)
  80. {
  81. if( msg_q_data == NULL )
  82. {
  83. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  84. return eMSG_Q_INVALID_PARAMETER;
  85. }
  86. msg_q* tmp_msg_q;
  87. tmp_msg_q = (msg_q*)calloc(1, sizeof(msg_q));
  88. if( tmp_msg_q == NULL )
  89. {
  90. LOC_LOGE("%s: Unable to allocate space for message queue!\n", __FUNCTION__);
  91. return eMSG_Q_FAILURE_GENERAL;
  92. }
  93. if( linked_list_init(&tmp_msg_q->msg_list) != 0 )
  94. {
  95. LOC_LOGE("%s: Unable to initialize storage list!\n", __FUNCTION__);
  96. free(tmp_msg_q);
  97. return eMSG_Q_FAILURE_GENERAL;
  98. }
  99. if( pthread_mutex_init(&tmp_msg_q->list_mutex, NULL) != 0 )
  100. {
  101. LOC_LOGE("%s: Unable to initialize list mutex!\n", __FUNCTION__);
  102. linked_list_destroy(&tmp_msg_q->msg_list);
  103. free(tmp_msg_q);
  104. return eMSG_Q_FAILURE_GENERAL;
  105. }
  106. if( pthread_cond_init(&tmp_msg_q->list_cond, NULL) != 0 )
  107. {
  108. LOC_LOGE("%s: Unable to initialize msg q cond var!\n", __FUNCTION__);
  109. linked_list_destroy(&tmp_msg_q->msg_list);
  110. pthread_mutex_destroy(&tmp_msg_q->list_mutex);
  111. free(tmp_msg_q);
  112. return eMSG_Q_FAILURE_GENERAL;
  113. }
  114. tmp_msg_q->unblocked = 0;
  115. *msg_q_data = tmp_msg_q;
  116. return eMSG_Q_SUCCESS;
  117. }
  118. /*===========================================================================
  119. FUNCTION: msg_q_init2
  120. ===========================================================================*/
  121. const void* msg_q_init2()
  122. {
  123. void* q = NULL;
  124. if (eMSG_Q_SUCCESS != msg_q_init(&q)) {
  125. q = NULL;
  126. }
  127. return q;
  128. }
  129. /*===========================================================================
  130. FUNCTION: msg_q_destroy
  131. ===========================================================================*/
  132. msq_q_err_type msg_q_destroy(void** msg_q_data)
  133. {
  134. if( msg_q_data == NULL )
  135. {
  136. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  137. return eMSG_Q_INVALID_HANDLE;
  138. }
  139. msg_q* p_msg_q = (msg_q*)*msg_q_data;
  140. linked_list_destroy(&p_msg_q->msg_list);
  141. pthread_mutex_destroy(&p_msg_q->list_mutex);
  142. pthread_cond_destroy(&p_msg_q->list_cond);
  143. p_msg_q->unblocked = 0;
  144. free(*msg_q_data);
  145. *msg_q_data = NULL;
  146. return eMSG_Q_SUCCESS;
  147. }
  148. /*===========================================================================
  149. FUNCTION: msg_q_snd
  150. ===========================================================================*/
  151. msq_q_err_type msg_q_snd(void* msg_q_data, void* msg_obj, void (*dealloc)(void*))
  152. {
  153. msq_q_err_type rv;
  154. if( msg_q_data == NULL )
  155. {
  156. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  157. return eMSG_Q_INVALID_HANDLE;
  158. }
  159. if( msg_obj == NULL )
  160. {
  161. LOC_LOGE("%s: Invalid msg_obj parameter!\n", __FUNCTION__);
  162. return eMSG_Q_INVALID_PARAMETER;
  163. }
  164. msg_q* p_msg_q = (msg_q*)msg_q_data;
  165. pthread_mutex_lock(&p_msg_q->list_mutex);
  166. LOC_LOGV("%s: Sending message with handle = %p\n", __FUNCTION__, msg_obj);
  167. if( p_msg_q->unblocked )
  168. {
  169. LOC_LOGE("%s: Message queue has been unblocked.\n", __FUNCTION__);
  170. pthread_mutex_unlock(&p_msg_q->list_mutex);
  171. return eMSG_Q_UNAVAILABLE_RESOURCE;
  172. }
  173. rv = convert_linked_list_err_type(linked_list_add(p_msg_q->msg_list, msg_obj, dealloc));
  174. /* Show data is in the message queue. */
  175. pthread_cond_signal(&p_msg_q->list_cond);
  176. pthread_mutex_unlock(&p_msg_q->list_mutex);
  177. LOC_LOGV("%s: Finished Sending message with handle = %p\n", __FUNCTION__, msg_obj);
  178. return rv;
  179. }
  180. /*===========================================================================
  181. FUNCTION: msg_q_rcv
  182. ===========================================================================*/
  183. msq_q_err_type msg_q_rcv(void* msg_q_data, void** msg_obj)
  184. {
  185. msq_q_err_type rv;
  186. if( msg_q_data == NULL )
  187. {
  188. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  189. return eMSG_Q_INVALID_HANDLE;
  190. }
  191. if( msg_obj == NULL )
  192. {
  193. LOC_LOGE("%s: Invalid msg_obj parameter!\n", __FUNCTION__);
  194. return eMSG_Q_INVALID_PARAMETER;
  195. }
  196. msg_q* p_msg_q = (msg_q*)msg_q_data;
  197. pthread_mutex_lock(&p_msg_q->list_mutex);
  198. if( p_msg_q->unblocked )
  199. {
  200. LOC_LOGE("%s: Message queue has been unblocked.\n", __FUNCTION__);
  201. pthread_mutex_unlock(&p_msg_q->list_mutex);
  202. return eMSG_Q_UNAVAILABLE_RESOURCE;
  203. }
  204. /* Wait for data in the message queue */
  205. while( linked_list_empty(p_msg_q->msg_list) && !p_msg_q->unblocked )
  206. {
  207. pthread_cond_wait(&p_msg_q->list_cond, &p_msg_q->list_mutex);
  208. }
  209. rv = convert_linked_list_err_type(linked_list_remove(p_msg_q->msg_list, msg_obj));
  210. pthread_mutex_unlock(&p_msg_q->list_mutex);
  211. LOC_LOGV("%s: Received message %p rv = %d\n", __FUNCTION__, *msg_obj, rv);
  212. return rv;
  213. }
  214. /*===========================================================================
  215. FUNCTION: msg_q_rmv
  216. ===========================================================================*/
  217. msq_q_err_type msg_q_rmv(void* msg_q_data, void** msg_obj)
  218. {
  219. msq_q_err_type rv;
  220. if (msg_q_data == NULL) {
  221. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  222. return eMSG_Q_INVALID_HANDLE;
  223. }
  224. if (msg_obj == NULL) {
  225. LOC_LOGE("%s: Invalid msg_obj parameter!\n", __FUNCTION__);
  226. return eMSG_Q_INVALID_PARAMETER;
  227. }
  228. msg_q* p_msg_q = (msg_q*)msg_q_data;
  229. pthread_mutex_lock(&p_msg_q->list_mutex);
  230. if (p_msg_q->unblocked) {
  231. LOC_LOGE("%s: Message queue has been unblocked.\n", __FUNCTION__);
  232. pthread_mutex_unlock(&p_msg_q->list_mutex);
  233. return eMSG_Q_UNAVAILABLE_RESOURCE;
  234. }
  235. if (linked_list_empty(p_msg_q->msg_list)) {
  236. LOC_LOGW("%s: list is empty !!\n", __FUNCTION__);
  237. pthread_mutex_unlock(&p_msg_q->list_mutex);
  238. return eLINKED_LIST_EMPTY;
  239. }
  240. rv = convert_linked_list_err_type(linked_list_remove(p_msg_q->msg_list, msg_obj));
  241. pthread_mutex_unlock(&p_msg_q->list_mutex);
  242. LOC_LOGV("%s: Removed message %p rv = %d\n", __FUNCTION__, *msg_obj, rv);
  243. return rv;
  244. }
  245. /*===========================================================================
  246. FUNCTION: msg_q_flush
  247. ===========================================================================*/
  248. msq_q_err_type msg_q_flush(void* msg_q_data)
  249. {
  250. msq_q_err_type rv;
  251. if ( msg_q_data == NULL )
  252. {
  253. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  254. return eMSG_Q_INVALID_HANDLE;
  255. }
  256. msg_q* p_msg_q = (msg_q*)msg_q_data;
  257. LOC_LOGD("%s: Flushing Message Queue\n", __FUNCTION__);
  258. pthread_mutex_lock(&p_msg_q->list_mutex);
  259. /* Remove all elements from the list */
  260. rv = convert_linked_list_err_type(linked_list_flush(p_msg_q->msg_list));
  261. pthread_mutex_unlock(&p_msg_q->list_mutex);
  262. LOC_LOGD("%s: Message Queue flushed\n", __FUNCTION__);
  263. return rv;
  264. }
  265. /*===========================================================================
  266. FUNCTION: msg_q_unblock
  267. ===========================================================================*/
  268. msq_q_err_type msg_q_unblock(void* msg_q_data)
  269. {
  270. if ( msg_q_data == NULL )
  271. {
  272. LOC_LOGE("%s: Invalid msg_q_data parameter!\n", __FUNCTION__);
  273. return eMSG_Q_INVALID_HANDLE;
  274. }
  275. msg_q* p_msg_q = (msg_q*)msg_q_data;
  276. pthread_mutex_lock(&p_msg_q->list_mutex);
  277. if( p_msg_q->unblocked )
  278. {
  279. LOC_LOGE("%s: Message queue has been unblocked.\n", __FUNCTION__);
  280. pthread_mutex_unlock(&p_msg_q->list_mutex);
  281. return eMSG_Q_UNAVAILABLE_RESOURCE;
  282. }
  283. LOC_LOGD("%s: Unblocking Message Queue\n", __FUNCTION__);
  284. /* Unblocking message queue */
  285. p_msg_q->unblocked = 1;
  286. /* Allow all the waiters to wake up */
  287. pthread_cond_broadcast(&p_msg_q->list_cond);
  288. pthread_mutex_unlock(&p_msg_q->list_mutex);
  289. LOC_LOGD("%s: Message Queue unblocked\n", __FUNCTION__);
  290. return eMSG_Q_SUCCESS;
  291. }