|
@@ -119,13 +119,15 @@ enum ipa3_usb_state {
|
|
|
enum ipa3_usb_transport_type {
|
|
|
IPA_USB_TRANSPORT_TETH,
|
|
|
IPA_USB_TRANSPORT_DPL,
|
|
|
+ IPA_USB_TRANSPORT_TETH_2,
|
|
|
IPA_USB_TRANSPORT_MAX
|
|
|
};
|
|
|
|
|
|
/* Get transport type from tethering protocol */
|
|
|
#define IPA3_USB_GET_TTYPE(__teth_prot) \
|
|
|
(((__teth_prot) == IPA_USB_DIAG) ? \
|
|
|
- IPA_USB_TRANSPORT_DPL : IPA_USB_TRANSPORT_TETH)
|
|
|
+ IPA_USB_TRANSPORT_DPL : (((__teth_prot) == IPA_USB_RMNET_CV2X) ? \
|
|
|
+ IPA_USB_TRANSPORT_TETH_2 : IPA_USB_TRANSPORT_TETH))
|
|
|
|
|
|
/* Does the given transport type is DPL? */
|
|
|
#define IPA3_USB_IS_TTYPE_DPL(__ttype) \
|
|
@@ -178,7 +180,7 @@ struct ipa3_usb_context {
|
|
|
struct ipa3_usb_teth_prot_context
|
|
|
teth_prot_ctx[IPA_USB_MAX_TETH_PROT_SIZE];
|
|
|
int num_init_prot; /* without dpl */
|
|
|
- struct teth_bridge_init_params teth_bridge_params;
|
|
|
+ struct teth_bridge_init_params teth_bridge_params[IPA_TETH_BRIDGE_MAX];
|
|
|
struct completion dev_ready_comp;
|
|
|
u32 qmi_req_id;
|
|
|
spinlock_t state_lock;
|
|
@@ -577,6 +579,8 @@ static char *ipa3_usb_teth_prot_to_string(enum ipa_usb_teth_prot teth_prot)
|
|
|
case IPA_USB_RMNET:
|
|
|
case IPA_USB_MBIM:
|
|
|
return "teth_bridge";
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
+ return "teth_bridge_cv2x";
|
|
|
case IPA_USB_DIAG:
|
|
|
return "dpl";
|
|
|
default:
|
|
@@ -592,6 +596,8 @@ static char *ipa3_usb_teth_bridge_prot_to_string(
|
|
|
switch (teth_prot) {
|
|
|
case IPA_USB_RMNET:
|
|
|
return "rmnet";
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
+ return "rmnet_cv2x";
|
|
|
case IPA_USB_MBIM:
|
|
|
return "mbim";
|
|
|
default:
|
|
@@ -601,11 +607,18 @@ static char *ipa3_usb_teth_bridge_prot_to_string(
|
|
|
return "unsupported";
|
|
|
}
|
|
|
|
|
|
-static int ipa3_usb_init_teth_bridge(void)
|
|
|
+static int ipa3_usb_init_teth_bridge(enum ipa_usb_teth_prot teth_prot)
|
|
|
{
|
|
|
int result;
|
|
|
|
|
|
- result = ipa3_teth_bridge_init(&ipa3_usb_ctx->teth_bridge_params);
|
|
|
+ if (teth_prot == IPA_USB_RMNET_CV2X)
|
|
|
+ result =
|
|
|
+ ipa3_teth_bridge_init(
|
|
|
+ &ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_2]);
|
|
|
+ else
|
|
|
+ result =
|
|
|
+ ipa3_teth_bridge_init(
|
|
|
+ &ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_1]);
|
|
|
if (result) {
|
|
|
IPA_USB_ERR("Failed to initialize teth_bridge\n");
|
|
|
return result;
|
|
@@ -619,15 +632,26 @@ static int ipa3_usb_register_pm(enum ipa3_usb_transport_type ttype)
|
|
|
struct ipa3_usb_transport_type_ctx *ttype_ctx =
|
|
|
&ipa3_usb_ctx->ttype_ctx[ttype];
|
|
|
int result;
|
|
|
+ enum ipa_client_type consumer;
|
|
|
+
|
|
|
+ /*
|
|
|
+ * One PM resource for teth1,
|
|
|
+ * One PM resource for teth2 (CV2X),
|
|
|
+ * One for DPL,
|
|
|
+ */
|
|
|
|
|
|
- /* there is one PM resource for teth and one for DPL */
|
|
|
- if (!IPA3_USB_IS_TTYPE_DPL(ttype) && ipa3_usb_ctx->num_init_prot > 0)
|
|
|
+ if (!IPA3_USB_IS_TTYPE_DPL(ttype) && (ipa3_usb_ctx->num_init_prot > 0)
|
|
|
+ && (ttype != IPA_USB_TRANSPORT_TETH_2))
|
|
|
return 0;
|
|
|
|
|
|
memset(&ttype_ctx->pm_ctx.reg_params, 0,
|
|
|
sizeof(ttype_ctx->pm_ctx.reg_params));
|
|
|
- ttype_ctx->pm_ctx.reg_params.name = (ttype == IPA_USB_TRANSPORT_DPL) ?
|
|
|
- "USB DPL" : "USB";
|
|
|
+ ttype_ctx->pm_ctx.reg_params.name =
|
|
|
+ (ttype == IPA_USB_TRANSPORT_DPL) ?
|
|
|
+ "USB DPL" :
|
|
|
+ (ttype == IPA_USB_TRANSPORT_TETH_2) ?
|
|
|
+ "USB2" : "USB";
|
|
|
+
|
|
|
ttype_ctx->pm_ctx.reg_params.callback = ipa3_usb_pm_cb;
|
|
|
ttype_ctx->pm_ctx.reg_params.user_data = ttype_ctx;
|
|
|
ttype_ctx->pm_ctx.reg_params.group = IPA_PM_GROUP_DEFAULT;
|
|
@@ -639,9 +663,12 @@ static int ipa3_usb_register_pm(enum ipa3_usb_transport_type ttype)
|
|
|
goto fail_pm_reg;
|
|
|
}
|
|
|
|
|
|
+ consumer = (ttype == IPA_USB_TRANSPORT_DPL) ?
|
|
|
+ IPA_CLIENT_USB_DPL_CONS :
|
|
|
+ (ttype == IPA_USB_TRANSPORT_TETH_2) ?
|
|
|
+ IPA_CLIENT_USB2_CONS : IPA_CLIENT_USB_CONS;
|
|
|
result = ipa_pm_associate_ipa_cons_to_client(ttype_ctx->pm_ctx.hdl,
|
|
|
- (ttype == IPA_USB_TRANSPORT_DPL) ?
|
|
|
- IPA_CLIENT_USB_DPL_CONS : IPA_CLIENT_USB_CONS);
|
|
|
+ consumer);
|
|
|
if (result) {
|
|
|
IPA_USB_ERR("fail to associate cons with PM %d\n", result);
|
|
|
goto fail_pm_cons;
|
|
@@ -855,9 +882,11 @@ static int ipa_usb_init_teth_prot_internal(enum ipa_usb_teth_prot teth_prot,
|
|
|
goto bad_params;
|
|
|
}
|
|
|
ipa3_usb_ctx->teth_prot_ctx[teth_prot].user_data = user_data;
|
|
|
- result = ipa3_usb_init_teth_bridge();
|
|
|
+
|
|
|
+ result = ipa3_usb_init_teth_bridge(teth_prot);
|
|
|
if (result)
|
|
|
goto teth_prot_init_fail;
|
|
|
+
|
|
|
ipa3_usb_ctx->teth_prot_ctx[teth_prot].state =
|
|
|
IPA_USB_TETH_PROT_INITIALIZED;
|
|
|
ipa3_usb_ctx->num_init_prot++;
|
|
@@ -873,6 +902,26 @@ static int ipa_usb_init_teth_prot_internal(enum ipa_usb_teth_prot teth_prot,
|
|
|
ipa3_register_client_callback(&ipa_usb_set_lock_unlock,
|
|
|
&ipa3_usb_get_teth_port_state, IPA_CLIENT_USB_PROD);
|
|
|
break;
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
+ if (ipa3_usb_ctx->teth_prot_ctx[teth_prot].state !=
|
|
|
+ IPA_USB_TETH_PROT_INVALID) {
|
|
|
+ IPA_USB_DBG("%s already initialized\n",
|
|
|
+ ipa3_usb_teth_prot_to_string(teth_prot));
|
|
|
+ result = -EPERM;
|
|
|
+ goto bad_params;
|
|
|
+ }
|
|
|
+ ipa3_usb_ctx->teth_prot_ctx[teth_prot].user_data = user_data;
|
|
|
+
|
|
|
+ result = ipa3_usb_init_teth_bridge(teth_prot);
|
|
|
+ if (result)
|
|
|
+ goto teth_prot_init_fail;
|
|
|
+
|
|
|
+ ipa3_usb_ctx->teth_prot_ctx[teth_prot].state =
|
|
|
+ IPA_USB_TETH_PROT_INITIALIZED;
|
|
|
+ IPA_USB_DBG("initialized %s %s\n",
|
|
|
+ ipa3_usb_teth_prot_to_string(teth_prot),
|
|
|
+ ipa3_usb_teth_bridge_prot_to_string(teth_prot));
|
|
|
+ break;
|
|
|
case IPA_USB_DIAG:
|
|
|
if (ipa3_usb_ctx->teth_prot_ctx[teth_prot].state !=
|
|
|
IPA_USB_TETH_PROT_INVALID) {
|
|
@@ -900,7 +949,8 @@ static int ipa_usb_init_teth_prot_internal(enum ipa_usb_teth_prot teth_prot,
|
|
|
|
|
|
teth_prot_init_fail:
|
|
|
if ((IPA3_USB_IS_TTYPE_DPL(ttype))
|
|
|
- || (ipa3_usb_ctx->num_init_prot == 0)) {
|
|
|
+ || (ipa3_usb_ctx->num_init_prot == 0)
|
|
|
+ || (teth_prot == IPA_USB_RMNET_CV2X)) {
|
|
|
ipa3_usb_deregister_pm(ttype);
|
|
|
}
|
|
|
bad_params:
|
|
@@ -971,6 +1021,7 @@ static bool ipa3_usb_check_chan_params(struct ipa_usb_xdci_chan_params *params)
|
|
|
}
|
|
|
break;
|
|
|
case IPA_USB_RMNET:
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
case IPA_USB_MBIM:
|
|
|
if (ipa3_usb_ctx->teth_prot_ctx[params->teth_prot].state ==
|
|
|
IPA_USB_TETH_PROT_INVALID) {
|
|
@@ -1041,7 +1092,6 @@ static int ipa3_usb_smmu_map_xdci_channel(
|
|
|
ipa3_usb_ctx->smmu_reg_map.cnt--;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
result = ipa3_smmu_map_peer_buff(params->xfer_ring_base_addr_iova,
|
|
|
params->xfer_ring_len, map, params->sgt_xfer_rings,
|
|
|
IPA_SMMU_CB_AP);
|
|
@@ -1118,11 +1168,19 @@ static int ipa3_usb_request_xdci_channel(
|
|
|
case IPA_USB_RMNET:
|
|
|
case IPA_USB_MBIM:
|
|
|
chan_params.priv =
|
|
|
- ipa3_usb_ctx->teth_bridge_params.private_data;
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_1].private_data;
|
|
|
chan_params.notify =
|
|
|
- ipa3_usb_ctx->teth_bridge_params.usb_notify_cb;
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_1].usb_notify_cb;
|
|
|
chan_params.skip_ep_cfg =
|
|
|
- ipa3_usb_ctx->teth_bridge_params.skip_ep_cfg;
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_1].skip_ep_cfg;
|
|
|
+ break;
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
+ chan_params.priv =
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_2].private_data;
|
|
|
+ chan_params.notify =
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_2].usb_notify_cb;
|
|
|
+ chan_params.skip_ep_cfg =
|
|
|
+ ipa3_usb_ctx->teth_bridge_params[IPA_TETH_BRIDGE_2].skip_ep_cfg;
|
|
|
break;
|
|
|
case IPA_USB_DIAG:
|
|
|
chan_params.priv = NULL;
|
|
@@ -1330,6 +1388,16 @@ static int ipa3_usb_connect_teth_bridge(
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+static int ipa3_get_tethering_mode(enum ipa_usb_teth_prot teth_prot)
|
|
|
+{
|
|
|
+ if (teth_prot == IPA_USB_RMNET)
|
|
|
+ return TETH_TETHERING_MODE_RMNET;
|
|
|
+ else if (teth_prot == IPA_USB_RMNET_CV2X)
|
|
|
+ return TETH_TETHERING_MODE_RMNET_2;
|
|
|
+ else
|
|
|
+ return TETH_TETHERING_MODE_MBIM;
|
|
|
+}
|
|
|
+
|
|
|
static int ipa3_usb_connect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
{
|
|
|
int result;
|
|
@@ -1398,6 +1466,7 @@ static int ipa3_usb_connect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
ipa3_usb_teth_prot_to_string(teth_prot));
|
|
|
break;
|
|
|
case IPA_USB_RMNET:
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
case IPA_USB_MBIM:
|
|
|
if (teth_prot_ptr->state ==
|
|
|
IPA_USB_TETH_PROT_CONNECTED) {
|
|
@@ -1405,7 +1474,8 @@ static int ipa3_usb_connect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
ipa3_usb_teth_prot_to_string(teth_prot));
|
|
|
break;
|
|
|
}
|
|
|
- result = ipa3_usb_init_teth_bridge();
|
|
|
+
|
|
|
+ result = ipa3_usb_init_teth_bridge(teth_prot);
|
|
|
if (result)
|
|
|
return result;
|
|
|
|
|
@@ -1416,14 +1486,19 @@ static int ipa3_usb_connect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
teth_bridge_params.usb_ipa_pipe_hdl =
|
|
|
teth_conn_params->usb_to_ipa_clnt_hdl;
|
|
|
teth_bridge_params.tethering_mode =
|
|
|
- (teth_prot == IPA_USB_RMNET) ?
|
|
|
- (TETH_TETHERING_MODE_RMNET):(TETH_TETHERING_MODE_MBIM);
|
|
|
- teth_bridge_params.client_type = IPA_CLIENT_USB_PROD;
|
|
|
+ ipa3_get_tethering_mode(teth_prot);
|
|
|
+
|
|
|
+ if (teth_prot == IPA_USB_RMNET_CV2X)
|
|
|
+ teth_bridge_params.client_type = IPA_CLIENT_USB2_PROD;
|
|
|
+ else
|
|
|
+ teth_bridge_params.client_type = IPA_CLIENT_USB_PROD;
|
|
|
+
|
|
|
result = ipa3_usb_connect_teth_bridge(&teth_bridge_params);
|
|
|
if (result) {
|
|
|
ipa3_usb_ctx->ttype_ctx[ttype].user_data = NULL;
|
|
|
return result;
|
|
|
}
|
|
|
+
|
|
|
ipa3_usb_ctx->teth_prot_ctx[teth_prot].state =
|
|
|
IPA_USB_TETH_PROT_CONNECTED;
|
|
|
ipa3_usb_notify_do(ttype, IPA_USB_DEVICE_READY);
|
|
@@ -1455,11 +1530,15 @@ static int ipa3_usb_connect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
-static int ipa3_usb_disconnect_teth_bridge(void)
|
|
|
+static int ipa3_usb_disconnect_teth_bridge(enum ipa_usb_teth_prot teth_prot)
|
|
|
{
|
|
|
int result;
|
|
|
|
|
|
- result = ipa3_teth_bridge_disconnect(IPA_CLIENT_USB_PROD);
|
|
|
+ if (teth_prot == IPA_USB_RMNET_CV2X)
|
|
|
+ result = ipa3_teth_bridge_disconnect(IPA_CLIENT_USB2_PROD);
|
|
|
+ else
|
|
|
+ result = ipa3_teth_bridge_disconnect(IPA_CLIENT_USB_PROD);
|
|
|
+
|
|
|
if (result) {
|
|
|
IPA_USB_ERR("failed to disconnect teth_bridge\n");
|
|
|
return result;
|
|
@@ -1503,6 +1582,7 @@ static int ipa3_usb_disconnect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
ipa3_usb_teth_prot_to_string(teth_prot));
|
|
|
break;
|
|
|
case IPA_USB_RMNET:
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
case IPA_USB_MBIM:
|
|
|
if (teth_prot_ptr->state != IPA_USB_TETH_PROT_CONNECTED) {
|
|
|
IPA_USB_DBG("%s (%s) is not connected\n",
|
|
@@ -1510,7 +1590,8 @@ static int ipa3_usb_disconnect_teth_prot(enum ipa_usb_teth_prot teth_prot)
|
|
|
ipa3_usb_teth_bridge_prot_to_string(teth_prot));
|
|
|
return -EPERM;
|
|
|
}
|
|
|
- result = ipa3_usb_disconnect_teth_bridge();
|
|
|
+
|
|
|
+ result = ipa3_usb_disconnect_teth_bridge(teth_prot);
|
|
|
if (result)
|
|
|
break;
|
|
|
|
|
@@ -1550,8 +1631,7 @@ static int ipa3_usb_xdci_connect_internal(
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
- ttype = (params->teth_prot == IPA_USB_DIAG) ? IPA_USB_TRANSPORT_DPL :
|
|
|
- IPA_USB_TRANSPORT_TETH;
|
|
|
+ ttype = IPA3_USB_GET_TTYPE(params->teth_prot);
|
|
|
|
|
|
if (!ipa3_usb_check_legal_op(IPA_USB_OP_CONNECT, ttype)) {
|
|
|
IPA_USB_ERR("Illegal operation\n");
|
|
@@ -1695,7 +1775,9 @@ static int ipa3_usb_get_status_dbg_info(struct ipa3_usb_status_dbg_info *status)
|
|
|
for (i = 0 ; i < IPA_USB_MAX_TETH_PROT_SIZE ; i++) {
|
|
|
if (ipa3_usb_ctx->teth_prot_ctx[i].state ==
|
|
|
IPA_USB_TETH_PROT_INITIALIZED) {
|
|
|
- if ((i == IPA_USB_RMNET) || (i == IPA_USB_MBIM))
|
|
|
+ if ((i == IPA_USB_RMNET) ||
|
|
|
+ (i == IPA_USB_MBIM) ||
|
|
|
+ (i == IPA_USB_RMNET_CV2X))
|
|
|
status->inited_prots[status->num_init_prot++] =
|
|
|
ipa3_usb_teth_bridge_prot_to_string(i);
|
|
|
else
|
|
@@ -1705,6 +1787,7 @@ static int ipa3_usb_get_status_dbg_info(struct ipa3_usb_status_dbg_info *status)
|
|
|
IPA_USB_TETH_PROT_CONNECTED) {
|
|
|
switch (i) {
|
|
|
case IPA_USB_RMNET:
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
case IPA_USB_MBIM:
|
|
|
status->teth_connected_prot =
|
|
|
ipa3_usb_teth_bridge_prot_to_string(i);
|
|
@@ -2171,6 +2254,24 @@ static int ipa_usb_deinit_teth_prot_internal(enum ipa_usb_teth_prot teth_prot)
|
|
|
ipa3_usb_teth_prot_to_string(teth_prot),
|
|
|
ipa3_usb_teth_bridge_prot_to_string(teth_prot));
|
|
|
break;
|
|
|
+ case IPA_USB_RMNET_CV2X:
|
|
|
+ if (ipa3_usb_ctx->teth_prot_ctx[teth_prot].state !=
|
|
|
+ IPA_USB_TETH_PROT_INITIALIZED) {
|
|
|
+ IPA_USB_ERR("%s (%s) is not initialized\n",
|
|
|
+ ipa3_usb_teth_prot_to_string(teth_prot),
|
|
|
+ ipa3_usb_teth_bridge_prot_to_string(teth_prot));
|
|
|
+ result = -EINVAL;
|
|
|
+ goto bad_params;
|
|
|
+ }
|
|
|
+
|
|
|
+ ipa3_usb_ctx->teth_prot_ctx[teth_prot].user_data =
|
|
|
+ NULL;
|
|
|
+ ipa3_usb_ctx->teth_prot_ctx[teth_prot].state =
|
|
|
+ IPA_USB_TETH_PROT_INVALID;
|
|
|
+ IPA_USB_DBG("deinitialized %s (%s)\n",
|
|
|
+ ipa3_usb_teth_prot_to_string(teth_prot),
|
|
|
+ ipa3_usb_teth_bridge_prot_to_string(teth_prot));
|
|
|
+ break;
|
|
|
case IPA_USB_DIAG:
|
|
|
if (teth_prot_ptr->state !=
|
|
|
IPA_USB_TETH_PROT_INITIALIZED) {
|
|
@@ -2191,7 +2292,8 @@ static int ipa_usb_deinit_teth_prot_internal(enum ipa_usb_teth_prot teth_prot)
|
|
|
}
|
|
|
|
|
|
if (IPA3_USB_IS_TTYPE_DPL(ttype) ||
|
|
|
- (ipa3_usb_ctx->num_init_prot == 0)) {
|
|
|
+ (ipa3_usb_ctx->num_init_prot == 0) ||
|
|
|
+ (teth_prot == IPA_USB_RMNET_CV2X)) {
|
|
|
if (!ipa3_usb_set_state(IPA_USB_INVALID, false, ttype))
|
|
|
IPA_USB_ERR(
|
|
|
"failed to change state to invalid\n");
|
|
@@ -2583,6 +2685,9 @@ int ipa3_usb_init(void)
|
|
|
pm_ctx = &ipa3_usb_ctx->ttype_ctx[IPA_USB_TRANSPORT_TETH].pm_ctx;
|
|
|
pm_ctx->hdl = ~0;
|
|
|
pm_ctx->remote_wakeup_work = &ipa3_usb_notify_remote_wakeup_work;
|
|
|
+ pm_ctx = &ipa3_usb_ctx->ttype_ctx[IPA_USB_TRANSPORT_TETH_2].pm_ctx;
|
|
|
+ pm_ctx->hdl = ~0;
|
|
|
+ pm_ctx->remote_wakeup_work = &ipa3_usb_notify_remote_wakeup_work;
|
|
|
pm_ctx = &ipa3_usb_ctx->ttype_ctx[IPA_USB_TRANSPORT_DPL].pm_ctx;
|
|
|
pm_ctx->hdl = ~0;
|
|
|
pm_ctx->remote_wakeup_work = &ipa3_usb_dpl_notify_remote_wakeup_work;
|