123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751 |
- // SPDX-License-Identifier: GPL-2.0-only
- /*
- * Copyright (c) 2015-2019, The Linux Foundation. All rights reserved.
- */
- #include <linux/debugfs.h>
- #include <linux/export.h>
- #include <linux/delay.h>
- #include <linux/module.h>
- #include <linux/mutex.h>
- #include <linux/ipa.h>
- #include <linux/msm_gsi.h>
- #include <linux/ipa_mhi.h>
- #include "../ipa_common_i.h"
- #include "ipa_i.h"
- #include "ipa_qmi_service.h"
- #define IPA_MHI_DRV_NAME "ipa_mhi"
- #define IPA_MHI_DBG(fmt, args...) \
- do { \
- pr_debug(IPA_MHI_DRV_NAME " %s:%d " fmt, \
- __func__, __LINE__, ## args); \
- IPA_IPC_LOGGING(ipa_get_ipc_logbuf(), \
- IPA_MHI_DRV_NAME " %s:%d " fmt, ## args); \
- IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
- IPA_MHI_DRV_NAME " %s:%d " fmt, ## args); \
- } while (0)
- #define IPA_MHI_DBG_LOW(fmt, args...) \
- do { \
- pr_debug(IPA_MHI_DRV_NAME " %s:%d " fmt, \
- __func__, __LINE__, ## args); \
- IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
- IPA_MHI_DRV_NAME " %s:%d " fmt, ## args); \
- } while (0)
- #define IPA_MHI_ERR(fmt, args...) \
- do { \
- pr_err(IPA_MHI_DRV_NAME " %s:%d " fmt, \
- __func__, __LINE__, ## args); \
- IPA_IPC_LOGGING(ipa_get_ipc_logbuf(), \
- IPA_MHI_DRV_NAME " %s:%d " fmt, ## args); \
- IPA_IPC_LOGGING(ipa_get_ipc_logbuf_low(), \
- IPA_MHI_DRV_NAME " %s:%d " fmt, ## args); \
- } while (0)
- #define IPA_MHI_FUNC_ENTRY() \
- IPA_MHI_DBG("ENTRY\n")
- #define IPA_MHI_FUNC_EXIT() \
- IPA_MHI_DBG("EXIT\n")
- #define IPA_MHI_MAX_UL_CHANNELS 1
- #define IPA_MHI_MAX_DL_CHANNELS 2
- /* bit #40 in address should be asserted for MHI transfers over pcie */
- #define IPA_MHI_HOST_ADDR_COND(addr) \
- ((params->assert_bit40)?(IPA_MHI_HOST_ADDR(addr)):(addr))
- enum ipa3_mhi_polling_mode {
- IPA_MHI_POLLING_MODE_DB_MODE,
- IPA_MHI_POLLING_MODE_POLL_MODE,
- };
- bool ipa3_mhi_stop_gsi_channel(enum ipa_client_type client)
- {
- int res;
- int ipa_ep_idx;
- struct ipa3_ep_context *ep;
- IPA_MHI_FUNC_ENTRY();
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx == -1) {
- IPA_MHI_ERR("Invalid client.\n");
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- IPA_MHI_DBG_LOW("Stopping GSI channel %ld\n", ep->gsi_chan_hdl);
- res = gsi_stop_channel(ep->gsi_chan_hdl);
- if (res != 0 &&
- res != -GSI_STATUS_AGAIN &&
- res != -GSI_STATUS_TIMED_OUT) {
- IPA_MHI_ERR("GSI stop channel failed %d\n",
- res);
- WARN_ON(1);
- return false;
- }
- if (res == 0) {
- IPA_MHI_DBG_LOW("GSI channel %ld STOP\n",
- ep->gsi_chan_hdl);
- return true;
- }
- return false;
- }
- static int ipa3_mhi_reset_gsi_channel(enum ipa_client_type client)
- {
- int res;
- int clnt_hdl;
- IPA_MHI_FUNC_ENTRY();
- clnt_hdl = ipa3_get_ep_mapping(client);
- if (clnt_hdl < 0)
- return -EFAULT;
- res = ipa3_reset_gsi_channel(clnt_hdl);
- if (res) {
- IPA_MHI_ERR("ipa3_reset_gsi_channel failed %d\n", res);
- return -EFAULT;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- }
- int ipa3_mhi_reset_channel_internal(enum ipa_client_type client)
- {
- int res;
- IPA_MHI_FUNC_ENTRY();
- res = ipa3_mhi_reset_gsi_channel(client);
- if (res) {
- IPAERR("ipa3_mhi_reset_gsi_channel failed\n");
- ipa_assert();
- return res;
- }
- res = ipa3_disable_data_path(ipa3_get_ep_mapping(client));
- if (res) {
- IPA_MHI_ERR("ipa3_disable_data_path failed %d\n", res);
- return res;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- }
- int ipa3_mhi_start_channel_internal(enum ipa_client_type client)
- {
- int res;
- int ipa_ep_idx;
- IPA_MHI_FUNC_ENTRY();
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx < 0) {
- IPA_MHI_ERR("Invalid client %d\n", client);
- return -EINVAL;
- }
- res = ipa3_enable_data_path(ipa_ep_idx);
- if (res) {
- IPA_MHI_ERR("ipa3_enable_data_path failed %d\n", res);
- return res;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- }
- static int ipa3_mhi_get_ch_poll_cfg(enum ipa_client_type client,
- struct ipa_mhi_ch_ctx *ch_ctx_host, int ring_size)
- {
- switch (ch_ctx_host->pollcfg) {
- case 0:
- /*set default polling configuration according to MHI spec*/
- if (IPA_CLIENT_IS_PROD(client))
- return 7;
- else
- return (ring_size/2)/8;
- break;
- default:
- return ch_ctx_host->pollcfg;
- }
- }
- static int ipa_mhi_start_gsi_channel(enum ipa_client_type client,
- int ipa_ep_idx, struct start_gsi_channel *params)
- {
- int res = 0;
- struct gsi_evt_ring_props ev_props;
- struct ipa_mhi_msi_info *msi;
- struct gsi_chan_props ch_props;
- union __packed gsi_channel_scratch ch_scratch;
- struct ipa3_ep_context *ep;
- const struct ipa_gsi_ep_config *ep_cfg;
- struct ipa_ep_cfg_ctrl ep_cfg_ctrl;
- bool burst_mode_enabled = false;
- IPA_MHI_FUNC_ENTRY();
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- msi = params->msi;
- ep_cfg = ipa3_get_gsi_ep_info(client);
- if (!ep_cfg) {
- IPA_MHI_ERR("Wrong parameter, ep_cfg is NULL\n");
- return -EPERM;
- }
- /* allocate event ring only for the first time pipe is connected */
- if (params->state == IPA_HW_MHI_CHANNEL_STATE_INVALID) {
- memset(&ev_props, 0, sizeof(ev_props));
- ev_props.intf = GSI_EVT_CHTYPE_MHI_EV;
- ev_props.intr = GSI_INTR_MSI;
- ev_props.re_size = GSI_EVT_RING_RE_SIZE_16B;
- ev_props.ring_len = params->ev_ctx_host->rlen;
- ev_props.ring_base_addr = IPA_MHI_HOST_ADDR_COND(
- params->ev_ctx_host->rbase);
- ev_props.int_modt = params->ev_ctx_host->intmodt *
- IPA_SLEEP_CLK_RATE_KHZ;
- ev_props.int_modc = params->ev_ctx_host->intmodc;
- ev_props.intvec = ((msi->data & ~msi->mask) |
- (params->ev_ctx_host->msivec & msi->mask));
- ev_props.msi_addr = IPA_MHI_HOST_ADDR_COND(
- (((u64)msi->addr_hi << 32) | msi->addr_low));
- ev_props.rp_update_addr = IPA_MHI_HOST_ADDR_COND(
- params->event_context_addr +
- offsetof(struct ipa_mhi_ev_ctx, rp));
- ev_props.exclusive = true;
- ev_props.err_cb = params->ev_err_cb;
- ev_props.user_data = params->channel;
- ev_props.evchid_valid = true;
- ev_props.evchid = params->evchid;
- IPA_MHI_DBG("allocating event ring ep:%u evchid:%u\n",
- ipa_ep_idx, ev_props.evchid);
- res = gsi_alloc_evt_ring(&ev_props, ipa3_ctx->gsi_dev_hdl,
- &ep->gsi_evt_ring_hdl);
- if (res) {
- IPA_MHI_ERR("gsi_alloc_evt_ring failed %d\n", res);
- goto fail_alloc_evt;
- }
- IPA_MHI_DBG("client %d, caching event ring hdl %lu\n",
- client,
- ep->gsi_evt_ring_hdl);
- *params->cached_gsi_evt_ring_hdl =
- ep->gsi_evt_ring_hdl;
- } else {
- IPA_MHI_DBG("event ring already exists: evt_ring_hdl=%lu\n",
- *params->cached_gsi_evt_ring_hdl);
- ep->gsi_evt_ring_hdl = *params->cached_gsi_evt_ring_hdl;
- }
- if (params->ev_ctx_host->wp == params->ev_ctx_host->rbase) {
- IPA_MHI_ERR("event ring wp is not updated. base=wp=0x%llx\n",
- params->ev_ctx_host->wp);
- goto fail_alloc_ch;
- }
- IPA_MHI_DBG("Ring event db: evt_ring_hdl=%lu host_wp=0x%llx\n",
- ep->gsi_evt_ring_hdl, params->ev_ctx_host->wp);
- res = gsi_ring_evt_ring_db(ep->gsi_evt_ring_hdl,
- params->ev_ctx_host->wp);
- if (res) {
- IPA_MHI_ERR("fail to ring evt ring db %d. hdl=%lu wp=0x%llx\n",
- res, ep->gsi_evt_ring_hdl, params->ev_ctx_host->wp);
- goto fail_alloc_ch;
- }
- memset(&ch_props, 0, sizeof(ch_props));
- ch_props.prot = GSI_CHAN_PROT_MHI;
- ch_props.dir = IPA_CLIENT_IS_PROD(client) ?
- GSI_CHAN_DIR_TO_GSI : GSI_CHAN_DIR_FROM_GSI;
- ch_props.ch_id = ep_cfg->ipa_gsi_chan_num;
- ch_props.evt_ring_hdl = *params->cached_gsi_evt_ring_hdl;
- ch_props.re_size = GSI_CHAN_RE_SIZE_16B;
- ch_props.ring_len = params->ch_ctx_host->rlen;
- ch_props.ring_base_addr = IPA_MHI_HOST_ADDR_COND(
- params->ch_ctx_host->rbase);
- /* Burst mode is not supported on DPL pipes */
- if ((client != IPA_CLIENT_MHI_DPL_CONS) &&
- (params->ch_ctx_host->brstmode == IPA_MHI_BURST_MODE_DEFAULT ||
- params->ch_ctx_host->brstmode == IPA_MHI_BURST_MODE_ENABLE)) {
- burst_mode_enabled = true;
- }
- if (ipa3_ctx->ipa_hw_type >= IPA_HW_v4_0 &&
- !burst_mode_enabled)
- ch_props.use_db_eng = GSI_CHAN_DIRECT_MODE;
- else
- ch_props.use_db_eng = GSI_CHAN_DB_MODE;
- ch_props.db_in_bytes = 1;
- ch_props.max_prefetch = GSI_ONE_PREFETCH_SEG;
- ch_props.low_weight = 1;
- ch_props.prefetch_mode = ep_cfg->prefetch_mode;
- ch_props.empty_lvl_threshold = ep_cfg->prefetch_threshold;
- ch_props.err_cb = params->ch_err_cb;
- ch_props.chan_user_data = params->channel;
- res = gsi_alloc_channel(&ch_props, ipa3_ctx->gsi_dev_hdl,
- &ep->gsi_chan_hdl);
- if (res) {
- IPA_MHI_ERR("gsi_alloc_channel failed %d\n",
- res);
- goto fail_alloc_ch;
- }
- memset(&ch_scratch, 0, sizeof(ch_scratch));
- ch_scratch.mhi.mhi_host_wp_addr = IPA_MHI_HOST_ADDR_COND(
- params->channel_context_addr +
- offsetof(struct ipa_mhi_ch_ctx, wp));
- ch_scratch.mhi.assert_bit40 = params->assert_bit40;
- /*
- * Update scratch for MCS smart prefetch:
- * Starting IPA4.5, smart prefetch implemented by H/W.
- * At IPA 4.0/4.1/4.2, we do not use MCS smart prefetch
- * so keep the fields zero.
- */
- if (ipa3_ctx->ipa_hw_type < IPA_HW_v4_0) {
- ch_scratch.mhi.max_outstanding_tre =
- ep_cfg->ipa_if_tlv * ch_props.re_size;
- ch_scratch.mhi.outstanding_threshold =
- min(ep_cfg->ipa_if_tlv / 2, 8) * ch_props.re_size;
- }
- ch_scratch.mhi.oob_mod_threshold = 4;
- if (burst_mode_enabled) {
- ch_scratch.mhi.burst_mode_enabled = burst_mode_enabled;
- ch_scratch.mhi.polling_configuration =
- ipa3_mhi_get_ch_poll_cfg(client, params->ch_ctx_host,
- (ch_props.ring_len / ch_props.re_size));
- ch_scratch.mhi.polling_mode = IPA_MHI_POLLING_MODE_DB_MODE;
- } else {
- ch_scratch.mhi.burst_mode_enabled = false;
- }
- res = gsi_write_channel_scratch(ep->gsi_chan_hdl,
- ch_scratch);
- if (res) {
- IPA_MHI_ERR("gsi_write_channel_scratch failed %d\n",
- res);
- goto fail_ch_scratch;
- }
- *params->mhi = ch_scratch.mhi;
- if (IPA_CLIENT_IS_PROD(ep->client) && ep->skip_ep_cfg) {
- memset(&ep_cfg_ctrl, 0, sizeof(struct ipa_ep_cfg_ctrl));
- ep_cfg_ctrl.ipa_ep_delay = true;
- ep->ep_delay_set = true;
- res = ipa3_cfg_ep_ctrl(ipa_ep_idx, &ep_cfg_ctrl);
- if (res)
- IPA_MHI_ERR("client (ep: %d) failed result=%d\n",
- ipa_ep_idx, res);
- else
- IPA_MHI_DBG("client (ep: %d) success\n", ipa_ep_idx);
- } else {
- ep->ep_delay_set = false;
- }
- IPA_MHI_DBG("Starting channel\n");
- res = gsi_start_channel(ep->gsi_chan_hdl);
- if (res) {
- IPA_MHI_ERR("gsi_start_channel failed %d\n", res);
- goto fail_ch_start;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- fail_ch_start:
- fail_ch_scratch:
- gsi_dealloc_channel(ep->gsi_chan_hdl);
- fail_alloc_ch:
- gsi_dealloc_evt_ring(ep->gsi_evt_ring_hdl);
- ep->gsi_evt_ring_hdl = ~0;
- fail_alloc_evt:
- return res;
- }
- int ipa3_mhi_init_engine(struct ipa_mhi_init_engine *params)
- {
- int res;
- struct gsi_device_scratch gsi_scratch;
- const struct ipa_gsi_ep_config *gsi_ep_info;
- IPA_MHI_FUNC_ENTRY();
- if (!params) {
- IPA_MHI_ERR("null args\n");
- return -EINVAL;
- }
- if ((IPA_MHI_MAX_UL_CHANNELS + IPA_MHI_MAX_DL_CHANNELS) >
- ((ipa3_ctx->mhi_evid_limits[1] -
- ipa3_ctx->mhi_evid_limits[0]) + 1)) {
- IPAERR("Not enough event rings for MHI\n");
- ipa_assert();
- return -EINVAL;
- }
- /* Initialize IPA MHI engine */
- gsi_ep_info = ipa3_get_gsi_ep_info(IPA_CLIENT_MHI_PROD);
- if (!gsi_ep_info) {
- IPAERR("MHI PROD has no ep allocated\n");
- ipa_assert();
- }
- memset(&gsi_scratch, 0, sizeof(gsi_scratch));
- gsi_scratch.mhi_base_chan_idx_valid = true;
- gsi_scratch.mhi_base_chan_idx = gsi_ep_info->ipa_gsi_chan_num +
- params->gsi.first_ch_idx;
- res = gsi_write_device_scratch(ipa3_ctx->gsi_dev_hdl,
- &gsi_scratch);
- if (res) {
- IPA_MHI_ERR("failed to write device scratch %d\n", res);
- goto fail_init_engine;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- fail_init_engine:
- return res;
- }
- /**
- * ipa3_connect_mhi_pipe() - Connect pipe to IPA and start corresponding
- * MHI channel
- * @in: connect parameters
- * @clnt_hdl: [out] client handle for this pipe
- *
- * This function is called by IPA MHI client driver on MHI channel start.
- * This function is called after MHI engine was started.
- *
- * Return codes: 0 : success
- * negative : error
- */
- int ipa3_connect_mhi_pipe(struct ipa_mhi_connect_params_internal *in,
- u32 *clnt_hdl)
- {
- struct ipa3_ep_context *ep;
- int ipa_ep_idx;
- int res;
- enum ipa_client_type client;
- IPA_MHI_FUNC_ENTRY();
- if (!in || !clnt_hdl) {
- IPA_MHI_ERR("NULL args\n");
- return -EINVAL;
- }
- in->start.gsi.evchid += ipa3_ctx->mhi_evid_limits[0];
- client = in->sys->client;
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx == -1) {
- IPA_MHI_ERR("Invalid client.\n");
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- if (ep->valid == 1) {
- IPA_MHI_ERR("EP already allocated.\n");
- return -EPERM;
- }
- memset(ep, 0, offsetof(struct ipa3_ep_context, sys));
- ep->valid = 1;
- ep->skip_ep_cfg = in->sys->skip_ep_cfg;
- ep->client = client;
- ep->client_notify = in->sys->notify;
- ep->priv = in->sys->priv;
- ep->keep_ipa_awake = in->sys->keep_ipa_awake;
- res = ipa_mhi_start_gsi_channel(client,
- ipa_ep_idx, &in->start.gsi);
- if (res) {
- IPA_MHI_ERR("ipa_mhi_start_gsi_channel failed %d\n",
- res);
- goto fail_start_channel;
- }
- res = ipa3_enable_data_path(ipa_ep_idx);
- if (res) {
- IPA_MHI_ERR("enable data path failed res=%d clnt=%d.\n", res,
- ipa_ep_idx);
- goto fail_ep_cfg;
- }
- if (!ep->skip_ep_cfg) {
- if (ipa3_cfg_ep(ipa_ep_idx, &in->sys->ipa_ep_cfg)) {
- IPAERR("fail to configure EP.\n");
- goto fail_ep_cfg;
- }
- if (ipa3_cfg_ep_status(ipa_ep_idx, &ep->status)) {
- IPAERR("fail to configure status of EP.\n");
- goto fail_ep_cfg;
- }
- IPA_MHI_DBG("ep configuration successful\n");
- } else {
- IPA_MHI_DBG("skipping ep configuration\n");
- }
- *clnt_hdl = ipa_ep_idx;
- if (!ep->skip_ep_cfg && IPA_CLIENT_IS_PROD(client))
- ipa3_install_dflt_flt_rules(ipa_ep_idx);
- ipa3_ctx->skip_ep_cfg_shadow[ipa_ep_idx] = ep->skip_ep_cfg;
- IPA_MHI_DBG("client %d (ep: %d) connected\n", client,
- ipa_ep_idx);
- IPA_MHI_FUNC_EXIT();
- return 0;
- fail_ep_cfg:
- ipa3_disable_data_path(ipa_ep_idx);
- fail_start_channel:
- memset(ep, 0, offsetof(struct ipa3_ep_context, sys));
- return -EPERM;
- }
- /**
- * ipa3_disconnect_mhi_pipe() - Disconnect pipe from IPA and reset corresponding
- * MHI channel
- * @clnt_hdl: client handle for this pipe
- *
- * This function is called by IPA MHI client driver on MHI channel reset.
- * This function is called after MHI channel was started.
- * This function is doing the following:
- * - Send command to uC/GSI to reset corresponding MHI channel
- * - Configure IPA EP control
- *
- * Return codes: 0 : success
- * negative : error
- */
- int ipa3_disconnect_mhi_pipe(u32 clnt_hdl)
- {
- struct ipa3_ep_context *ep;
- int res;
- struct ipa_ep_cfg_ctrl ep_cfg_ctrl;
- IPA_MHI_FUNC_ENTRY();
- if (clnt_hdl >= ipa3_ctx->ipa_num_pipes) {
- IPAERR("invalid handle %d\n", clnt_hdl);
- return -EINVAL;
- }
- if (ipa3_ctx->ep[clnt_hdl].valid == 0) {
- IPAERR("pipe was not connected %d\n", clnt_hdl);
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[clnt_hdl];
- if (ep->ep_delay_set) {
- memset(&ep_cfg_ctrl, 0, sizeof(struct ipa_ep_cfg_ctrl));
- ep_cfg_ctrl.ipa_ep_delay = false;
- res = ipa3_cfg_ep_ctrl(clnt_hdl,
- &ep_cfg_ctrl);
- if (res) {
- IPAERR
- ("client(ep:%d) failed to remove delay res=%d\n",
- clnt_hdl, res);
- } else {
- IPADBG("client (ep: %d) delay removed\n",
- clnt_hdl);
- ep->ep_delay_set = false;
- }
- }
- res = gsi_dealloc_channel(ep->gsi_chan_hdl);
- if (res) {
- IPAERR("gsi_dealloc_channel failed %d\n", res);
- goto fail_reset_channel;
- }
- ep->valid = 0;
- ipa3_delete_dflt_flt_rules(clnt_hdl);
- IPA_MHI_DBG("client (ep: %d) disconnected\n", clnt_hdl);
- IPA_MHI_FUNC_EXIT();
- return 0;
- fail_reset_channel:
- return res;
- }
- int ipa3_mhi_resume_channels_internal(enum ipa_client_type client,
- bool LPTransitionRejected, bool brstmode_enabled,
- union __packed gsi_channel_scratch ch_scratch, u8 index)
- {
- int res;
- int ipa_ep_idx;
- struct ipa3_ep_context *ep;
- union __packed gsi_channel_scratch gsi_ch_scratch;
- IPA_MHI_FUNC_ENTRY();
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx < 0) {
- IPA_MHI_ERR("Invalid client %d\n", client);
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- if (brstmode_enabled && !LPTransitionRejected) {
- res = gsi_read_channel_scratch(ep->gsi_chan_hdl,
- &gsi_ch_scratch);
- if (res) {
- IPA_MHI_ERR("read ch scratch fail %d\n", res);
- return res;
- }
- /*
- * set polling mode bit to DB mode before
- * resuming the channel
- *
- * For MHI-->IPA pipes:
- * when resuming due to transition to M0,
- * set the polling mode bit to 0.
- * In other cases, restore it's value form
- * when you stopped the channel.
- * Here, after successful resume client move to M0 state.
- * So, by default setting polling mode bit to 0.
- *
- * For IPA-->MHI pipe:
- * always restore the polling mode bit.
- */
- if (IPA_CLIENT_IS_PROD(client))
- ch_scratch.mhi.polling_mode =
- IPA_MHI_POLLING_MODE_DB_MODE;
- else
- ch_scratch.mhi.polling_mode =
- gsi_ch_scratch.mhi.polling_mode;
- /* Use GSI update API to not affect non-SWI fields
- * inside the scratch while in suspend-resume operation
- */
- res = gsi_update_mhi_channel_scratch(
- ep->gsi_chan_hdl, ch_scratch.mhi);
- if (res) {
- IPA_MHI_ERR("write ch scratch fail %d\n"
- , res);
- return res;
- }
- }
- res = gsi_start_channel(ep->gsi_chan_hdl);
- if (res) {
- IPA_MHI_ERR("failed to resume channel error %d\n", res);
- return res;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- }
- int ipa3_mhi_query_ch_info(enum ipa_client_type client,
- struct gsi_chan_info *ch_info)
- {
- int ipa_ep_idx;
- int res;
- struct ipa3_ep_context *ep;
- IPA_MHI_FUNC_ENTRY();
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx < 0) {
- IPA_MHI_ERR("Invalid client %d\n", client);
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- res = gsi_query_channel_info(ep->gsi_chan_hdl, ch_info);
- if (res) {
- IPA_MHI_ERR("gsi_query_channel_info failed\n");
- return res;
- }
- IPA_MHI_FUNC_EXIT();
- return 0;
- }
- bool ipa3_has_open_aggr_frame(enum ipa_client_type client)
- {
- u32 aggr_state_active;
- int ipa_ep_idx;
- aggr_state_active = ipahal_read_reg(IPA_STATE_AGGR_ACTIVE);
- IPA_MHI_DBG_LOW("IPA_STATE_AGGR_ACTIVE_OFST 0x%x\n", aggr_state_active);
- ipa_ep_idx = ipa_get_ep_mapping(client);
- if (ipa_ep_idx == -1) {
- ipa_assert();
- return false;
- }
- if ((1 << ipa_ep_idx) & aggr_state_active)
- return true;
- return false;
- }
- int ipa3_mhi_destroy_channel(enum ipa_client_type client)
- {
- int res;
- int ipa_ep_idx;
- struct ipa3_ep_context *ep;
- ipa_ep_idx = ipa3_get_ep_mapping(client);
- if (ipa_ep_idx < 0) {
- IPA_MHI_ERR("Invalid client %d\n", client);
- return -EINVAL;
- }
- ep = &ipa3_ctx->ep[ipa_ep_idx];
- IPA_ACTIVE_CLIENTS_INC_EP(client);
- IPA_MHI_DBG("reset event ring (hdl: %lu, ep: %d)\n",
- ep->gsi_evt_ring_hdl, ipa_ep_idx);
- res = gsi_reset_evt_ring(ep->gsi_evt_ring_hdl);
- if (res) {
- IPAERR(" failed to reset evt ring %lu, err %d\n"
- , ep->gsi_evt_ring_hdl, res);
- goto fail;
- }
- IPA_MHI_DBG("dealloc event ring (hdl: %lu, ep: %d)\n",
- ep->gsi_evt_ring_hdl, ipa_ep_idx);
- res = gsi_dealloc_evt_ring(
- ep->gsi_evt_ring_hdl);
- if (res) {
- IPAERR("dealloc evt ring %lu failed, err %d\n"
- , ep->gsi_evt_ring_hdl, res);
- goto fail;
- }
- IPA_ACTIVE_CLIENTS_DEC_EP(client);
- return 0;
- fail:
- IPA_ACTIVE_CLIENTS_DEC_EP(client);
- return res;
- }
- MODULE_LICENSE("GPL v2");
- MODULE_DESCRIPTION("IPA MHI driver");
|