|
@@ -73,6 +73,8 @@ enum ipa_ap_ingress_ep_enum {
|
|
|
|
|
|
#define IPA_WWAN_CONS_DESC_FIFO_SZ 256
|
|
|
|
|
|
+#define LAN_STATS_FOR_ALL_CLIENTS 0xFFFFFFFF
|
|
|
+
|
|
|
static void rmnet_ipa_free_msg(void *buff, u32 len, u32 type);
|
|
|
static void rmnet_ipa_get_stats_and_update(void);
|
|
|
|
|
@@ -4427,6 +4429,54 @@ static inline int rmnet_ipa3_delete_lan_client_info
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
+/* Query must be free-d by the caller */
|
|
|
+static int rmnet_ipa_get_hw_fnr_stats_v2(
|
|
|
+ struct ipa_lan_client_cntr_index *client,
|
|
|
+ struct wan_ioctl_query_per_client_stats *data,
|
|
|
+ struct ipa_ioc_flt_rt_query *query)
|
|
|
+{
|
|
|
+ int num_counters;
|
|
|
+
|
|
|
+ query->start_id = client->ul_cnt_idx;
|
|
|
+ query->end_id = client->dl_cnt_idx;
|
|
|
+
|
|
|
+ query->reset = data->reset_stats;
|
|
|
+ num_counters = query->end_id - query->start_id + 1;
|
|
|
+
|
|
|
+ if (num_counters != 2) {
|
|
|
+ IPAWANERR("Dont support more than 2 counter\n");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ IPAWANDBG(" Start/End %u/%u, num counters = %d\n",
|
|
|
+ query->start_id, query->end_id, num_counters);
|
|
|
+
|
|
|
+ query->stats = (uint64_t)kcalloc(
|
|
|
+ num_counters,
|
|
|
+ sizeof(struct ipa_flt_rt_stats),
|
|
|
+ GFP_KERNEL);
|
|
|
+ if (!query->stats) {
|
|
|
+ IPAERR("Failed to allocate memory for query stats\n");
|
|
|
+ return -ENOMEM;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (ipa_get_flt_rt_stats(query)) {
|
|
|
+ IPAERR("Failed to request stats from h/w\n");
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ IPAWANDBG("ul: bytes = %llu, pkts = %u, pkts_hash = %u\n",
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[0].num_bytes,
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[0].num_pkts,
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[0].num_pkts_hash);
|
|
|
+ IPAWANDBG("dl: bytes = %llu, pkts = %u, pkts_hash = %u\n",
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[1].num_bytes,
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[1].num_pkts,
|
|
|
+ ((struct ipa_flt_rt_stats *)query->stats)[1].num_pkts_hash);
|
|
|
+
|
|
|
+ return 0;
|
|
|
+}
|
|
|
+
|
|
|
/* rmnet_ipa3_set_lan_client_info() -
|
|
|
* @data - IOCTL data
|
|
|
*
|
|
@@ -4442,6 +4492,8 @@ int rmnet_ipa3_set_lan_client_info(
|
|
|
struct wan_ioctl_lan_client_info *data)
|
|
|
{
|
|
|
struct ipa_lan_client *lan_client = NULL;
|
|
|
+ struct ipa_lan_client_cntr_index
|
|
|
+ *client_index = NULL;
|
|
|
struct ipa_tether_device_info *teth_ptr = NULL;
|
|
|
|
|
|
|
|
@@ -4463,6 +4515,14 @@ int rmnet_ipa3_set_lan_client_info(
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
+ /* This should be done when allocation of hw fnr counters happens */
|
|
|
+ if (!(data->ul_cnt_idx > 0 &&
|
|
|
+ data->dl_cnt_idx == (data->ul_cnt_idx + 1))) {
|
|
|
+ IPAWANERR("Invalid counter indices %u, %u\n",
|
|
|
+ data->ul_cnt_idx, data->dl_cnt_idx);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
mutex_lock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
if (data->client_init) {
|
|
|
/* check if the client is already inited. */
|
|
@@ -4477,6 +4537,7 @@ int rmnet_ipa3_set_lan_client_info(
|
|
|
|
|
|
teth_ptr = &rmnet_ipa3_ctx->tether_device[data->device_type];
|
|
|
lan_client = &teth_ptr->lan_client[data->client_idx];
|
|
|
+ client_index = &teth_ptr->lan_client_indices[data->client_idx];
|
|
|
|
|
|
memcpy(lan_client->mac, data->mac, IPA_MAC_ADDR_SIZE);
|
|
|
|
|
@@ -4490,6 +4551,13 @@ int rmnet_ipa3_set_lan_client_info(
|
|
|
if (!rmnet_ipa3_ctx->tether_device[data->device_type].hdr_len)
|
|
|
rmnet_ipa3_ctx->tether_device[data->device_type].hdr_len =
|
|
|
data->hdr_len;
|
|
|
+ client_index->ul_cnt_idx = data->ul_cnt_idx;
|
|
|
+ client_index->dl_cnt_idx = data->dl_cnt_idx;
|
|
|
+
|
|
|
+ IPAWANDBG("Device type %d, ul/dl = %d/%d\n",
|
|
|
+ data->device_type,
|
|
|
+ data->ul_cnt_idx,
|
|
|
+ data->dl_cnt_idx);
|
|
|
|
|
|
lan_client->inited = true;
|
|
|
|
|
@@ -4857,8 +4925,6 @@ int rmnet_ipa3_query_per_client_stats(
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
-
|
|
|
-
|
|
|
#ifdef CONFIG_DEBUG_FS
|
|
|
static void rmnet_ipa_debugfs_init(void)
|
|
|
{
|
|
@@ -4913,6 +4979,145 @@ int ipa3_wwan_platform_driver_register(void)
|
|
|
}
|
|
|
EXPORT_SYMBOL(ipa3_wwan_platform_driver_register);
|
|
|
|
|
|
+int rmnet_ipa3_query_per_client_stats_v2(
|
|
|
+ struct wan_ioctl_query_per_client_stats *data)
|
|
|
+{
|
|
|
+ int lan_clnt_idx, i, j;
|
|
|
+ struct ipa_lan_client *lan_client = NULL;
|
|
|
+ struct ipa_lan_client_cntr_index
|
|
|
+ *lan_client_index = NULL;
|
|
|
+ struct ipa_tether_device_info *teth_ptr = NULL;
|
|
|
+ struct ipa_ioc_flt_rt_query query_f;
|
|
|
+ struct ipa_ioc_flt_rt_query *query = &query_f;
|
|
|
+ struct ipa_flt_rt_stats *fnr_stats = NULL;
|
|
|
+ int ret = 1;
|
|
|
+
|
|
|
+ /* Check if Device type is valid. */
|
|
|
+ if (data->device_type >= IPACM_MAX_CLIENT_DEVICE_TYPES ||
|
|
|
+ data->device_type < 0) {
|
|
|
+ IPAWANERR("Invalid Device type: %d\n", data->device_type);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Check if num_clients is valid. */
|
|
|
+ if (data->num_clients != IPA_MAX_NUM_HW_PATH_CLIENTS &&
|
|
|
+ data->num_clients != 1) {
|
|
|
+ IPAWANERR("Invalid number of clients: %d\n", data->num_clients);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ mutex_lock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+
|
|
|
+ /* Check if Source pipe is valid. */
|
|
|
+ if (rmnet_ipa3_ctx->tether_device
|
|
|
+ [data->device_type].ul_src_pipe == -1) {
|
|
|
+ IPAWANERR("Device not initialized: %d\n", data->device_type);
|
|
|
+ mutex_unlock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Check if we have clients connected. */
|
|
|
+ if (rmnet_ipa3_ctx->tether_device[data->device_type].num_clients == 0) {
|
|
|
+ IPAWANERR("No clients connected: %d\n", data->device_type);
|
|
|
+ mutex_unlock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (data->num_clients == 1) {
|
|
|
+ /* Check if the client info is valid.*/
|
|
|
+ lan_clnt_idx = rmnet_ipa3_get_lan_client_info(
|
|
|
+ data->device_type,
|
|
|
+ data->client_info[0].mac);
|
|
|
+ if (lan_clnt_idx < 0) {
|
|
|
+ IPAWANERR("Client info not available return.\n");
|
|
|
+ mutex_unlock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+ return -EINVAL;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ /* Max number of clients. */
|
|
|
+ /* Check if disconnect flag is set and
|
|
|
+ * see if all the clients info are cleared.
|
|
|
+ */
|
|
|
+ if (data->disconnect_clnt &&
|
|
|
+ rmnet_ipa3_check_any_client_inited(data->device_type)) {
|
|
|
+ IPAWANERR("CLient not inited. Try again.\n");
|
|
|
+ mutex_unlock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+ return -EAGAIN;
|
|
|
+ }
|
|
|
+ lan_clnt_idx = LAN_STATS_FOR_ALL_CLIENTS;
|
|
|
+ }
|
|
|
+
|
|
|
+ IPAWANDBG("Query stats for client index (0x%x)\n",
|
|
|
+ lan_clnt_idx);
|
|
|
+
|
|
|
+ teth_ptr = &rmnet_ipa3_ctx->tether_device[data->device_type];
|
|
|
+ lan_client = teth_ptr->lan_client;
|
|
|
+ lan_client_index = teth_ptr->lan_client_indices;
|
|
|
+
|
|
|
+ if (lan_clnt_idx == LAN_STATS_FOR_ALL_CLIENTS) {
|
|
|
+ i = 0;
|
|
|
+ j = IPA_MAX_NUM_HW_PATH_CLIENTS;
|
|
|
+ } else {
|
|
|
+ i = lan_clnt_idx;
|
|
|
+ j = i + 1;
|
|
|
+ }
|
|
|
+
|
|
|
+ for (; i < j; i++) {
|
|
|
+ if (!lan_client[i].inited && !data->disconnect_clnt)
|
|
|
+ continue;
|
|
|
+
|
|
|
+ IPAWANDBG("Client MAC %02x:%02x:%02x:%02x:%02x:%02x\n",
|
|
|
+ lan_client[i].mac[0],
|
|
|
+ lan_client[i].mac[1],
|
|
|
+ lan_client[i].mac[2],
|
|
|
+ lan_client[i].mac[3],
|
|
|
+ lan_client[i].mac[4],
|
|
|
+ lan_client[i].mac[5]);
|
|
|
+ IPAWANDBG("Lan client %d inited\n", i);
|
|
|
+ IPAWANDBG("Query stats ul/dl indices = %u/%u\n",
|
|
|
+ lan_client_index[i].ul_cnt_idx,
|
|
|
+ lan_client_index[i].dl_cnt_idx);
|
|
|
+ memset(query, 0, sizeof(query_f));
|
|
|
+ ret = rmnet_ipa_get_hw_fnr_stats_v2(&lan_client_index[i],
|
|
|
+ data, query);
|
|
|
+ if (ret) {
|
|
|
+ IPAWANERR("Failed: Client type %d, idx %d\n",
|
|
|
+ data->device_type, i);
|
|
|
+ kfree((void *)query->stats);
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+ fnr_stats = &((struct ipa_flt_rt_stats *)
|
|
|
+ query->stats)[0];
|
|
|
+ data->client_info[i].ipv4_tx_bytes =
|
|
|
+ fnr_stats->num_bytes;
|
|
|
+ fnr_stats = &((struct ipa_flt_rt_stats *)
|
|
|
+ query->stats)[1];
|
|
|
+ data->client_info[i].ipv4_rx_bytes =
|
|
|
+ fnr_stats->num_bytes;
|
|
|
+ memcpy(data->client_info[i].mac,
|
|
|
+ lan_client[i].mac,
|
|
|
+ IPA_MAC_ADDR_SIZE);
|
|
|
+
|
|
|
+ IPAWANDBG("Client ipv4_tx_bytes = %lu, ipv4_rx_bytes = %lu\n",
|
|
|
+ data->client_info[i].ipv4_tx_bytes,
|
|
|
+ data->client_info[i].ipv4_rx_bytes);
|
|
|
+
|
|
|
+ kfree((void *)query->stats);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Legacy per-client stats */
|
|
|
+ IPAWANDBG("Disconnect clnt: %s",
|
|
|
+ data->disconnect_clnt?"Yes":"No");
|
|
|
+
|
|
|
+ if (data->disconnect_clnt) {
|
|
|
+ rmnet_ipa3_delete_lan_client_info(data->device_type,
|
|
|
+ lan_clnt_idx);
|
|
|
+ }
|
|
|
+
|
|
|
+ mutex_unlock(&rmnet_ipa3_ctx->per_client_stats_guard);
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
int ipa3_wwan_init(void)
|
|
|
{
|
|
|
int i, j;
|