| /* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 and |
| * only version 2 as published by the Free Software Foundation. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| |
| #ifdef CONFIG_DEBUG_FS |
| |
| #include <linux/debugfs.h> |
| #include "ipa_i.h" |
| |
| |
| #define IPA_MAX_MSG_LEN 1024 |
| static struct dentry *dent; |
| static struct dentry *dfile_gen_reg; |
| static struct dentry *dfile_ep_reg; |
| static struct dentry *dfile_hdr; |
| static struct dentry *dfile_ip4_rt; |
| static struct dentry *dfile_ip6_rt; |
| static struct dentry *dfile_ip4_flt; |
| static struct dentry *dfile_ip6_flt; |
| static char dbg_buff[IPA_MAX_MSG_LEN]; |
| static s8 ep_reg_idx; |
| |
| static ssize_t ipa_read_gen_reg(struct file *file, char __user *ubuf, |
| size_t count, loff_t *ppos) |
| { |
| int nbytes; |
| |
| if (ipa_ctx->ipa_hw_type == IPA_HW_v1_0) |
| nbytes = scnprintf(dbg_buff, IPA_MAX_MSG_LEN, |
| "IPA_VERSION=0x%x\n" |
| "IPA_COMP_HW_VERSION=0x%x\n" |
| "IPA_ROUTE=0x%x\n" |
| "IPA_FILTER=0x%x\n" |
| "IPA_SHARED_MEM_SIZE=0x%x\n" |
| "IPA_HEAD_OF_LINE_BLOCK_EN=0x%x\n", |
| ipa_read_reg(ipa_ctx->mmio, IPA_VERSION_OFST), |
| ipa_read_reg(ipa_ctx->mmio, IPA_COMP_HW_VERSION_OFST), |
| ipa_read_reg(ipa_ctx->mmio, IPA_ROUTE_OFST_v1), |
| ipa_read_reg(ipa_ctx->mmio, IPA_FILTER_OFST_v1), |
| ipa_read_reg(ipa_ctx->mmio, |
| IPA_SHARED_MEM_SIZE_OFST_v1), |
| ipa_read_reg(ipa_ctx->mmio, |
| IPA_HEAD_OF_LINE_BLOCK_EN_OFST) |
| ); |
| else |
| nbytes = scnprintf(dbg_buff, IPA_MAX_MSG_LEN, |
| "IPA_VERSION=0x%x\n" |
| "IPA_COMP_HW_VERSION=0x%x\n" |
| "IPA_ROUTE=0x%x\n" |
| "IPA_FILTER=0x%x\n" |
| "IPA_SHARED_MEM_SIZE=0x%x\n", |
| ipa_read_reg(ipa_ctx->mmio, IPA_VERSION_OFST), |
| ipa_read_reg(ipa_ctx->mmio, IPA_COMP_HW_VERSION_OFST), |
| ipa_read_reg(ipa_ctx->mmio, IPA_ROUTE_OFST_v2), |
| ipa_read_reg(ipa_ctx->mmio, IPA_FILTER_OFST_v2), |
| ipa_read_reg(ipa_ctx->mmio, IPA_SHARED_MEM_SIZE_OFST_v2) |
| ); |
| |
| return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, nbytes); |
| } |
| |
| static ssize_t ipa_write_ep_reg(struct file *file, const char __user *buf, |
| size_t count, loff_t *ppos) |
| { |
| unsigned long missing; |
| s8 option = 0; |
| |
| if (sizeof(dbg_buff) < count + 1) |
| return -EFAULT; |
| |
| missing = copy_from_user(dbg_buff, buf, count); |
| if (missing) |
| return -EFAULT; |
| |
| dbg_buff[count] = '\0'; |
| if (kstrtos8(dbg_buff, 0, &option)) |
| return -EFAULT; |
| |
| if (option >= IPA_NUM_PIPES) { |
| IPAERR("bad pipe specified %u\n", option); |
| return count; |
| } |
| |
| ep_reg_idx = option; |
| |
| return count; |
| } |
| |
| static ssize_t ipa_read_ep_reg(struct file *file, char __user *ubuf, |
| size_t count, loff_t *ppos) |
| { |
| int nbytes; |
| int i; |
| int start_idx; |
| int end_idx; |
| int size = 0; |
| int ret; |
| loff_t pos; |
| |
| /* negative ep_reg_idx means all registers */ |
| if (ep_reg_idx < 0) { |
| start_idx = 0; |
| end_idx = IPA_NUM_PIPES; |
| } else { |
| start_idx = ep_reg_idx; |
| end_idx = start_idx + 1; |
| } |
| pos = *ppos; |
| for (i = start_idx; i < end_idx; i++) { |
| |
| if (ipa_ctx->ipa_hw_type == IPA_HW_v1_0) { |
| nbytes = scnprintf(dbg_buff, IPA_MAX_MSG_LEN, |
| "IPA_ENDP_INIT_NAT_%u=0x%x\n" |
| "IPA_ENDP_INIT_HDR_%u=0x%x\n" |
| "IPA_ENDP_INIT_MODE_%u=0x%x\n" |
| "IPA_ENDP_INIT_AGGR_%u=0x%x\n" |
| "IPA_ENDP_INIT_ROUTE_%u=0x%x\n", |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_NAT_n_OFST_v1(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_HDR_n_OFST_v1(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_MODE_n_OFST_v1(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_AGGR_n_OFST_v1(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_ROUTE_n_OFST_v1(i))); |
| } else { |
| nbytes = scnprintf(dbg_buff, IPA_MAX_MSG_LEN, |
| "IPA_ENDP_INIT_NAT_%u=0x%x\n" |
| "IPA_ENDP_INIT_HDR_%u=0x%x\n" |
| "IPA_ENDP_INIT_MODE_%u=0x%x\n" |
| "IPA_ENDP_INIT_AGGR_%u=0x%x\n" |
| "IPA_ENDP_INIT_ROUTE_%u=0x%x\n", |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_NAT_n_OFST_v2(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_HDR_n_OFST_v2(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_MODE_n_OFST_v2(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_AGGR_n_OFST_v2(i)), |
| i, ipa_read_reg(ipa_ctx->mmio, |
| IPA_ENDP_INIT_ROUTE_n_OFST_v2(i))); |
| } |
| |
| *ppos = pos; |
| ret = simple_read_from_buffer(ubuf, count, ppos, dbg_buff, |
| nbytes); |
| if (ret < 0) |
| return ret; |
| |
| size += ret; |
| ubuf += nbytes; |
| count -= nbytes; |
| } |
| |
| *ppos = pos + size; |
| return size; |
| } |
| |
| static ssize_t ipa_read_hdr(struct file *file, char __user *ubuf, size_t count, |
| loff_t *ppos) |
| { |
| int nbytes = 0; |
| int cnt = 0; |
| int i = 0; |
| struct ipa_hdr_entry *entry; |
| |
| mutex_lock(&ipa_ctx->lock); |
| list_for_each_entry(entry, &ipa_ctx->hdr_tbl.head_hdr_entry_list, |
| link) { |
| nbytes = scnprintf(dbg_buff + cnt, IPA_MAX_MSG_LEN - cnt, |
| "name:%s len=%d ref=%d partial=%d lcl=%d ofst=%u ", |
| entry->name, |
| entry->hdr_len, entry->ref_cnt, |
| entry->is_partial, |
| ipa_ctx->hdr_tbl_lcl, |
| entry->offset_entry->offset >> 2); |
| for (i = 0; i < entry->hdr_len; i++) { |
| scnprintf(dbg_buff + cnt + nbytes + i * 2, |
| IPA_MAX_MSG_LEN - cnt - nbytes - i * 2, |
| "%02x", entry->hdr[i]); |
| } |
| scnprintf(dbg_buff + cnt + nbytes + entry->hdr_len * 2, |
| IPA_MAX_MSG_LEN - cnt - nbytes - entry->hdr_len * 2, |
| "\n"); |
| cnt += nbytes + entry->hdr_len * 2 + 1; |
| } |
| mutex_unlock(&ipa_ctx->lock); |
| |
| return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, cnt); |
| } |
| |
| static int ipa_attrib_dump(char *buff, size_t sz, |
| struct ipa_rule_attrib *attrib, enum ipa_ip_type ip) |
| { |
| int nbytes = 0; |
| int cnt = 0; |
| uint32_t addr[4]; |
| uint32_t mask[4]; |
| int i; |
| |
| if (attrib->attrib_mask & IPA_FLT_TOS) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "tos:%d ", |
| attrib->u.v4.tos); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_PROTOCOL) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "protocol:%d ", |
| attrib->u.v4.protocol); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_SRC_ADDR) { |
| if (ip == IPA_IP_v4) { |
| addr[0] = htonl(attrib->u.v4.src_addr); |
| mask[0] = htonl(attrib->u.v4.src_addr_mask); |
| nbytes = scnprintf(buff + cnt, sz - cnt, |
| "src_addr:%pI4 src_addr_mask:%pI4 ", |
| addr + 0, mask + 0); |
| cnt += nbytes; |
| } else if (ip == IPA_IP_v6) { |
| for (i = 0; i < 4; i++) { |
| addr[i] = htonl(attrib->u.v6.src_addr[i]); |
| mask[i] = htonl(attrib->u.v6.src_addr_mask[i]); |
| } |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, |
| "src_addr:%pI6 src_addr_mask:%pI6 ", |
| addr + 0, mask + 0); |
| cnt += nbytes; |
| } else { |
| WARN_ON(1); |
| } |
| } |
| if (attrib->attrib_mask & IPA_FLT_DST_ADDR) { |
| if (ip == IPA_IP_v4) { |
| addr[0] = htonl(attrib->u.v4.dst_addr); |
| mask[0] = htonl(attrib->u.v4.dst_addr_mask); |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, |
| "dst_addr:%pI4 dst_addr_mask:%pI4 ", |
| addr + 0, mask + 0); |
| cnt += nbytes; |
| } else if (ip == IPA_IP_v6) { |
| for (i = 0; i < 4; i++) { |
| addr[i] = htonl(attrib->u.v6.dst_addr[i]); |
| mask[i] = htonl(attrib->u.v6.dst_addr_mask[i]); |
| } |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, |
| "dst_addr:%pI6 dst_addr_mask:%pI6 ", |
| addr + 0, mask + 0); |
| cnt += nbytes; |
| } else { |
| WARN_ON(1); |
| } |
| } |
| if (attrib->attrib_mask & IPA_FLT_SRC_PORT_RANGE) { |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, "src_port_range:%u %u ", |
| attrib->src_port_lo, |
| attrib->src_port_hi); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_DST_PORT_RANGE) { |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, "dst_port_range:%u %u ", |
| attrib->dst_port_lo, |
| attrib->dst_port_hi); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_TYPE) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "type:%d ", |
| attrib->type); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_CODE) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "code:%d ", |
| attrib->code); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_SPI) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "spi:%x ", |
| attrib->spi); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_SRC_PORT) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "src_port:%u ", |
| attrib->src_port); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_DST_PORT) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "dst_port:%u ", |
| attrib->dst_port); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_TC) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "tc:%d ", |
| attrib->u.v6.tc); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_FLOW_LABEL) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "flow_label:%x ", |
| attrib->u.v6.flow_label); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_NEXT_HDR) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "next_hdr:%d ", |
| attrib->u.v6.next_hdr); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_META_DATA) { |
| nbytes = |
| scnprintf(buff + cnt, sz - cnt, |
| "metadata:%x metadata_mask:%x", |
| attrib->meta_data, attrib->meta_data_mask); |
| cnt += nbytes; |
| } |
| if (attrib->attrib_mask & IPA_FLT_FRAGMENT) { |
| nbytes = scnprintf(buff + cnt, sz - cnt, "frg "); |
| cnt += nbytes; |
| } |
| nbytes = scnprintf(buff + cnt, sz - cnt, "\n"); |
| cnt += nbytes; |
| |
| return cnt; |
| } |
| |
| static int ipa_open_dbg(struct inode *inode, struct file *file) |
| { |
| file->private_data = inode->i_private; |
| return 0; |
| } |
| |
| static ssize_t ipa_read_rt(struct file *file, char __user *ubuf, size_t count, |
| loff_t *ppos) |
| { |
| int nbytes = 0; |
| int cnt = 0; |
| int i = 0; |
| struct ipa_rt_tbl *tbl; |
| struct ipa_rt_entry *entry; |
| struct ipa_rt_tbl_set *set; |
| enum ipa_ip_type ip = (enum ipa_ip_type)file->private_data; |
| u32 hdr_ofst; |
| |
| set = &ipa_ctx->rt_tbl_set[ip]; |
| |
| mutex_lock(&ipa_ctx->lock); |
| list_for_each_entry(tbl, &set->head_rt_tbl_list, link) { |
| i = 0; |
| list_for_each_entry(entry, &tbl->head_rt_rule_list, link) { |
| if (entry->hdr) |
| hdr_ofst = entry->hdr->offset_entry->offset; |
| else |
| hdr_ofst = 0; |
| nbytes = scnprintf(dbg_buff + cnt, |
| IPA_MAX_MSG_LEN - cnt, |
| "tbl_idx:%d tbl_name:%s tbl_ref:%u rule_idx:%d dst:%d ep:%d S:%u hdr_ofst[words]:%u attrib_mask:%08x ", |
| entry->tbl->idx, entry->tbl->name, |
| entry->tbl->ref_cnt, i, entry->rule.dst, |
| ipa_get_ep_mapping(ipa_ctx->mode, |
| entry->rule.dst), |
| !ipa_ctx->hdr_tbl_lcl, |
| hdr_ofst >> 2, |
| entry->rule.attrib.attrib_mask); |
| cnt += nbytes; |
| cnt += ipa_attrib_dump(dbg_buff + cnt, |
| IPA_MAX_MSG_LEN - cnt, |
| &entry->rule.attrib, |
| ip); |
| i++; |
| } |
| } |
| mutex_unlock(&ipa_ctx->lock); |
| |
| return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, cnt); |
| } |
| |
| static ssize_t ipa_read_flt(struct file *file, char __user *ubuf, size_t count, |
| loff_t *ppos) |
| { |
| int nbytes = 0; |
| int cnt = 0; |
| int i; |
| int j; |
| struct ipa_flt_tbl *tbl; |
| struct ipa_flt_entry *entry; |
| enum ipa_ip_type ip = (enum ipa_ip_type)file->private_data; |
| struct ipa_rt_tbl *rt_tbl; |
| |
| tbl = &ipa_ctx->glob_flt_tbl[ip]; |
| mutex_lock(&ipa_ctx->lock); |
| i = 0; |
| list_for_each_entry(entry, &tbl->head_flt_rule_list, link) { |
| rt_tbl = (struct ipa_rt_tbl *)entry->rule.rt_tbl_hdl; |
| nbytes = scnprintf(dbg_buff + cnt, IPA_MAX_MSG_LEN - cnt, |
| "ep_idx:global rule_idx:%d act:%d rt_tbl_idx:%d attrib_mask:%08x ", |
| i, entry->rule.action, rt_tbl->idx, |
| entry->rule.attrib.attrib_mask); |
| cnt += nbytes; |
| cnt += ipa_attrib_dump(dbg_buff + cnt, IPA_MAX_MSG_LEN - cnt, |
| &entry->rule.attrib, ip); |
| i++; |
| } |
| |
| for (j = 0; j < IPA_NUM_PIPES; j++) { |
| tbl = &ipa_ctx->flt_tbl[j][ip]; |
| i = 0; |
| list_for_each_entry(entry, &tbl->head_flt_rule_list, link) { |
| rt_tbl = (struct ipa_rt_tbl *)entry->rule.rt_tbl_hdl; |
| nbytes = scnprintf(dbg_buff + cnt, |
| IPA_MAX_MSG_LEN - cnt, |
| "ep_idx:%d rule_idx:%d act:%d rt_tbl_idx:%d attrib_mask:%08x ", |
| j, i, entry->rule.action, rt_tbl->idx, |
| entry->rule.attrib.attrib_mask); |
| cnt += nbytes; |
| cnt += |
| ipa_attrib_dump(dbg_buff + cnt, |
| IPA_MAX_MSG_LEN - cnt, |
| &entry->rule.attrib, |
| ip); |
| i++; |
| } |
| } |
| mutex_unlock(&ipa_ctx->lock); |
| |
| return simple_read_from_buffer(ubuf, count, ppos, dbg_buff, cnt); |
| } |
| |
| const struct file_operations ipa_gen_reg_ops = { |
| .read = ipa_read_gen_reg, |
| }; |
| |
| const struct file_operations ipa_ep_reg_ops = { |
| .read = ipa_read_ep_reg, |
| .write = ipa_write_ep_reg, |
| }; |
| |
| const struct file_operations ipa_hdr_ops = { |
| .read = ipa_read_hdr, |
| }; |
| |
| const struct file_operations ipa_rt_ops = { |
| .read = ipa_read_rt, |
| .open = ipa_open_dbg, |
| }; |
| |
| const struct file_operations ipa_flt_ops = { |
| .read = ipa_read_flt, |
| .open = ipa_open_dbg, |
| }; |
| |
| void ipa_debugfs_init(void) |
| { |
| const mode_t read_only_mode = S_IRUSR | S_IRGRP | S_IROTH; |
| const mode_t read_write_mode = S_IRUSR | S_IRGRP | S_IROTH | |
| S_IWUSR | S_IWGRP | S_IWOTH; |
| |
| dent = debugfs_create_dir("ipa", 0); |
| if (IS_ERR(dent)) { |
| IPAERR("fail to create folder in debug_fs.\n"); |
| return; |
| } |
| |
| dfile_gen_reg = debugfs_create_file("gen_reg", read_only_mode, dent, 0, |
| &ipa_gen_reg_ops); |
| if (!dfile_gen_reg || IS_ERR(dfile_gen_reg)) { |
| IPAERR("fail to create file for debug_fs gen_reg\n"); |
| goto fail; |
| } |
| |
| dfile_ep_reg = debugfs_create_file("ep_reg", read_write_mode, dent, 0, |
| &ipa_ep_reg_ops); |
| if (!dfile_ep_reg || IS_ERR(dfile_ep_reg)) { |
| IPAERR("fail to create file for debug_fs ep_reg\n"); |
| goto fail; |
| } |
| |
| dfile_hdr = debugfs_create_file("hdr", read_only_mode, dent, 0, |
| &ipa_hdr_ops); |
| if (!dfile_hdr || IS_ERR(dfile_hdr)) { |
| IPAERR("fail to create file for debug_fs hdr\n"); |
| goto fail; |
| } |
| |
| dfile_ip4_rt = debugfs_create_file("ip4_rt", read_only_mode, dent, |
| (void *)IPA_IP_v4, &ipa_rt_ops); |
| if (!dfile_ip4_rt || IS_ERR(dfile_ip4_rt)) { |
| IPAERR("fail to create file for debug_fs ip4 rt\n"); |
| goto fail; |
| } |
| |
| dfile_ip6_rt = debugfs_create_file("ip6_rt", read_only_mode, dent, |
| (void *)IPA_IP_v6, &ipa_rt_ops); |
| if (!dfile_ip6_rt || IS_ERR(dfile_ip6_rt)) { |
| IPAERR("fail to create file for debug_fs ip6:w" " rt\n"); |
| goto fail; |
| } |
| |
| dfile_ip4_flt = debugfs_create_file("ip4_flt", read_only_mode, dent, |
| (void *)IPA_IP_v4, &ipa_flt_ops); |
| if (!dfile_ip4_flt || IS_ERR(dfile_ip4_flt)) { |
| IPAERR("fail to create file for debug_fs ip4 flt\n"); |
| goto fail; |
| } |
| |
| dfile_ip6_flt = debugfs_create_file("ip6_flt", read_only_mode, dent, |
| (void *)IPA_IP_v6, &ipa_flt_ops); |
| if (!dfile_ip6_flt || IS_ERR(dfile_ip6_flt)) { |
| IPAERR("fail to create file for debug_fs ip6 flt\n"); |
| goto fail; |
| } |
| |
| return; |
| |
| fail: |
| debugfs_remove_recursive(dent); |
| } |
| |
| void ipa_debugfs_remove(void) |
| { |
| if (IS_ERR(dent)) { |
| IPAERR("ipa_debugfs_remove: folder was not created.\n"); |
| return; |
| } |
| debugfs_remove_recursive(dent); |
| } |
| |
| #else /* !CONFIG_DEBUG_FS */ |
| void ipa_debugfs_init(void) {} |
| void ipa_debugfs_remove(void) {} |
| #endif |
| |