// SPDX-License-Identifier: GPL-2.0 /* * Copyright (C) 2020 MediaTek Inc. */ #include #include #include #include #include #include #include #include #include #include #include #include #include // MTK common IPI/MBOX #include #include #include "vcp.h" #include "vcp_hwvoter_dbg.h" #ifdef pr_fmt #undef pr_fmt #endif #define pr_fmt(fmt) "[hw_voter]: " fmt static unsigned char ipi_ackdata[16]; #if IS_ENABLED(CONFIG_PROC_FS) /********************************** * hw_voter debug ***********************************/ static int vcp_hw_voter_dbg_proc_show(struct seq_file *m, void *v) { return 0; } static ssize_t vcp_hw_voter_dbg_proc_write( struct file *file, const char __user *buffer, size_t count, loff_t *data) { char desc[64]; unsigned int len = 0; int ret = 0; int n; struct hwvoter_ipi_test_t ipi_data; len = (count < (sizeof(desc) - 1)) ? count : (sizeof(desc) - 1); if (copy_from_user(desc, buffer, len)) return 0; desc[len] = '\0'; pr_notice("%s: %s\n", __func__, desc); n = sscanf(desc, "%d %d %d %d %d", &ipi_data.type, &ipi_data.op, &ipi_data.clk_category, &ipi_data.clk_id, &ipi_data.val); if (n != 4 && n != 5) { pr_notice("invalid cmd length %d\n", n); return count; } ipi_data.cmd = HW_VOTER_DBG_CMD_TEST; ret = mtk_ipi_send_compl( &vcp_ipidev, IPI_OUT_C_VCP_HWVOTER_DEBUG, IPI_SEND_WAIT, &ipi_data, PIN_OUT_C_SIZE_HWVOTER, 500); if (ret != IPI_ACTION_DONE) pr_notice("[%s]: VCP send IPI failed - %d\n", __func__, ret); return count; } #define PROC_FOPS_RW(name) \ static int vcp_ ## name ## _proc_open(\ struct inode *inode, \ struct file *file) \ { \ return single_open(file, \ vcp_ ## name ## _proc_show, \ PDE_DATA(inode)); \ } \ static const struct proc_ops \ vcp_ ## name ## _proc_fops = {\ .proc_open = vcp_ ## name ## _proc_open, \ .proc_read = seq_read, \ .proc_lseek = seq_lseek, \ .proc_release = single_release, \ .proc_write = vcp_ ## name ## _proc_write, \ } #define PROC_FOPS_RO(name) \ static int vcp_ ## name ## _proc_open(\ struct inode *inode,\ struct file *file)\ {\ return single_open(file, \ vcp_ ## name ## _proc_show, \ PDE_DATA(inode)); \ } \ static const struct proc_ops \ vcp_ ## name ## _proc_fops = {\ .proc_open = vcp_ ## name ## _proc_open,\ .proc_read = seq_read,\ .proc_lseek = seq_lseek,\ .proc_release = single_release,\ } #define PROC_FOPS_WO(name) \ static int scp_ ## name ## _proc_open(\ struct inode *inode, \ struct file *file) \ { \ return single_open(file, \ vcp_ ## name ## _proc_show, \ PDE_DATA(inode)); \ } \ static const struct proc_ops \ vcp_ ## name ## _proc_fops = {\ .proc_open = vcp_ ## name ## _proc_open, \ .proc_lseek = seq_lseek, \ .proc_release = single_release, \ .proc_write = vcp_ ## name ## _proc_write, \ } #define PROC_ENTRY(name) {__stringify(name), &vcp_ ## name ## _proc_fops} PROC_FOPS_RW(hw_voter_dbg); int vcp_hw_voter_create_procfs(void) { struct proc_dir_entry *dir = NULL; int i, ret = 0; struct pentry { const char *name; const struct proc_ops *fops; }; const struct pentry entries[] = { PROC_ENTRY(hw_voter_dbg) }; dir = proc_mkdir("vcp_hw_voter", NULL); if (!dir) { pr_notice("fail to create /proc/vcp_hw_voter\n"); return -ENOMEM; } for (i = 0; i < ARRAY_SIZE(entries); i++) { if (!proc_create(entries[i].name, 0664, dir, entries[i].fops)) { pr_notice("%s: create /proc/vcp_hw_voter/%s failed\n", __func__, entries[i].name); ret = -ENOMEM; } } return ret; } #endif /* CONFIG_PROC_FS */ int vcp_hw_voter_dbg_init(void) { #if IS_ENABLED(CONFIG_PROC_FS) int ret = 0; pr_notice("call %s\n", __func__); ret = mtk_ipi_register( &vcp_ipidev, IPI_OUT_C_VCP_HWVOTER_DEBUG, NULL, NULL, (void *)ipi_ackdata); if (ret) { pr_notice("vcp hwvoter mtk_ipi_register fail, %d\n", ret); WARN_ON(1); } else pr_notice("vcp hwvoter mtk_ipi_register done\n"); if (vcp_hw_voter_create_procfs()) { pr_notice("vcp_hw_voter_create_procfs fail, %d\n", ret); WARN_ON(1); } else pr_notice("vcp_hw_voter_create_procfs done\n"); #else pr_notice("no %s due to CONFIG_PROC_FS not defined\n", __func__); #endif return 0; }