hal: Add support for PCM in offload path
- Translate PCM Offload format to pcm codec id - Set 16/24 sub format id in compress params format - Calculate offload fragment size for pcm - Move offload calculation functions to platform file - Update audio policy manager isOffloadSupported for pcm offload profile. Use audio.offload.pcm.enable Change-Id: I2064d646b66e8283fce2cc736e96fb93cba0a552
This commit is contained in:
parent
c4ad7c0214
commit
c6a3a9e6ed
|
@ -53,16 +53,14 @@
|
||||||
#include "voice_extn.h"
|
#include "voice_extn.h"
|
||||||
|
|
||||||
#include "sound/compress_params.h"
|
#include "sound/compress_params.h"
|
||||||
|
#include "sound/asound.h"
|
||||||
|
|
||||||
#define MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE (256 * 1024)
|
|
||||||
#define MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE (2 * 1024)
|
|
||||||
#define COMPRESS_OFFLOAD_FRAGMENT_SIZE_FOR_AV_STREAMING (2 * 1024)
|
|
||||||
#define COMPRESS_OFFLOAD_FRAGMENT_SIZE (32 * 1024)
|
|
||||||
#define COMPRESS_OFFLOAD_NUM_FRAGMENTS 4
|
#define COMPRESS_OFFLOAD_NUM_FRAGMENTS 4
|
||||||
/* ToDo: Check and update a proper value in msec */
|
/* ToDo: Check and update a proper value in msec */
|
||||||
#define COMPRESS_OFFLOAD_PLAYBACK_LATENCY 96
|
#define COMPRESS_OFFLOAD_PLAYBACK_LATENCY 96
|
||||||
#define COMPRESS_PLAYBACK_VOLUME_MAX 0x2000
|
#define COMPRESS_PLAYBACK_VOLUME_MAX 0x2000
|
||||||
|
|
||||||
|
|
||||||
#define USECASE_AUDIO_PLAYBACK_PRIMARY USECASE_AUDIO_PLAYBACK_DEEP_BUFFER
|
#define USECASE_AUDIO_PLAYBACK_PRIMARY USECASE_AUDIO_PLAYBACK_DEEP_BUFFER
|
||||||
|
|
||||||
struct pcm_config pcm_config_deep_buffer = {
|
struct pcm_config pcm_config_deep_buffer = {
|
||||||
|
@ -155,35 +153,6 @@ static unsigned int audio_device_ref_count;
|
||||||
|
|
||||||
static int set_voice_volume_l(struct audio_device *adev, float volume);
|
static int set_voice_volume_l(struct audio_device *adev, float volume);
|
||||||
|
|
||||||
/* Read offload buffer size from a property.
|
|
||||||
* If value is not power of 2 round it to
|
|
||||||
* power of 2.
|
|
||||||
*/
|
|
||||||
static uint32_t get_offload_buffer_size(audio_offload_info_t* info)
|
|
||||||
{
|
|
||||||
char value[PROPERTY_VALUE_MAX] = {0};
|
|
||||||
uint32_t fragment_size = COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
|
||||||
if((property_get("audio.offload.buffer.size.kb", value, "")) &&
|
|
||||||
atoi(value)) {
|
|
||||||
fragment_size = atoi(value) * 1024;
|
|
||||||
//ring buffer size needs to be 4k aligned.
|
|
||||||
CHECK(!(fragment_size * COMPRESS_OFFLOAD_NUM_FRAGMENTS % 4096));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (info != NULL && info->has_video && info->is_streaming) {
|
|
||||||
fragment_size = COMPRESS_OFFLOAD_FRAGMENT_SIZE_FOR_AV_STREAMING;
|
|
||||||
ALOGV("%s: offload fragment size reduced for AV streaming to %d",
|
|
||||||
__func__, out->compr_config.fragment_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(fragment_size < MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE)
|
|
||||||
fragment_size = MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
|
||||||
else if(fragment_size > MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE)
|
|
||||||
fragment_size = MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
|
||||||
ALOGVV("%s: fragment_size %d", __func__, fragment_size);
|
|
||||||
return fragment_size;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int check_and_set_gapless_mode(struct audio_device *adev) {
|
static int check_and_set_gapless_mode(struct audio_device *adev) {
|
||||||
|
|
||||||
|
|
||||||
|
@ -214,7 +183,9 @@ static int check_and_set_gapless_mode(struct audio_device *adev) {
|
||||||
static bool is_supported_format(audio_format_t format)
|
static bool is_supported_format(audio_format_t format)
|
||||||
{
|
{
|
||||||
if (format == AUDIO_FORMAT_MP3 ||
|
if (format == AUDIO_FORMAT_MP3 ||
|
||||||
format == AUDIO_FORMAT_AAC)
|
format == AUDIO_FORMAT_AAC ||
|
||||||
|
format == AUDIO_FORMAT_PCM_16_BIT_OFFLOAD ||
|
||||||
|
format == AUDIO_FORMAT_PCM_24_BIT_OFFLOAD)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
|
@ -231,6 +202,10 @@ static int get_snd_codec_id(audio_format_t format)
|
||||||
case AUDIO_FORMAT_AAC:
|
case AUDIO_FORMAT_AAC:
|
||||||
id = SND_AUDIOCODEC_AAC;
|
id = SND_AUDIOCODEC_AAC;
|
||||||
break;
|
break;
|
||||||
|
case AUDIO_FORMAT_PCM_16_BIT_OFFLOAD:
|
||||||
|
case AUDIO_FORMAT_PCM_24_BIT_OFFLOAD:
|
||||||
|
id = SND_AUDIOCODEC_PCM;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
ALOGE("%s: Unsupported audio format :%x", __func__, format);
|
ALOGE("%s: Unsupported audio format :%x", __func__, format);
|
||||||
}
|
}
|
||||||
|
@ -2161,8 +2136,10 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
out->usecase = USECASE_AUDIO_PLAYBACK_OFFLOAD;
|
out->usecase = USECASE_AUDIO_PLAYBACK_OFFLOAD;
|
||||||
if (config->offload_info.channel_mask)
|
if (config->offload_info.channel_mask)
|
||||||
out->channel_mask = config->offload_info.channel_mask;
|
out->channel_mask = config->offload_info.channel_mask;
|
||||||
else if (config->channel_mask)
|
else if (config->channel_mask) {
|
||||||
out->channel_mask = config->channel_mask;
|
out->channel_mask = config->channel_mask;
|
||||||
|
config->offload_info.channel_mask = config->channel_mask;
|
||||||
|
}
|
||||||
out->format = config->offload_info.format;
|
out->format = config->offload_info.format;
|
||||||
out->sample_rate = config->offload_info.sample_rate;
|
out->sample_rate = config->offload_info.sample_rate;
|
||||||
|
|
||||||
|
@ -2179,7 +2156,13 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
else
|
else
|
||||||
out->compr_config.codec->id =
|
out->compr_config.codec->id =
|
||||||
get_snd_codec_id(config->offload_info.format);
|
get_snd_codec_id(config->offload_info.format);
|
||||||
out->compr_config.fragment_size = get_offload_buffer_size(&config->offload_info);
|
if (audio_is_offload_pcm(config->offload_info.format)) {
|
||||||
|
out->compr_config.fragment_size =
|
||||||
|
platform_get_pcm_offload_buffer_size(&config->offload_info);
|
||||||
|
} else {
|
||||||
|
out->compr_config.fragment_size =
|
||||||
|
platform_get_compress_offload_buffer_size(&config->offload_info);
|
||||||
|
}
|
||||||
out->compr_config.fragments = COMPRESS_OFFLOAD_NUM_FRAGMENTS;
|
out->compr_config.fragments = COMPRESS_OFFLOAD_NUM_FRAGMENTS;
|
||||||
out->compr_config.codec->sample_rate =
|
out->compr_config.codec->sample_rate =
|
||||||
compress_get_alsa_rate(config->offload_info.sample_rate);
|
compress_get_alsa_rate(config->offload_info.sample_rate);
|
||||||
|
@ -2190,6 +2173,11 @@ static int adev_open_output_stream(struct audio_hw_device *dev,
|
||||||
out->compr_config.codec->ch_out = out->compr_config.codec->ch_in;
|
out->compr_config.codec->ch_out = out->compr_config.codec->ch_in;
|
||||||
out->compr_config.codec->format = SND_AUDIOSTREAMFORMAT_RAW;
|
out->compr_config.codec->format = SND_AUDIOSTREAMFORMAT_RAW;
|
||||||
|
|
||||||
|
if (config->offload_info.format == AUDIO_FORMAT_PCM_16_BIT_OFFLOAD)
|
||||||
|
out->compr_config.codec->format = SNDRV_PCM_FORMAT_S16_LE;
|
||||||
|
else if(config->offload_info.format == AUDIO_FORMAT_PCM_24_BIT_OFFLOAD)
|
||||||
|
out->compr_config.codec->format = SNDRV_PCM_FORMAT_S24_LE;
|
||||||
|
|
||||||
if (flags & AUDIO_OUTPUT_FLAG_NON_BLOCKING)
|
if (flags & AUDIO_OUTPUT_FLAG_NON_BLOCKING)
|
||||||
out->non_blocking = 1;
|
out->non_blocking = 1;
|
||||||
|
|
||||||
|
|
|
@ -31,12 +31,30 @@
|
||||||
#include "platform.h"
|
#include "platform.h"
|
||||||
#include "audio_extn.h"
|
#include "audio_extn.h"
|
||||||
#include "voice_extn.h"
|
#include "voice_extn.h"
|
||||||
|
#include "sound/compress_params.h"
|
||||||
|
|
||||||
#define MIXER_XML_PATH "/system/etc/mixer_paths.xml"
|
#define MIXER_XML_PATH "/system/etc/mixer_paths.xml"
|
||||||
#define MIXER_XML_PATH_AUXPCM "/system/etc/mixer_paths_auxpcm.xml"
|
#define MIXER_XML_PATH_AUXPCM "/system/etc/mixer_paths_auxpcm.xml"
|
||||||
#define LIB_ACDB_LOADER "libacdbloader.so"
|
#define LIB_ACDB_LOADER "libacdbloader.so"
|
||||||
#define AUDIO_DATA_BLOCK_MIXER_CTL "HDMI EDID"
|
#define AUDIO_DATA_BLOCK_MIXER_CTL "HDMI EDID"
|
||||||
|
|
||||||
|
#define MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE (256 * 1024)
|
||||||
|
#define MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE (2 * 1024)
|
||||||
|
#define COMPRESS_OFFLOAD_FRAGMENT_SIZE_FOR_AV_STREAMING (2 * 1024)
|
||||||
|
#define COMPRESS_OFFLOAD_FRAGMENT_SIZE (32 * 1024)
|
||||||
|
|
||||||
|
/* Used in calculating fragment size for pcm offload */
|
||||||
|
#define PCM_OFFLOAD_BUFFER_DURATION_FOR_AV 2000 /* 2 secs */
|
||||||
|
#define PCM_OFFLOAD_BUFFER_DURATION_FOR_AV_STREAMING 100 /* 100 millisecs */
|
||||||
|
|
||||||
|
/* MAX PCM fragment size cannot be increased further due
|
||||||
|
* to flinger's cblk size of 1mb,and it has to be a multiple of
|
||||||
|
* 24 - lcm of channels supported by DSP
|
||||||
|
*/
|
||||||
|
#define MAX_PCM_OFFLOAD_FRAGMENT_SIZE (240 * 1024)
|
||||||
|
#define MIN_PCM_OFFLOAD_FRAGMENT_SIZE (32 * 1024)
|
||||||
|
|
||||||
|
#define ALIGN( num, to ) (((num) + (to-1)) & (~(to-1)))
|
||||||
/*
|
/*
|
||||||
* This file will have a maximum of 38 bytes:
|
* This file will have a maximum of 38 bytes:
|
||||||
*
|
*
|
||||||
|
@ -1694,3 +1712,69 @@ bool platform_listen_update_status(snd_device_t snd_device)
|
||||||
else
|
else
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Read offload buffer size from a property.
|
||||||
|
* If value is not power of 2 round it to
|
||||||
|
* power of 2.
|
||||||
|
*/
|
||||||
|
uint32_t platform_get_compress_offload_buffer_size(audio_offload_info_t* info)
|
||||||
|
{
|
||||||
|
char value[PROPERTY_VALUE_MAX] = {0};
|
||||||
|
uint32_t fragment_size = COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
if((property_get("audio.offload.buffer.size.kb", value, "")) &&
|
||||||
|
atoi(value)) {
|
||||||
|
fragment_size = atoi(value) * 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info != NULL && info->has_video && info->is_streaming) {
|
||||||
|
fragment_size = COMPRESS_OFFLOAD_FRAGMENT_SIZE_FOR_AV_STREAMING;
|
||||||
|
ALOGV("%s: offload fragment size reduced for AV streaming to %d",
|
||||||
|
__func__, out->compr_config.fragment_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
fragment_size = ALIGN( fragment_size, 1024);
|
||||||
|
|
||||||
|
if(fragment_size < MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE)
|
||||||
|
fragment_size = MIN_COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
else if(fragment_size > MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE)
|
||||||
|
fragment_size = MAX_COMPRESS_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
ALOGV("%s: fragment_size %d", __func__, fragment_size);
|
||||||
|
return fragment_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint32_t platform_get_pcm_offload_buffer_size(audio_offload_info_t* info)
|
||||||
|
{
|
||||||
|
uint32_t fragment_size = MIN_PCM_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
uint32_t bits_per_sample = 16;
|
||||||
|
|
||||||
|
if (info->format == AUDIO_FORMAT_PCM_24_BIT_OFFLOAD) {
|
||||||
|
bits_per_sample = 32;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!info->has_video) {
|
||||||
|
fragment_size = MAX_PCM_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
|
||||||
|
} else if (info->has_video && info->is_streaming) {
|
||||||
|
fragment_size = (PCM_OFFLOAD_BUFFER_DURATION_FOR_AV_STREAMING
|
||||||
|
* info->sample_rate
|
||||||
|
* bits_per_sample
|
||||||
|
* popcount(info->channel_mask))/1000;
|
||||||
|
|
||||||
|
} else if (info->has_video) {
|
||||||
|
fragment_size = (PCM_OFFLOAD_BUFFER_DURATION_FOR_AV
|
||||||
|
* info->sample_rate
|
||||||
|
* bits_per_sample
|
||||||
|
* popcount(info->channel_mask))/1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
fragment_size = ALIGN( fragment_size, 1024);
|
||||||
|
|
||||||
|
if(fragment_size < MIN_PCM_OFFLOAD_FRAGMENT_SIZE)
|
||||||
|
fragment_size = MIN_PCM_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
else if(fragment_size > MAX_PCM_OFFLOAD_FRAGMENT_SIZE)
|
||||||
|
fragment_size = MAX_PCM_OFFLOAD_FRAGMENT_SIZE;
|
||||||
|
|
||||||
|
ALOGV("%s: fragment_size %d", __func__, fragment_size);
|
||||||
|
return fragment_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
@ -62,4 +62,8 @@ bool platform_listen_update_status(snd_device_t snd_device);
|
||||||
/* From platform_info_parser.c */
|
/* From platform_info_parser.c */
|
||||||
int platform_info_init(void);
|
int platform_info_init(void);
|
||||||
|
|
||||||
|
struct audio_offload_info_t;
|
||||||
|
uint32_t platform_get_compress_offload_buffer_size(audio_offload_info_t* info);
|
||||||
|
uint32_t platform_get_pcm_offload_buffer_size(audio_offload_info_t* info);
|
||||||
|
|
||||||
#endif // AUDIO_PLATFORM_API_H
|
#endif // AUDIO_PLATFORM_API_H
|
||||||
|
|
|
@ -1116,16 +1116,6 @@ bool AudioPolicyManager::isOffloadSupported(const audio_offload_info_t& offloadI
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Check if offload has been disabled
|
|
||||||
char propValue[PROPERTY_VALUE_MAX];
|
|
||||||
if (property_get("audio.offload.disable", propValue, "0")) {
|
|
||||||
if (atoi(propValue) != 0) {
|
|
||||||
ALOGV("offload disabled by audio.offload.disable=%s", propValue );
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if stream type is music, then only allow offload as of now.
|
// Check if stream type is music, then only allow offload as of now.
|
||||||
if (offloadInfo.stream_type != AUDIO_STREAM_MUSIC)
|
if (offloadInfo.stream_type != AUDIO_STREAM_MUSIC)
|
||||||
{
|
{
|
||||||
|
@ -1133,7 +1123,38 @@ bool AudioPolicyManager::isOffloadSupported(const audio_offload_info_t& offloadI
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
//TODO: enable audio offloading with video when ready
|
char propValue[PROPERTY_VALUE_MAX];
|
||||||
|
bool pcmOffload = false;
|
||||||
|
if (audio_is_offload_pcm(offloadInfo.format)) {
|
||||||
|
if(property_get("audio.offload.pcm.enable", propValue, NULL)) {
|
||||||
|
bool prop_enabled = atoi(propValue) || !strncmp("true", propValue, 4);
|
||||||
|
if (prop_enabled) {
|
||||||
|
ALOGW("PCM offload property is enabled");
|
||||||
|
pcmOffload = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!pcmOffload) {
|
||||||
|
ALOGV("PCM offload disabled by property audio.offload.pcm.enable");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!pcmOffload) {
|
||||||
|
// Check if offload has been disabled
|
||||||
|
if (property_get("audio.offload.disable", propValue, "0")) {
|
||||||
|
if (atoi(propValue) != 0) {
|
||||||
|
ALOGV("offload disabled by audio.offload.disable=%s", propValue );
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//check if it's multi-channel AAC format
|
||||||
|
if (AudioSystem::popCount(offloadInfo.channel_mask) > 2
|
||||||
|
&& offloadInfo.format == AUDIO_FORMAT_AAC) {
|
||||||
|
ALOGV("offload disabled for multi-channel AAC format");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if (offloadInfo.has_video)
|
if (offloadInfo.has_video)
|
||||||
{
|
{
|
||||||
if(property_get("av.offload.enable", propValue, NULL)) {
|
if(property_get("av.offload.enable", propValue, NULL)) {
|
||||||
|
@ -1142,18 +1163,26 @@ bool AudioPolicyManager::isOffloadSupported(const audio_offload_info_t& offloadI
|
||||||
ALOGW("offload disabled by av.offload.enable = %s ", propValue );
|
ALOGW("offload disabled by av.offload.enable = %s ", propValue );
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
if(offloadInfo.is_streaming &&
|
|
||||||
property_get("av.streaming.offload.enable", propValue, NULL)) {
|
if(offloadInfo.is_streaming) {
|
||||||
|
if (property_get("av.streaming.offload.enable", propValue, NULL)) {
|
||||||
bool prop_enabled = atoi(propValue) || !strncmp("true", propValue, 4);
|
bool prop_enabled = atoi(propValue) || !strncmp("true", propValue, 4);
|
||||||
if (!prop_enabled) {
|
if (!prop_enabled) {
|
||||||
ALOGW("offload disabled by av.streaming.offload.enable = %s ", propValue );
|
ALOGW("offload disabled by av.streaming.offload.enable = %s ", propValue );
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
//Do not offload AV streamnig if the property is not defined
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ALOGV("isOffloadSupported: has_video == true, property\
|
ALOGV("isOffloadSupported: has_video == true, property\
|
||||||
set to enable offload");
|
set to enable offload");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//If duration is less than minimum value defined in property, return false
|
//If duration is less than minimum value defined in property, return false
|
||||||
if (property_get("audio.offload.min.duration.secs", propValue, NULL)) {
|
if (property_get("audio.offload.min.duration.secs", propValue, NULL)) {
|
||||||
|
@ -1165,7 +1194,7 @@ bool AudioPolicyManager::isOffloadSupported(const audio_offload_info_t& offloadI
|
||||||
ALOGV("Offload denied by duration < default min(=%u)", OFFLOAD_DEFAULT_MIN_DURATION_SECS);
|
ALOGV("Offload denied by duration < default min(=%u)", OFFLOAD_DEFAULT_MIN_DURATION_SECS);
|
||||||
//duration checks only valid for MP3/AAC formats,
|
//duration checks only valid for MP3/AAC formats,
|
||||||
//do not check duration for other audio formats, e.g. dolby AAC/AC3 and amrwb+ formats
|
//do not check duration for other audio formats, e.g. dolby AAC/AC3 and amrwb+ formats
|
||||||
if (offloadInfo.format == AUDIO_FORMAT_MP3 || offloadInfo.format == AUDIO_FORMAT_AAC)
|
if (offloadInfo.format == AUDIO_FORMAT_MP3 || offloadInfo.format == AUDIO_FORMAT_AAC || pcmOffload)
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue