1142 lines
30 KiB
C
1142 lines
30 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright (c) 2019 MediaTek Inc.
|
|
*/
|
|
|
|
#include <linux/delay.h>
|
|
#include <linux/sched.h>
|
|
#include <uapi/linux/sched/types.h>
|
|
#include <linux/semaphore.h>
|
|
#include <linux/module.h>
|
|
#include <linux/wait.h>
|
|
#include <linux/kthread.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/types.h>
|
|
#include <linux/ktime.h>
|
|
#include <linux/of.h>
|
|
#include <linux/of_irq.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/interrupt.h>
|
|
#if defined(CONFIG_MTK_ION)
|
|
#include "ion_drv.h"
|
|
#include "mtk_ion.h"
|
|
#endif
|
|
#if defined(CONFIG_MTK_M4U)
|
|
#include "m4u.h"
|
|
#endif
|
|
|
|
#include "disp_drv_platform.h"
|
|
#include "debug.h"
|
|
#include "ddp_debug.h"
|
|
#include "disp_drv_log.h"
|
|
#include "disp_lcm.h"
|
|
#include "disp_utils.h"
|
|
#include "disp_session.h"
|
|
#include "primary_display.h"
|
|
#include "disp_helper.h"
|
|
#if defined(CONFIG_MTK_CMDQ)
|
|
#include "cmdq_def.h"
|
|
#include "cmdq_record.h"
|
|
#include "cmdq_reg.h"
|
|
#include "cmdq_core.h"
|
|
#endif
|
|
#include "ddp_manager.h"
|
|
#include "disp_lcm.h"
|
|
#include "ddp_clkmgr.h"
|
|
/* #include "mmdvfs_mgr.h" */
|
|
#include "disp_drv_log.h"
|
|
#include "ddp_log.h"
|
|
#include "disp_lowpower.h"
|
|
/* device tree */
|
|
#include <linux/of.h>
|
|
#include <linux/of_irq.h>
|
|
#include <linux/of_address.h>
|
|
#include <linux/io.h>
|
|
/* #include "mach/eint.h" */
|
|
#if defined(CONFIG_MTK_LEGACY)
|
|
# include <mach/mtk_gpio.h>
|
|
# include <cust_gpio_usage.h>
|
|
# include <cust_eint.h>
|
|
#else
|
|
# include "disp_dts_gpio.h"
|
|
# include <linux/gpio.h>
|
|
#endif /* CONFIG_MTK_LEGACY */
|
|
|
|
#include "disp_recovery.h"
|
|
#include "disp_partial.h"
|
|
#include "ddp_dsi.h"
|
|
#include "ddp_reg_mmsys.h"
|
|
#include "ddp_reg.h"
|
|
|
|
/* For abnormal check */
|
|
static struct task_struct *primary_display_check_task;
|
|
static struct task_struct *primary_display_recovery_thread;
|
|
/* used for blocking check task */
|
|
static wait_queue_head_t _check_task_wq;
|
|
/* For Check Task */
|
|
static atomic_t _check_task_wakeup = ATOMIC_INIT(0);
|
|
|
|
/* For EXT TE EINT Check */
|
|
static wait_queue_head_t esd_ext_te_wq;
|
|
/* For EXT TE EINT Check */
|
|
static atomic_t esd_ext_te_event = ATOMIC_INIT(0);
|
|
static unsigned int esd_check_mode;
|
|
static unsigned int esd_check_enable;
|
|
static int te_irq;
|
|
|
|
#if defined(CONFIG_MTK_DUAL_DISPLAY_SUPPORT) && \
|
|
(CONFIG_MTK_DUAL_DISPLAY_SUPPORT == 2)
|
|
/********** external display dual LCM ESD check *****************/
|
|
/* For abnormal check */
|
|
static struct task_struct *external_display_check_task;
|
|
/* used for blocking check task */
|
|
static wait_queue_head_t extd_check_task_wq;
|
|
/* For Check Task */
|
|
static atomic_t extd_check_task_wakeup = ATOMIC_INIT(0);
|
|
|
|
/* For EXT TE EINT Check */
|
|
static wait_queue_head_t esd_ext_te_1_wq;
|
|
/* For EXT TE EINT Check */
|
|
static atomic_t esd_ext_te_1_event = ATOMIC_INIT(0);
|
|
static unsigned int extd_esd_check_mode;
|
|
static unsigned int extd_esd_check_enable;
|
|
#endif
|
|
|
|
atomic_t enable_ovl0_recovery = ATOMIC_INIT(0);
|
|
atomic_t enable_ovl0_2l_recovery = ATOMIC_INIT(0);
|
|
|
|
unsigned int get_esd_check_mode(void)
|
|
{
|
|
return esd_check_mode;
|
|
}
|
|
|
|
void set_esd_check_mode(unsigned int mode)
|
|
{
|
|
esd_check_mode = mode;
|
|
}
|
|
|
|
static unsigned int _can_switch_check_mode(void)
|
|
{
|
|
int ret = 0;
|
|
struct LCM_PARAMS *params;
|
|
|
|
params = primary_get_lcm()->params;
|
|
if (params->dsi.customization_esd_check_enable == 0 &&
|
|
params->dsi.lcm_esd_check_table[0].cmd != 0 &&
|
|
disp_helper_get_option(DISP_OPT_ESD_CHECK_SWITCH))
|
|
ret = 1;
|
|
return ret;
|
|
}
|
|
|
|
static unsigned int _lcm_need_esd_check(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (primary_get_lcm()->params->dsi.esd_check_enable == 1) {
|
|
#ifdef CONFIG_OF
|
|
if (islcmconnected == 1)
|
|
ret = 1;
|
|
#else
|
|
ret = 1;
|
|
#endif
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* For Cmd Mode Read LCM Check
|
|
* Config cmdq_handle_config_esd
|
|
*
|
|
* @return value:
|
|
* 0: success
|
|
*/
|
|
int _esd_check_config_handle_cmd(struct cmdqRecStruct *qhandle)
|
|
{
|
|
int ret = 0;
|
|
disp_path_handle phandle = primary_get_dpmgr_handle();
|
|
|
|
/* 1.reset */
|
|
cmdqRecReset(qhandle);
|
|
primary_display_manual_lock();
|
|
|
|
/* 2.write first instruction */
|
|
/*
|
|
* cmd mode: wait CMDQ_SYNC_TOKEN_STREAM_EOF
|
|
* (wait trigger thread done)
|
|
*/
|
|
cmdqRecWaitNoClear(qhandle, CMDQ_SYNC_TOKEN_STREAM_EOF);
|
|
|
|
/*
|
|
* 3.clear CMDQ_SYNC_TOKEN_ESD_EOF
|
|
* (trigger thread need wait this sync token)
|
|
*/
|
|
cmdqRecClearEventToken(qhandle, CMDQ_SYNC_TOKEN_ESD_EOF);
|
|
|
|
/* 4.write instruction(read from lcm) */
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_ESD_CHECK_READ, 0);
|
|
|
|
/* 5.set CMDQ_SYNC_TOKE_ESD_EOF(trigger thread can work now) */
|
|
cmdqRecSetEventToken(qhandle, CMDQ_SYNC_TOKEN_ESD_EOF);
|
|
|
|
primary_display_manual_unlock();
|
|
|
|
/* 6.flush instruction */
|
|
dprec_logger_start(DPREC_LOGGER_ESD_CMDQ, 0, 0);
|
|
ret = cmdqRecFlush(qhandle);
|
|
dprec_logger_done(DPREC_LOGGER_ESD_CMDQ, 0, 0);
|
|
|
|
DISPINFO("[ESD]%s ret=%d\n", __func__, ret);
|
|
|
|
if (ret)
|
|
ret = 1;
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* For Vdo Mode Read LCM Check
|
|
* Config cmdq_handle_config_esd
|
|
* return value: 0:success, 1:fail
|
|
*/
|
|
int _esd_check_config_handle_vdo(struct cmdqRecStruct *qhandle)
|
|
{
|
|
int ret = 0;
|
|
disp_path_handle phandle = primary_get_dpmgr_handle();
|
|
|
|
/* 1.reset */
|
|
cmdqRecReset(qhandle);
|
|
|
|
/* wait stream eof first */
|
|
/* cmdqRecWait(qhandle, CMDQ_EVENT_DISP_RDMA0_EOF); */
|
|
cmdqRecWait(qhandle, CMDQ_EVENT_MUTEX0_STREAM_EOF);
|
|
|
|
primary_display_manual_lock();
|
|
|
|
/* 2.stop dsi vdo mode */
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_STOP_VDO_MODE, 0);
|
|
|
|
/* 3.write instruction(read from lcm) */
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_ESD_CHECK_READ, 0);
|
|
|
|
/* 4.start dsi vdo mode */
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_START_VDO_MODE, 0);
|
|
cmdqRecClearEventToken(qhandle, CMDQ_EVENT_MUTEX0_STREAM_EOF);
|
|
/* cmdqRecClearEventToken(qhandle, CMDQ_EVENT_DISP_RDMA0_EOF); */
|
|
/* 5.trigger path */
|
|
dpmgr_path_trigger(phandle, qhandle, CMDQ_ENABLE);
|
|
|
|
/* mutex sof wait*/
|
|
ddp_mutex_set_sof_wait(dpmgr_path_get_mutex(phandle), qhandle, 0);
|
|
|
|
primary_display_manual_unlock();
|
|
|
|
/* 6.flush instruction */
|
|
dprec_logger_start(DPREC_LOGGER_ESD_CMDQ, 0, 0);
|
|
ret = cmdqRecFlush(qhandle);
|
|
dprec_logger_done(DPREC_LOGGER_ESD_CMDQ, 0, 0);
|
|
|
|
DISPINFO("[ESD]%s ret=%d\n", __func__, ret);
|
|
|
|
if (ret)
|
|
ret = 1;
|
|
return ret;
|
|
}
|
|
|
|
/* For EXT TE EINT Check */
|
|
static irqreturn_t _esd_check_ext_te_irq_handler(int irq, void *data)
|
|
{
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_vdo_eint,
|
|
MMPROFILE_FLAG_PULSE, 0, 0);
|
|
atomic_set(&esd_ext_te_event, 1);
|
|
wake_up_interruptible(&esd_ext_te_wq);
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
void primary_display_switch_esd_mode(int mode)
|
|
{
|
|
if (mode == GPIO_EINT_MODE)
|
|
enable_irq(te_irq);
|
|
else if (mode == GPIO_DSI_MODE)
|
|
disable_irq(te_irq);
|
|
}
|
|
|
|
int do_esd_check_eint(void)
|
|
{
|
|
int ret = 0;
|
|
mmp_event mmp_te = ddp_mmp_get_events()->esd_extte;
|
|
|
|
DISPINFO("[ESD]ESD check eint\n");
|
|
mmprofile_log_ex(mmp_te, MMPROFILE_FLAG_PULSE,
|
|
primary_display_is_video_mode(), GPIO_EINT_MODE);
|
|
primary_display_switch_esd_mode(GPIO_EINT_MODE);
|
|
|
|
if (wait_event_interruptible_timeout(esd_ext_te_wq,
|
|
atomic_read(&esd_ext_te_event),
|
|
HZ / 2) > 0)
|
|
ret = 0; /* esd check pass */
|
|
else
|
|
ret = 1; /* esd check fail */
|
|
|
|
atomic_set(&esd_ext_te_event, 0);
|
|
|
|
primary_display_switch_esd_mode(GPIO_DSI_MODE);
|
|
|
|
return ret;
|
|
}
|
|
|
|
int do_esd_check_dsi_te(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (dpmgr_wait_event_timeout(primary_get_dpmgr_handle(),
|
|
DISP_PATH_EVENT_IF_VSYNC, HZ / 2) > 0)
|
|
ret = 0; /* esd check pass */
|
|
else
|
|
ret = 1; /* esd check fail */
|
|
|
|
return ret;
|
|
}
|
|
|
|
int do_esd_check_read(void)
|
|
{
|
|
int ret = 0;
|
|
struct cmdqRecStruct *qhandle = NULL;
|
|
disp_path_handle phandle = primary_get_dpmgr_handle();
|
|
mmp_event mmp_te = ddp_mmp_get_events()->esd_extte;
|
|
|
|
DISPCHECK("[ESD]ESD check read\n");
|
|
mmprofile_log_ex(mmp_te, MMPROFILE_FLAG_PULSE,
|
|
primary_display_is_video_mode(), GPIO_DSI_MODE);
|
|
|
|
/* only cmd mode read & with disable mmsys clk will kick */
|
|
if (disp_helper_get_option(DISP_OPT_IDLEMGR_ENTER_ULPS) &&
|
|
!primary_display_is_video_mode())
|
|
primary_display_idlemgr_kick((char *)__func__, 1);
|
|
|
|
/* 0.create esd check cmdq */
|
|
ret = cmdqRecCreate(CMDQ_SCENARIO_DISP_ESD_CHECK, &qhandle);
|
|
if (ret) {
|
|
DISP_PR_ERR("%s:%d, create cmdq handle fail!ret=%d\n",
|
|
__func__, __LINE__, ret);
|
|
return -1;
|
|
}
|
|
cmdqRecReset(qhandle);
|
|
|
|
primary_display_manual_lock();
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_ESD_ALLC_SLOT, 0);
|
|
primary_display_manual_unlock();
|
|
|
|
/* 1.use cmdq to read from lcm */
|
|
if (primary_display_is_video_mode())
|
|
ret = _esd_check_config_handle_vdo(qhandle);
|
|
else
|
|
ret = _esd_check_config_handle_cmd(qhandle);
|
|
|
|
primary_display_manual_lock();
|
|
|
|
if (ret == 1) { /* cmdq fail */
|
|
if (need_wait_esd_eof()) {
|
|
/*
|
|
* Need set esd check eof sync_token to
|
|
* let trigger loop go.
|
|
*/
|
|
cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_ESD_EOF);
|
|
}
|
|
/* do dsi reset */
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_DSI_RESET, 0);
|
|
goto destroy_cmdq;
|
|
}
|
|
|
|
/* 2.check data(*cpu check now) */
|
|
ret = dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_ESD_CHECK_CMP, 0);
|
|
if (ret)
|
|
ret = 1; /* esd check fail */
|
|
|
|
destroy_cmdq:
|
|
dpmgr_path_build_cmdq(phandle, qhandle, CMDQ_ESD_FREE_SLOT, 0);
|
|
|
|
primary_display_manual_unlock();
|
|
|
|
/* 3.destroy esd config thread */
|
|
cmdqRecDestroy(qhandle);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* primary_display_esd_check - ESD CHECK FUNCTION
|
|
* return 1: esd check fail
|
|
* return 0: esd check pass
|
|
*/
|
|
int primary_display_esd_check(void)
|
|
{
|
|
int ret = 0;
|
|
unsigned int mode;
|
|
mmp_event mmp_te = ddp_mmp_get_events()->esd_extte;
|
|
mmp_event mmp_rd = ddp_mmp_get_events()->esd_rdlcm;
|
|
mmp_event mmp_chk = ddp_mmp_get_events()->esd_check_t;
|
|
struct LCM_PARAMS *params;
|
|
|
|
dprec_logger_start(DPREC_LOGGER_ESD_CHECK, 0, 0);
|
|
mmprofile_log_ex(mmp_chk, MMPROFILE_FLAG_START, 0, 0);
|
|
DISPINFO("[ESD]ESD check begin\n");
|
|
|
|
primary_display_manual_lock();
|
|
if (primary_get_state() == DISP_SLEPT) {
|
|
mmprofile_log_ex(mmp_chk, MMPROFILE_FLAG_PULSE, 1, 0);
|
|
DISPINFO("[ESD]Primary DISP slept. Skip esd check\n");
|
|
primary_display_manual_unlock();
|
|
goto done;
|
|
}
|
|
primary_display_manual_unlock();
|
|
|
|
/* Esd Check: EXT TE */
|
|
params = primary_get_lcm()->params;
|
|
if (params->dsi.customization_esd_check_enable == 0) {
|
|
/* use TE for esd check */
|
|
mmprofile_log_ex(mmp_te, MMPROFILE_FLAG_START, 0, 0);
|
|
|
|
if (primary_display_is_video_mode()) {
|
|
mode = get_esd_check_mode();
|
|
if (mode == GPIO_EINT_MODE) {
|
|
ret = do_esd_check_eint();
|
|
if (_can_switch_check_mode())
|
|
set_esd_check_mode(GPIO_DSI_MODE);
|
|
} else {
|
|
ret = do_esd_check_read();
|
|
if (_can_switch_check_mode())
|
|
set_esd_check_mode(GPIO_EINT_MODE);
|
|
}
|
|
} else
|
|
ret = do_esd_check_eint();
|
|
|
|
mmprofile_log_ex(mmp_te, MMPROFILE_FLAG_END, 0, ret);
|
|
|
|
goto done;
|
|
}
|
|
|
|
/* Esd Check: Read from lcm */
|
|
mmprofile_log_ex(mmp_rd, MMPROFILE_FLAG_START,
|
|
0, primary_display_cmdq_enabled());
|
|
|
|
if (primary_display_cmdq_enabled() == 0) {
|
|
DISPCHECK("[ESD]not support cpu read do esd check\n");
|
|
mmprofile_log_ex(mmp_rd, MMPROFILE_FLAG_END, 0, ret);
|
|
goto done;
|
|
}
|
|
|
|
mmprofile_log_ex(mmp_rd, MMPROFILE_FLAG_PULSE,
|
|
0, primary_display_is_video_mode());
|
|
|
|
/* only cmd mode read & with disable mmsys clk will kick */
|
|
ret = do_esd_check_read();
|
|
|
|
mmprofile_log_ex(mmp_rd, MMPROFILE_FLAG_END, 0, ret);
|
|
|
|
done:
|
|
DISPINFO("[ESD]ESD check %s\n", ret ? "fail" : "pass");
|
|
mmprofile_log_ex(mmp_chk, MMPROFILE_FLAG_END, 0, ret);
|
|
dprec_logger_done(DPREC_LOGGER_ESD_CHECK, 0, 0);
|
|
return ret;
|
|
}
|
|
|
|
static int primary_display_check_recovery_worker_kthread(void *data)
|
|
{
|
|
struct sched_param param = { .sched_priority = 87 };
|
|
int ret = 0;
|
|
int i = 0;
|
|
int esd_try_cnt = 5;
|
|
int recovery_done = 0;
|
|
|
|
DISPFUNC();
|
|
sched_setscheduler(current, SCHED_RR, ¶m);
|
|
|
|
while (1) {
|
|
msleep(2000); /* 2s */
|
|
ret = wait_event_interruptible(_check_task_wq,
|
|
atomic_read(&_check_task_wakeup));
|
|
if (ret < 0) {
|
|
DISPINFO("[ESD]check thread waked up accidently\n");
|
|
continue;
|
|
}
|
|
|
|
_primary_path_switch_dst_lock();
|
|
|
|
/* 1.esd check & recovery */
|
|
if (!esd_check_enable) {
|
|
_primary_path_switch_dst_unlock();
|
|
continue;
|
|
}
|
|
|
|
i = 0; /* repeat */
|
|
do {
|
|
ret = primary_display_esd_check();
|
|
if (!ret) /* 0:success */
|
|
break;
|
|
|
|
DISP_PR_ERR("[ESD]check fail, do esd recovery.try=%d\n",
|
|
i);
|
|
primary_display_esd_recovery();
|
|
recovery_done = 1;
|
|
} while (++i < esd_try_cnt);
|
|
|
|
if (ret == 1) {
|
|
const int len = 120;
|
|
char msg[len];
|
|
int n = 0;
|
|
|
|
n = scnprintf(msg, len,
|
|
"[ESD]LCM recover fail. Try time:%d. Disable esd check\n",
|
|
esd_try_cnt);
|
|
DISP_PR_ERR("%s", msg);
|
|
primary_display_esd_check_enable(0);
|
|
} else if (recovery_done == 1) {
|
|
DISPCHECK("[ESD]esd recovery success\n");
|
|
recovery_done = 0;
|
|
}
|
|
|
|
_primary_path_switch_dst_unlock();
|
|
|
|
/* 2.other check & recovery */
|
|
|
|
if (kthread_should_stop())
|
|
break;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* ESD RECOVERY */
|
|
int primary_display_esd_recovery(void)
|
|
{
|
|
enum DISP_STATUS ret = DISP_STATUS_OK;
|
|
struct LCM_PARAMS *lcm_param = NULL;
|
|
mmp_event mmp_r = ddp_mmp_get_events()->esd_recovery_t;
|
|
|
|
DISPFUNC();
|
|
dprec_logger_start(DPREC_LOGGER_ESD_RECOVERY, 0, 0);
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_START, 0, 0);
|
|
DISPCHECK("[ESD]ESD recovery begin\n");
|
|
|
|
primary_display_manual_lock();
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE,
|
|
primary_display_is_video_mode(), 1);
|
|
|
|
lcm_param = disp_lcm_get_params(primary_get_lcm());
|
|
if (primary_get_state() == DISP_SLEPT) {
|
|
DISPCHECK("[ESD]Primary DISP is slept, skip esd recovery\n");
|
|
goto done;
|
|
}
|
|
|
|
primary_display_idlemgr_kick((char *)__func__, 0);
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 2);
|
|
|
|
/* blocking flush before stop trigger loop */
|
|
_blocking_flush();
|
|
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 3);
|
|
|
|
DISPINFO("[ESD]display cmdq trigger loop stop[begin]\n");
|
|
_cmdq_stop_trigger_loop();
|
|
DISPINFO("[ESD]display cmdq trigger loop stop[end]\n");
|
|
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 4);
|
|
|
|
DISPDBG("[ESD]stop dpmgr path[begin]\n");
|
|
dpmgr_path_stop(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
DISPCHECK("[ESD]stop dpmgr path[end]\n");
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 0xff);
|
|
|
|
if (dpmgr_path_is_busy(primary_get_dpmgr_handle())) {
|
|
DISPCHECK("[ESD]primary display path is busy after stop\n");
|
|
dpmgr_wait_event_timeout(primary_get_dpmgr_handle(),
|
|
DISP_PATH_EVENT_FRAME_DONE, HZ * 1);
|
|
DISPCHECK("[ESD]wait frame done ret:%d\n", ret);
|
|
}
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 5);
|
|
|
|
DISPDBG("[ESD]reset display path[begin]\n");
|
|
dpmgr_path_reset(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
DISPCHECK("[ESD]reset display path[end]\n");
|
|
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 6);
|
|
|
|
DISPDBG("[POWER]lcm suspend[begin]\n");
|
|
/* after dsi_stop, we should enable the dsi basic irq. */
|
|
dsi_basic_irq_enable(DISP_MODULE_DSI0, NULL);
|
|
disp_lcm_suspend(primary_get_lcm());
|
|
DISPCHECK("[POWER]lcm suspend[end]\n");
|
|
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 7);
|
|
|
|
DISPDBG("[ESD]dsi power reset[begine]\n");
|
|
dpmgr_path_dsi_power_off(primary_get_dpmgr_handle(), NULL);
|
|
dpmgr_path_dsi_power_on(primary_get_dpmgr_handle(), NULL);
|
|
if (!primary_display_is_video_mode())
|
|
dpmgr_path_ioctl(primary_get_dpmgr_handle(), NULL,
|
|
DDP_DSI_ENABLE_TE, NULL);
|
|
DISPCHECK("[ESD]dsi power reset[end]\n");
|
|
|
|
DISPDBG("[ESD]lcm recover[begin]\n");
|
|
disp_lcm_esd_recover(primary_get_lcm());
|
|
DISPCHECK("[ESD]lcm recover[end]\n");
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 8);
|
|
|
|
DISPDBG("[ESD]start dpmgr path[begin]\n");
|
|
if (disp_partial_is_support()) {
|
|
struct disp_ddp_path_config *data_config =
|
|
dpmgr_path_get_last_config(primary_get_dpmgr_handle());
|
|
|
|
primary_display_config_full_roi(data_config,
|
|
primary_get_dpmgr_handle(), NULL);
|
|
}
|
|
dpmgr_path_start(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
DISPCHECK("[ESD]start dpmgr path[end]\n");
|
|
|
|
if (dpmgr_path_is_busy(primary_get_dpmgr_handle())) {
|
|
DISP_PR_ERR("[ESD]Main display busy before triggering SOF\n");
|
|
ret = -1;
|
|
/* goto done; */
|
|
}
|
|
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 9);
|
|
DISPDBG("[ESD]start cmdq trigger loop[begin]\n");
|
|
_cmdq_start_trigger_loop();
|
|
DISPCHECK("[ESD]start cmdq trigger loop[end]\n");
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 10);
|
|
if (primary_display_is_video_mode()) {
|
|
/*
|
|
* for video mode, we need to force trigger here
|
|
* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW
|
|
* when trigger loop start
|
|
*/
|
|
dpmgr_path_trigger(primary_get_dpmgr_handle(), NULL,
|
|
CMDQ_DISABLE);
|
|
}
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_PULSE, 0, 11);
|
|
|
|
/*
|
|
* (in suspend) when we stop trigger loop
|
|
* if no other thread is running, cmdq may disable its clock
|
|
* all cmdq event will be cleared after suspend
|
|
*/
|
|
cmdqCoreSetEvent(CMDQ_EVENT_DISP_WDMA0_EOF);
|
|
|
|
/* set dirty to trigger one frame -- cmd mode */
|
|
if (!primary_display_is_video_mode()) {
|
|
cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
|
|
mdelay(40);
|
|
}
|
|
|
|
done:
|
|
primary_display_manual_unlock();
|
|
DISPCHECK("[ESD]ESD recovery end\n");
|
|
mmprofile_log_ex(mmp_r, MMPROFILE_FLAG_END, 0, 0);
|
|
dprec_logger_done(DPREC_LOGGER_ESD_RECOVERY, 0, 0);
|
|
return ret;
|
|
}
|
|
|
|
/* add for display recovery */
|
|
int primary_display_ovl_recovery(void)
|
|
{
|
|
enum DISP_STATUS ret = DISP_STATUS_OK;
|
|
struct disp_ddp_path_config *pconfig;
|
|
struct ddp_io_golden_setting_arg io_gs;
|
|
|
|
DISPFUNC();
|
|
DISPCHECK("[%s]begin\n", __func__);
|
|
|
|
primary_display_manual_lock();
|
|
|
|
if (primary_get_state() == DISP_SLEPT) {
|
|
DISPCHECK("[%s]primary display is slept?\n", __func__);
|
|
goto done;
|
|
}
|
|
|
|
primary_display_idlemgr_kick((char *)__func__, 0);
|
|
|
|
if (pgc->session_mode != DISP_SESSION_DIRECT_LINK_MODE) {
|
|
DISPCHECK("[%s]path is not DL, skip recovery\n", __func__);
|
|
goto done;
|
|
}
|
|
|
|
DISPDBG("[%s]cmdq trigger loop stop[begin]\n", __func__);
|
|
_cmdq_stop_trigger_loop();
|
|
DISPCHECK("[%s]cmdq trigger loop stop[end]\n", __func__);
|
|
|
|
DISPDBG("[%s]stop dpmgr path[begin]\n", __func__);
|
|
dpmgr_path_stop(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
DISPCHECK("[%s]stop dpmgr path[end]\n", __func__);
|
|
|
|
if (dpmgr_path_is_busy(primary_get_dpmgr_handle()))
|
|
DISP_PR_ERR("[%s]display path is busy after stop\n",
|
|
__func__);
|
|
|
|
DISPDBG("[%s]reset display path[begin]\n", __func__);
|
|
|
|
ddp_path_mmsys_sw_reset(0, 14); /* reset OVL0_2L */
|
|
ddp_path_mmsys_sw_reset(0, 13); /* reset OVL0 */
|
|
ddp_path_mmsys_sw_reset(1, 18); /* reset PVRIC core */
|
|
|
|
dpmgr_path_reset(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
|
|
DISPCHECK("[%s]reset display path[end]\n", __func__);
|
|
dsi_basic_irq_enable(DISP_MODULE_DSI0, NULL);
|
|
|
|
pconfig = dpmgr_path_get_last_config(pgc->dpmgr_handle);
|
|
pconfig->rdma_dirty = 1;
|
|
pconfig->ovl_dirty = 1;
|
|
pconfig->dst_dirty = 1;
|
|
pconfig->sbch_enable = 0;
|
|
|
|
dpmgr_path_connect(pgc->dpmgr_handle, CMDQ_DISABLE);
|
|
ret = dpmgr_path_config(pgc->dpmgr_handle, pconfig, NULL);
|
|
|
|
memset(&io_gs, 0, sizeof(struct ddp_io_golden_setting_arg));
|
|
io_gs.dst_mod_type = DST_MOD_REAL_TIME;
|
|
io_gs.is_decouple_mode = 0;
|
|
dpmgr_path_ioctl(pgc->dpmgr_handle, NULL,
|
|
DDP_OVL_GOLDEN_SETTING, &io_gs);
|
|
|
|
DISPDBG("[%s]start dpmgr path[begin]\n", __func__);
|
|
if (disp_partial_is_support()) {
|
|
struct disp_ddp_path_config *data_config =
|
|
dpmgr_path_get_last_config(primary_get_dpmgr_handle());
|
|
|
|
primary_display_config_full_roi(data_config,
|
|
primary_get_dpmgr_handle(), NULL);
|
|
}
|
|
dpmgr_path_start(primary_get_dpmgr_handle(), CMDQ_DISABLE);
|
|
DISPCHECK("[%s]start dpmgr path[end]\n", __func__);
|
|
|
|
if (dpmgr_path_is_busy(primary_get_dpmgr_handle())) {
|
|
DISP_PR_ERR("[%s]not trigger display but already busy\n",
|
|
__func__);
|
|
ret = -1;
|
|
/* goto done; */
|
|
}
|
|
|
|
DISPDBG("[%s]start cmdq trigger loop[begin]\n", __func__);
|
|
_cmdq_start_trigger_loop();
|
|
DISPCHECK("[%s]start cmdq trigger loop[end]\n", __func__);
|
|
if (primary_display_is_video_mode()) {
|
|
/*
|
|
* for video mode, we need to force trigger here
|
|
* for cmd mode, just set DPREC_EVENT_CMDQ_SET_EVENT_ALLOW
|
|
* when trigger loop start
|
|
*/
|
|
dpmgr_path_trigger(primary_get_dpmgr_handle(),
|
|
NULL, CMDQ_DISABLE);
|
|
|
|
}
|
|
|
|
/*
|
|
* (in suspend) when we stop trigger loop
|
|
* if no other thread is running, cmdq may disable its clock
|
|
* all cmdq event will be cleared after suspend
|
|
*/
|
|
cmdqCoreSetEvent(CMDQ_EVENT_DISP_WDMA0_EOF);
|
|
|
|
/* set dirty to trigger one frame -- cmd mode */
|
|
if (!primary_display_is_video_mode()) {
|
|
cmdqCoreSetEvent(CMDQ_SYNC_TOKEN_CONFIG_DIRTY);
|
|
mdelay(40);
|
|
}
|
|
|
|
done:
|
|
primary_display_manual_unlock();
|
|
DISPCHECK("[%s] end\n", __func__);
|
|
return ret;
|
|
}
|
|
|
|
void primary_display_set_recovery_module(enum DISP_MODULE_ENUM module)
|
|
{
|
|
switch (module) {
|
|
case DISP_MODULE_OVL0:
|
|
atomic_set(&enable_ovl0_recovery, 1);
|
|
break;
|
|
case DISP_MODULE_OVL0_2L:
|
|
atomic_set(&enable_ovl0_2l_recovery, 1);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static int primary_display_recovery_kthread(void *data)
|
|
{
|
|
dpmgr_enable_event(primary_get_dpmgr_handle(),
|
|
DISP_PATH_EVENT_DISP_RECOVERY);
|
|
while (1) {
|
|
dpmgr_wait_event(primary_get_dpmgr_handle(),
|
|
DISP_PATH_EVENT_DISP_RECOVERY);
|
|
|
|
if (atomic_read(&enable_ovl0_recovery) &&
|
|
ovl_need_mmsys_sw_reset(DISP_MODULE_OVL0)) {
|
|
atomic_set(&enable_ovl0_recovery, 0);
|
|
atomic_set(&enable_ovl0_2l_recovery, 0);
|
|
DISP_PR_ERR("Detect %s malfunction, do recovery\n",
|
|
ddp_get_module_name(DISP_MODULE_OVL0));
|
|
primary_display_ovl_recovery();
|
|
}
|
|
|
|
if (atomic_read(&enable_ovl0_2l_recovery) &&
|
|
ovl_need_mmsys_sw_reset(DISP_MODULE_OVL0_2L)) {
|
|
atomic_set(&enable_ovl0_recovery, 0);
|
|
atomic_set(&enable_ovl0_2l_recovery, 0);
|
|
DISP_PR_ERR("Detect %s malfunction, do recovery\n",
|
|
ddp_get_module_name(DISP_MODULE_OVL0_2L));
|
|
primary_display_ovl_recovery();
|
|
}
|
|
|
|
if (kthread_should_stop())
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void primary_display_request_eint(void)
|
|
{
|
|
struct LCM_PARAMS *params;
|
|
struct device_node *node;
|
|
|
|
params = primary_get_lcm()->params;
|
|
if (params->dsi.customization_esd_check_enable)
|
|
return;
|
|
|
|
node = of_find_compatible_node(NULL, NULL, "mediatek, DSI_TE-eint");
|
|
if (!node) {
|
|
DISP_PR_ERR("[ESD] cannot find DSI_TE-eint DT node\n");
|
|
return;
|
|
}
|
|
|
|
te_irq = irq_of_parse_and_map(node, 0);
|
|
if (request_irq(te_irq, _esd_check_ext_te_irq_handler,
|
|
IRQF_TRIGGER_RISING, "DSI_TE-eint", NULL)) {
|
|
DISP_PR_ERR("[ESD] EINT IRQ LINE not available\n");
|
|
return;
|
|
}
|
|
|
|
disable_irq(te_irq);
|
|
disp_dts_gpio_select_state(DTS_GPIO_STATE_LCM_DSI_TE);
|
|
}
|
|
|
|
void primary_display_check_recovery_init(void)
|
|
{
|
|
/* primary display check thread init */
|
|
primary_display_check_task =
|
|
kthread_create(primary_display_check_recovery_worker_kthread,
|
|
NULL, "disp_check");
|
|
init_waitqueue_head(&_check_task_wq);
|
|
|
|
if (disp_helper_get_option(DISP_OPT_ESD_CHECK_RECOVERY)) {
|
|
wake_up_process(primary_display_check_task);
|
|
if (_lcm_need_esd_check()) {
|
|
/* default check mode: EINT */
|
|
init_waitqueue_head(&esd_ext_te_wq);
|
|
primary_display_request_eint();
|
|
set_esd_check_mode(GPIO_EINT_MODE);
|
|
primary_display_esd_check_enable(1);
|
|
}
|
|
}
|
|
if (disp_helper_get_stage() == DISP_HELPER_STAGE_NORMAL) {
|
|
primary_display_recovery_thread =
|
|
kthread_create(primary_display_recovery_kthread,
|
|
NULL, "primary_display_path_recovery");
|
|
wake_up_process(primary_display_recovery_thread);
|
|
}
|
|
}
|
|
|
|
void primary_display_esd_check_enable(int enable)
|
|
{
|
|
if (!_lcm_need_esd_check()) {
|
|
DISPCHECK("[ESD]do not support esd check\n");
|
|
return;
|
|
}
|
|
|
|
if (enable) {
|
|
DISPCHECK("[ESD]enable esd check\n");
|
|
esd_check_enable = 1;
|
|
atomic_set(&_check_task_wakeup, 1);
|
|
wake_up_interruptible(&_check_task_wq);
|
|
} else {
|
|
esd_check_enable = 0;
|
|
atomic_set(&_check_task_wakeup, 0);
|
|
DISPCHECK("[ESD]disable esd check\n");
|
|
}
|
|
}
|
|
|
|
unsigned int need_wait_esd_eof(void)
|
|
{
|
|
int ret = 1;
|
|
|
|
/*
|
|
* 1.esd check disable
|
|
* 2.vdo mode
|
|
* 3.cmd mode te
|
|
*/
|
|
if (_lcm_need_esd_check() == 0)
|
|
ret = 0;
|
|
|
|
if (primary_display_is_video_mode())
|
|
ret = 0;
|
|
|
|
if (primary_get_lcm()->params->dsi.customization_esd_check_enable == 0)
|
|
ret = 0;
|
|
|
|
return ret;
|
|
}
|
|
|
|
#if defined(CONFIG_MTK_DUAL_DISPLAY_SUPPORT) && \
|
|
(CONFIG_MTK_DUAL_DISPLAY_SUPPORT == 2)
|
|
/******** external display dual LCM feature ***********
|
|
******** esd check ***********************************
|
|
*/
|
|
static unsigned int extd_need_do_esd_check(void)
|
|
{
|
|
int ret = 0;
|
|
struct disp_lcm_handle *plcm = NULL;
|
|
|
|
extd_disp_get_interface((struct disp_lcm_handle **)&plcm);
|
|
|
|
if (plcm && plcm->params->dsi.esd_check_enable == 1)
|
|
ret = 1;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* For external display EXT TE EINT Check */
|
|
static irqreturn_t extd_esd_check_ext_te_irq_handler(int irq, void *data)
|
|
{
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_vdo_eint,
|
|
MMPROFILE_FLAG_PULSE, 0, 0);
|
|
atomic_set(&esd_ext_te_1_event, 1);
|
|
wake_up_interruptible(&esd_ext_te_1_wq);
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static int extd_esd_check_eint(void)
|
|
{
|
|
int ret = 0;
|
|
|
|
if (wait_event_interruptible_timeout(esd_ext_te_1_wq,
|
|
atomic_read(&esd_ext_te_1_event),
|
|
HZ / 2) > 0)
|
|
ret = 0; /* esd check pass */
|
|
else
|
|
ret = 1; /* esd check fail */
|
|
|
|
atomic_set(&esd_ext_te_1_event, 0);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static unsigned int get_extd_esd_check_mode(void)
|
|
{
|
|
return extd_esd_check_mode;
|
|
}
|
|
|
|
static void set_extd_esd_check_mode(unsigned int mode)
|
|
{
|
|
extd_esd_check_mode = mode;
|
|
}
|
|
|
|
void external_display_esd_check_enable(int enable)
|
|
{
|
|
if (extd_need_do_esd_check()) {
|
|
if (enable) {
|
|
extd_esd_check_enable = 1;
|
|
DISPCHECK("[EXTD-ESD]enable esd check\n");
|
|
atomic_set(&extd_check_task_wakeup, 1);
|
|
wake_up_interruptible(&extd_check_task_wq);
|
|
} else {
|
|
extd_esd_check_enable = 0;
|
|
atomic_set(&extd_check_task_wakeup, 0);
|
|
DISPCHECK("[EXTD-ESD]disable esd check\n");
|
|
|
|
}
|
|
} else {
|
|
DISPCHECK("[EXTD-ESD]do not support esd check\n");
|
|
}
|
|
}
|
|
|
|
int external_display_switch_esd_mode(int mode)
|
|
{
|
|
int ret = 0;
|
|
struct device_node *node;
|
|
int irq;
|
|
u32 ints[2] = { 0, 0 };
|
|
|
|
node = of_find_compatible_node(NULL, NULL, "mediatek, dsi_te_1-eint");
|
|
if (node == NULL) {
|
|
DISP_PR_ERR("[EXTD-ESD][%s]can't find DSI_TE eint DT node\n",
|
|
__func__);
|
|
return ret;
|
|
}
|
|
|
|
if (mode == GPIO_EINT_MODE) {
|
|
/* register irq handler */
|
|
of_property_read_u32_array(node, "debounce",
|
|
ints, ARRAY_SIZE(ints));
|
|
/* mt_gpio_set_debounce(ints[0], ints[1]); */
|
|
irq = irq_of_parse_and_map(node, 0);
|
|
if (request_irq(irq, extd_esd_check_ext_te_irq_handler,
|
|
IRQF_TRIGGER_RISING, "dsi_te_1-eint", NULL))
|
|
DISP_PR_ERR("[EXTD-ESD]EINT IRQ LINE NOT AVAILABLE!\n");
|
|
} else if (mode == GPIO_DSI_MODE) {
|
|
/* 1. unregister irq handler */
|
|
irq = irq_of_parse_and_map(node, 0);
|
|
free_irq(irq, NULL);
|
|
/*disp_dts_gpio_select_state(DTS_GPIO_STATE_TE_MODE_TE);*/
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* ESD CHECK FUNCTION */
|
|
/* return 1: esd check fail */
|
|
/* return 0: esd check pass */
|
|
int external_display_esd_check(void)
|
|
{
|
|
int ret = 0;
|
|
unsigned int mode;
|
|
struct disp_lcm_handle *plcm = NULL;
|
|
|
|
dprec_logger_start(DPREC_LOGGER_ESD_CHECK, 0, 0);
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_check_t,
|
|
MMPROFILE_FLAG_START, 0, 0);
|
|
DISPCHECK("[EXTD-ESD]ESD check begin\n");
|
|
|
|
if (ext_disp_is_alive() != EXTD_RESUME) {
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_check_t,
|
|
MMPROFILE_FLAG_PULSE, 1, 0);
|
|
DISPCHECK("[EXTD-ESD]EXTD DISP is slept. skip esd check\n");
|
|
goto done;
|
|
}
|
|
|
|
/* Esd Check : EXT TE */
|
|
extd_disp_get_interface((struct disp_lcm_handle **)&plcm);
|
|
if (!plcm || plcm->params->dsi.customization_esd_check_enable != 0)
|
|
goto done;
|
|
/* use te for esd check */
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_extte, MMPROFILE_FLAG_START,
|
|
0, 0);
|
|
|
|
mode = get_extd_esd_check_mode();
|
|
if (mode == GPIO_EINT_MODE) {
|
|
DISPCHECK("[EXTD-ESD]ESD check eint\n");
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_extte,
|
|
MMPROFILE_FLAG_PULSE, ext_disp_is_video_mode(),
|
|
mode);
|
|
external_display_switch_esd_mode(mode);
|
|
DISPCHECK("[EXTD-ESD]ESD check begin ~\n");
|
|
ret = extd_esd_check_eint();
|
|
DISPCHECK("[EXTD-ESD]ESD check end, ret:%d\n", ret);
|
|
mode = GPIO_DSI_MODE; /* used for mode switch */
|
|
external_display_switch_esd_mode(mode);
|
|
} else if (mode == GPIO_DSI_MODE) {
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_extte,
|
|
MMPROFILE_FLAG_PULSE, ext_disp_is_video_mode(),
|
|
mode);
|
|
|
|
DISPCHECK("[EXTD-ESD]ESD check read\n");
|
|
/*ret = do_esd_check_read();*/
|
|
mode = GPIO_EINT_MODE; /* used for mode switch */
|
|
}
|
|
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_extte, MMPROFILE_FLAG_END,
|
|
0, ret);
|
|
|
|
done:
|
|
DISPCHECK("[EXTD-ESD]ESD check end, ret = %d\n", ret);
|
|
mmprofile_log_ex(ddp_mmp_get_events()->esd_check_t, MMPROFILE_FLAG_END,
|
|
0, ret);
|
|
dprec_logger_done(DPREC_LOGGER_ESD_CHECK, 0, 0);
|
|
return ret;
|
|
}
|
|
|
|
static int external_display_check_recovery_worker_kthread(void *data)
|
|
{
|
|
struct sched_param param = {.sched_priority = 87 };
|
|
int ret = 0;
|
|
int i = 0;
|
|
int esd_try_cnt = 5; /* 20; */
|
|
int recovery_done = 0;
|
|
const int len = 160;
|
|
char msg[len];
|
|
int n = 0;
|
|
|
|
DISPFUNC();
|
|
sched_setscheduler(current, SCHED_RR, ¶m);
|
|
|
|
while (1) {
|
|
msleep(2000); /* 2s */
|
|
ret = wait_event_interruptible(extd_check_task_wq,
|
|
atomic_read(&extd_check_task_wakeup));
|
|
if (ret < 0) {
|
|
n = scnprintf(msg, len,
|
|
"[ext_disp_check]check thread waked up accidently\n");
|
|
DISPCHECK("%s", msg);
|
|
continue;
|
|
}
|
|
pr_debug("[EXTD ext_disp_check]check thread waked up!\n");
|
|
ext_disp_esd_check_lock();
|
|
/* esd check and recovery */
|
|
|
|
if (!extd_esd_check_enable) {
|
|
ext_disp_esd_check_unlock();
|
|
continue;
|
|
}
|
|
|
|
i = 0; /* repeat */
|
|
do {
|
|
ret = external_display_esd_check();
|
|
if (ret != 1)
|
|
break;
|
|
n = scnprintf(msg, len,
|
|
"[EXTD-ESD]esd check fail, will do esd recovery. try=%d\n",
|
|
i);
|
|
DISP_PR_ERR("%s", msg);
|
|
ext_disp_esd_recovery();
|
|
recovery_done = 1;
|
|
} while (++i < esd_try_cnt);
|
|
|
|
if (ret == 1) {
|
|
n = scnprintf(msg, len,
|
|
"[EXTD-ESD]after esd recovery %d times, still fail, disable esd check\n",
|
|
esd_try_cnt);
|
|
DISP_PR_ERR("%s", msg);
|
|
external_display_esd_check_enable(0);
|
|
} else if (recovery_done == 1) {
|
|
DISPCHECK("[EXTD-ESD]esd recovery success\n");
|
|
recovery_done = 0;
|
|
}
|
|
ext_disp_esd_check_unlock();
|
|
|
|
if (kthread_should_stop())
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void external_display_check_recovery_init(void)
|
|
{
|
|
/* primary display check thread init */
|
|
if (external_display_check_task == NULL) {
|
|
external_display_check_task = kthread_create(
|
|
external_display_check_recovery_worker_kthread,
|
|
NULL, "extd_esd_check");
|
|
init_waitqueue_head(&extd_check_task_wq);
|
|
wake_up_process(external_display_check_task);
|
|
}
|
|
|
|
if (disp_helper_get_option(DISP_OPT_ESD_CHECK_RECOVERY) &&
|
|
extd_need_do_esd_check()) {
|
|
/* esd check init */
|
|
init_waitqueue_head(&esd_ext_te_1_wq);
|
|
set_extd_esd_check_mode(GPIO_EINT_MODE);
|
|
/*external_display_esd_check_enable(1);*/
|
|
}
|
|
}
|
|
#endif
|