kernel_samsung_a34x-permissive/drivers/gpu/drm/mediatek/mtk_debug.c
2024-04-28 15:51:13 +02:00

3122 lines
77 KiB
C

/* SPDX-License-Identifier: GPL-2.0 */
/*
* Copyright (c) 2019 MediaTek Inc.
*/
#include <linux/syscalls.h>
#include <linux/string.h>
#include <linux/time.h>
#include <linux/wait.h>
#if IS_ENABLED(CONFIG_DEBUG_FS)
#include <linux/debugfs.h>
#endif
#if IS_ENABLED(CONFIG_PROC_FS)
#include <linux/proc_fs.h>
#endif
#include <linux/sched/clock.h>
#include <linux/of_address.h>
#include <drm/drmP.h>
#include <drm/drm_mipi_dsi.h>
#include "mtk_dump.h"
#include "mtk_debug.h"
#include "mtk_drm_crtc.h"
#include "mtk_drm_mmp.h"
#include "mtk_drm_drv.h"
#include "mtk_drm_assert.h"
#include "mtk_drm_helper.h"
#include "mtk_layering_rule.h"
#include "mtk_drm_lowpower.h"
#if defined(CONFIG_MTK_IOMMU_V2)
#include "mt_iommu.h"
#include "mtk_iommu_ext.h"
#endif
#include "mtk_drm_gem.h"
#include "mtk_drm_fb.h"
#include "mtk_drm_fbdev.h"
#include "mtk_disp_aal.h"
#ifdef CONFIG_MTK_HDMI_SUPPORT
#include "mtk_dp_debug.h"
#endif
#include "mtk_drm_arr.h"
#include "mtk_drm_graphics_base.h"
#ifdef CONFIG_MTK_MT6382_BDG
#include "mtk_disp_bdg.h"
#endif
#include "mtk_notify.h"
#define DISP_REG_CONFIG_MMSYS_CG_SET(idx) (0x104 + 0x10 * (idx))
#define DISP_REG_CONFIG_MMSYS_CG_CLR(idx) (0x108 + 0x10 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_EN(idx) (0x200 + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_RST(idx) (0x204 + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_CON0(idx) (0x208 + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_CON1(idx) (0x20c + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_RD_ADDR(idx) (0x210 + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_WR_ADDR(idx) (0x214 + 0x20 * (idx))
#define DISP_REG_CONFIG_DISP_FAKE_ENG_STATE(idx) (0x218 + 0x20 * (idx))
#define DISP_REG_CONFIG_RDMA_SHARE_SRAM_CON (0x654)
#define DISP_RDMA_FAKE_SMI_SEL(idx) (BIT(4 + idx))
#define SMI_LARB_VC_PRI_MODE (0x020)
#define SMI_LARB_NON_SEC_CON(port) (0x380 + 4 * (port))
#define GET_M4U_PORT 0x1F
#define MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH 128
#define INIT_TID 1
#if IS_ENABLED(CONFIG_DEBUG_FS)
static struct dentry *mtkfb_dbgfs;
#endif
#if IS_ENABLED(CONFIG_PROC_FS)
static struct proc_dir_entry *mtkfb_procfs;
static struct proc_dir_entry *disp_lowpower_proc;
static struct proc_dir_entry *mtkfb_debug_procfs;
#endif
static struct drm_device *drm_dev;
bool g_mobile_log;
bool g_fence_log;
bool g_irq_log;
bool g_detail_log;
bool g_trace_log;
unsigned int mipi_volt;
unsigned int disp_met_en;
static unsigned int m_old_pq_persist_property[32];
unsigned int m_new_pq_persist_property[32];
int gCaptureOVLEn;
int gCaptureWDMAEn;
int gCapturePriLayerDownX = 10;
int gCapturePriLayerDownY = 10;
u64 vfp_backup;
int hwc_pid;
static atomic_t lfr_dbg;
static atomic_t lfr_params;
static struct completion cwb_cmp;
struct logger_buffer {
char **buffer_ptr;
unsigned int len;
unsigned int id;
const unsigned int cnt;
const unsigned int size;
};
static DEFINE_SPINLOCK(dprec_logger_spinlock);
static char **err_buffer;
static char **fence_buffer;
static char **dbg_buffer;
static char **dump_buffer;
static char **status_buffer;
static struct logger_buffer dprec_logger_buffer[DPREC_LOGGER_PR_NUM] = {
{0, 0, 0, ERROR_BUFFER_COUNT, LOGGER_BUFFER_SIZE},
{0, 0, 0, FENCE_BUFFER_COUNT, LOGGER_BUFFER_SIZE},
{0, 0, 0, DEBUG_BUFFER_COUNT, LOGGER_BUFFER_SIZE},
{0, 0, 0, DUMP_BUFFER_COUNT, LOGGER_BUFFER_SIZE},
{0, 0, 0, STATUS_BUFFER_COUNT, LOGGER_BUFFER_SIZE},
};
static bool is_buffer_init;
static char *debug_buffer;
static int draw_RGBA8888_buffer(char *va, int w, int h,
char r, char g, char b, char a)
{
int i, j;
int Bpp = mtk_get_format_bpp(DRM_FORMAT_RGBA8888);
for (i = 0; i < h; i++)
for (j = 0; j < w; j++) {
int x = j * Bpp + i * w * Bpp;
va[x++] = a;
va[x++] = b;
va[x++] = g;
va[x++] = r;
}
return 0;
}
static int prepare_fake_layer_buffer(struct drm_crtc *crtc)
{
unsigned int i;
size_t size;
struct mtk_drm_gem_obj *mtk_gem;
struct drm_mode_fb_cmd2 mode = { 0 };
struct mtk_drm_crtc *mtk_crtc = to_mtk_crtc(crtc);
struct mtk_drm_fake_layer *fake_layer = &mtk_crtc->fake_layer;
if (fake_layer->init)
return 0;
mode.width = crtc->state->adjusted_mode.hdisplay;
mode.height = crtc->state->adjusted_mode.vdisplay;
mode.pixel_format = DRM_FORMAT_RGBA8888;
mode.pitches[0] = mode.width
* mtk_get_format_bpp(mode.pixel_format);
size = mode.width * mode.height
* mtk_get_format_bpp(mode.pixel_format);
for (i = 0; i < PRIMARY_OVL_PHY_LAYER_NR; i++) {
mtk_gem = mtk_drm_gem_create(crtc->dev, size, true);
draw_RGBA8888_buffer(mtk_gem->kvaddr, mode.width, mode.height,
(!((i + 0) % 3)) * 255 / (i / 3 + 1),
(!((i + 1) % 3)) * 255 / (i / 3 + 1),
(!((i + 2) % 3)) * 255 / (i / 3 + 1), 100);
fake_layer->fake_layer_buf[i] =
mtk_drm_framebuffer_create(crtc->dev, &mode,
&mtk_gem->base);
}
fake_layer->init = true;
DDPMSG("%s init done\n", __func__);
return 0;
}
static unsigned long long get_current_time_us(void)
{
unsigned long long time = sched_clock();
struct timeval t;
/* return do_div(time,1000); */
return time;
do_gettimeofday(&t);
return (t.tv_sec & 0xFFF) * 1000000 + t.tv_usec;
}
static char *_logger_pr_type_spy(enum DPREC_LOGGER_PR_TYPE type)
{
switch (type) {
case DPREC_LOGGER_ERROR:
return "error";
case DPREC_LOGGER_FENCE:
return "fence";
case DPREC_LOGGER_DEBUG:
return "dbg";
case DPREC_LOGGER_DUMP:
return "dump";
case DPREC_LOGGER_STATUS:
return "status";
default:
return "unknown";
}
}
static void init_log_buffer(void)
{
int i, buf_size, buf_idx;
char *temp_buf;
if (is_buffer_init)
return;
/*1. Allocate Error, Fence, Debug and Dump log buffer slot*/
err_buffer = kzalloc(sizeof(char *) * ERROR_BUFFER_COUNT, GFP_KERNEL);
if (!err_buffer)
goto err;
fence_buffer = kzalloc(sizeof(char *) * FENCE_BUFFER_COUNT, GFP_KERNEL);
if (!fence_buffer)
goto err;
dbg_buffer = kzalloc(sizeof(char *) * DEBUG_BUFFER_COUNT, GFP_KERNEL);
if (!dbg_buffer)
goto err;
dump_buffer = kzalloc(sizeof(char *) * DUMP_BUFFER_COUNT, GFP_KERNEL);
if (!dump_buffer)
goto err;
status_buffer = kzalloc(sizeof(char *) * DUMP_BUFFER_COUNT, GFP_KERNEL);
if (!status_buffer)
goto err;
/*2. Allocate log ring buffer.*/
buf_size = sizeof(char) * (DEBUG_BUFFER_SIZE - 4096);
temp_buf = kzalloc(buf_size, GFP_KERNEL);
if (!temp_buf)
goto err;
/*3. Dispatch log ring buffer to each buffer slot*/
buf_idx = 0;
for (i = 0; i < ERROR_BUFFER_COUNT; i++) {
err_buffer[i] = (temp_buf + buf_idx * LOGGER_BUFFER_SIZE);
buf_idx++;
}
dprec_logger_buffer[0].buffer_ptr = err_buffer;
for (i = 0; i < FENCE_BUFFER_COUNT; i++) {
fence_buffer[i] = (temp_buf + buf_idx * LOGGER_BUFFER_SIZE);
buf_idx++;
}
dprec_logger_buffer[1].buffer_ptr = fence_buffer;
for (i = 0; i < DEBUG_BUFFER_COUNT; i++) {
dbg_buffer[i] = (temp_buf + buf_idx * LOGGER_BUFFER_SIZE);
buf_idx++;
}
dprec_logger_buffer[2].buffer_ptr = dbg_buffer;
for (i = 0; i < DUMP_BUFFER_COUNT; i++) {
dump_buffer[i] = (temp_buf + buf_idx * LOGGER_BUFFER_SIZE);
buf_idx++;
}
dprec_logger_buffer[3].buffer_ptr = dump_buffer;
for (i = 0; i < STATUS_BUFFER_COUNT; i++) {
status_buffer[i] = (temp_buf + buf_idx * LOGGER_BUFFER_SIZE);
buf_idx++;
}
dprec_logger_buffer[4].buffer_ptr = status_buffer;
is_buffer_init = true;
DDPINFO("[DISP]%s success\n", __func__);
return;
err:
DDPPR_ERR("[DISP]%s: log buffer allocation fail\n", __func__);
}
int mtk_dprec_logger_pr(unsigned int type, char *fmt, ...)
{
int n = 0;
unsigned long flags = 0;
uint64_t time = get_current_time_us();
unsigned long rem_nsec;
char **buf_arr;
char *buf = NULL;
int len = 0;
if (type >= DPREC_LOGGER_PR_NUM)
return -1;
if (!is_buffer_init)
return -1;
spin_lock_irqsave(&dprec_logger_spinlock, flags);
if (dprec_logger_buffer[type].len < 128) {
dprec_logger_buffer[type].id++;
dprec_logger_buffer[type].id = dprec_logger_buffer[type].id %
dprec_logger_buffer[type].cnt;
dprec_logger_buffer[type].len = dprec_logger_buffer[type].size;
}
buf_arr = dprec_logger_buffer[type].buffer_ptr;
buf = buf_arr[dprec_logger_buffer[type].id] +
dprec_logger_buffer[type].size - dprec_logger_buffer[type].len;
len = dprec_logger_buffer[type].len;
if (buf) {
va_list args;
rem_nsec = do_div(time, 1000000000);
n += snprintf(buf + n, len - n, "[%5lu.%06lu]",
(unsigned long)time, rem_nsec / 1000);
va_start(args, fmt);
n += vscnprintf(buf + n, len - n, fmt, args);
va_end(args);
}
dprec_logger_buffer[type].len -= n;
spin_unlock_irqrestore(&dprec_logger_spinlock, flags);
return n;
}
int mtk_dprec_logger_get_buf(enum DPREC_LOGGER_PR_TYPE type, char *stringbuf,
int len)
{
int n = 0;
int i;
char **buf_arr;
int c;
if (type >= DPREC_LOGGER_PR_NUM || type < 0 || len < 0) {
DDPPR_ERR("%s invalid DPREC_LOGGER_PR_TYPE\n", __func__);
return 0;
}
if (!is_buffer_init)
return 0;
c = dprec_logger_buffer[type].id;
buf_arr = dprec_logger_buffer[type].buffer_ptr;
for (i = 0; i < dprec_logger_buffer[type].cnt; i++) {
c++;
c %= dprec_logger_buffer[type].cnt;
n += scnprintf(stringbuf + n, len - n,
"dprec log buffer[%s][%d]\n",
_logger_pr_type_spy(type), c);
n += scnprintf(stringbuf + n, len - n, "%s\n", buf_arr[c]);
}
return n;
}
extern int mtk_drm_setbacklight(struct drm_crtc *crtc, unsigned int level);
int mtkfb_set_backlight_level(unsigned int level)
{
struct drm_crtc *crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return 0;
}
mtk_drm_setbacklight(crtc, level);
return 0;
}
EXPORT_SYMBOL(mtkfb_set_backlight_level);
int mtkfb_set_aod_backlight_level(unsigned int level)
{
struct drm_crtc *crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
pr_info("find crtc fail\n");
return 0;
}
mtk_drm_aod_setbacklight(crtc, level);
return 0;
}
EXPORT_SYMBOL(mtkfb_set_aod_backlight_level);
void mtk_disp_mipi_ccci_callback(unsigned int en, unsigned int usrdata)
{
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc_mipi_freq_switch(crtc, en, usrdata);
return;
}
EXPORT_SYMBOL(mtk_disp_mipi_ccci_callback);
void mtk_disp_osc_ccci_callback(unsigned int en, unsigned int usrdata)
{
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc_osc_freq_switch(crtc, en, usrdata);
return;
}
EXPORT_SYMBOL(mtk_disp_osc_ccci_callback);
int display_enter_tui(void)
{
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return -1;
}
return mtk_crtc_enter_tui(crtc);
}
EXPORT_SYMBOL(display_enter_tui);
int display_exit_tui(void)
{
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return -1;
}
return mtk_crtc_exit_tui(crtc);
}
EXPORT_SYMBOL(display_exit_tui);
static int debug_get_info(unsigned char *stringbuf, int buf_len)
{
int n = 0;
struct mtk_drm_private *private;
if (!drm_dev) {
DDPPR_ERR("%s:%d, drm_dev is NULL\n",
__func__, __LINE__);
return -1;
}
if (!drm_dev->dev_private) {
DDPPR_ERR("%s:%d, drm_dev->dev_private is NULL\n",
__func__, __LINE__);
return -1;
}
private = drm_dev->dev_private;
#if 0
DISPFUNC();
n += mtkfb_get_debug_state(stringbuf + n, buf_len - n);
n += primary_display_get_debug_state(stringbuf + n, buf_len - n);
n += disp_sync_get_debug_info(stringbuf + n, buf_len - n);
n += dprec_logger_get_result_string_all(stringbuf + n, buf_len - n);
n += disp_helper_get_option_list(stringbuf + n, buf_len - n);
#endif
n += mtk_drm_primary_display_get_debug_state(private, stringbuf + n,
buf_len - n);
n += mtk_drm_dump_wk_lock(private, stringbuf + n,
buf_len - n);
n += mtk_drm_helper_get_opt_list(private->helper_opt, stringbuf + n,
buf_len - n);
n += mtk_dprec_logger_get_buf(DPREC_LOGGER_ERROR, stringbuf + n,
buf_len - n);
n += mtk_dprec_logger_get_buf(DPREC_LOGGER_FENCE, stringbuf + n,
buf_len - n);
n += mtk_dprec_logger_get_buf(DPREC_LOGGER_DUMP, stringbuf + n,
buf_len - n);
n += mtk_dprec_logger_get_buf(DPREC_LOGGER_DEBUG, stringbuf + n,
buf_len - n);
n += mtk_dprec_logger_get_buf(DPREC_LOGGER_STATUS, stringbuf + n,
buf_len - n);
stringbuf[n++] = 0;
return n;
}
static void mtk_fake_engine_iommu_enable(struct drm_device *dev,
unsigned int idx)
{
int port, ret;
unsigned int value;
struct device_node *larb_node;
void __iomem *baddr;
struct mtk_drm_private *priv = dev->dev_private;
/* get larb reg */
larb_node = of_parse_phandle(priv->mmsys_dev->of_node,
"fake-engine", idx * 2);
if (!larb_node) {
DDPPR_ERR("Cannot find larb node\n");
return;
}
baddr = of_iomap(larb_node, 0);
of_node_put(larb_node);
/* get port num */
ret = of_property_read_u32_index(priv->mmsys_dev->of_node,
"fake-engine", idx * 2 + 1, &port);
if (ret < 0) {
DDPPR_ERR("Node %s cannot find fake-engine data!\n",
priv->mmsys_dev->of_node->full_name);
return;
}
port &= GET_M4U_PORT;
value = readl(baddr + SMI_LARB_NON_SEC_CON(port));
value = (value & ~0x1) | (0x1 & 0x1);
writel_relaxed(value, baddr + SMI_LARB_NON_SEC_CON(port));
}
static void mtk_fake_engine_share_port_config(struct drm_crtc *crtc,
unsigned int idx, bool en)
{
unsigned int value;
struct device_node *larb_node;
static void __iomem **baddr;
struct mtk_drm_crtc *mtk_crtc = to_mtk_crtc(crtc);
struct mtk_drm_private *priv = crtc->dev->dev_private;
const struct mtk_fake_eng_data *fake_eng_data =
priv->data->fake_eng_data;
int i;
if (!baddr) {
baddr = devm_kmalloc_array(crtc->dev->dev,
fake_eng_data->fake_eng_num,
sizeof(void __iomem *),
GFP_KERNEL);
if (!baddr) {
DDPPR_ERR("%s: devm_kmalloc_array failed\n", __func__);
return;
}
for (i = 0; i < fake_eng_data->fake_eng_num; i++) {
larb_node = of_parse_phandle(priv->mmsys_dev->of_node,
"fake-engine", i * 2);
if (!larb_node) {
DDPPR_ERR("Cannot find larb node\n");
return;
}
baddr[i] = of_iomap(larb_node, 0);
of_node_put(larb_node);
}
}
if (en) {
value = readl(baddr[idx] + SMI_LARB_VC_PRI_MODE);
value = (value & ~0x3) | (0x0 & 0x3);
writel_relaxed(value, baddr + SMI_LARB_VC_PRI_MODE);
value = readl(mtk_crtc->config_regs +
DISP_REG_CONFIG_RDMA_SHARE_SRAM_CON);
value |= DISP_RDMA_FAKE_SMI_SEL(idx);
writel_relaxed(value, mtk_crtc->config_regs +
DISP_REG_CONFIG_RDMA_SHARE_SRAM_CON);
} else {
value = readl(baddr[idx] + SMI_LARB_VC_PRI_MODE);
value = (value & ~0x3) | (0x1 & 0x3);
writel_relaxed(value, baddr + SMI_LARB_VC_PRI_MODE);
value = readl(mtk_crtc->config_regs +
DISP_REG_CONFIG_RDMA_SHARE_SRAM_CON);
value &= ~(DISP_RDMA_FAKE_SMI_SEL(idx));
writel_relaxed(value, mtk_crtc->config_regs +
DISP_REG_CONFIG_RDMA_SHARE_SRAM_CON);
}
}
void fake_engine(struct drm_crtc *crtc, unsigned int idx, unsigned int en,
unsigned int wr_en, unsigned int rd_en, unsigned int wr_pat1,
unsigned int wr_pat2, unsigned int latency,
unsigned int preultra_cnt,
unsigned int ultra_cnt)
{
int burst = 7;
int test_len = 255;
int loop = 1;
int preultra_en = 0;
int ultra_en = 0;
int dis_wr = !wr_en;
int dis_rd = !rd_en;
int delay_cnt = 0;
struct mtk_drm_crtc *mtk_crtc = to_mtk_crtc(crtc);
struct mtk_drm_private *priv = crtc->dev->dev_private;
const struct mtk_fake_eng_data *fake_eng_data;
const struct mtk_fake_eng_reg *fake_eng;
static struct mtk_drm_gem_obj **gem;
int i;
fake_eng_data = priv->data->fake_eng_data;
if (!fake_eng_data) {
DDPPR_ERR("this platform not support any fake engine\n");
return;
}
if (idx > fake_eng_data->fake_eng_num - 1) {
DDPPR_ERR("this platform not support fake engine %d\n", idx);
return;
}
fake_eng = &fake_eng_data->fake_eng_reg[idx];
if (preultra_cnt > 0) {
preultra_en = 1;
preultra_cnt--;
}
if (ultra_cnt > 0) {
ultra_en = 1;
ultra_cnt--;
}
if (en) {
if (!gem) {
gem = devm_kmalloc_array(crtc->dev->dev,
fake_eng_data->fake_eng_num,
sizeof(struct mtk_drm_gem_obj *),
GFP_KERNEL);
if (!gem) {
DDPPR_ERR("%s: devm_kmalloc_array failed\n", __func__);
return;
}
for (i = 0; i < fake_eng_data->fake_eng_num; i++) {
gem[i] = mtk_drm_gem_create(crtc->dev,
1024*1024, true);
mtk_fake_engine_iommu_enable(crtc->dev, i);
DDPMSG("fake_engine_%d va=0x%08lx, pa=0x%08x\n",
i, (unsigned long)gem[i]->kvaddr,
(unsigned int)gem[i]->dma_addr);
}
}
if (fake_eng->share_port)
mtk_fake_engine_share_port_config(crtc, idx, en);
writel_relaxed(BIT(fake_eng->CG_bit), mtk_crtc->config_regs +
DISP_REG_CONFIG_MMSYS_CG_CLR(fake_eng->CG_idx));
writel_relaxed((unsigned int)gem[idx]->dma_addr,
mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_RD_ADDR(idx));
writel_relaxed((unsigned int)gem[idx]->dma_addr + 4096,
mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_WR_ADDR(idx));
writel_relaxed((wr_pat1 << 24) | (loop << 22) | test_len,
mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_CON0(idx));
writel_relaxed((ultra_en << 23) | (ultra_cnt << 20) |
(preultra_en << 19) | (preultra_cnt << 16) |
(burst << 12) | (dis_wr << 11) | (dis_rd << 10) |
latency, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_CON1(idx));
writel_relaxed(1, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_RST(idx));
writel_relaxed(0, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_RST(idx));
writel_relaxed(0x3, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_EN(idx));
if (wr_pat2 != wr_pat1)
writel_relaxed((wr_pat2 << 24) | (loop << 22) |
test_len,
mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_CON0(idx));
DDPMSG("fake_engine_%d enable\n", idx);
} else {
writel_relaxed(0x1, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_EN(idx));
while ((readl(mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_STATE(idx))
& 0x1) == 0x1) {
delay_cnt++;
udelay(1);
if (delay_cnt > 1000) {
DDPPR_ERR("Wait fake_engine_%d idle timeout\n",
idx);
break;
}
}
writel_relaxed(0x0, mtk_crtc->config_regs +
DISP_REG_CONFIG_DISP_FAKE_ENG_EN(idx));
writel_relaxed(BIT(fake_eng->CG_bit), mtk_crtc->config_regs +
DISP_REG_CONFIG_MMSYS_CG_SET(fake_eng->CG_idx));
if (fake_eng->share_port)
mtk_fake_engine_share_port_config(crtc, idx, en);
DDPMSG("fake_engine_%d disable\n", idx);
}
}
void dump_fake_engine(void __iomem *config_regs)
{
DDPDUMP("=================Dump Fake_engine================\n");
mtk_serial_dump_reg(config_regs, 0x100, 1);
mtk_serial_dump_reg(config_regs, 0x110, 1);
mtk_serial_dump_reg(config_regs, 0x200, 4);
mtk_serial_dump_reg(config_regs, 0x210, 3);
mtk_serial_dump_reg(config_regs, 0x220, 4);
mtk_serial_dump_reg(config_regs, 0x230, 3);
}
static void mtk_ddic_send_cb(struct cmdq_cb_data data)
{
struct mtk_cmdq_cb_data *cb_data = data.data;
cmdq_pkt_destroy(cb_data->cmdq_handle);
kfree(cb_data);
CRTC_MMP_MARK(0, ddic_send_cmd, 1, 1);
}
int mtk_drm_set_frame_skip(bool skip)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
pr_info("find crtc fail\n");
return 0;
}
mtk_crtc = to_mtk_crtc(crtc);
if (skip) {
mtk_crtc->skip_frame = true;
pr_info("skip frame set 1\n");
} else {
mtk_crtc->skip_frame = false;
pr_info("skip frame set 0\n");
}
return 0;
}
EXPORT_SYMBOL(mtk_drm_set_frame_skip);
int mtk_crtc_lock_control(bool en)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_drm_private *private;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("crtc error\n");
return -EINVAL;
}
private = crtc->dev->dev_private;
mtk_crtc = to_mtk_crtc(crtc);
if (!private) {
DDPPR_ERR("private error\n");
return -EINVAL;
}
if (en) {
mutex_lock(&private->commit.lock);
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
#ifndef CONFIG_ARCH_HAS_SYSCALL_WRAPPER
mtk_crtc->customer_lock_tid = (int) sys_gettid();
#else
mtk_crtc->customer_lock_tid = current->pid;
#endif
} else {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mtk_crtc->customer_lock_tid = 0;
mutex_unlock(&private->commit.lock);
}
return 0;
}
EXPORT_SYMBOL(mtk_crtc_lock_control);
int set_lcm_wrapper(struct mtk_ddic_dsi_msg *cmd_msg, int blocking)
{
int ret = 0;
int current_tid = 0;
bool recursive_check = false;
bool need_lock = false;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_ddp_comp *output_comp;
#ifndef CONFIG_ARCH_HAS_SYSCALL_WRAPPER
current_tid = sys_gettid();
#else
current_tid = current->pid;
#endif
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
pr_info("find crtc fail\n");
return -EINVAL;
}
mtk_crtc = to_mtk_crtc(crtc);
if (!mtk_crtc) {
pr_info("find mtk_crtc fail\n");
return -EINVAL;
}
output_comp = mtk_ddp_comp_request_output(mtk_crtc);
if (unlikely(!output_comp)) {
pr_info("%s:invalid output comp\n", __func__);
return -EINVAL;
}
recursive_check = (((int)mtk_crtc->need_lock_tid !=
(int)current_tid) &&
(int)current_tid != INIT_TID);
need_lock = (current_tid != mtk_crtc->customer_lock_tid);
DDPMSG("%s + %d_%d_%d_%d_%d_%d\n", __func__, current_tid, mtk_crtc->need_lock_tid,
mtk_crtc->customer_lock_tid, recursive_check,
need_lock, mtk_crtc->set_lcm_scn);
if (recursive_check)
ret = mtk_ddic_dsi_send_cmd(cmd_msg, blocking, need_lock);
else if (mtk_crtc->set_lcm_scn > SET_LCM_CMDQ_AVAILABLE &&
mtk_crtc->set_lcm_scn < SET_LCM_CMDQ_FRAME_DONE)
ret = mtk_ddp_comp_io_cmd(output_comp, NULL,
SET_LCM_CMDQ, cmd_msg);
else if (mtk_crtc->set_lcm_scn > SET_LCM_CMDQ_FRAME_DONE)
ret = mtk_ddic_dsi_send_cmd(cmd_msg, blocking, recursive_check);
else
ret = mtk_ddp_comp_io_cmd(output_comp, NULL,
SET_LCM_DCS_CMD, cmd_msg);
return ret;
}
int read_lcm_wrapper(struct mtk_ddic_dsi_msg *cmd_msg)
{
int ret = 0;
int current_tid = 0;
bool recursive_check = false;
bool need_lock = false;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_ddp_comp *output_comp;
#ifndef CONFIG_ARCH_HAS_SYSCALL_WRAPPER
current_tid = sys_gettid();
#else
current_tid = current->pid;
#endif
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
pr_info("find crtc fail\n");
return -EINVAL;
}
mtk_crtc = to_mtk_crtc(crtc);
if (!mtk_crtc) {
pr_info("find mtk_crtc fail\n");
return -EINVAL;
}
output_comp = mtk_ddp_comp_request_output(mtk_crtc);
if (unlikely(!output_comp)) {
pr_info("%s:invalid output comp\n", __func__);
return -EINVAL;
}
recursive_check = (((int)mtk_crtc->need_lock_tid !=
(int)current_tid) &&
(int)current_tid != INIT_TID);
need_lock = (current_tid != mtk_crtc->customer_lock_tid);
DDPMSG("%s + %d_%d_%d_%d_%d_%d\n", __func__, current_tid, mtk_crtc->need_lock_tid,
mtk_crtc->customer_lock_tid, recursive_check, need_lock, mtk_crtc->set_lcm_scn);
if (recursive_check)
ret = mtk_ddic_dsi_read_cmd(cmd_msg, need_lock);
else if (mtk_crtc->set_lcm_scn > SET_LCM_CMDQ_FRAME_DONE)
ret = mtk_ddic_dsi_read_cmd(cmd_msg, recursive_check);
else
ret = mtk_ddp_comp_io_cmd(output_comp, NULL,
READ_LCM_DCS_CMD, cmd_msg);
return ret;
}
int mtk_ddic_dsi_send_cmd(struct mtk_ddic_dsi_msg *cmd_msg,
bool blocking, bool need_lock)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_drm_private *private;
struct mtk_ddp_comp *output_comp;
struct cmdq_pkt *cmdq_handle;
struct cmdq_client *gce_client;
bool is_frame_mode;
bool use_lpm = false;
struct mtk_cmdq_cb_data *cb_data;
int index = 0;
int ret = 0;
DDPMSG("%s +\n", __func__);
/* This cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
index = drm_crtc_index(crtc);
CRTC_MMP_EVENT_START(index, ddic_send_cmd, (unsigned long)crtc,
blocking);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
CRTC_MMP_EVENT_END(index, ddic_send_cmd, 0, 0);
return -EINVAL;
}
private = crtc->dev->dev_private;
mtk_crtc = to_mtk_crtc(crtc);
if (need_lock) {
mutex_lock(&private->commit.lock);
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
}
if (!mtk_crtc->enabled) {
DDPMSG("crtc%d disable skip %s\n",
drm_crtc_index(&mtk_crtc->base), __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_send_cmd, 0, 1);
return -EINVAL;
} else if (mtk_crtc->ddp_mode == DDP_NO_USE) {
DDPMSG("skip %s, ddp_mode: NO_USE\n",
__func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_send_cmd, 0, 2);
return -EINVAL;
}
output_comp = mtk_ddp_comp_request_output(mtk_crtc);
if (unlikely(!output_comp)) {
DDPPR_ERR("%s:invalid output comp\n", __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_send_cmd, 0, 3);
return -EINVAL;
}
is_frame_mode = mtk_crtc_is_frame_trigger_mode(&mtk_crtc->base);
if (cmd_msg)
use_lpm = cmd_msg->flags & MIPI_DSI_MSG_USE_LPM;
CRTC_MMP_MARK(index, ddic_send_cmd, 1, 0);
/* Kick idle */
mtk_drm_idlemgr_kick(__func__, crtc, 0);
CRTC_MMP_MARK(index, ddic_send_cmd, 2, 0);
/* only use CLIENT_DSI_CFG for VM CMD scenario */
/* use CLIENT_CFG otherwise */
gce_client = (!is_frame_mode && use_lpm) ?
mtk_crtc->gce_obj.client[CLIENT_DSI_CFG] :
mtk_crtc->gce_obj.client[CLIENT_CFG];
mtk_crtc_pkt_create(&cmdq_handle, crtc, gce_client);
if (mtk_crtc_with_sub_path(crtc, mtk_crtc->ddp_mode))
mtk_crtc_wait_frame_done(mtk_crtc, cmdq_handle,
DDP_SECOND_PATH, 0);
else
mtk_crtc_wait_frame_done(mtk_crtc, cmdq_handle,
DDP_FIRST_PATH, 0);
if (is_frame_mode) {
cmdq_pkt_clear_event(cmdq_handle,
mtk_crtc->gce_obj.event[EVENT_STREAM_BLOCK]);
cmdq_pkt_wfe(cmdq_handle,
mtk_crtc->gce_obj.event[EVENT_CABC_EOF]);
// cmdq_pkt_clear_event(cmdq_handle,
// mtk_crtc->gce_obj.event[EVENT_STREAM_DIRTY]);
}
/* DSI_SEND_DDIC_CMD */
if (output_comp)
ret = mtk_ddp_comp_io_cmd(output_comp, cmdq_handle,
DSI_SEND_DDIC_CMD, cmd_msg);
if (is_frame_mode) {
// cmdq_pkt_set_event(cmdq_handle,
// mtk_crtc->gce_obj.event[EVENT_STREAM_DIRTY]);
cmdq_pkt_set_event(cmdq_handle,
mtk_crtc->gce_obj.event[EVENT_CABC_EOF]);
cmdq_pkt_set_event(cmdq_handle,
mtk_crtc->gce_obj.event[EVENT_STREAM_BLOCK]);
}
if (blocking) {
cmdq_pkt_flush(cmdq_handle);
cmdq_pkt_destroy(cmdq_handle);
} else {
cb_data = kmalloc(sizeof(*cb_data), GFP_KERNEL);
if (!cb_data) {
DDPPR_ERR("%s:cb data creation failed\n", __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_send_cmd, 0, 4);
return -EINVAL;
}
cb_data->cmdq_handle = cmdq_handle;
cmdq_pkt_flush_threaded(cmdq_handle, mtk_ddic_send_cb, cb_data);
}
DDPMSG("%s -\n", __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_send_cmd, (unsigned long)crtc,
blocking);
return ret;
}
int mtk_ddic_dsi_read_cmd(struct mtk_ddic_dsi_msg *cmd_msg, bool need_lock)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_drm_private *private;
struct mtk_ddp_comp *output_comp;
int index = 0;
int ret = 0;
DDPMSG("%s +\n", __func__);
/* This cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
index = drm_crtc_index(crtc);
CRTC_MMP_EVENT_START(index, ddic_read_cmd, (unsigned long)crtc, 0);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
CRTC_MMP_EVENT_END(index, ddic_read_cmd, 0, 0);
return -EINVAL;
}
private = crtc->dev->dev_private;
mtk_crtc = to_mtk_crtc(crtc);
if (need_lock) {
mutex_lock(&private->commit.lock);
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
}
if (!mtk_crtc->enabled) {
DDPMSG("crtc%d disable skip %s\n",
drm_crtc_index(&mtk_crtc->base), __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_read_cmd, 0, 1);
return -EINVAL;
} else if (mtk_crtc->ddp_mode == DDP_NO_USE) {
DDPMSG("skip %s, ddp_mode: NO_USE\n",
__func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_read_cmd, 0, 2);
return -EINVAL;
}
output_comp = mtk_ddp_comp_request_output(mtk_crtc);
if (unlikely(!output_comp)) {
DDPPR_ERR("%s:invalid output comp\n", __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_read_cmd, 0, 3);
return -EINVAL;
}
CRTC_MMP_MARK(index, ddic_read_cmd, 1, 0);
/* Kick idle */
mtk_drm_idlemgr_kick(__func__, crtc, 0);
CRTC_MMP_MARK(index, ddic_read_cmd, 2, 0);
/* DSI_READ_DDIC_CMD */
if (output_comp)
ret = mtk_ddp_comp_io_cmd(output_comp, NULL, DSI_READ_DDIC_CMD,
cmd_msg);
CRTC_MMP_MARK(index, ddic_read_cmd, 3, 0);
DDPMSG("%s -\n", __func__);
if (need_lock) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
mutex_unlock(&private->commit.lock);
}
CRTC_MMP_EVENT_END(index, ddic_read_cmd, (unsigned long)crtc, 4);
return ret;
}
void ddic_dsi_send_cmd_test(unsigned int case_num)
{
unsigned int i = 0, j = 0;
int ret;
struct mtk_ddic_dsi_msg *cmd_msg =
vmalloc(sizeof(struct mtk_ddic_dsi_msg));
u8 tx[10] = {0};
u8 tx_1[10] = {0};
DDPMSG("%s start case_num:%d\n", __func__, case_num);
if (!cmd_msg) {
DDPPR_ERR("cmd msg is NULL\n");
return;
}
memset(cmd_msg, 0, sizeof(struct mtk_ddic_dsi_msg));
switch (case_num) {
case 1:
{
/* Send 0x34 */
cmd_msg->channel = 0;
cmd_msg->flags = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x05;
tx[0] = 0x34;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
break;
}
case 2:
{
/* Send 0x35:0x00 */
cmd_msg->channel = 0;
cmd_msg->flags = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x15;
tx[0] = 0x35;
tx[1] = 0x00;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 2;
break;
}
case 3:
{
/* Send 0x28 */
cmd_msg->channel = 0;
cmd_msg->flags |= MIPI_DSI_MSG_USE_LPM;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x05;
tx[0] = 0x28;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
break;
}
case 4:
{
/* Send 0x29 */
cmd_msg->channel = 0;
cmd_msg->flags |= MIPI_DSI_MSG_USE_LPM;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x05;
tx[0] = 0x29;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
break;
}
case 5:
{
/* Multiple cmd UT case */
cmd_msg->channel = 0;
cmd_msg->flags = 0;
/* cmd_msg->flags |= MIPI_DSI_MSG_USE_LPM; */
cmd_msg->tx_cmd_num = 2;
/* Send 0x34 */
cmd_msg->type[0] = 0x05;
tx[0] = 0x34;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
/* Send 0x28 */
cmd_msg->type[1] = 0x05;
tx_1[0] = 0x28;
cmd_msg->tx_buf[1] = tx_1;
cmd_msg->tx_len[1] = 1;
break;
}
case 6:
{
/* Multiple cmd UT case */
cmd_msg->channel = 0;
cmd_msg->flags = 0;
/* cmd_msg->flags |= MIPI_DSI_MSG_USE_LPM; */
cmd_msg->tx_cmd_num = 2;
/* Send 0x35 */
cmd_msg->type[0] = 0x15;
tx[0] = 0x35;
tx[1] = 0x00;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 2;
/* Send 0x29 */
cmd_msg->type[1] = 0x05;
tx_1[0] = 0x29;
cmd_msg->tx_buf[1] = tx_1;
cmd_msg->tx_len[1] = 1;
break;
}
default:
DDPMSG("%s no this test case:%d\n", __func__, case_num);
break;
}
DDPMSG("send lcm tx_cmd_num:%d\n", (int)cmd_msg->tx_cmd_num);
for (i = 0; i < (int)cmd_msg->tx_cmd_num; i++) {
DDPMSG("send lcm tx_len[%d]=%d\n",
i, (int)cmd_msg->tx_len[i]);
for (j = 0; j < (int)cmd_msg->tx_len[i]; j++) {
DDPMSG(
"send lcm type[%d]=0x%x, tx_buf[%d]--byte:%d,val:0x%x\n",
i, cmd_msg->type[i], i, j,
*(char *)(cmd_msg->tx_buf[i] + j));
}
}
ret = mtk_ddic_dsi_send_cmd(cmd_msg, true, true);
if (ret != 0) {
DDPPR_ERR("mtk_ddic_dsi_send_cmd error\n");
goto done;
}
done:
vfree(cmd_msg);
DDPMSG("%s end -\n", __func__);
}
void ddic_dsi_read_cmd_test(unsigned int case_num)
{
unsigned int i = 0, j = 0;
unsigned int ret_dlen = 0;
int ret;
struct mtk_ddic_dsi_msg *cmd_msg =
vmalloc(sizeof(struct mtk_ddic_dsi_msg));
u8 tx[10] = {0};
u8 tx_1[10] = {0};
u8 tx_2[10] = {0};
u8 tx_3[10] = {0};
DDPMSG("%s start case_num:%d\n", __func__, case_num);
if (!cmd_msg) {
DDPPR_ERR("cmd msg is NULL\n");
return;
}
memset(cmd_msg, 0, sizeof(struct mtk_ddic_dsi_msg));
switch (case_num) {
case 1:
{
/* Read 0x0A = 0x1C */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x06;
tx[0] = 0x0A;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_cmd_num = 1;
cmd_msg->rx_buf[0] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 1;
break;
}
case 1001:
{
/* Read 0x0A = 0x1C */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 20;
cmd_msg->rx_cmd_num = 20;
tx[0] = 0x0A;
for (i = 0; i < 20; i++) {
cmd_msg->type[i] = 0x06;
cmd_msg->tx_buf[i] = tx;
cmd_msg->tx_len[i] = 1;
cmd_msg->rx_buf[i] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[i], 0, 4);
cmd_msg->rx_len[i] = 1;
}
break;
}
case 2:
{
/* Read 0xe8 = 0x00,0x01,0x23,0x00 */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x06;
tx[0] = 0xe8;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_cmd_num = 1;
cmd_msg->rx_buf[0] = kmalloc(8 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 4;
break;
}
case 1002:
{
/* Read 0xe8 = 0x00,0x01,0x23,0x00 */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 20;
cmd_msg->rx_cmd_num = 20;
tx[0] = 0xe8;
for (i = 0; i < 20; i++) {
cmd_msg->type[i] = 0x06;
cmd_msg->tx_buf[i] = tx;
cmd_msg->tx_len[i] = 1;
cmd_msg->rx_buf[i] = kmalloc(8 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[i], 0, 4);
cmd_msg->rx_len[i] = 4;
}
break;
}
case 3:
{
/*
* Read 0xb6 =
* 0x30,0x6b,0x00,0x06,0x03,0x0A,0x13,0x1A,0x6C,0x18
*/
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x06;
tx[0] = 0xb6;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_cmd_num = 1;
cmd_msg->rx_buf[0] = kmalloc(20 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 20);
cmd_msg->rx_len[0] = 10;
break;
}
case 4:
{
/* Read 0x0e = 0x80 */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 1;
cmd_msg->type[0] = 0x06;
tx[0] = 0x0e;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_cmd_num = 1;
cmd_msg->rx_buf[0] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 1;
break;
}
case 5:
{
/* multiple cmd Read*/
/*0x0A = 0x1C;*/
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 2;
cmd_msg->rx_cmd_num = 2;
cmd_msg->type[0] = 0x06;
tx[0] = 0x0A;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_buf[0] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 1;
/*0x0e = 0x80 */
cmd_msg->type[1] = 0x06;
tx_1[0] = 0x0e;
cmd_msg->tx_buf[1] = tx_1;
cmd_msg->tx_len[1] = 1;
cmd_msg->rx_buf[1] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[1], 0, 4);
cmd_msg->rx_len[1] = 1;
break;
}
case 6:
{
/* multiple cmd Read*/
/*0x0A = 0x1C; */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 3;
cmd_msg->rx_cmd_num = 3;
cmd_msg->type[0] = 0x06;
tx[0] = 0x0A;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_buf[0] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 1;
/*0x0e = 0x80 */
cmd_msg->type[1] = 0x06;
tx_1[0] = 0x0e;
cmd_msg->tx_buf[1] = tx_1;
cmd_msg->tx_len[1] = 1;
cmd_msg->rx_buf[1] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[1], 0, 4);
cmd_msg->rx_len[1] = 1;
/* Read 0xe8 = 0x00,0x01,0x23,0x00 */
cmd_msg->type[2] = 0x06;
tx_2[0] = 0xe8;
cmd_msg->tx_buf[2] = tx_2;
cmd_msg->tx_len[2] = 1;
cmd_msg->rx_buf[2] = kmalloc(8 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[2], 0, 4);
cmd_msg->rx_len[2] = 4;
break;
}
case 7:
{
/* multiple cmd Read*/
/*0x0A = 0x1C; */
cmd_msg->channel = 0;
cmd_msg->tx_cmd_num = 4;
cmd_msg->rx_cmd_num = 4;
cmd_msg->type[0] = 0x06;
tx[0] = 0x0A;
cmd_msg->tx_buf[0] = tx;
cmd_msg->tx_len[0] = 1;
cmd_msg->rx_buf[0] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[0], 0, 4);
cmd_msg->rx_len[0] = 1;
/*0x0e = 0x80 */
cmd_msg->type[1] = 0x06;
tx_1[0] = 0x0e;
cmd_msg->tx_buf[1] = tx_1;
cmd_msg->tx_len[1] = 1;
cmd_msg->rx_buf[1] = kmalloc(4 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[1], 0, 4);
cmd_msg->rx_len[1] = 1;
/* Read 0xe8 = 0x00,0x01,0x23,0x00 */
cmd_msg->type[2] = 0x06;
tx_2[0] = 0xe8;
cmd_msg->tx_buf[2] = tx_2;
cmd_msg->tx_len[2] = 1;
cmd_msg->rx_buf[2] = kmalloc(8 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[2], 0, 4);
cmd_msg->rx_len[2] = 4;
/*
* Read 0xb6 =
* 0x30,0x6b,0x00,0x06,0x03,0x0A,0x13,0x1A,0x6C,0x18
*/
cmd_msg->type[3] = 0x06;
tx_3[0] = 0xb6;
cmd_msg->tx_buf[3] = tx_3;
cmd_msg->tx_len[3] = 1;
cmd_msg->rx_buf[3] = kmalloc(20 * sizeof(unsigned char),
GFP_ATOMIC);
memset(cmd_msg->rx_buf[3], 0, 20);
cmd_msg->rx_len[3] = 10;
break;
}
default:
DDPMSG("%s no this test case:%d\n", __func__, case_num);
break;
}
ret = mtk_ddic_dsi_read_cmd(cmd_msg, true);
if (ret != 0) {
DDPPR_ERR("%s error\n", __func__);
goto done;
}
for (i = 0; i < cmd_msg->rx_cmd_num; i++) {
ret_dlen = cmd_msg->rx_len[i];
DDPMSG("read lcm addr:0x%x--dlen:%d--cmd_idx:%d\n",
*(char *)(cmd_msg->tx_buf[i]), ret_dlen, i);
for (j = 0; j < ret_dlen; j++) {
DDPMSG("read lcm addr:0x%x--byte:%d,val:0x%x\n",
*(char *)(cmd_msg->tx_buf[i]), j,
*(char *)(cmd_msg->rx_buf[i] + j));
}
}
done:
for (i = 0; i < cmd_msg->rx_cmd_num; i++)
kfree(cmd_msg->rx_buf[i]);
vfree(cmd_msg);
DDPMSG("%s end -\n", __func__);
}
int mtk_dprec_mmp_dump_ovl_layer(struct mtk_plane_state *plane_state)
{
if (gCaptureOVLEn) {
mtk_drm_mmp_ovl_layer(plane_state, gCapturePriLayerDownX,
gCapturePriLayerDownY);
return 0;
}
DDPINFO("%s, gCapturePriLayerEnable is %d\n",
__func__, gCaptureOVLEn);
return -1;
}
int mtk_dprec_mmp_dump_cwb_buffer(struct drm_crtc *crtc,
void *buffer, unsigned int buf_idx)
{
struct mtk_drm_crtc *mtk_crtc = to_mtk_crtc(crtc);
if (gCaptureWDMAEn && mtk_crtc->cwb_info) {
mtk_drm_mmp_cwb_buffer(crtc, mtk_crtc->cwb_info,
buffer, buf_idx);
return 0;
}
DDPDBG("%s, gCaptureWDMAEn is %d\n",
__func__, gCaptureWDMAEn);
return -1;
}
static void user_copy_done_function(void *buffer,
enum CWB_BUFFER_TYPE type)
{
DDPMSG("[capture] I get buffer:0x%x, type:%d\n",
buffer, type);
complete(&cwb_cmp);
}
static const struct mtk_cwb_funcs user_cwb_funcs = {
.copy_done = user_copy_done_function,
};
static void mtk_drm_cwb_info_init(struct drm_crtc *crtc)
{
struct mtk_drm_crtc *mtk_crtc = to_mtk_crtc(crtc);
int crtc_idx = drm_crtc_index(&mtk_crtc->base);
struct mtk_drm_private *priv = mtk_crtc->base.dev->dev_private;
struct mtk_cwb_info *cwb_info = mtk_crtc->cwb_info;
struct drm_mode_fb_cmd2 mode = {0};
struct mtk_drm_gem_obj *mtk_gem;
if (!cwb_info) {
DDPPR_ERR("%s: cwb_info not found\n", __func__);
return;
} else if (!cwb_info->enable)
return;
cwb_info->count = 0;
cwb_info->src_roi.width =
crtc->state->adjusted_mode.hdisplay;
cwb_info->src_roi.height =
crtc->state->adjusted_mode.vdisplay;
cwb_info->scn = WDMA_WRITE_BACK;
if (crtc_idx == 0)
cwb_info->comp = priv->ddp_comp[DDP_COMPONENT_WDMA0];
if (!cwb_info->buffer[0].dst_roi.width ||
!cwb_info->buffer[0].dst_roi.height) {
mtk_rect_make(&cwb_info->buffer[0].dst_roi, 0, 0,
MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH,
MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH);
mtk_rect_make(&cwb_info->buffer[1].dst_roi, 0, 0,
MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH,
MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH);
}
/*alloc && config two fb*/
if (!cwb_info->buffer[0].fb) {
mode.width = MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH;
mode.height = cwb_info->src_roi.height;
mode.pixel_format = DRM_FORMAT_RGB888;
mode.pitches[0] = mode.width * 3;
mtk_gem = mtk_drm_gem_create(
crtc->dev, mode.width * mode.height * 3, true);
cwb_info->buffer[0].addr_mva = (u32)mtk_gem->dma_addr;
cwb_info->buffer[0].addr_va = (u64)mtk_gem->kvaddr;
cwb_info->buffer[0].fb =
mtk_drm_framebuffer_create(
crtc->dev, &mode, &mtk_gem->base);
DDPMSG("[capture] b[0].addr_mva:0x%x, addr_va:0x%llx\n",
cwb_info->buffer[0].addr_mva,
cwb_info->buffer[0].addr_va);
mtk_gem = mtk_drm_gem_create(
crtc->dev, mode.width * mode.height * 3, true);
cwb_info->buffer[1].addr_mva = (u32)mtk_gem->dma_addr;
cwb_info->buffer[1].addr_va = (u64)mtk_gem->kvaddr;
cwb_info->buffer[1].fb =
mtk_drm_framebuffer_create(
crtc->dev, &mode, &mtk_gem->base);
DDPMSG("[capture] b[1].addr_mva:0x%x, addr_va:0x%llx\n",
cwb_info->buffer[1].addr_mva,
cwb_info->buffer[1].addr_va);
}
DDPMSG("[capture] enable capture, roi:(%d,%d,%d,%d)\n",
cwb_info->buffer[0].dst_roi.x,
cwb_info->buffer[0].dst_roi.y,
cwb_info->buffer[0].dst_roi.width,
cwb_info->buffer[0].dst_roi.height);
}
bool mtk_drm_cwb_enable(int en,
const struct mtk_cwb_funcs *funcs,
enum CWB_BUFFER_TYPE type)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_cwb_info *cwb_info;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return false;
}
mtk_crtc = to_mtk_crtc(crtc);
if (!mtk_crtc->cwb_info) {
mtk_crtc->cwb_info = kzalloc(sizeof(struct mtk_cwb_info),
GFP_KERNEL);
DDPMSG("%s: need allocate memory\n", __func__);
}
if (!mtk_crtc->cwb_info) {
DDPPR_ERR("%s: allocate memory fail\n", __func__);
return false;
}
cwb_info = mtk_crtc->cwb_info;
if (cwb_info->enable == en) {
DDPMSG("[capture] en:%d already effective\n", en);
return true;
}
cwb_info->funcs = funcs;
cwb_info->type = type;
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
cwb_info->enable = en;
if (en)
mtk_drm_cwb_info_init(crtc);
else
DDPMSG("[capture] disable capture");
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
return true;
}
bool mtk_drm_set_cwb_roi(struct mtk_rect rect)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_cwb_info *cwb_info;
struct mtk_drm_gem_obj *mtk_gem;
struct drm_mode_fb_cmd2 mode = {0};
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return false;
}
mtk_crtc = to_mtk_crtc(crtc);
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
if (!mtk_crtc->cwb_info) {
mtk_crtc->cwb_info = kzalloc(sizeof(struct mtk_cwb_info),
GFP_KERNEL);
DDPMSG("%s: need allocate memory\n", __func__);
}
if (!mtk_crtc->cwb_info) {
DDPPR_ERR("%s: allocate memory fail\n", __func__);
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
return false;
}
cwb_info = mtk_crtc->cwb_info;
cwb_info->src_roi.width =
crtc->state->adjusted_mode.hdisplay;
cwb_info->src_roi.height =
crtc->state->adjusted_mode.vdisplay;
if (rect.x >= cwb_info->src_roi.width ||
rect.y >= cwb_info->src_roi.height ||
!rect.width || !rect.height) {
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
return false;
}
if (rect.width > MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH)
rect.width = MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH;
if (rect.x + rect.width > cwb_info->src_roi.width)
rect.width = cwb_info->src_roi.width - rect.x;
if (rect.y + rect.height > cwb_info->src_roi.height)
rect.height = cwb_info->src_roi.height - rect.y;
if (!cwb_info->buffer[0].fb) {
mode.width = MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH;
mode.height = cwb_info->src_roi.height;
mode.pixel_format = DRM_FORMAT_RGB888;
mode.pitches[0] = mode.width * 3;
mtk_gem = mtk_drm_gem_create(
crtc->dev, mode.width * mode.height * 3, true);
cwb_info->buffer[0].addr_mva = (u32)mtk_gem->dma_addr;
cwb_info->buffer[0].addr_va = (u64)mtk_gem->kvaddr;
cwb_info->buffer[0].fb =
mtk_drm_framebuffer_create(
crtc->dev, &mode, &mtk_gem->base);
DDPMSG("[capture] b[0].addr_mva:0x%x, addr_va:0x%llx\n",
cwb_info->buffer[0].addr_mva,
cwb_info->buffer[0].addr_va);
mtk_gem = mtk_drm_gem_create(
crtc->dev, mode.width * mode.height * 3, true);
cwb_info->buffer[1].addr_mva = (u32)mtk_gem->dma_addr;
cwb_info->buffer[1].addr_va = (u64)mtk_gem->kvaddr;
cwb_info->buffer[1].fb =
mtk_drm_framebuffer_create(
crtc->dev, &mode, &mtk_gem->base);
DDPMSG("[capture] b[1].addr_mva:0x%x, addr_va:0x%llx\n",
cwb_info->buffer[1].addr_mva,
cwb_info->buffer[1].addr_va);
}
/* update roi */
mtk_rect_make(&cwb_info->buffer[0].dst_roi,
rect.x, rect.y, rect.width, rect.height);
mtk_rect_make(&cwb_info->buffer[1].dst_roi,
rect.x, rect.y, rect.width, rect.height);
DDPMSG("[capture] change roi:(%d,%d,%d,%d)\n",
cwb_info->buffer[0].dst_roi.x,
cwb_info->buffer[0].dst_roi.y,
cwb_info->buffer[0].dst_roi.width,
cwb_info->buffer[0].dst_roi.height);
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
return true;
}
void mtk_drm_cwb_backup_copy_size(void)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_cwb_info *cwb_info;
struct mtk_ddp_comp *comp;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
cwb_info = mtk_crtc->cwb_info;
if (!cwb_info)
return;
if (!cwb_info->comp) {
DDPPR_ERR("[capture] cwb enable, but has not comp\n");
return;
}
comp = cwb_info->comp;
mtk_ddp_comp_io_cmd(comp, NULL, WDMA_READ_DST_SIZE, cwb_info);
}
bool mtk_drm_set_cwb_user_buf(void *user_buffer, enum CWB_BUFFER_TYPE type)
{
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_cwb_info *cwb_info;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return false;
}
mtk_crtc = to_mtk_crtc(crtc);
cwb_info = mtk_crtc->cwb_info;
if (!cwb_info)
return false;
DDP_MUTEX_LOCK(&mtk_crtc->cwb_lock, __func__, __LINE__);
cwb_info->type = type;
cwb_info->user_buffer = user_buffer;
DDP_MUTEX_UNLOCK(&mtk_crtc->cwb_lock, __func__, __LINE__);
DDPMSG("[capture] User set buffer:0x%x, type:%d\n",
user_buffer, type);
return true;
}
int mtk_drm_ioctl_pq_get_persist_property(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int i, ret = 0;
unsigned int pq_persist_property[32];
struct mtk_drm_private *private = dev->dev_private;
struct drm_crtc *crtc = private->crtc[0];
memset(pq_persist_property, 0, sizeof(pq_persist_property));
memcpy(pq_persist_property, (unsigned int *)data, sizeof(pq_persist_property));
for (i = 0; i < DISP_PQ_PROPERTY_MAX; i++) {
m_old_pq_persist_property[i] = m_new_pq_persist_property[i];
m_new_pq_persist_property[i] = pq_persist_property[i];
}
DDPFUNC("+");
if (m_old_pq_persist_property[DISP_PQ_COLOR_BYPASS] !=
m_new_pq_persist_property[DISP_PQ_COLOR_BYPASS])
disp_color_set_bypass(crtc, m_new_pq_persist_property[DISP_PQ_COLOR_BYPASS]);
if (m_old_pq_persist_property[DISP_PQ_CCORR_BYPASS] !=
m_new_pq_persist_property[DISP_PQ_CCORR_BYPASS])
disp_ccorr_set_bypass(crtc, m_new_pq_persist_property[DISP_PQ_CCORR_BYPASS]);
if (m_old_pq_persist_property[DISP_PQ_GAMMA_BYPASS] !=
m_new_pq_persist_property[DISP_PQ_GAMMA_BYPASS])
disp_gamma_set_bypass(crtc, m_new_pq_persist_property[DISP_PQ_GAMMA_BYPASS]);
if (m_old_pq_persist_property[DISP_PQ_DITHER_BYPASS] !=
m_new_pq_persist_property[DISP_PQ_DITHER_BYPASS])
disp_dither_set_bypass(crtc, m_new_pq_persist_property[DISP_PQ_DITHER_BYPASS]);
if (m_old_pq_persist_property[DISP_PQ_AAL_BYPASS] !=
m_new_pq_persist_property[DISP_PQ_AAL_BYPASS])
disp_aal_set_bypass(crtc, m_new_pq_persist_property[DISP_PQ_AAL_BYPASS]);
DDPFUNC("-");
return ret;
}
static void process_dbg_opt(const char *opt)
{
DDPINFO("display_debug cmd %s\n", opt);
if (strncmp(opt, "helper", 6) == 0) {
/*ex: echo helper:DISP_OPT_BYPASS_OVL,0 > /d/mtkfb */
char option[100] = "";
char *tmp;
int value, i, limited;
enum MTK_DRM_HELPER_OPT helper_opt;
struct mtk_drm_private *priv = drm_dev->dev_private;
int ret;
tmp = (char *)(opt + 7);
limited = strlen(tmp);
for (i = 0; i < 100; i++) {
if (i >= limited)
return;
if (tmp[i] != ',' && tmp[i] != ' ')
option[i] = tmp[i];
else
break;
}
tmp += i + 1;
ret = sscanf(tmp, "%d\n", &value);
if (ret != 1) {
DDPPR_ERR("error to parse cmd %s: %s %s ret=%d\n", opt,
option, tmp, ret);
return;
}
DDPMSG("will set option %s to %d\n", option, value);
mtk_drm_helper_set_opt_by_name(priv->helper_opt, option, value);
helper_opt =
mtk_drm_helper_name_to_opt(priv->helper_opt, option);
mtk_update_layering_opt_by_disp_opt(helper_opt, value);
} else if (strncmp(opt, "mobile:", 7) == 0) {
if (strncmp(opt + 7, "on", 2) == 0)
g_mobile_log = 1;
else if (strncmp(opt + 7, "off", 3) == 0)
g_mobile_log = 0;
} else if (strncmp(opt, "fence:", 6) == 0) {
if (strncmp(opt + 6, "on", 2) == 0)
g_fence_log = 1;
else if (strncmp(opt + 6, "off", 3) == 0)
g_fence_log = 0;
} else if (strncmp(opt, "irq:", 4) == 0) {
if (strncmp(opt + 4, "on", 2) == 0)
g_irq_log = 1;
else if (strncmp(opt + 4, "off", 3) == 0)
g_irq_log = 0;
} else if (strncmp(opt, "detail:", 7) == 0) {
if (strncmp(opt + 7, "on", 2) == 0)
g_detail_log = 1;
else if (strncmp(opt + 7, "off", 3) == 0)
g_detail_log = 0;
} else if (strncmp(opt, "trace:", 6) == 0) {
if (strncmp(opt + 6, "on", 2) == 0)
g_trace_log = 1;
else if (strncmp(opt + 6, "off", 3) == 0)
g_trace_log = 0;
} else if (strncmp(opt, "lcd:", 4) == 0) {
struct mtk_drm_private *priv = drm_dev->dev_private;
if (strncmp(opt + 4, "on", 2) == 0) {
noti_uevent_user(&priv->uevent_data, 1);
} else if (strncmp(opt + 4, "off", 3) == 0) {
noti_uevent_user(&priv->uevent_data, 0);
}
} else if (strncmp(opt, "diagnose", 8) == 0) {
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
drm_for_each_crtc(crtc, drm_dev) {
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
continue;
}
mtk_crtc = to_mtk_crtc(crtc);
if (!crtc->enabled
|| mtk_crtc->ddp_mode == DDP_NO_USE)
continue;
mtk_drm_crtc_analysis(crtc);
mtk_drm_crtc_dump(crtc);
}
#ifdef CONFIG_MTK_MT6382_BDG
} else if (strncmp(opt, "bdg_dump", 8) == 0) {
bdg_dsi_dump_reg(DISP_BDG_DSI0);
} else if (strncmp(opt, "set_data_rate:", 14) == 0) {
unsigned int data_rate = 0;
int ret = -1;
ret = sscanf(opt, "set_data_rate:%d\n",
&data_rate);
if (ret != 1) {
DDPMSG("[error]%d error to parse set_data_rate cmd %s\n",
__LINE__, opt);
return;
}
set_bdg_data_rate(data_rate);
} else if (!strncmp(opt, "set_mask_spi:", 13)) {
unsigned int addr = 0, val = 0, mask = 0;
int ret = -1;
ret = sscanf(opt, "set_mask_spi:addr=0x%x,mask=0x%x,val=0x%x\n",
&addr, &mask, &val);
if (ret != 3) {
DDPMSG("[error]%d error to parse set_mt6382_spi cmd %s\n",
__LINE__, opt);
return;
}
ret = mtk_spi_mask_write(addr, mask, val);
if (ret < 0) {
DDPMSG("[error]write mt6382 fail,addr:0x%x, val:0x%x\n",
addr, val);
return;
}
} else if (!strncmp(opt, "set_mt6382_spi:", 15)) {
unsigned int addr = 0, val = 0;
int ret = -1;
ret = sscanf(opt, "set_mt6382_spi:addr=0x%x,val=0x%x\n",
&addr, &val);
if (ret != 2) {
DDPMSG("[error]%d error to parse set_mt6382_spi cmd %s\n",
__LINE__, opt);
return;
}
ret = mtk_spi_write(addr, val);
if (ret < 0) {
DDPMSG("[error]write mt6382 fail,addr:0x%x, val:0x%x\n",
addr, val);
return;
}
} else if (!strncmp(opt, "read_mt6382_spi:", 16)) {
unsigned int addr = 0, val = 0;
int ret = -1;
ret = sscanf(opt, "read_mt6382_spi:addr=0x%x\n", &addr);
if (ret != 1) {
DDPMSG("[error]%d error to parse read_mt6382_spi cmd %s\n",
__LINE__, opt);
return;
}
val = mtk_spi_read(addr);
DDPMSG("mt6382 read addr:0x%08x, val:0x%08x\n", addr, val);
} else if (strncmp(opt, "check", 5) == 0) {
if (check_stopstate(NULL) == 0)
bdg_tx_start(DISP_BDG_DSI0, NULL);
mdelay(100);
return;
#endif
} else if (strncmp(opt, "repaint", 7) == 0) {
drm_trigger_repaint(DRM_REPAINT_FOR_IDLE, drm_dev);
} else if (strncmp(opt, "dalprintf", 9) == 0) {
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
DAL_Printf("DAL printf\n");
} else if (strncmp(opt, "dalclean", 8) == 0) {
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
DAL_Clean();
} else if (strncmp(opt, "ata_check", 9) == 0) {
struct drm_crtc *crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPMSG("find crtc fail\n");
return;
}
mtk_crtc_lcm_ATA(crtc);
} else if (strncmp(opt, "path_switch:", 11) == 0) {
struct drm_crtc *crtc;
int path_sel, ret;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
ret = sscanf(opt, "path_switch:%d\n", &path_sel);
mtk_crtc_path_switch(crtc, path_sel, 1);
} else if (strncmp(opt, "enable_idlemgr:", 15) == 0) {
char *p = (char *)opt + 15;
unsigned int flg = 0;
struct drm_crtc *crtc;
int ret;
ret = kstrtouint(p, 0, &flg);
if (ret) {
DDPPR_ERR("%d error to parse cmd %s\n", __LINE__, opt);
return;
}
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_drm_set_idlemgr(crtc, flg, 1);
} else if (strncmp(opt, "idle_wait:", 10) == 0) {
unsigned long long idle_check_interval = 0;
struct drm_crtc *crtc;
int ret;
ret = sscanf(opt, "idle_wait:%llu\n", &idle_check_interval);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n", __LINE__, opt);
return;
}
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
idle_check_interval = max(idle_check_interval, 17ULL);
mtk_drm_set_idle_check_interval(crtc, idle_check_interval);
DDPMSG("change idle interval to %llu ms\n",
idle_check_interval);
} else if (strncmp(opt, "hrt_bw", 6) == 0) {
DDPINFO("HRT test+\n");
#ifdef MTK_FB_MMDVFS_SUPPORT
mtk_disp_hrt_bw_dbg();
#endif
DDPINFO("HRT test-\n");
} else if (strncmp(opt, "lcm0_reset", 10) == 0) {
struct mtk_ddp_comp *comp;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
int enable;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
comp = mtk_ddp_comp_request_output(mtk_crtc);
if (!comp || !comp->funcs || !comp->funcs->io_cmd) {
DDPINFO("cannot find output component\n");
return;
}
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
DDPMSG("lcm0_reset tset\n");
enable = 1;
comp->funcs->io_cmd(comp, NULL, LCM_RESET, &enable);
msleep(20);
enable = 0;
comp->funcs->io_cmd(comp, NULL, LCM_RESET, &enable);
msleep(20);
enable = 1;
comp->funcs->io_cmd(comp, NULL, LCM_RESET, &enable);
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
} else if (strncmp(opt, "backlight:", 10) == 0) {
unsigned int level;
int ret;
ret = sscanf(opt, "backlight:%u\n", &level);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
mtkfb_set_backlight_level(level);
} else if (!strncmp(opt, "aod_bl:", 7)) {
unsigned int level;
int ret;
ret = sscanf(opt, "aod_bl:%u\n", &level);
if (ret != 1) {
pr_info("%d fail to parse cmd %s\n",
__LINE__, opt);
return;
}
mtkfb_set_aod_backlight_level(level);
} else if (strncmp(opt, "dump_fake_engine", 16) == 0) {
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
dump_fake_engine(mtk_crtc->config_regs);
} else if (!strncmp(opt, "fake_engine:", 12)) {
unsigned int en, idx, wr_en, rd_en, wr_pat1, wr_pat2, latency,
preultra_cnt, ultra_cnt;
struct drm_crtc *crtc;
int ret = 0;
ret = sscanf(opt, "fake_engine:%d,%d,%d,%d,%d,%d,%d,%d,%d\n",
&idx, &en, &wr_en, &rd_en, &wr_pat1, &wr_pat2,
&latency, &preultra_cnt, &ultra_cnt);
if (ret != 9) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_drm_idlemgr_kick(__func__, crtc, 1);
mtk_drm_set_idlemgr(crtc, 0, 1);
fake_engine(crtc, idx, en, wr_en, rd_en, wr_pat1, wr_pat2,
latency, preultra_cnt, ultra_cnt);
} else if (strncmp(opt, "checkt", 6) == 0) { /* check trigger */
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
mtk_crtc_check_trigger(mtk_crtc, false, true);
} else if (strncmp(opt, "checkd", 6) == 0) { /* check trigger delay */
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
mtk_crtc_check_trigger(mtk_crtc, true, true);
} else if (!strncmp(opt, "fake_layer:", 11)) {
unsigned int mask;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
int ret = 0;
ret = sscanf(opt, "fake_layer:0x%x\n", &mask);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_drm_idlemgr_kick(__func__, crtc, 1);
mtk_drm_set_idlemgr(crtc, 0, 1);
prepare_fake_layer_buffer(crtc);
mtk_crtc = to_mtk_crtc(crtc);
if (!mask && mtk_crtc->fake_layer.fake_layer_mask)
mtk_crtc->fake_layer.first_dis = true;
mtk_crtc->fake_layer.fake_layer_mask = mask;
DDPINFO("fake_layer:0x%x enable\n", mask);
} else if (!strncmp(opt, "mipi_ccci:", 10)) {
unsigned int en, ret;
ret = sscanf(opt, "mipi_ccci:%d\n", &en);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
DDPINFO("mipi_ccci:%d\n", en);
mtk_disp_mipi_ccci_callback(en, 0);
} else if (strncmp(opt, "aal:", 4) == 0) {
disp_aal_debug(opt + 4);
} else if (strncmp(opt, "aee:", 4) == 0) {
DDPAEE("trigger aee dump of mmproile\n");
} else if (strncmp(opt, "send_ddic_test:", 15) == 0) {
unsigned int case_num, ret;
ret = sscanf(opt, "send_ddic_test:%d\n", &case_num);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
DDPMSG("send_ddic_test:%d\n", case_num);
ddic_dsi_send_cmd_test(case_num);
} else if (strncmp(opt, "read_ddic_test:", 15) == 0) {
unsigned int case_num, ret;
ret = sscanf(opt, "read_ddic_test:%d\n", &case_num);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
DDPMSG("read_ddic_test:%d\n", case_num);
ddic_dsi_read_cmd_test(case_num);
} else if (!strncmp(opt, "chg_mipi:", 9)) {
int ret;
unsigned int rate;
struct drm_crtc *crtc;
ret = sscanf(opt, "chg_mipi:%u\n", &rate);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
DDPMSG("chg_mipi:%u 1\n", rate);
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
DDPMSG("chg_mipi:%u 2\n", rate);
mtk_mipi_clk_change(crtc, rate);
} else if (strncmp(opt, "mipi_volt:", 10) == 0) {
char *p = (char *)opt + 10;
int ret;
ret = kstrtouint(p, 0, &mipi_volt);
if (ret) {
DDPMSG("%d error to parse cmd %s\n", __LINE__, opt);
return;
}
DDPMSG("mipi_volt change :%d\n",
mipi_volt);
} else if (strncmp(opt, "dump_layer:", 11) == 0) {
int ret;
unsigned int dump_en;
unsigned int downSampleX, downSampleY;
DDPMSG("get dump\n");
ret = sscanf(opt, "dump_layer:%d,%d,%d\n", &dump_en,
&downSampleX, &downSampleY);
if (ret != 3) {
DDPMSG("error to parse cmd\n");
return;
}
if (downSampleX)
gCapturePriLayerDownX = downSampleX;
if (downSampleY)
gCapturePriLayerDownY = downSampleY;
gCaptureOVLEn = dump_en;
} else if (strncmp(opt, "dump_user_buffer:", 17) == 0) {
int ret;
unsigned int dump_en;
DDPMSG("get dump\n");
ret = sscanf(opt, "dump_user_buffer:%d\n", &dump_en);
if (ret != 1) {
DDPMSG("error to parse cmd\n");
return;
}
gCaptureWDMAEn = dump_en;
#ifdef CONFIG_MTK_HDMI_SUPPORT
} else if (strncmp(opt, "dptx:", 5) == 0) {
mtk_dp_debug(opt + 5);
} else if (strncmp(opt, "dpintf_dump:", 12) == 0) {
struct mtk_ddp_comp *comp;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
drm_for_each_crtc(crtc, drm_dev) {
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
continue;
}
DDPINFO("------find crtc------");
mtk_crtc = to_mtk_crtc(crtc);
if (!crtc->enabled
|| mtk_crtc->ddp_mode == DDP_NO_USE)
continue;
mtk_crtc = to_mtk_crtc(crtc);
comp = mtk_ddp_comp_request_output(mtk_crtc);
mtk_dp_intf_dump(comp);
}
#endif
} else if (strncmp(opt, "pan_disp_test:", 13) == 0) {
int frame_num, bpp, ret;
ret = sscanf(opt, "pan_disp_test:%d,%d\n", &frame_num, &bpp);
if (ret != 2) {
DDPMSG("%d error to parse cmd %s\n", __LINE__, opt);
return;
}
pan_display_test(frame_num, bpp);
} else if (strncmp(opt, "arr4_enable", 11) == 0) {
struct mtk_ddp_comp *comp;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
int lfr_enable = 1;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
comp = mtk_ddp_comp_request_output(mtk_crtc);
if (comp && comp->funcs && comp->funcs->io_cmd)
comp->funcs->io_cmd(comp, NULL, DSI_LFR_SET, &lfr_enable);
} else if (strncmp(opt, "LFR_update", 10) == 0) {
struct mtk_ddp_comp *comp;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
comp = mtk_ddp_comp_request_output(mtk_crtc);
if (comp && comp->funcs && comp->funcs->io_cmd)
comp->funcs->io_cmd(comp, NULL, DSI_LFR_UPDATE, NULL);
} else if (strncmp(opt, "LFR_status_check", 16) == 0) {
//unsigned int data = mtk_dbg_get_LFR_value();
struct mtk_ddp_comp *comp;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
comp = mtk_ddp_comp_request_output(mtk_crtc);
if (comp && comp->funcs && comp->funcs->io_cmd)
comp->funcs->io_cmd(comp, NULL, DSI_LFR_STATUS_CHECK, NULL);
} else if (strncmp(opt, "tui:", 4) == 0) {
unsigned int en, ret;
ret = sscanf(opt, "tui:%d\n", &en);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n",
__LINE__, opt);
return;
}
if (en)
display_enter_tui();
else
display_exit_tui();
} else if (strncmp(opt, "cwb_en:", 7) == 0) {
unsigned int ret, enable;
/* this debug cmd only for crtc0 */
ret = sscanf(opt, "cwb_en:%d\n", &enable);
if (ret != 1) {
DDPMSG("error to parse cmd\n");
return;
}
mtk_drm_cwb_enable(enable, &user_cwb_funcs, IMAGE_ONLY);
} else if (strncmp(opt, "cwb_roi:", 8) == 0) {
unsigned int ret, offset_x, offset_y, clip_w, clip_h;
struct mtk_rect rect;
/* this debug cmd only for crtc0 */
ret = sscanf(opt, "cwb_roi:%d,%d,%d,%d\n", &offset_x,
&offset_y, &clip_w, &clip_h);
if (ret != 4) {
DDPMSG("error to parse cmd\n");
return;
}
rect.x = offset_x;
rect.y = offset_y;
rect.width = clip_w;
rect.height = clip_h;
mtk_drm_set_cwb_roi(rect);
} else if (strncmp(opt, "cwb:", 4) == 0) {
unsigned int ret, enable, offset_x, offset_y;
unsigned int clip_w, clip_h;
struct mtk_rect rect;
/* this debug cmd only for crtc0 */
ret = sscanf(opt, "cwb:%d,%d,%d,%d,%d\n", &enable,
&offset_x, &offset_y,
&clip_w, &clip_h);
if (ret != 5) {
DDPMSG("error to parse cmd\n");
return;
}
rect.x = offset_x;
rect.y = offset_y;
rect.width = clip_w;
rect.height = clip_h;
mtk_drm_set_cwb_roi(rect);
mtk_drm_cwb_enable(enable, &user_cwb_funcs, IMAGE_ONLY);
} else if (strncmp(opt, "cwb_get_buffer", 14) == 0) {
u8 *user_buffer;
struct drm_crtc *crtc;
struct mtk_drm_crtc *mtk_crtc;
struct mtk_cwb_info *cwb_info;
int width, height, size, ret;
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
mtk_crtc = to_mtk_crtc(crtc);
cwb_info = mtk_crtc->cwb_info;
if (!cwb_info)
return;
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
width = MTK_CWB_NO_EFFECT_HRT_MAX_WIDTH;
height = cwb_info->src_roi.height;
size = sizeof(u8) * width * height * 3;
user_buffer = kzalloc(size, GFP_KERNEL);
mtk_drm_set_cwb_user_buf((void *)user_buffer, IMAGE_ONLY);
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
DDPMSG("[capture] wait frame complete\n");
ret = wait_for_completion_interruptible_timeout(&cwb_cmp,
msecs_to_jiffies(3000));
if (ret > 0)
DDPMSG("[capture] frame complete done\n");
else {
DDPMSG("[capture] wait frame timeout(3s)\n");
DDP_MUTEX_LOCK(&mtk_crtc->lock, __func__, __LINE__);
mtk_drm_set_cwb_user_buf((void *)NULL, IMAGE_ONLY);
DDP_MUTEX_UNLOCK(&mtk_crtc->lock, __func__, __LINE__);
}
kfree(user_buffer);
reinit_completion(&cwb_cmp);
} else if (strncmp(opt, "drm:", 4) == 0) {
disp_drm_debug(opt + 4);
} else if (strncmp(opt, "fake_wcg", 8) == 0) {
unsigned int fake_hdr_en = 0;
struct drm_crtc *crtc;
struct mtk_panel_params *params = NULL;
int ret;
ret = sscanf(opt, "fake_wcg:%u\n", &fake_hdr_en);
if (ret != 1) {
DDPPR_ERR("%d error to parse cmd %s\n", __LINE__, opt);
return;
}
/* this debug cmd only for crtc0 */
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return;
}
params = mtk_drm_get_lcm_ext_params(crtc);
if (!params) {
DDPPR_ERR("[Fake HDR] find lcm ext fail\n");
return;
}
params->lcm_color_mode = (fake_hdr_en) ?
MTK_DRM_COLOR_MODE_DISPLAY_P3 : MTK_DRM_COLOR_MODE_NATIVE;
DDPINFO("set panel color_mode to %d\n", params->lcm_color_mode);
}
}
static void process_dbg_cmd(char *cmd)
{
char *tok;
DDPINFO("[mtkfb_dbg] %s\n", cmd);
while ((tok = strsep(&cmd, " ")) != NULL)
process_dbg_opt(tok);
}
static int debug_open(struct inode *inode, struct file *file)
{
file->private_data = inode->i_private;
return 0;
}
static ssize_t debug_read(struct file *file, char __user *ubuf, size_t count,
loff_t *ppos)
{
int debug_bufmax;
static int n;
if (*ppos != 0 || !is_buffer_init)
goto out;
if (!debug_buffer) {
debug_buffer = vmalloc(sizeof(char) * DEBUG_BUFFER_SIZE);
if (!debug_buffer)
return -ENOMEM;
memset(debug_buffer, 0, sizeof(char) * DEBUG_BUFFER_SIZE);
}
debug_bufmax = DEBUG_BUFFER_SIZE - 1;
n = debug_get_info(debug_buffer, debug_bufmax);
out:
if (n < 0)
return -EINVAL;
return simple_read_from_buffer(ubuf, count, ppos, debug_buffer, n);
}
#if defined(CONFIG_SMCDSD_PANEL)
int mtkfb_debug_show(struct seq_file *m, void *unused)
{
int debug_bufmax;
static int n;
if (!is_buffer_init)
goto out;
if (!debug_buffer) {
debug_buffer = vmalloc(sizeof(char) * DEBUG_BUFFER_SIZE);
if (!debug_buffer)
return -ENOMEM;
memset(debug_buffer, 0, sizeof(char) * DEBUG_BUFFER_SIZE);
}
debug_bufmax = DEBUG_BUFFER_SIZE - 1;
n = debug_get_info(debug_buffer, debug_bufmax);
out:
if (n < 0)
return -EINVAL;
//return simple_read_from_buffer(ubuf, count, ppos, debug_buffer, n);
seq_puts(m, "------ DISPLAY DEBUG INFO (/proc/mtkfb) ------\n");
if (debug_buffer)
seq_puts(m, debug_buffer);
return 0;
}
EXPORT_SYMBOL(mtkfb_debug_show);
#endif
static ssize_t debug_write(struct file *file, const char __user *ubuf,
size_t count, loff_t *ppos)
{
const int debug_bufmax = 512 - 1;
size_t ret;
char cmd_buffer[512];
ret = count;
if (count > debug_bufmax)
count = debug_bufmax;
if (copy_from_user(&cmd_buffer, ubuf, count))
return -EFAULT;
cmd_buffer[count] = 0;
process_dbg_cmd(cmd_buffer);
return ret;
}
static const struct file_operations debug_fops = {
.read = debug_read, .write = debug_write, .open = debug_open,
};
static int idletime_set(void *data, u64 val)
{
struct drm_crtc *crtc;
u64 ret = 0;
if (val < 33)
val = 33;
if (val > 1000000)
val = 1000000;
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return -ENODEV;
}
ret = mtk_drm_set_idle_check_interval(crtc, val);
if (ret == 0)
return -ENODEV;
return 0;
}
static int idletime_get(void *data, u64 *val)
{
struct drm_crtc *crtc;
if (!drm_dev) {
DDPMSG("%s:%d, drm_dev is NULL\n",
__func__, __LINE__);
return -ENODEV;
}
crtc = list_first_entry(&(drm_dev)->mode_config.crtc_list,
typeof(*crtc), head);
if (!crtc) {
DDPPR_ERR("find crtc fail\n");
return -ENODEV;
}
*val = mtk_drm_get_idle_check_interval(crtc);
if (*val == 0)
return -ENODEV;
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(idletime_fops, idletime_get, idletime_set, "%llu\n");
int disp_met_set(void *data, u64 val)
{
/*1 enable ; 0 disable*/
disp_met_en = val;
return 0;
}
static int disp_met_get(void *data, u64 *val)
{
*val = disp_met_en;
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(disp_met_fops, disp_met_get, disp_met_set, "%llu\n");
int disp_lfr_dbg_set(void *data, u64 val)
{
atomic_set(&lfr_dbg, val);
return 0;
}
static int disp_lfr_dbg_get(void *data, u64 *val)
{
*val = atomic_read(&lfr_dbg);
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(disp_lfr_dbg_fops, disp_lfr_dbg_get,
disp_lfr_dbg_set, "%llu\n");
int disp_lfr_params_set(void *data, u64 val)
{
atomic_set(&lfr_params, val);
return 0;
}
static int disp_lfr_params_get(void *data, u64 *val)
{
*val = atomic_read(&lfr_params);
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(disp_lfr_params_fops, disp_lfr_params_get,
disp_lfr_params_set, "%llu\n");
unsigned int mtk_dbg_get_lfr_mode_value(void)
{
unsigned int lfr_mode = (atomic_read(&lfr_params) & 0x03);
lfr_mode = lfr_mode & 0x03;
return lfr_mode;
}
unsigned int mtk_dbg_get_lfr_type_value(void)
{
unsigned int lfr_type = (atomic_read(&lfr_params));
lfr_type = (lfr_type & 0x0C) >> 2;
return lfr_type;
}
unsigned int mtk_dbg_get_lfr_enable_value(void)
{
unsigned int lfr_enable = atomic_read(&lfr_params);
lfr_enable = (lfr_enable & 0x10) >> 4;
return lfr_enable;
}
unsigned int mtk_dbg_get_lfr_update_value(void)
{
unsigned int lfr_update = atomic_read(&lfr_params);
lfr_update = (lfr_update & 0x20) >> 5;
return lfr_update;
}
unsigned int mtk_dbg_get_lfr_vse_dis_value(void)
{
unsigned int lfr_vse_dis = atomic_read(&lfr_params);
lfr_vse_dis = (lfr_vse_dis & 0x40) >> 6;
return lfr_vse_dis;
}
unsigned int mtk_dbg_get_lfr_skip_num_value(void)
{
unsigned int lfr_skip_num = atomic_read(&lfr_params);
lfr_skip_num = (lfr_skip_num & 0x3F00) >> 8;
return lfr_skip_num;
}
unsigned int mtk_dbg_get_lfr_dbg_value(void)
{
return atomic_read(&lfr_dbg);
}
static void backup_vfp_for_lp_cust(u64 vfp)
{
vfp_backup = vfp;
}
static u64 get_backup_vfp(void)
{
return vfp_backup;
}
static int idlevfp_set(void *data, u64 val)
{
if (val > 4095)
val = 4095;
backup_vfp_for_lp_cust((unsigned int)val);
return 0;
}
static int idlevfp_get(void *data, u64 *val)
{
*val = (u64)get_backup_vfp();
return 0;
}
DEFINE_SIMPLE_ATTRIBUTE(idlevfp_fops, idlevfp_get, idlevfp_set, "%llu\n");
void disp_dbg_probe(void)
{
#if IS_ENABLED(CONFIG_DEBUG_FS)
struct dentry *d_folder;
struct dentry *d_file;
mtkfb_dbgfs = debugfs_create_file("mtkfb", S_IFREG | 0444, NULL,
NULL, &debug_fops);
d_folder = debugfs_create_dir("displowpower", NULL);
if (d_folder) {
d_file = debugfs_create_file("idletime", S_IFREG | 0644,
d_folder, NULL, &idletime_fops);
}
d_folder = debugfs_create_dir("mtkfb_debug", NULL);
if (d_folder) {
d_file = debugfs_create_file("disp_met", S_IFREG | 0644,
d_folder, NULL, &disp_met_fops);
}
if (d_folder) {
d_file = debugfs_create_file("disp_lfr_dbg",
S_IFREG | 0644, d_folder, NULL, &disp_lfr_dbg_fops);
d_file = debugfs_create_file("disp_lfr_params",
S_IFREG | 0644, d_folder, NULL, &disp_lfr_params_fops);
}
init_log_buffer();
drm_mmp_init();
#endif
#if IS_ENABLED(CONFIG_PROC_FS)
mtkfb_procfs = proc_create("mtkfb", S_IFREG | 0440,
NULL,
&debug_fops);
if (!mtkfb_procfs) {
pr_info("[%s %d]failed to create mtkfb in /proc/disp_ddp\n",
__func__, __LINE__);
goto out;
}
disp_lowpower_proc = proc_mkdir("displowpower", NULL);
if (!disp_lowpower_proc) {
pr_info("[%s %d]failed to create dir: /proc/displowpower\n",
__func__, __LINE__);
goto out;
}
if (!proc_create("idletime", S_IFREG | 0440,
disp_lowpower_proc, &idletime_fops)) {
pr_info("[%s %d]failed to create idletime in /proc/displowpower\n",
__func__, __LINE__);
goto out;
}
if (!proc_create("idlevfp", S_IFREG | 0440,
disp_lowpower_proc, &idlevfp_fops)) {
pr_info("[%s %d]failed to create idlevfp in /proc/displowpower\n",
__func__, __LINE__);
goto out;
}
mtkfb_debug_procfs = proc_mkdir("mtkfb_debug", NULL);
if (!mtkfb_debug_procfs) {
pr_info("[%s %d]failed to create dir: /proc/mtkfb_debug\n",
__func__, __LINE__);
goto out;
}
if (!proc_create("disp_met", S_IFREG | 0440,
mtkfb_debug_procfs, &disp_met_fops)) {
pr_info("[%s %d]failed to create idlevfp in /proc/mtkfb_debug/disp_met\n",
__func__, __LINE__);
goto out;
}
if (!proc_create("disp_lfr_dbg", S_IFREG | 0440,
mtkfb_debug_procfs, &disp_lfr_dbg_fops)) {
pr_info("[%s %d]failed to create idlevfp in /proc/mtkfb_debug/disp_lfr_dbg\n",
__func__, __LINE__);
goto out;
}
if (!proc_create("disp_lfr_params", S_IFREG | 0444,
mtkfb_debug_procfs, &disp_lfr_params_fops)) {
pr_info("[%s %d]failed to create idlevfp in /proc/mtkfb_debug/disp_lfr_params\n",
__func__, __LINE__);
goto out;
}
#endif
#if IS_ENABLED(CONFIG_MTK_HDMI_SUPPORT)
mtk_dp_debugfs_init();
#endif
out:
return;
}
void disp_dbg_init(struct drm_device *dev)
{
drm_dev = dev;
init_completion(&cwb_cmp);
}
void disp_dbg_deinit(void)
{
if (debug_buffer)
vfree(debug_buffer);
#if IS_ENABLED(CONFIG_DEBUG_FS)
debugfs_remove(mtkfb_dbgfs);
#endif
#if IS_ENABLED(CONFIG_PROC_FS)
if (mtkfb_procfs) {
proc_remove(mtkfb_procfs);
mtkfb_procfs = NULL;
}
if (disp_lowpower_proc) {
proc_remove(disp_lowpower_proc);
disp_lowpower_proc = NULL;
}
#endif
#if IS_ENABLED(CONFIG_MTK_HDMI_SUPPORT)
mtk_dp_debugfs_deinit();
#endif
}
void get_disp_dbg_buffer(unsigned long *addr, unsigned long *size,
unsigned long *start)
{
init_log_buffer();
if (is_buffer_init) {
*addr = (unsigned long)err_buffer[0];
*size = (DEBUG_BUFFER_SIZE - 4096);
*start = 0;
} else {
*addr = 0;
*size = 0;
*start = 0;
}
}
int mtk_disp_ioctl_debug_log_switch(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
unsigned int switch_log = *(unsigned int *)data;
DDPMSG("%d:%s():switch_log=%d\n", __LINE__, __func__, switch_log);
if (switch_log == MTK_DRM_MOBILE_LOG)
g_mobile_log = 1;
else if (switch_log == MTK_DRM_DETAIL_LOG)
g_detail_log = 1;
else if (switch_log == MTK_DRM_FENCE_LOG)
g_fence_log = 1;
else if (switch_log == MTK_DRM_IRQ_LOG)
g_irq_log = 1;
return 0;
}