audio_tws_aps.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  1. /*
  2. * Copyright (c) 2016 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief audio stream.
  9. */
  10. #include <os_common_api.h>
  11. #include <mem_manager.h>
  12. #include <msg_manager.h>
  13. #include <audio_hal.h>
  14. #include <audio_system.h>
  15. #include <audio_track.h>
  16. #include <media_type.h>
  17. #include <string.h>
  18. #include <stdlib.h>
  19. #include <stdio.h>
  20. #include <stream.h>
  21. #ifdef CONFIG_TWS
  22. #include <btservice_api.h>
  23. #endif
  24. #define SYS_LOG_NO_NEWLINE
  25. #ifdef SYS_LOG_DOMAIN
  26. #undef SYS_LOG_DOMAIN
  27. #endif
  28. #define SYS_LOG_DOMAIN "audio_aps"
  29. extern void audio_aps_monitor_set_aps(void *audio_handle, uint8_t status, int level);
  30. extern void audio_aps_monitor_normal(aps_monitor_info_t *handle, int stream_length,
  31. uint8_t aps_max_level, uint8_t aps_min_level,
  32. uint8_t aps_level);
  33. static int _audio_is_ready(void *aps_monitor)
  34. {
  35. aps_monitor_info_t *handle = (aps_monitor_info_t *)aps_monitor;
  36. io_stream_t stream = NULL;
  37. if (!handle || !handle->audio_track)
  38. return 0;
  39. stream = audio_track_get_stream(handle->audio_track);
  40. SYS_LOG_INF("steam_len %d ", stream_get_length(stream));
  41. if (stream_get_space(stream) <= stream_get_length(stream)) {
  42. SYS_LOG_INF(" ok ");
  43. return 1;
  44. }
  45. return 0;
  46. }
  47. static int _audio_trigger_start(void *aps_monitor)
  48. {
  49. aps_monitor_info_t *handle = (aps_monitor_info_t *)aps_monitor;
  50. if (!handle || !handle->audio_track)
  51. return 0;
  52. SYS_LOG_INF(" ok ");
  53. return audio_track_start(handle->audio_track);
  54. }
  55. static uint32_t _audio_get_samples_cnt(void *aps_monitor, uint8_t *audio_mode)
  56. {
  57. uint32_t sample = 0;
  58. aps_monitor_info_t *handle = (aps_monitor_info_t *)aps_monitor;
  59. if (!handle || !handle->audio_track)
  60. return 0;
  61. sample = hal_aout_channel_get_sample_cnt(handle->audio_track->audio_handle);
  62. if (handle->audio_track->audio_mode == AUDIO_MODE_MONO) {
  63. *audio_mode = 1;
  64. } else {
  65. *audio_mode = 2;
  66. }
  67. return sample;
  68. }
  69. static int _audio_get_error_state(void *aps_monitor)
  70. {
  71. aps_monitor_info_t *handle = (aps_monitor_info_t *)aps_monitor;
  72. if (!handle)
  73. return 0;
  74. return hal_aout_channel_check_fifo_underflow(handle->audio_track->audio_handle);
  75. }
  76. static int _audio_get_aps_level(void *aps_monitor)
  77. {
  78. aps_monitor_info_t *handle = (aps_monitor_info_t *)aps_monitor;
  79. return handle->current_level;
  80. }
  81. static media_runtime_observer_t audio_observer = {
  82. .media_handle = NULL,
  83. .is_ready = _audio_is_ready,
  84. .trigger_start = _audio_trigger_start,
  85. .get_samples_cnt = _audio_get_samples_cnt,
  86. .get_error_state = _audio_get_error_state,
  87. .get_aps_level = _audio_get_aps_level,
  88. };
  89. void audio_aps_monitor_tws_init(void *tws_observer)
  90. {
  91. aps_monitor_info_t *handle = audio_aps_monitor_get_instance();
  92. tws_runtime_observer_t *observer = (tws_runtime_observer_t *)tws_observer;
  93. if (tws_observer) {
  94. audio_observer.media_handle = handle;
  95. observer->set_media_observer(&audio_observer);
  96. handle->tws_observer = tws_observer;
  97. handle->role = observer->get_role();
  98. hal_aout_channel_enable_sample_cnt(handle->audio_track->audio_handle, true);
  99. }
  100. audio_aps_monitor_set_aps(handle->audio_track->audio_handle, APS_OPR_FAST_SET, handle->aps_default_level);
  101. }
  102. void audio_aps_tws_notify_decode_err(uint16_t err_cnt)
  103. {
  104. aps_monitor_info_t *handle = audio_aps_monitor_get_instance();
  105. tws_runtime_observer_t *observer = handle->tws_observer;
  106. if (observer && observer->trigger_restart) {
  107. observer->trigger_restart(err_cnt);
  108. }
  109. }
  110. void audio_aps_monitor_tws_deinit(void *tws_observer)
  111. {
  112. aps_monitor_info_t *handle = audio_aps_monitor_get_instance();
  113. tws_runtime_observer_t *observer = (tws_runtime_observer_t *)tws_observer;
  114. if (tws_observer) {
  115. hal_aout_channel_enable_sample_cnt(handle->audio_track->audio_handle, false);
  116. observer->set_media_observer(NULL);
  117. handle->tws_observer = NULL;
  118. }
  119. }
  120. void audio_aps_monitor_master(aps_monitor_info_t *handle, int stream_length, uint8_t aps_max_level, uint8_t aps_min_level, uint8_t aps_level)
  121. {
  122. tws_runtime_observer_t *tws_observer = (tws_runtime_observer_t *)handle->tws_observer;
  123. struct audio_track_t *audio_track = handle->audio_track;
  124. uint16_t mid_threshold = 0;
  125. uint16_t diff_threshold = 0;
  126. int local_compensate_samples;
  127. int remote_compensate_samples;
  128. aps_max_level = aps_max_level - 1;
  129. aps_min_level = aps_min_level + 1;
  130. local_compensate_samples = audio_track_get_fill_samples(audio_track);
  131. tws_observer->exchange_samples(&local_compensate_samples, &remote_compensate_samples);
  132. if (!handle->need_aps) {
  133. if (tws_observer && tws_observer->aps_nogotiate) {
  134. /* Not need adjust aps, but need notify tws module if needed */
  135. handle->dest_level = handle->current_level;
  136. tws_observer->aps_nogotiate(&handle->dest_level, &handle->current_level);
  137. }
  138. return;
  139. }
  140. /* printk("---in stream --- %d out stream %d\n",stream_length, stream_tell(info->pcm_stream)); */
  141. diff_threshold = (handle->aps_increase_water_mark - handle->aps_reduce_water_mark);
  142. mid_threshold = handle->aps_increase_water_mark - (diff_threshold / 2);
  143. switch (handle->aps_status) {
  144. case APS_STATUS_DEFAULT:
  145. if (stream_length > handle->aps_increase_water_mark) {
  146. SYS_LOG_DBG("inc aps\n");
  147. handle->dest_level = aps_max_level;
  148. handle->aps_status = APS_STATUS_INC;
  149. } else if (stream_length < handle->aps_reduce_water_mark) {
  150. SYS_LOG_DBG("fast dec aps\n");
  151. handle->dest_level = aps_min_level;
  152. handle->aps_status = APS_STATUS_DEC;
  153. } else {
  154. /* keep default */
  155. handle->dest_level = aps_level;
  156. handle->aps_status = APS_STATUS_DEFAULT;
  157. }
  158. break;
  159. case APS_STATUS_INC:
  160. if (stream_length < handle->aps_reduce_water_mark) {
  161. SYS_LOG_DBG("fast dec aps\n");
  162. handle->dest_level = aps_min_level;
  163. handle->aps_status = APS_STATUS_DEC;
  164. } else if (stream_length <= mid_threshold) {
  165. SYS_LOG_DBG("default aps\n");
  166. handle->dest_level = aps_level;
  167. handle->aps_status = APS_STATUS_DEFAULT;
  168. } else {
  169. handle->dest_level = aps_max_level;
  170. handle->aps_status = APS_STATUS_INC;
  171. }
  172. break;
  173. case APS_STATUS_DEC:
  174. if (stream_length > handle->aps_increase_water_mark) {
  175. SYS_LOG_DBG("fast inc aps\n");
  176. handle->dest_level = aps_max_level;
  177. handle->aps_status = APS_STATUS_INC;
  178. } else if (stream_length >= mid_threshold) {
  179. SYS_LOG_DBG("default aps\n");
  180. handle->dest_level = aps_level;
  181. handle->aps_status = APS_STATUS_DEFAULT;
  182. } else {
  183. handle->dest_level = aps_min_level;
  184. handle->aps_status = APS_STATUS_DEC;
  185. }
  186. break;
  187. }
  188. if (tws_observer && tws_observer->aps_nogotiate) {
  189. tws_observer->aps_nogotiate(&handle->dest_level, &handle->current_level);
  190. }
  191. if (tws_observer && handle->dest_level != handle->current_level) {
  192. /* Slowly adjust */
  193. if (handle->dest_level > handle->current_level) {
  194. handle->current_level++;
  195. } else if (handle->dest_level < handle->current_level) {
  196. handle->current_level--;
  197. }
  198. /* printk("%s: %d\n", __func__, handle->current_level); */
  199. tws_observer->aps_change_notify(handle->current_level);
  200. hal_aout_channel_set_aps(audio_track->audio_handle, handle->current_level, APS_LEVEL_AUDIOPLL);
  201. }
  202. /* printk("master: stream_length %d aps_level %d dest_level %d \n",stream_length, handle->current_level, handle->dest_level); */
  203. }
  204. void audio_aps_monitor_slave(aps_monitor_info_t *handle, int stream_length, uint8_t aps_max_level, uint8_t aps_min_level, uint8_t slave_aps_level)
  205. {
  206. tws_runtime_observer_t *tws_observer = (tws_runtime_observer_t *)handle->tws_observer;
  207. struct audio_track_t *audio_track = NULL;
  208. int sample_diff = 0;
  209. uint16_t bt_clock = 0;
  210. uint8_t master_aps_level = 0;
  211. int req_aps = slave_aps_level;
  212. static int monitor_cnt;
  213. static uint16_t pre_bt_clock;
  214. int diff_samples = 0;
  215. int local_compensate_samples;
  216. int remote_compensate_samples;
  217. audio_track = handle->audio_track;
  218. tws_observer->get_samples_diff(&sample_diff, &master_aps_level, &bt_clock);
  219. if ((pre_bt_clock != bt_clock) || (sample_diff > 20) || (sample_diff < -20)) {
  220. if (sample_diff < -15) {
  221. req_aps = master_aps_level - 3;
  222. } else if (sample_diff < -10) {
  223. req_aps = master_aps_level - 2;
  224. } else if (sample_diff < -5) {
  225. req_aps = master_aps_level - 1;
  226. } else if (sample_diff > 15) {
  227. req_aps = master_aps_level + 3;
  228. } else if (sample_diff > 10) {
  229. req_aps = master_aps_level + 2;
  230. } else if (sample_diff > 5) {
  231. req_aps = master_aps_level + 1;
  232. } else {
  233. req_aps = master_aps_level;
  234. }
  235. if (req_aps > aps_max_level)
  236. req_aps = aps_max_level;
  237. if (req_aps < aps_min_level)
  238. req_aps = aps_min_level;
  239. pre_bt_clock = bt_clock;
  240. } else {
  241. /* Other time follow master level */
  242. req_aps = master_aps_level;
  243. }
  244. if (slave_aps_level != req_aps) {
  245. handle->dest_level = req_aps;
  246. handle->current_level = handle->dest_level;
  247. /* printk("%s: %d\n", __func__, handle->current_level); */
  248. hal_aout_channel_set_aps(audio_track->audio_handle, handle->current_level, APS_LEVEL_AUDIOPLL);
  249. }
  250. /* printk("diff %d ml %d sl %d cl %x\n",sample_diff, master_aps_level, handle->current_level, bt_clock); */
  251. if (sample_diff > 50 || sample_diff < -50) {
  252. if (monitor_cnt++ > 1000) {
  253. tws_observer->trigger_restart(TWS_RESTART_SAMPLE_DIFF);
  254. monitor_cnt = 0;
  255. return;
  256. }
  257. /* printk("slave: sample_diff %d master_aps_level %d aps_level %d dest_level %d stream_length %d\n",sample_diff, master_aps_level, handle->current_level,handle->dest_level, stream_length); */
  258. } else {
  259. monitor_cnt = 0;
  260. }
  261. local_compensate_samples = audio_track_get_fill_samples(audio_track);
  262. tws_observer->exchange_samples(&local_compensate_samples, &remote_compensate_samples);
  263. diff_samples = remote_compensate_samples - local_compensate_samples;
  264. if (diff_samples) {
  265. SYS_LOG_INF("diff %d(local %d + remote %d)", diff_samples, local_compensate_samples, remote_compensate_samples);
  266. }
  267. audio_track_compensate_samples(audio_track, diff_samples);
  268. if ((diff_samples > 0x3FFFFFFF) ||
  269. (diff_samples < (-0x3FFFFFFF))) {
  270. /* Too larger, restart */
  271. tws_observer->trigger_restart(TWS_RESTART_SAMPLE_DIFF);
  272. }
  273. }
  274. int32_t audio_tws_set_stream_info(uint8_t format, uint16_t first_pktnum, uint8_t sample_rate)
  275. {
  276. aps_monitor_info_t *handle = audio_aps_monitor_get_instance();
  277. if(handle->audio_track)
  278. audio_track_set_waitto_start(handle->audio_track, false);
  279. return 0;
  280. }
  281. uint16_t audio_tws_get_playback_first_pktnum(void)
  282. {
  283. return 1;
  284. }
  285. int32_t audio_tws_set_pkt_info(uint16_t pkt_num, uint16_t pkt_len, uint16_t pcm_len)
  286. {
  287. return 0;
  288. }