summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--audio/Android.mk6
-rw-r--r--audio/audio_hw.c77
-rw-r--r--audio/audio_policy.conf4
3 files changed, 82 insertions, 5 deletions
diff --git a/audio/Android.mk b/audio/Android.mk
index 4ec8ae5..b678403 100644
--- a/audio/Android.mk
+++ b/audio/Android.mk
@@ -27,7 +27,11 @@ LOCAL_C_INCLUDES += \
LOCAL_SHARED_LIBRARIES := liblog libcutils libtinyalsa libaudioutils libdl libsecril-client
LOCAL_MODULE_TAGS := optional
-LOCAL_CFLAGS += -DFORCE_OUT_SAMPLING_RATE=44100
+ifeq ($(TARGET_TUNA_AUDIO_FORCE_SAMPLE_RATE),)
+LOCAL_CFLAGS += -DUSE_VARIABLE_SAMPLING_RATE
+else
+LOCAL_CFLAGS += -DFORCE_OUT_SAMPLING_RATE=$(TARGET_TUNA_AUDIO_FORCE_SAMPLE_RATE)
+endif
include $(BUILD_SHARED_LIBRARY)
diff --git a/audio/audio_hw.c b/audio/audio_hw.c
index 0715fff..5e36799 100644
--- a/audio/audio_hw.c
+++ b/audio/audio_hw.c
@@ -240,9 +240,11 @@
/* If sample rate converter is required, then use triple-buffering to
* help mask the variance in cycle times. Otherwise use double-buffering.
*/
+/* TODO: Figure out a better check for this
#elif DEFAULT_OUT_SAMPLING_RATE != MM_FULL_POWER_SAMPLING_RATE
#define PLAYBACK_SHORT_PERIOD_COUNT 3
#define OUT_RESAMPLER
+*/
#else
#define PLAYBACK_SHORT_PERIOD_COUNT 2
#endif
@@ -325,7 +327,7 @@ enum tty_modes {
/* deep buffer */
struct pcm_config pcm_config_mm = {
.channels = 2,
- .rate = MM_FULL_POWER_SAMPLING_RATE,
+ .rate = MM_FULL_POWER_SAMPLING_RATE, /* changed based on audio policy setting */
.period_size = DEEP_BUFFER_LONG_PERIOD_SIZE,
.period_count = PLAYBACK_DEEP_BUFFER_LONG_PERIOD_COUNT,
.format = PCM_FORMAT_S16_LE,
@@ -336,7 +338,7 @@ struct pcm_config pcm_config_mm = {
/* low latency */
struct pcm_config pcm_config_tones = {
.channels = 2,
- .rate = MM_FULL_POWER_SAMPLING_RATE,
+ .rate = MM_FULL_POWER_SAMPLING_RATE, /* changed based on audio policy setting */
.period_size = SHORT_PERIOD_SIZE,
.period_count = PLAYBACK_SHORT_PERIOD_COUNT,
.format = PCM_FORMAT_S16_LE,
@@ -716,6 +718,10 @@ struct tuna_stream_out {
bool muted;
struct tuna_audio_device *dev;
+
+#ifdef USE_VARIABLE_SAMPLING_RATE
+ unsigned int sample_rate;
+#endif
};
#define MAX_PREPROCESSORS 3 /* maximum one AGC + one NS + one AEC per input stream */
@@ -1424,7 +1430,14 @@ static int start_output_stream_low_latency(struct tuna_stream_out *out)
if (adev->out_device & ~(AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET | AUDIO_DEVICE_OUT_AUX_DIGITAL)) {
/* Something not a dock in use */
out->config[PCM_NORMAL] = pcm_config_tones;
+#ifndef USE_VARIABLE_SAMPLING_RATE
out->config[PCM_NORMAL].rate = MM_FULL_POWER_SAMPLING_RATE;
+#else
+ if (out->sample_rate % 48 == 0)
+ out->config[PCM_NORMAL].rate = MM_FULL_POWER_SAMPLING_RATE;
+ else
+ out->config[PCM_NORMAL].rate = MM_LOW_POWER_SAMPLING_RATE;
+#endif
out->pcm[PCM_NORMAL] = pcm_open(CARD_TUNA_DEFAULT, PORT_TONES,
flags, &out->config[PCM_NORMAL]);
}
@@ -1432,7 +1445,14 @@ static int start_output_stream_low_latency(struct tuna_stream_out *out)
if (adev->out_device & AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET) {
/* SPDIF output in use */
out->config[PCM_SPDIF] = pcm_config_tones;
+#ifndef USE_VARIABLE_SAMPLING_RATE
out->config[PCM_SPDIF].rate = MM_FULL_POWER_SAMPLING_RATE;
+#else
+ if (out->sample_rate % 48 == 0)
+ out->config[PCM_SPDIF].rate = MM_FULL_POWER_SAMPLING_RATE;
+ else
+ out->config[PCM_SPDIF].rate = MM_LOW_POWER_SAMPLING_RATE;
+#endif
out->pcm[PCM_SPDIF] = pcm_open(CARD_TUNA_DEFAULT, PORT_SPDIF,
flags, &out->config[PCM_SPDIF]);
}
@@ -1489,7 +1509,14 @@ static int start_output_stream_deep_buffer(struct tuna_stream_out *out)
out->use_long_periods = true;
out->config[PCM_NORMAL] = pcm_config_mm;
+#ifndef USE_VARIABLE_SAMPLING_RATE
out->config[PCM_NORMAL].rate = MM_FULL_POWER_SAMPLING_RATE;
+#else
+ if (out->sample_rate % 48 == 0)
+ out->config[PCM_NORMAL].rate = MM_FULL_POWER_SAMPLING_RATE;
+ else
+ out->config[PCM_NORMAL].rate = MM_LOW_POWER_SAMPLING_RATE;
+#endif
out->pcm[PCM_NORMAL] = pcm_open(CARD_TUNA_DEFAULT, PORT_MM,
PCM_OUT | PCM_MMAP | PCM_NOIRQ, &out->config[PCM_NORMAL]);
if (out->pcm[PCM_NORMAL] && !pcm_is_ready(out->pcm[PCM_NORMAL])) {
@@ -1664,15 +1691,26 @@ static int get_playback_delay(struct tuna_stream_out *out,
/* adjust render time stamp with delay added by current driver buffer.
* Add the duration of current frame as we want the render time of the last
* sample being written. */
+#ifndef USE_VARIABLE_SAMPLING_RATE
buffer->delay_ns = (long)(((int64_t)(kernel_frames + frames)* 1000000000)/
MM_FULL_POWER_SAMPLING_RATE);
+#else
+ buffer->delay_ns = (long)(((int64_t)(kernel_frames + frames)* 1000000000)/
+ out->sample_rate); // ?
+#endif
return 0;
}
static uint32_t out_get_sample_rate(const struct audio_stream *stream __unused)
{
+#ifdef USE_VARIABLE_SAMPLING_RATE
+ struct tuna_stream_out *out = (struct tuna_stream_out *)stream;
+
+ return out->sample_rate; // TODO: out->config[PCM_*].rate?
+#else
return DEFAULT_OUT_SAMPLING_RATE;
+#endif
}
static uint32_t out_get_sample_rate_hdmi(const struct audio_stream *stream)
@@ -1695,7 +1733,11 @@ static size_t out_get_buffer_size_low_latency(const struct audio_stream *stream)
multiple of 16 frames, as audioflinger expects audio buffers to
be a multiple of 16 frames. Note: we use the default rate here
from pcm_config_tones.rate. */
+#ifndef USE_VARIABLE_SAMPLING_RATE
size_t size = (SHORT_PERIOD_SIZE * DEFAULT_OUT_SAMPLING_RATE) / pcm_config_tones.rate;
+#else
+ size_t size = SHORT_PERIOD_SIZE; //?
+#endif
size = ((size + 15) / 16) * 16;
return size * audio_stream_out_frame_size((const struct audio_stream_out *)stream);
}
@@ -1708,8 +1750,12 @@ static size_t out_get_buffer_size_deep_buffer(const struct audio_stream *stream)
multiple of 16 frames, as audioflinger expects audio buffers to
be a multiple of 16 frames. Note: we use the default rate here
from pcm_config_mm.rate. */
+#ifndef USE_VARIABLE_SAMPLING_RATE
size_t size = (DEEP_BUFFER_SHORT_PERIOD_SIZE * DEFAULT_OUT_SAMPLING_RATE) /
pcm_config_mm.rate;
+#else
+ size_t size = DEEP_BUFFER_SHORT_PERIOD_SIZE; //?
+#endif
size = ((size + 15) / 16) * 16;
return size * audio_stream_out_frame_size((const struct audio_stream_out *)stream);
}
@@ -1922,7 +1968,11 @@ static uint32_t out_get_latency_low_latency(const struct audio_stream_out *strea
struct tuna_stream_out *out = (struct tuna_stream_out *)stream;
/* Note: we use the default rate here from pcm_config_mm.rate */
+#ifndef USE_VARIABLE_SAMPLING_RATE
return (SHORT_PERIOD_SIZE * PLAYBACK_SHORT_PERIOD_COUNT * 1000) / pcm_config_tones.rate;
+#else
+ return (SHORT_PERIOD_SIZE * PLAYBACK_SHORT_PERIOD_COUNT * 1000) / out->sample_rate; // ?
+#endif
}
static uint32_t out_get_latency_deep_buffer(const struct audio_stream_out *stream)
@@ -1930,8 +1980,13 @@ static uint32_t out_get_latency_deep_buffer(const struct audio_stream_out *strea
struct tuna_stream_out *out = (struct tuna_stream_out *)stream;
/* Note: we use the default rate here from pcm_config_mm.rate */
+#ifndef USE_VARIABLE_SAMPLING_RATE
return (DEEP_BUFFER_LONG_PERIOD_SIZE * PLAYBACK_DEEP_BUFFER_LONG_PERIOD_COUNT * 1000) /
pcm_config_mm.rate;
+#else
+ return (DEEP_BUFFER_LONG_PERIOD_SIZE * PLAYBACK_DEEP_BUFFER_LONG_PERIOD_COUNT * 1000) /
+ out->sample_rate; // ?
+#endif
}
static uint32_t out_get_latency_hdmi(const struct audio_stream_out *stream)
@@ -2129,9 +2184,15 @@ static ssize_t out_write_deep_buffer(struct audio_stream_out *stream, const void
kernel_frames = pcm_get_buffer_size(out->pcm[PCM_NORMAL]) - kernel_frames;
if (kernel_frames > out->write_threshold) {
+#ifndef USE_VARIABLE_SAMPLING_RATE
unsigned long time = (unsigned long)
(((int64_t)(kernel_frames - out->write_threshold) * 1000000) /
MM_FULL_POWER_SAMPLING_RATE);
+#else
+ unsigned long time = (unsigned long)
+ (((int64_t)(kernel_frames - out->write_threshold) * 1000000) /
+ out->sample_rate); // ?
+#endif
if (time < MIN_WRITE_SLEEP_US)
time = MIN_WRITE_SLEEP_US;
usleep(time);
@@ -3277,9 +3338,17 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
out = (struct tuna_stream_out *)calloc(1, sizeof(struct tuna_stream_out));
if (!out)
return -ENOMEM;
+ ALOGV("%s: enter: sample_rate(%d) channel_mask(%#x) devices(%#x) flags(%#x)",
+ __func__, config->sample_rate, config->channel_mask, devices, flags);
out->sup_channel_masks[0] = AUDIO_CHANNEL_OUT_STEREO;
out->channel_mask = AUDIO_CHANNEL_OUT_STEREO;
+#ifdef USE_VARIABLE_SAMPLING_RATE
+ if (config->sample_rate == 0) {
+ config->sample_rate = MM_LOW_POWER_SAMPLING_RATE;
+ }
+ out->sample_rate = config->sample_rate;
+#endif
if (flags & AUDIO_OUTPUT_FLAG_DIRECT &&
devices == AUDIO_DEVICE_OUT_AUX_DIGITAL) {
@@ -3313,6 +3382,8 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
ret = -ENOSYS;
goto err_open;
}
+ /* NOTE: This gets called with the highest (or last?)
+ * sampling rate listed in the audio policy */
output_type = OUTPUT_DEEP_BUF;
out->channel_mask = AUDIO_CHANNEL_OUT_STEREO;
out->stream.common.get_buffer_size = out_get_buffer_size_deep_buffer;
@@ -3326,6 +3397,8 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
ret = -ENOSYS;
goto err_open;
}
+ /* NOTE: This gets called with the highest (or last?)
+ * sampling rate listed in the audio policy */
output_type = OUTPUT_LOW_LATENCY;
out->stream.common.get_buffer_size = out_get_buffer_size_low_latency;
out->stream.common.get_sample_rate = out_get_sample_rate;
diff --git a/audio/audio_policy.conf b/audio/audio_policy.conf
index 3131fa7..a782448 100644
--- a/audio/audio_policy.conf
+++ b/audio/audio_policy.conf
@@ -23,14 +23,14 @@ audio_hw_modules {
primary {
outputs {
primary {
- sampling_rates 44100|48000
+ sampling_rates 44100
channel_masks AUDIO_CHANNEL_OUT_STEREO
formats AUDIO_FORMAT_PCM_16_BIT
devices AUDIO_DEVICE_OUT_EARPIECE|AUDIO_DEVICE_OUT_SPEAKER|AUDIO_DEVICE_OUT_WIRED_HEADSET|AUDIO_DEVICE_OUT_WIRED_HEADPHONE|AUDIO_DEVICE_OUT_ALL_SCO|AUDIO_DEVICE_OUT_AUX_DIGITAL|AUDIO_DEVICE_OUT_DGTL_DOCK_HEADSET
flags AUDIO_OUTPUT_FLAG_PRIMARY
}
deep_buffer {
- sampling_rates 44100|48000
+ sampling_rates 44100
channel_masks AUDIO_CHANNEL_OUT_STEREO
formats AUDIO_FORMAT_PCM_16_BIT
devices AUDIO_DEVICE_OUT_SPEAKER|AUDIO_DEVICE_OUT_WIRED_HEADSET|AUDIO_DEVICE_OUT_WIRED_HEADPHONE