2017-04-19 00:43:05 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017, The Linux Foundation. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are
|
|
|
|
* met:
|
|
|
|
* * Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* * Redistributions in binary form must reproduce the above
|
|
|
|
* copyright notice, this list of conditions and the following
|
|
|
|
* disclaimer in the documentation and/or other materials provided
|
|
|
|
* with the distribution.
|
|
|
|
* * Neither the name of The Linux Foundation nor the names of its
|
|
|
|
* contributors may be used to endorse or promote products derived
|
|
|
|
* from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
|
|
|
|
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
|
|
|
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE
|
|
|
|
* OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
|
|
|
|
* IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define LOG_TAG "ip_hdlr_intf"
|
|
|
|
/*#define LOG_NDEBUG 0*/
|
|
|
|
/*#define VERY_VERY_VERBOSE_LOGGING*/
|
|
|
|
#ifdef VERY_VERY_VERBOSE_LOGGING
|
|
|
|
#define ALOGVV ALOGV
|
|
|
|
#else
|
|
|
|
#define ALOGVV(a...) do { } while(0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef LINUX_ENABLED
|
|
|
|
#define LIB_PATH "/usr/lib/libaudio_ip_handler.so"
|
|
|
|
#else
|
|
|
|
#define LIB_PATH "/system/vendor/lib/libaudio_ip_handler.so"
|
|
|
|
#endif
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
#include <pthread.h>
|
2017-04-19 00:43:05 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <dlfcn.h>
|
|
|
|
#include <cutils/log.h>
|
|
|
|
#include <sound/asound.h>
|
|
|
|
|
|
|
|
#include "audio_hw.h"
|
|
|
|
#include "audio_defs.h"
|
|
|
|
#include "platform.h"
|
|
|
|
|
|
|
|
/* These values defined by ADSP */
|
|
|
|
#define ADSP_DEC_SERVICE_ID 1
|
|
|
|
#define ADSP_EVENT_ID_RTIC 0x00013239
|
|
|
|
#define ADSP_EVENT_ID_RTIC_FAIL 0x0001323A
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
struct ip_hdlr_stream {
|
|
|
|
struct listnode list;
|
|
|
|
void *stream;
|
|
|
|
audio_usecase_t usecase;
|
|
|
|
};
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
struct ip_hdlr_intf {
|
|
|
|
void *lib_hdl;
|
|
|
|
int (*init)(void **handle, char *lib_path, void **lib_handle);
|
|
|
|
int (*deinit)(void *handle);
|
|
|
|
int (*open)(void *handle, bool is_dsp_decode, void *aud_sess_handle);
|
|
|
|
int (*shm_info)(void *handle, int *fd);
|
|
|
|
int (*close)(void *handle);
|
|
|
|
int (*event)(void *handle, void *payload);
|
|
|
|
int (*reg_cb)(void *handle, void *ack_cb, void *fail_cb);
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
struct listnode stream_list;
|
|
|
|
pthread_mutex_t stream_list_lock;
|
2017-04-19 00:43:05 +00:00
|
|
|
int ref_cnt;
|
|
|
|
};
|
|
|
|
static struct ip_hdlr_intf *ip_hdlr = NULL;
|
|
|
|
|
|
|
|
/* RTIC ack information */
|
|
|
|
struct rtic_ack_info {
|
|
|
|
uint32_t token;
|
|
|
|
uint32_t status;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* RTIC ack format sent to ADSP */
|
|
|
|
struct rtic_ack_param {
|
|
|
|
uint32_t param_size;
|
|
|
|
struct rtic_ack_info rtic_ack;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* each event payload format */
|
|
|
|
struct reg_ev_pl {
|
|
|
|
uint32_t event_id;
|
|
|
|
uint32_t cfg_mask;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* event registration format */
|
|
|
|
struct reg_event {
|
|
|
|
uint16_t version;
|
|
|
|
uint16_t service_id;
|
|
|
|
uint32_t num_reg_events;
|
|
|
|
struct reg_ev_pl rtic;
|
|
|
|
struct reg_ev_pl rtic_fail;
|
|
|
|
};
|
|
|
|
|
|
|
|
/* event received from ADSP is in this format */
|
|
|
|
struct rtic_event {
|
|
|
|
uint16_t service_id;
|
|
|
|
uint16_t reserved;
|
|
|
|
uint32_t event_id;
|
|
|
|
uint32_t payload_size;
|
|
|
|
uint8_t payload[0];
|
|
|
|
};
|
|
|
|
|
2017-05-05 01:55:45 +00:00
|
|
|
bool audio_extn_ip_hdlr_intf_supported(audio_format_t format)
|
|
|
|
{
|
2017-07-03 21:04:45 +00:00
|
|
|
if (((format & AUDIO_FORMAT_MAIN_MASK) == AUDIO_FORMAT_DOLBY_TRUEHD) ||
|
|
|
|
((!property_get_bool("audio.offload.passthrough", false)) &&
|
|
|
|
(((format & AUDIO_FORMAT_MAIN_MASK) == AUDIO_FORMAT_E_AC3) ||
|
|
|
|
((format & AUDIO_FORMAT_MAIN_MASK) == AUDIO_FORMAT_AC3))))
|
2017-05-05 01:55:45 +00:00
|
|
|
return true;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
int audio_extn_ip_hdlr_intf_event(void *stream_handle, void *payload, void *ip_hdlr_handle)
|
|
|
|
{
|
|
|
|
ALOGVV("%s:[%d] handle = %p",__func__, ip_hdlr->ref_cnt, ip_hdlr_handle);
|
|
|
|
|
|
|
|
return ip_hdlr->event(ip_hdlr_handle, payload);
|
|
|
|
}
|
|
|
|
|
|
|
|
int audio_extn_ip_hdlr_intf_rtic_ack(void *aud_sess_handle, struct rtic_ack_info *info)
|
|
|
|
{
|
|
|
|
char mixer_ctl_name[MIXER_PATH_MAX_LENGTH] = {0};
|
|
|
|
int ret = 0;
|
|
|
|
int pcm_device_id = 0;
|
|
|
|
struct mixer_ctl *ctl = NULL;
|
|
|
|
struct stream_out *out = (struct stream_out *)aud_sess_handle;
|
|
|
|
struct rtic_ack_param param;
|
2017-05-23 01:26:24 +00:00
|
|
|
struct listnode *node, *tempnode;
|
|
|
|
struct ip_hdlr_stream *stream_info;
|
|
|
|
struct audio_device *adev = NULL;
|
|
|
|
audio_usecase_t usecase = 0;
|
|
|
|
|
|
|
|
pthread_mutex_lock(&ip_hdlr->stream_list_lock);
|
|
|
|
list_for_each_safe(node, tempnode, &ip_hdlr->stream_list) {
|
|
|
|
stream_info = node_to_item(node, struct ip_hdlr_stream, list);
|
|
|
|
/* send the error if rtic failure notifcation is received */
|
|
|
|
if ((stream_info->stream == aud_sess_handle) &&
|
|
|
|
(stream_info->usecase == USECASE_AUDIO_TRANSCODE_LOOPBACK)) {
|
|
|
|
struct stream_inout *inout = (struct stream_inout *)aud_sess_handle;
|
|
|
|
usecase = stream_info->usecase;
|
|
|
|
adev = inout->dev;
|
|
|
|
break;
|
|
|
|
} else if (stream_info->stream == aud_sess_handle) {
|
|
|
|
struct stream_out *out = (struct stream_out *)aud_sess_handle;
|
|
|
|
usecase = stream_info->usecase;
|
|
|
|
adev = out->dev;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&ip_hdlr->stream_list_lock);
|
2017-04-19 00:43:05 +00:00
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
if (adev == NULL) {
|
|
|
|
ALOGE("%s:[%d] Invalid adev", __func__, ip_hdlr->ref_cnt);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
pcm_device_id = platform_get_pcm_device_id(usecase, PCM_PLAYBACK);
|
2017-04-19 00:43:05 +00:00
|
|
|
|
|
|
|
ALOGVV("%s:[%d] token = %d, info->status = %d, pcm_id = %d",__func__,
|
|
|
|
ip_hdlr->ref_cnt, info->token, info->status, pcm_device_id);
|
|
|
|
|
|
|
|
/* set mixer control to send RTIC done information */
|
|
|
|
ret = snprintf(mixer_ctl_name, sizeof(mixer_ctl_name),
|
|
|
|
"Playback Event Ack %d", pcm_device_id);
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] snprintf failed",__func__, ip_hdlr->ref_cnt);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
2017-05-23 01:26:24 +00:00
|
|
|
ctl = mixer_get_ctl_by_name(adev->mixer, mixer_ctl_name);
|
2017-04-19 00:43:05 +00:00
|
|
|
if (!ctl) {
|
|
|
|
ALOGE("%s:[%d] Could not get ctl for mixer cmd - %s", __func__,
|
|
|
|
ip_hdlr->ref_cnt, mixer_ctl_name);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
param.param_size = sizeof(struct rtic_ack_info);
|
|
|
|
memcpy(¶m.rtic_ack, info, sizeof(struct rtic_ack_info));
|
|
|
|
ret = mixer_ctl_set_array(ctl, (void *)¶m, sizeof(param));
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] Could not set ctl for mixer cmd - %s, ret %d", __func__, ip_hdlr->ref_cnt,
|
|
|
|
mixer_ctl_name, ret);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int audio_extn_ip_hdlr_intf_rtic_fail(void *aud_sess_handle)
|
|
|
|
{
|
2017-05-23 01:26:24 +00:00
|
|
|
struct listnode *node, *tempnode;
|
|
|
|
struct ip_hdlr_stream *stream_info;
|
2017-04-19 00:43:05 +00:00
|
|
|
|
|
|
|
ALOGD("%s:[%d] sess_handle = %p",__func__, ip_hdlr->ref_cnt, aud_sess_handle);
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
pthread_mutex_lock(&ip_hdlr->stream_list_lock);
|
|
|
|
list_for_each_safe(node, tempnode, &ip_hdlr->stream_list) {
|
|
|
|
stream_info = node_to_item(node, struct ip_hdlr_stream, list);
|
|
|
|
/* send the error if rtic failure notifcation is received */
|
|
|
|
if ((stream_info->stream == aud_sess_handle) &&
|
|
|
|
(stream_info->usecase == USECASE_AUDIO_TRANSCODE_LOOPBACK)) {
|
|
|
|
struct stream_inout *inout = (struct stream_inout *)aud_sess_handle;
|
|
|
|
pthread_mutex_lock(&inout->pre_lock);
|
|
|
|
pthread_mutex_lock(&inout->lock);
|
|
|
|
pthread_mutex_unlock(&inout->pre_lock);
|
|
|
|
ALOGVV("%s:[%d] calling client callback", __func__, ip_hdlr->ref_cnt);
|
|
|
|
if (inout && inout->client_callback)
|
|
|
|
inout->client_callback(AUDIO_EXTN_STREAM_CBK_EVENT_ERROR, NULL, inout->client_cookie);
|
|
|
|
pthread_mutex_unlock(&inout->lock);
|
|
|
|
break;
|
|
|
|
} else if (stream_info->stream == aud_sess_handle) {
|
|
|
|
struct stream_out *out = (struct stream_out *)aud_sess_handle;
|
|
|
|
pthread_mutex_lock(&out->pre_lock);
|
|
|
|
pthread_mutex_lock(&out->lock);
|
|
|
|
pthread_mutex_unlock(&out->pre_lock);
|
|
|
|
ALOGVV("%s:[%d] calling client callback", __func__, ip_hdlr->ref_cnt);
|
|
|
|
if (out && out->client_callback)
|
|
|
|
out->client_callback(AUDIO_EXTN_STREAM_CBK_EVENT_ERROR, NULL, out->client_cookie);
|
|
|
|
pthread_mutex_unlock(&out->lock);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&ip_hdlr->stream_list_lock);
|
2017-04-19 00:43:05 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
static int audio_extn_ip_hdlr_intf_open_dsp(void *handle, void *stream_handle, audio_usecase_t usecase)
|
2017-04-19 00:43:05 +00:00
|
|
|
{
|
|
|
|
int ret = 0, fd = 0, pcm_device_id = 0;
|
|
|
|
struct audio_adsp_event *param;
|
|
|
|
struct reg_event *reg_ev;
|
|
|
|
size_t shm_size;
|
|
|
|
void *shm_buf;;
|
2017-05-23 01:26:24 +00:00
|
|
|
struct stream_out *out;
|
|
|
|
struct stream_inout *inout;
|
|
|
|
void *adsp_hdlr_stream_handle;
|
|
|
|
struct audio_device *dev = NULL;
|
2017-04-19 00:43:05 +00:00
|
|
|
struct mixer_ctl *ctl = NULL;
|
|
|
|
char mixer_ctl_name[MIXER_PATH_MAX_LENGTH] = {0};
|
|
|
|
|
|
|
|
param = (struct audio_adsp_event *)calloc(1, sizeof(struct audio_adsp_event));
|
|
|
|
if (!param)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
reg_ev = (struct reg_event *)calloc(1, sizeof(struct reg_event));
|
|
|
|
if (!reg_ev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
reg_ev->service_id = ADSP_DEC_SERVICE_ID;
|
|
|
|
reg_ev->num_reg_events = 2;
|
|
|
|
reg_ev->rtic.event_id = ADSP_EVENT_ID_RTIC;
|
|
|
|
reg_ev->rtic.cfg_mask = 1; /* event enabled */
|
|
|
|
reg_ev->rtic_fail.event_id = ADSP_EVENT_ID_RTIC_FAIL;
|
|
|
|
reg_ev->rtic_fail.cfg_mask = 1; /* event enabled */
|
|
|
|
|
|
|
|
param->event_type = AUDIO_STREAM_ENCDEC_EVENT;
|
|
|
|
param->payload_length = sizeof(struct reg_event);
|
|
|
|
param->payload = reg_ev;
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
if (usecase == USECASE_AUDIO_TRANSCODE_LOOPBACK) {
|
|
|
|
inout = (struct stream_inout *)stream_handle;
|
|
|
|
adsp_hdlr_stream_handle = inout->adsp_hdlr_stream_handle;
|
|
|
|
dev = inout->dev;
|
|
|
|
} else {
|
|
|
|
out = (struct stream_out *)stream_handle;
|
|
|
|
adsp_hdlr_stream_handle = out->adsp_hdlr_stream_handle;
|
|
|
|
dev = out->dev;
|
|
|
|
}
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
/* Register for event and its callback */
|
2017-05-23 01:26:24 +00:00
|
|
|
ret = audio_extn_adsp_hdlr_stream_register_event(adsp_hdlr_stream_handle, param,
|
2017-04-19 00:43:05 +00:00
|
|
|
audio_extn_ip_hdlr_intf_event,
|
|
|
|
handle);
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] failed to register event",__func__, ip_hdlr->ref_cnt, ret);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
ip_hdlr->reg_cb(handle, &audio_extn_ip_hdlr_intf_rtic_ack, &audio_extn_ip_hdlr_intf_rtic_fail);
|
|
|
|
ip_hdlr->shm_info(handle, &fd);
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
pcm_device_id = platform_get_pcm_device_id(usecase, PCM_PLAYBACK);
|
2017-04-19 00:43:05 +00:00
|
|
|
ret = snprintf(mixer_ctl_name, sizeof(mixer_ctl_name),
|
|
|
|
"Playback ION FD %d", pcm_device_id);
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] snprintf failed",__func__, ip_hdlr->ref_cnt, ret);
|
|
|
|
goto done;
|
|
|
|
}
|
2017-05-23 01:26:24 +00:00
|
|
|
ALOGV("%s: fd = %d pcm_id = %d", __func__, fd, pcm_device_id);
|
|
|
|
|
|
|
|
ctl = mixer_get_ctl_by_name(dev->mixer, mixer_ctl_name);
|
2017-04-19 00:43:05 +00:00
|
|
|
if (!ctl) {
|
|
|
|
ALOGE("%s:[%d] Could not get ctl for mixer cmd - %s", __func__,
|
|
|
|
ip_hdlr->ref_cnt, mixer_ctl_name);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
ret = mixer_ctl_set_array(ctl, &fd, sizeof(fd));
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] Could not set ctl for mixer cmd - %s, ret %d", __func__, ip_hdlr->ref_cnt,
|
|
|
|
mixer_ctl_name, ret);
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
|
|
|
|
done:
|
|
|
|
free(param);
|
|
|
|
free(reg_ev);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
int audio_extn_ip_hdlr_intf_open(void *handle, bool is_dsp_decode,
|
|
|
|
void *aud_sess_handle, audio_usecase_t usecase)
|
2017-04-19 00:43:05 +00:00
|
|
|
{
|
|
|
|
int ret = 0;
|
2017-05-23 01:26:24 +00:00
|
|
|
struct ip_hdlr_stream *stream_info;
|
2017-04-19 00:43:05 +00:00
|
|
|
|
|
|
|
if (!handle || !aud_sess_handle) {
|
|
|
|
ALOGE("%s:[%d] Invalid arguments, handle %p", __func__, ip_hdlr->ref_cnt, handle);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
stream_info = (struct ip_hdlr_stream *)calloc(1, sizeof(struct ip_hdlr_stream));
|
|
|
|
if (!stream_info)
|
|
|
|
return -ENOMEM;
|
|
|
|
stream_info->stream = aud_sess_handle;
|
|
|
|
stream_info->usecase = usecase;
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
ret = ip_hdlr->open(handle, is_dsp_decode, aud_sess_handle);
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] open failed", __func__, ip_hdlr->ref_cnt);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2017-05-23 01:26:24 +00:00
|
|
|
ALOGD("%s:[%d] handle = %p, sess_handle = %p, is_dsp_decode = %d, usecase = %d",
|
|
|
|
__func__, ip_hdlr->ref_cnt, handle, aud_sess_handle, is_dsp_decode, usecase);
|
2017-04-19 00:43:05 +00:00
|
|
|
if (is_dsp_decode) {
|
2017-05-23 01:26:24 +00:00
|
|
|
ret = audio_extn_ip_hdlr_intf_open_dsp(handle, aud_sess_handle, usecase);
|
2017-04-19 00:43:05 +00:00
|
|
|
if (ret < 0)
|
|
|
|
ip_hdlr->close(handle);
|
|
|
|
}
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
pthread_mutex_lock(&ip_hdlr->stream_list_lock);
|
|
|
|
list_add_tail(&ip_hdlr->stream_list, &stream_info->list);
|
|
|
|
pthread_mutex_unlock(&ip_hdlr->stream_list_lock);
|
2017-04-19 00:43:05 +00:00
|
|
|
done:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int audio_extn_ip_hdlr_intf_close(void *handle, bool is_dsp_decode, void *aud_sess_handle)
|
|
|
|
{
|
|
|
|
struct audio_adsp_event param;
|
2017-05-23 01:26:24 +00:00
|
|
|
void *adsp_hdlr_stream_handle;
|
|
|
|
struct listnode *node, *tempnode;
|
|
|
|
struct ip_hdlr_stream *stream_info;
|
|
|
|
audio_usecase_t usecase = 0;
|
2017-04-19 00:43:05 +00:00
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (!handle) {
|
|
|
|
ALOGE("%s:[%d] handle is NULL", __func__, ip_hdlr->ref_cnt);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = ip_hdlr->close(handle);
|
|
|
|
if (ret < 0)
|
|
|
|
ALOGE("%s:[%d] close failed", __func__, ip_hdlr->ref_cnt);
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
pthread_mutex_lock(&ip_hdlr->stream_list_lock);
|
|
|
|
list_for_each_safe(node, tempnode, &ip_hdlr->stream_list) {
|
|
|
|
stream_info = node_to_item(node, struct ip_hdlr_stream, list);
|
|
|
|
if (stream_info->stream == aud_sess_handle) {
|
|
|
|
usecase = stream_info->usecase;
|
|
|
|
list_remove(node);
|
|
|
|
free(stream_info);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pthread_mutex_unlock(&ip_hdlr->stream_list_lock);
|
|
|
|
ALOGD("%s:[%d] handle = %p, usecase = %d",__func__, ip_hdlr->ref_cnt, handle, usecase);
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
if (is_dsp_decode) {
|
2017-05-23 01:26:24 +00:00
|
|
|
if (usecase == USECASE_AUDIO_TRANSCODE_LOOPBACK) {
|
|
|
|
struct stream_inout *inout = (struct stream_inout *)aud_sess_handle;
|
|
|
|
adsp_hdlr_stream_handle = inout->adsp_hdlr_stream_handle;
|
|
|
|
} else {
|
|
|
|
struct stream_out *out = (struct stream_out *)aud_sess_handle;
|
|
|
|
adsp_hdlr_stream_handle = out->adsp_hdlr_stream_handle;
|
|
|
|
}
|
2017-04-19 00:43:05 +00:00
|
|
|
param.event_type = AUDIO_STREAM_ENCDEC_EVENT;
|
|
|
|
param.payload_length = 0;
|
|
|
|
/* Deregister the event */
|
2017-05-23 01:26:24 +00:00
|
|
|
ret = audio_extn_adsp_hdlr_stream_deregister_event(adsp_hdlr_stream_handle, ¶m);
|
2017-04-19 00:43:05 +00:00
|
|
|
if (ret < 0)
|
|
|
|
ALOGE("%s:[%d] event deregister failed", __func__, ip_hdlr->ref_cnt);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int audio_extn_ip_hdlr_intf_init(void **handle, char *lib_path, void **lib_handle)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (!ip_hdlr) {
|
|
|
|
ip_hdlr = (struct ip_hdlr_intf *)calloc(1, sizeof(struct ip_hdlr_intf));
|
|
|
|
if (!ip_hdlr)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
list_init(&ip_hdlr->stream_list);
|
|
|
|
pthread_mutex_init(&ip_hdlr->stream_list_lock, (const pthread_mutexattr_t *) NULL);
|
|
|
|
|
2017-04-19 00:43:05 +00:00
|
|
|
ip_hdlr->lib_hdl = dlopen(LIB_PATH, RTLD_NOW);
|
|
|
|
if (ip_hdlr->lib_hdl == NULL) {
|
|
|
|
ALOGE("%s: DLOPEN failed, %s", __func__, dlerror());
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
ip_hdlr->init =(int (*)(void **handle, char *lib_path,
|
|
|
|
void **lib_handle))dlsym(ip_hdlr->lib_hdl, "audio_ip_hdlr_init");
|
|
|
|
ip_hdlr->deinit = (int (*)(void *handle))dlsym(ip_hdlr->lib_hdl, "audio_ip_hdlr_deinit");
|
|
|
|
ip_hdlr->open = (int (*)(void *handle, bool is_dsp_decode,
|
|
|
|
void *sess_handle))dlsym(ip_hdlr->lib_hdl, "audio_ip_hdlr_open");
|
|
|
|
ip_hdlr->close =(int (*)(void *handle))dlsym(ip_hdlr->lib_hdl, "audio_ip_hdlr_close");
|
|
|
|
ip_hdlr->reg_cb =(int (*)(void *handle, void *ack_cb,
|
|
|
|
void *fail_cb))dlsym(ip_hdlr->lib_hdl, "audio_ip_hdlr_reg_cb");
|
|
|
|
ip_hdlr->shm_info =(int (*)(void *handle, int *fd))dlsym(ip_hdlr->lib_hdl,
|
|
|
|
"audio_ip_hdlr_shm_info");
|
|
|
|
ip_hdlr->event =(int (*)(void *handle, void *payload))dlsym(ip_hdlr->lib_hdl,
|
|
|
|
"audio_ip_hdlr_event");
|
|
|
|
if (!ip_hdlr->init || !ip_hdlr->deinit || !ip_hdlr->open ||
|
|
|
|
!ip_hdlr->close || !ip_hdlr->reg_cb || !ip_hdlr->shm_info ||
|
|
|
|
!ip_hdlr->event) {
|
|
|
|
ALOGE("%s: failed to get symbols", __func__);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto dlclose;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = ip_hdlr->init(handle, lib_path, lib_handle);
|
|
|
|
if (ret < 0) {
|
|
|
|
ALOGE("%s:[%d] init failed ret = %d", __func__, ip_hdlr->ref_cnt, ret);
|
|
|
|
ret = -EINVAL;
|
|
|
|
goto dlclose;
|
|
|
|
}
|
|
|
|
ip_hdlr->ref_cnt++;
|
|
|
|
ALOGD("%s:[%d] init done", __func__, ip_hdlr->ref_cnt);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
dlclose:
|
|
|
|
dlclose(ip_hdlr->lib_hdl);
|
|
|
|
err:
|
2017-05-23 01:26:24 +00:00
|
|
|
pthread_mutex_destroy(&ip_hdlr->stream_list_lock);
|
2017-04-19 00:43:05 +00:00
|
|
|
free(ip_hdlr);
|
|
|
|
ip_hdlr = NULL;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int audio_extn_ip_hdlr_intf_deinit(void *handle)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (!handle) {
|
|
|
|
ALOGE("%s:[%d] handle is NULL", __func__, ip_hdlr->ref_cnt);
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
ALOGD("%s:[%d] handle = %p",__func__, ip_hdlr->ref_cnt, handle);
|
|
|
|
ret = ip_hdlr->deinit(handle);
|
|
|
|
if (ret < 0)
|
|
|
|
ALOGE("%s:[%d] deinit failed ret = %d", __func__, ip_hdlr->ref_cnt, ret);
|
|
|
|
|
|
|
|
if (--ip_hdlr->ref_cnt == 0) {
|
|
|
|
if (ip_hdlr->lib_hdl)
|
|
|
|
dlclose(ip_hdlr->lib_hdl);
|
|
|
|
|
2017-05-23 01:26:24 +00:00
|
|
|
pthread_mutex_destroy(&ip_hdlr->stream_list_lock);
|
2017-04-19 00:43:05 +00:00
|
|
|
free(ip_hdlr);
|
2017-07-03 21:04:45 +00:00
|
|
|
ip_hdlr = NULL;
|
2017-04-19 00:43:05 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|