[PATCH v6 15/22] ASoC: qdsp6: audioreach: add q6apm support
Srinivas Kandagatla
srinivas.kandagatla at linaro.org
Thu Sep 16 17:02:18 CEST 2021
Thanks Pierre for review,
On 15/09/2021 17:02, Pierre-Louis Bossart wrote:
>
>> +static void audioreach_populate_graph(struct apm_graph_open_params *open,
>> + struct list_head *sg_list,
>> + int num_sub_graphs)
>> +{
>> + struct apm_mod_conn_list_params *mc_data = open->mod_conn_list_data;
>> + struct apm_module_list_params *ml_data = open->mod_list_data;
>> + struct apm_prop_list_params *mp_data = open->mod_prop_data;
>> + struct apm_container_params *c_data = open->cont_data;
>> + struct apm_sub_graph_params *sg_data = open->sg_data;
>> + int ncontainer = 0, nmodule = 0, nconn = 0;
>> + struct apm_mod_prop_obj *module_prop_obj;
>> + struct audioreach_container *container;
>> + struct apm_module_conn_obj *conn_obj;
>> + struct audioreach_module *module;
>> + struct audioreach_sub_graph *sg;
>> + struct apm_container_obj *cobj;
>> + struct apm_mod_list_obj *mlobj;
>> + int i = 0;
>> +
>> + mlobj = &ml_data->mod_list_obj[0];
>> +
>> + list_for_each_entry(sg, sg_list, node) {
>> + struct apm_sub_graph_data *sg_cfg = &sg_data->sg_cfg[i++];
>> +
>> + apm_populate_sub_graph_config(sg_cfg, sg);
>> +
>> + list_for_each_entry(container, &sg->container_list, node) {
>> + cobj = &c_data->cont_obj[ncontainer];
>> +
>> + apm_populate_container_config(cobj, container);
>> + apm_populate_module_list_obj(mlobj, container, sg->sub_graph_id);
>> +
>> + list_for_each_container_module(module, container) {
>> + uint32_t src_mod_inst_id;
>> +
>> + src_mod_inst_id = module->src_mod_inst_id;
>> +
>> + module_prop_obj = &mp_data->mod_prop_obj[nmodule];
>> + apm_populate_module_prop_obj(module_prop_obj, module);
>> +
>> + if (src_mod_inst_id /*&& dst_mod_inst_id*/) {
>
> remove left-over code or uncomment required condition?
Yes, this can be removed.
>
>> + conn_obj = &mc_data->conn_obj[nconn];
>> + apm_populate_connection_obj(conn_obj, module);
>> + nconn++;
>> + }
>> +
>> + nmodule++;
>> + }
>> + mlobj = (void *) mlobj + APM_MOD_LIST_OBJ_PSIZE(container->num_modules);
>> +
>> + ncontainer++;
>> + }
>> + }
>> +}
>
>> +static struct audioreach_graph *q6apm_get_audioreach_graph(struct q6apm *apm, uint32_t graph_id)
>> +{
>> + struct audioreach_graph_info *info;
>> + struct audioreach_graph *graph;
>> +
>> + mutex_lock(&apm->lock);
>> + graph = idr_find(&apm->graph_idr, graph_id);
>> + mutex_unlock(&apm->lock);
>> +
>> + if (graph) {
>> + kref_get(&graph->refcount);
>> + return graph;
>> + }
>> +
>> + info = idr_find(&apm->graph_info_idr, graph_id);
>> +
>> + if (!info)
>> + return ERR_PTR(-ENODEV);
>> +
>> + graph = kzalloc(sizeof(*graph), GFP_KERNEL);
>> + if (!graph)
>> + return ERR_PTR(-ENOMEM);
>> +
>> + graph->apm = apm;
>> + graph->info = info;
>> + graph->id = graph_id;
>> +
>> + graph->graph = audioreach_alloc_graph_pkt(apm, &info->sg_list, graph_id);
>> + if (IS_ERR(graph->graph)) {
>> + kfree(graph);
>> + return ERR_PTR(-ENOMEM);
>> + }
>> +
>> + mutex_lock(&apm->lock);
>> + idr_alloc(&apm->graph_idr, graph, graph_id, graph_id + 1, GFP_KERNEL);
>
> test for idr_alloc() success? You have error checks for idr_find() but
> not for the _alloc() case.
Yes, error handing is missing in this case, fixed in next version.
>
>> + mutex_unlock(&apm->lock);
>> +
>> + kref_init(&graph->refcount);
>> +
>> + q6apm_send_cmd_sync(apm, graph->graph, 0);
>> +
>> + return graph;
>> +}
>> +
>> +static int apm_callback(struct gpr_resp_pkt *data, void *priv, int op)
>> +{
>> + gpr_device_t *gdev = priv;
>> + struct q6apm *apm = dev_get_drvdata(&gdev->dev);
>> + struct device *dev = &gdev->dev;
>> + struct gpr_ibasic_rsp_result_t *result;
>> + struct gpr_hdr *hdr = &data->hdr;
>> + int ret = -EINVAL;
>> +
>> + result = data->payload;
>> +
>> + switch (hdr->opcode) {
>> + case APM_CMD_RSP_GET_SPF_STATE:
>> + apm->result.opcode = hdr->opcode;
>> + apm->result.status = 0;
>> + /* First word of result it state */
>> + apm->state = result->opcode;
>> + wake_up(&apm->wait);
>> + break;
>
> this would keep ret = -EINVAL and return it, that seems suspicious since
> this doesn't look as an error case?
>
>> + case GPR_BASIC_RSP_RESULT:
>> + switch (result->opcode) {
>> + case APM_CMD_GRAPH_START:
>> + case APM_CMD_GRAPH_OPEN:
>> + case APM_CMD_GRAPH_PREPARE:
>> + case APM_CMD_GRAPH_CLOSE:
>> + case APM_CMD_GRAPH_FLUSH:
>> + case APM_CMD_GRAPH_STOP:
>> + case APM_CMD_SET_CFG:
>> + apm->result.opcode = result->opcode;
>> + apm->result.status = result->status;
>> + if (result->status) {
>> + dev_err(dev, "Error (%d) Processing 0x%08x cmd\n",
>> + result->status, result->opcode);
>> + ret = -EINVAL;
>> + } else {
>> + ret = 0;
>
> That's also weird, maybe initialize to zero and return an error code in
> error cases, or fix the case APM_CMD_RSP_GET_SPF_STATE above.
this is now fixed by initializing the ret to 0 and setting it in error path.
--srini
>
>> + }
>> + wake_up(&apm->wait);
>> + break;
>> + default:
>> + break;
>> + }
>> + break;
>> + default:
>> + break;
>
> so all the defaults are errors?
>
>> + }
>> +
>> + return ret;
>> +}
More information about the Alsa-devel
mailing list