Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit c2b840c7 authored by qctecmdr's avatar qctecmdr Committed by Gerrit - the friendly Code Review server
Browse files

Merge "ASoC: Update Compress driver as per ASoC framework expecatations"

parents b797291c 6358c06e
Loading
Loading
Loading
Loading
+177 −0
Original line number Diff line number Diff line
@@ -137,6 +137,9 @@ struct msm_compr_audio {

	int32_t first_buffer;
	int32_t last_buffer;
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
	int32_t zero_buffer;
#endif
	int32_t partial_drain_delay;

	uint16_t session_id;
@@ -168,6 +171,9 @@ struct msm_compr_audio {
	atomic_t start;
	atomic_t eos;
	atomic_t drain;
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
	atomic_t partial_drain;
#endif
	atomic_t xrun;
	atomic_t close;
	atomic_t wait_on_close;
@@ -622,8 +628,10 @@ static void compr_event_handler(uint32_t opcode,
			if (atomic_cmpxchg(&prtd->drain, 1, 0) &&
			    prtd->last_buffer) {
				pr_debug("%s: wake up on drain\n", __func__);
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
				prtd->drain_ready = 1;
				wake_up(&prtd->drain_wait);
#endif
				prtd->last_buffer = 0;
			} else {
				atomic_set(&prtd->start, 0);
@@ -666,6 +674,103 @@ static void compr_event_handler(uint32_t opcode,
			break;
		}

#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
		if (prtd->zero_buffer) {
			pr_debug("write_done for zero buffer\n");
			prtd->zero_buffer = 0;

			/* move to next stream and reset vars */
			pr_debug("%s: Moving to next stream in gapless\n",
								__func__);
			ac->stream_id = NEXT_STREAM_ID(ac->stream_id);
			prtd->byte_offset = 0;
			prtd->app_pointer  = 0;
			prtd->first_buffer = 1;
			prtd->last_buffer = 0;
			/*
			 * Set gapless transition flag only if EOS hasn't been
			 * acknowledged already.
			 */
			if (atomic_read(&prtd->eos))
				prtd->gapless_state.gapless_transition = 1;
			prtd->marker_timestamp = 0;

			/*
			 * Don't reset these as these vars map to
			 * total_bytes_transferred and total_bytes_available
			 * directly, only total_bytes_transferred will be
			 * updated in the next avail() ioctl
			 *	prtd->copied_total = 0;
			 *	prtd->bytes_received = 0;
			 */
			atomic_set(&prtd->drain, 0);
			atomic_set(&prtd->xrun, 1);
			pr_debug("%s: issue CMD_RUN", __func__);
			q6asm_run_nowait(prtd->audio_client, 0, 0, 0);
			snd_compr_drain_notify(cstream);
			spin_unlock_irqrestore(&prtd->lock, flags);
			break;
		}

		bytes_available = prtd->bytes_received - prtd->copied_total;
		if (bytes_available == 0) {
			pr_debug("%s:bytes_available is 0\n", __func__);
			if (prtd->last_buffer)
				prtd->last_buffer = 0;

			if (atomic_read(&prtd->partial_drain) &&
				prtd->gapless_state.set_next_stream_id &&
				!prtd->zero_buffer) {

				pr_debug("%s:Partial Drain Case\n", __func__);
				pr_debug("%s:Send EOS command\n", __func__);
				/* send EOS */
				prtd->eos_ack = 0;
				atomic_set(&prtd->eos, 1);
				atomic_set(&prtd->drain, 0);
				q6asm_stream_cmd_nowait(ac, CMD_EOS, ac->stream_id);

				/* send a zero length buffer in case of partial drain*/
				atomic_set(&prtd->xrun, 0);
				pr_debug("%s:Send zero size buffer\n", __func__);
				msm_compr_send_buffer(prtd);
				prtd->zero_buffer = 1;
			} else {
				/*
				 * moving to next stream failed, so reset the gapless state
				 * set next stream id for the same session so that the same
				 * stream can be used for gapless playback
				 */
				pr_debug("%s:Drain Case\n", __func__);
				pr_debug("%s:Reset Gapless params \n", __func__);

				prtd->gapless_state.set_next_stream_id = false;
				prtd->gapless_state.gapless_transition = 0;

				pr_debug("%s:Send EOS command\n", __func__);
				prtd->eos_ack = 0;
				atomic_set(&prtd->eos, 1);
				atomic_set(&prtd->drain, 0);
				q6asm_stream_cmd_nowait(ac, CMD_EOS, ac->stream_id);

				prtd->cmd_interrupt = 0;
			}
		} else if (bytes_available < cstream->runtime->fragment_size) {
			pr_debug("%s:Partial Buffer Case \n", __func__);
			atomic_set(&prtd->xrun, 1);

			if (prtd->last_buffer)
				prtd->last_buffer = 0;
			if (atomic_read(&prtd->drain)) {
				if (bytes_available > 0) {
					pr_debug("%s: send %d partial bytes at the end",
						   __func__, bytes_available);
					atomic_set(&prtd->xrun, 0);
					prtd->last_buffer = 1;
					msm_compr_send_buffer(prtd);
				}
			}
#else
		bytes_available = prtd->bytes_received - prtd->copied_total;
		if (bytes_available < cstream->runtime->fragment_size) {
			pr_debug("WRITE_DONE Insufficient data to send. break out\n");
@@ -679,6 +784,7 @@ static void compr_event_handler(uint32_t opcode,
				wake_up(&prtd->drain_wait);
				atomic_set(&prtd->drain, 0);
			}
#endif
		} else if ((bytes_available == cstream->runtime->fragment_size)
			   && atomic_read(&prtd->drain)) {
			prtd->last_buffer = 1;
@@ -740,7 +846,40 @@ static void compr_event_handler(uint32_t opcode,
		    !prtd->gapless_state.set_next_stream_id) {
			pr_debug("ASM_DATA_CMDRSP_EOS wake up\n");
			prtd->eos_ack = 1;
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
			wake_up(&prtd->eos_wait);
#else
			pr_debug("%s:issue CMD_PAUSE stream_id %d",
					  __func__, ac->stream_id);
			q6asm_stream_cmd_nowait(ac, CMD_PAUSE, ac->stream_id);
			prtd->cmd_ack = 0;

			pr_debug("%s:DRAIN,don't wait for EOS ack\n", __func__);
			/*
			 * Don't reset these as these vars map to
			 * total_bytes_transferred and total_bytes_available.
			 * Just total_bytes_transferred will be updated
			 * in the next avail() ioctl.
			 * prtd->copied_total = 0;
			 * prtd->bytes_received = 0;
			 * do not reset prtd->bytes_sent as well as the same
			 * session is used for gapless playback
			 */
			prtd->byte_offset = 0;

			prtd->app_pointer  = 0;
			prtd->first_buffer = 1;
			prtd->last_buffer = 0;
			atomic_set(&prtd->drain, 0);
			atomic_set(&prtd->xrun, 1);

			pr_debug("%s:issue CMD_FLUSH ac->stream_id %d",
						  __func__, ac->stream_id);

			q6asm_run_nowait(prtd->audio_client, 0, 0, 0);

			snd_compr_drain_notify(cstream);
#endif
		}
		atomic_set(&prtd->eos, 0);
		stream_index = STREAM_ARRAY_INDEX(stream_id);
@@ -841,7 +980,9 @@ static void compr_event_handler(uint32_t opcode,
						prtd->last_buffer = 0;

					prtd->drain_ready = 1;
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
					wake_up(&prtd->drain_wait);
#endif
					atomic_set(&prtd->drain, 0);
				} else if (atomic_read(&prtd->xrun)) {
					pr_debug("%s: RUN ack, continue write cycle\n", __func__);
@@ -916,11 +1057,13 @@ static void compr_event_handler(uint32_t opcode,
		prtd->copied_total = prtd->bytes_received;
		snd_compr_fragment_elapsed(cstream);
		atomic_set(&prtd->error, 1);
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
		wake_up(&prtd->drain_wait);
		if (atomic_cmpxchg(&prtd->eos, 1, 0)) {
			pr_debug("%s:unblock eos wait queues", __func__);
			wake_up(&prtd->eos_wait);
		}
#endif
		spin_unlock_irqrestore(&prtd->lock, flags);
		break;
	default:
@@ -1754,6 +1897,9 @@ static int msm_compr_playback_open(struct snd_compr_stream *cstream)
	atomic_set(&prtd->eos, 0);
	atomic_set(&prtd->start, 0);
	atomic_set(&prtd->drain, 0);
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
	atomic_set(&prtd->partial_drain, 0);
#endif
	atomic_set(&prtd->xrun, 0);
	atomic_set(&prtd->close, 0);
	atomic_set(&prtd->wait_on_close, 0);
@@ -1849,6 +1995,9 @@ static int msm_compr_capture_open(struct snd_compr_stream *cstream)
	atomic_set(&prtd->eos, 0);
	atomic_set(&prtd->start, 0);
	atomic_set(&prtd->drain, 0);
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
	atomic_set(&prtd->partial_drain, 0);
#endif
	atomic_set(&prtd->xrun, 0);
	atomic_set(&prtd->close, 0);
	atomic_set(&prtd->wait_on_close, 0);
@@ -1911,7 +2060,9 @@ static int msm_compr_playback_free(struct snd_compr_stream *cstream)
		return 0;
	}
	prtd->cmd_interrupt = 1;
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
	wake_up(&prtd->drain_wait);
#endif
	pdata = snd_soc_component_get_drvdata(component);
	ac = prtd->audio_client;
	if (!pdata || !ac) {
@@ -2293,6 +2444,7 @@ static int msm_compr_set_params(struct snd_compr_stream *cstream,
	return ret;
}

#if IS_ENABLED(CONFIG_AUDIO_QGKI)
static int msm_compr_drain_buffer(struct msm_compr_audio *prtd,
				  unsigned long *flags)
{
@@ -2321,6 +2473,7 @@ static int msm_compr_drain_buffer(struct msm_compr_audio *prtd,
	}
	return rc;
}
#endif

static int msm_compr_wait_for_stream_avail(struct msm_compr_audio *prtd,
				    unsigned long *flags)
@@ -2373,7 +2526,9 @@ static int msm_compr_trigger(struct snd_compr_stream *cstream, int cmd)
	struct audio_client *ac = prtd->audio_client;
	unsigned long fe_id = rtd->dai_link->id;
	int rc = 0;
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
	int bytes_to_write;
#endif
	unsigned long flags;
	int stream_id;
	uint32_t stream_index;
@@ -2465,17 +2620,21 @@ static int msm_compr_trigger(struct snd_compr_stream *cstream, int cmd)
			 * cmd_int and do not wake up eos_wait during gapless
			 * transition
			 */
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
			if (!prtd->gapless_state.gapless_transition) {
				prtd->cmd_interrupt = 1;
				wake_up(&prtd->eos_wait);
			}
#endif
			atomic_set(&prtd->eos, 0);
		}
		if (atomic_read(&prtd->drain)) {
			pr_debug("%s: interrupt drain wait queues", __func__);
			prtd->cmd_interrupt = 1;
			prtd->drain_ready = 1;
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
			wake_up(&prtd->drain_wait);
#endif
			atomic_set(&prtd->drain, 0);
		}
		prtd->last_buffer = 0;
@@ -2521,10 +2680,20 @@ static int msm_compr_trigger(struct snd_compr_stream *cstream, int cmd)
		break;
	case SND_COMPR_TRIGGER_PARTIAL_DRAIN:
		pr_debug("%s: SND_COMPR_TRIGGER_PARTIAL_DRAIN\n", __func__);
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
		spin_lock_irqsave(&prtd->lock, flags);
		atomic_set(&prtd->partial_drain, 1);
#endif
		if (!prtd->gapless_state.use_dsp_gapless_mode) {
			pr_debug("%s: set partial drain as drain\n", __func__);
			cmd = SND_COMPR_TRIGGER_DRAIN;
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
			atomic_set(&prtd->partial_drain, 0);
#endif
		}
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
		spin_unlock_irqrestore(&prtd->lock, flags);
#endif
	case SND_COMPR_TRIGGER_DRAIN:
		pr_debug("%s: SNDRV_COMPRESS_DRAIN\n", __func__);
		/* Make sure all the data is sent to DSP before sending EOS */
@@ -2534,9 +2703,13 @@ static int msm_compr_trigger(struct snd_compr_stream *cstream, int cmd)
			pr_err("%s: stream is not in started state\n",
				__func__);
			rc = -EPERM;
#if !IS_ENABLED(CONFIG_AUDIO_QGKI)
			atomic_set(&prtd->partial_drain, 0);
#endif
			spin_unlock_irqrestore(&prtd->lock, flags);
			break;
		}
#if IS_ENABLED(CONFIG_AUDIO_QGKI)
		if (prtd->bytes_received > prtd->copied_total) {
			pr_debug("%s: wait till all the data is sent to dsp\n",
				__func__);
@@ -2743,6 +2916,10 @@ static int msm_compr_trigger(struct snd_compr_stream *cstream, int cmd)

			q6asm_run_nowait(prtd->audio_client, 0, 0, 0);
		}
#else
		atomic_set(&prtd->drain, 1);
		spin_unlock_irqrestore(&prtd->lock, flags);
#endif
		prtd->cmd_interrupt = 0;
		break;
	case SND_COMPR_TRIGGER_NEXT_TRACK: