android_kernel_modules_leno.../intel_media/common/psb_drv.c

5227 lines
148 KiB
C

/**************************************************************************
* Copyright (c) 2007, Intel Corporation.
* All Rights Reserved.
* Copyright (c) 2008, Tungsten Graphics, Inc. Cedar Park, TX., USA.
* All Rights Reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License along with
* this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
*
**************************************************************************/
#include <drm/drmP.h>
#include <drm/drm.h>
#include <linux/cpu.h>
#include <linux/notifier.h>
#include <linux/spinlock.h>
#include <drm/drm_pciids.h>
#include <linux/version.h>
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3,8,0))
#include <asm/intel_scu_pmic.h>
#else
#include <asm/intel_scu_ipc.h>
#endif
#include <asm/intel-mid.h>
#include "psb_drm.h"
#include "psb_drv.h"
#include "psb_fb.h"
#include "psb_reg.h"
#include "psb_intel_reg.h"
#include "psb_msvdx.h"
#include "pnw_topaz.h"
#include "mdfld_dsi_dbi_dsr.h"
#include "mdfld_csc.h"
#include "mdfld_dsi_pkg_sender.h"
#include "mdfld_dsi_dbi.h"
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
#include "mdfld_dsi_dpi.h"
#endif
//ASUS_BSP: [DDS] ---
#include "pvr_drm_shared.h"
#include "psb_powermgmt.h"
#ifdef CONFIG_MDFLD_DSI_DPU
#include "mdfld_dsi_dbi_dpu.h"
#endif
#ifdef CONFIG_GFX_RTPM
#include <linux/pm_runtime.h>
#endif
#ifdef CONFIG_MDFD_GL3
#include "mdfld_gl3.h"
#endif
#include "otm_hdmi.h"
#include "android_hdmi.h"
/*IMG headers*/
#include "pvr_drm_shared.h"
#include "img_types.h"
#include "pvr_bridge.h"
#include "linkage.h"
#include <linux/HWVersion.h>
extern int Read_HW_ID(void);
struct workqueue_struct *te_wq;
struct workqueue_struct *vsync_wq;
#define HDMI_MONITOR_NAME_LENGTH 20
int drm_psb_debug = PSB_D_WARN;
int drm_psb_enable_cabc = 1;
int drm_psb_enable_gamma;
int drm_psb_enable_color_conversion;
static int drm_psb_trap_pagefaults;
bool gbdispstatus = true;
int drm_psb_disable_vsync = 1;
int drm_psb_no_fb;
int drm_psb_force_pipeb;
int drm_msvdx_pmpolicy = PSB_PMPOLICY_POWERDOWN;
int drm_psb_cpurelax;
int drm_psb_udelaydivider = 1;
int drm_topaz_pmpolicy = PSB_PMPOLICY_POWERDOWN;
int drm_topaz_sbuswa;
int drm_psb_ospm = 1;
int drm_psb_gl3_enable = 1;
int drm_psb_topaz_clockgating;
int gfxrtdelay = 2 * 1000;
int drm_psb_3D_vblank = 1;
int drm_psb_smart_vsync = 1;
int drm_psb_te_timer_delay = (DRM_HZ / 40);
char HDMI_EDID[HDMI_MONITOR_NAME_LENGTH];
int hdmi_state;
u32 DISP_PLANEB_STATUS = ~DISPLAY_PLANE_ENABLE;
int drm_psb_use_cases_control = PSB_ALL_UC_ENABLE;
int drm_psb_dump_pm_history;
int gamma_setting[129] = {0};
int csc_setting[6] = {0};
int gamma_number = 129;
int csc_number = 6;
#ifdef CONFIG_CTP_DPST
int dpst_level = 3;
#endif
int asus_panel_id = 0x0;
//ASUS_BSP: Louis +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
int panel_id = 0;
int hpd = 0;
int panel_turn_on = DDS_NONE;
#endif
#ifdef CONFIG_SUPPORT_OTM8018B_MIPI_480X854_DISPLAY
bool esd_thread_enable = true;
#endif
int drm_hdmi_hpd_auto;
int default_hdmi_scaling_mode = DRM_MODE_SCALE_CENTER;
int drm_psb_msvdx_tiling = 1;
int drm_msvdx_bottom_half;
struct drm_device *g_drm_dev;
EXPORT_SYMBOL(g_drm_dev);
#ifdef CONFIG_SUPPORT_MIPI_H8C7_CMD_DISPLAY
extern struct platform_driver h8c7_lcd_driver;
#endif
#ifdef CONFIG_SUPPORT_VB_MIPI_DISPLAY
extern struct platform_driver vb_lcd_driver;
#endif
#ifdef CONFIG_R63311_MIPI_VIDEO_MODE
extern struct platform_driver jdi_r63311_lcd_driver;
#endif
#ifdef CONFIG_SUPPORT_TMD_MIPI_600X1024_DISPLAY
extern struct platform_driver tmd_lcd_driver;
#endif
#ifdef CONFIG_SUPPORT_OTM8018B_MIPI_480X854_DISPLAY
extern struct platform_driver pf450cl_vid_lcd_driver;
#endif
static int psb_probe(struct pci_dev *pdev, const struct pci_device_id *ent);
MODULE_PARM_DESC(debug, "Enable debug output");
MODULE_PARM_DESC(no_fb, "Disable FBdev");
MODULE_PARM_DESC(trap_pagefaults, "Error and reset on MMU pagefaults");
MODULE_PARM_DESC(disable_vsync, "Disable vsync interrupts");
MODULE_PARM_DESC(force_pipeb, "Forces PIPEB to become primary fb");
MODULE_PARM_DESC(ta_mem_size, "TA memory size in kiB");
MODULE_PARM_DESC(ospm, "switch for ospm support");
MODULE_PARM_DESC(gl3_enabled, "Enable GL3 cache");
MODULE_PARM_DESC(rtpm, "Specifies Runtime PM delay for GFX");
MODULE_PARM_DESC(msvdx_pmpolicy, "msvdx power management policy btw frames");
MODULE_PARM_DESC(topaz_pmpolicy, "topaz power managerment policy btw frames");
MODULE_PARM_DESC(topaz_sbuswa, "WA for topaz sysbus write");
MODULE_PARM_DESC(hdmi_edid, "EDID info for HDMI monitor");
MODULE_PARM_DESC(hdmi_state, "Whether HDMI Monitor is connected or not");
MODULE_PARM_DESC(vblank_sync, "whether sync to vblank interrupt when do 3D flip");
MODULE_PARM_DESC(smart_vsync, "Enable Smart Vsync for Display");
MODULE_PARM_DESC(te_delay, "swap delay after TE interrpt");
MODULE_PARM_DESC(cpu_relax, "replace udelay with cpu_relax for video");
MODULE_PARM_DESC(udelay_divider, "divide the usec value of video udelay");
MODULE_PARM_DESC(enable_color_conversion, "Enable display side color conversion");
MODULE_PARM_DESC(enable_gamma, "Enable display side gamma");
MODULE_PARM_DESC(use_cases_control, "Use to enable and disable use cases");
MODULE_PARM_DESC(pm_history, "whether to dump pm history when SGX HWR");
#ifdef CONFIG_CTP_DPST
MODULE_PARM_DESC(dpst_level, "dpst aggressive level: 0~5");
#endif
MODULE_PARM_DESC(hdmi_hpd_auto, "HDMI hot-plug auto test flag");
MODULE_PARM_DESC(default_hdmi_scaling_mode, "Default HDMI scaling mode");
MODULE_PARM_DESC(asus_panel_id, "ASUS panel ID");
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
MODULE_PARM_DESC(panel_id, "panel_id");
MODULE_PARM_DESC(hpd, "hpd");
#endif
//ASUS_BSP: [DDS] ---
module_param_named(debug, drm_psb_debug, int, 0600);
module_param_named(psb_enable_cabc, drm_psb_enable_cabc, int, 0600);
module_param_named(enable_color_conversion, drm_psb_enable_color_conversion, int, 0600);
module_param_named(enable_gamma, drm_psb_enable_gamma, int, 0600);
/* [SC1] change parameter name */
module_param_named(no_fb, drm_psb_no_fb, int, 0600);
module_param_named(trap_pagefaults, drm_psb_trap_pagefaults, int, 0600);
module_param_named(force_pipeb, drm_psb_force_pipeb, int, 0600);
module_param_named(msvdx_pmpolicy, drm_msvdx_pmpolicy, int, 0600);
module_param_named(cpu_relax, drm_psb_cpurelax, int, 0600);
module_param_named(udelay_divider, drm_psb_udelaydivider, int, 0600);
module_param_named(topaz_pmpolicy, drm_topaz_pmpolicy, int, 0600);
module_param_named(topaz_sbuswa, drm_topaz_sbuswa, int, 0600);
module_param_named(ospm, drm_psb_ospm, int, 0600);
module_param_named(gl3_enabled, drm_psb_gl3_enable, int, 0600);
module_param_named(rtpm, gfxrtdelay, int, 0600);
module_param_named(topaz_clockgating, drm_psb_topaz_clockgating, int, 0600);
module_param_string(hdmi_edid, HDMI_EDID, 20, 0600);
module_param_named(hdmi_state, hdmi_state, int, 0600);
module_param_named(vblank_sync, drm_psb_3D_vblank, int, 0600);
module_param_named(smart_vsync, drm_psb_smart_vsync, int, 0600);
module_param_named(te_delay, drm_psb_te_timer_delay, int, 0600);
module_param_named(msvdx_tiling_memory, drm_psb_msvdx_tiling, int, 0600);
module_param_named(msvdx_bottom_half, drm_msvdx_bottom_half, int, 0600);
module_param_named(psb_use_cases_control, drm_psb_use_cases_control, int, 0600);
module_param_named(pm_history, drm_psb_dump_pm_history, int, 0600);
module_param_array_named(gamma_adjust, gamma_setting, int, &gamma_number, 0600);
module_param_array_named(csc_adjust, csc_setting, int, &csc_number, 0600);
#ifdef CONFIG_CTP_DPST
module_param_named(dpst_level, dpst_level, int, 0600);
#endif
module_param_named(hdmi_hpd_auto, drm_hdmi_hpd_auto, int, 0600);
module_param_named(default_hdmi_scaling_mode, default_hdmi_scaling_mode,
int, 0600);
module_param_named(asus_panel_id, asus_panel_id, int, 0644);
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
module_param_named(panel_id, panel_id, int, 0644);
module_param_named(hpd, hpd, int, 0644);
module_param_named(panel_turn_on, panel_turn_on, int, 0644);
#endif
//ASUS_BSP: [DDS] ---
#ifdef CONFIG_SUPPORT_OTM8018B_MIPI_480X854_DISPLAY
module_param_named(esd_thread_enable, esd_thread_enable, bool, 0644);
#endif
#ifndef MODULE
/* Make ospm configurable via cmdline firstly, and others can be enabled if needed. */
static int __init config_ospm(char *arg)
{
/* ospm turn on/off control can be passed in as a cmdline parameter */
/* to enable this feature add ospm=1 to cmdline */
/* to disable this feature add ospm=0 to cmdline */
if (!arg)
return -EINVAL;
if (!strcasecmp(arg, "0"))
drm_psb_ospm = 0;
else if (!strcasecmp(arg, "1"))
drm_psb_ospm = 1;
return 0;
}
static int __init config_gl3(char *arg)
{
if (!arg)
return -EINVAL;
if (!strcasecmp(arg, "0"))
drm_psb_gl3_enable = 0;
else if (!strcasecmp(arg, "1"))
drm_psb_gl3_enable = 1;
return 0;
}
early_param("ospm", config_ospm);
early_param("gl3_enabled", config_gl3);
#endif
static struct pci_device_id pciidlist[] = {
#ifdef SGX535
{0x8086, 0x4100, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4101, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4102, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4103, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4104, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4105, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4106, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
{0x8086, 0x4107, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MRST_4100},
#endif
#if defined (MEDFIELD) || defined (CLOVERTRAIL_PHONE)
{0x8086, 0x0130, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0131, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0132, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0133, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0134, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0135, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0136, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x0137, PCI_ANY_ID, PCI_ANY_ID, 0, 0, CHIP_MDFLD_0130},
{0x8086, 0x08c0, PCI_ANY_ID, PCI_ANY_ID, PCI_CLASS_DISPLAY_VGA << 8, 0xFFFF00, CHIP_MDFLD_0130},
{0x8086, 0x08c7, PCI_ANY_ID, PCI_ANY_ID, PCI_CLASS_DISPLAY_VGA << 8, 0xFFFF00, CHIP_MDFLD_0130},
{0x8086, 0x08c8, PCI_ANY_ID, PCI_ANY_ID, PCI_CLASS_DISPLAY_VGA << 8, 0xFFFF00, CHIP_MDFLD_0130},
#endif
{0, 0, 0}
};
MODULE_DEVICE_TABLE(pci, pciidlist);
/*
* Standard IOCTLs.
*/
#define DRM_IOCTL_PSB_KMS_OFF \
DRM_IO(DRM_PSB_KMS_OFF + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_KMS_ON \
DRM_IO(DRM_PSB_KMS_ON + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_VT_LEAVE \
DRM_IO(DRM_PSB_VT_LEAVE + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_VT_ENTER \
DRM_IO(DRM_PSB_VT_ENTER + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_EXTENSION \
DRM_IOWR(DRM_PSB_EXTENSION + DRM_COMMAND_BASE, \
union drm_psb_extension_arg)
#define DRM_IOCTL_PSB_SIZES \
DRM_IOR(DRM_PSB_SIZES + DRM_COMMAND_BASE, \
struct drm_psb_sizes_arg)
#define DRM_IOCTL_PSB_FUSE_REG \
DRM_IOWR(DRM_PSB_FUSE_REG + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_VBT \
DRM_IOWR(DRM_PSB_VBT + DRM_COMMAND_BASE, \
struct gct_ioctl_arg)
#define DRM_IOCTL_PSB_DC_STATE \
DRM_IOW(DRM_PSB_DC_STATE + DRM_COMMAND_BASE, \
struct drm_psb_dc_state_arg)
#define DRM_IOCTL_PSB_ADB \
DRM_IOWR(DRM_PSB_ADB + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_MODE_OPERATION \
DRM_IOWR(DRM_PSB_MODE_OPERATION + DRM_COMMAND_BASE, \
struct drm_psb_mode_operation_arg)
#define DRM_IOCTL_PSB_STOLEN_MEMORY \
DRM_IOWR(DRM_PSB_STOLEN_MEMORY + DRM_COMMAND_BASE, \
struct drm_psb_stolen_memory_arg)
#define DRM_IOCTL_PSB_REGISTER_RW \
DRM_IOWR(DRM_PSB_REGISTER_RW + DRM_COMMAND_BASE, \
struct drm_psb_register_rw_arg)
#define DRM_IOCTL_PSB_GTT_MAP \
DRM_IOWR(DRM_PSB_GTT_MAP + DRM_COMMAND_BASE, \
struct psb_gtt_mapping_arg)
#define DRM_IOCTL_PSB_GTT_UNMAP \
DRM_IOW(DRM_PSB_GTT_UNMAP + DRM_COMMAND_BASE, \
struct psb_gtt_mapping_arg)
#define DRM_IOCTL_PSB_GETPAGEADDRS \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_GETPAGEADDRS,\
struct drm_psb_getpageaddrs_arg)
#define DRM_IOCTL_PSB_HIST_ENABLE \
DRM_IOWR(DRM_PSB_HIST_ENABLE + DRM_COMMAND_BASE, \
uint32_t)
#define DRM_IOCTL_PSB_HIST_STATUS \
DRM_IOWR(DRM_PSB_HIST_STATUS + DRM_COMMAND_BASE, \
struct drm_psb_hist_status_arg)
#define DRM_IOCTL_PSB_UPDATE_GUARD \
DRM_IOWR(DRM_PSB_UPDATE_GUARD + DRM_COMMAND_BASE, \
uint32_t)
#define DRM_IOCTL_PSB_INIT_COMM \
DRM_IOWR(DRM_PSB_INIT_COMM + DRM_COMMAND_BASE, \
uint32_t)
#define DRM_IOCTL_PSB_DPST \
DRM_IOWR(DRM_PSB_DPST + DRM_COMMAND_BASE, \
uint32_t)
#define DRM_IOCTL_PSB_GAMMA \
DRM_IOWR(DRM_PSB_GAMMA + DRM_COMMAND_BASE, \
struct drm_psb_dpst_lut_arg)
#define DRM_IOCTL_DPST_LEVEL \
DRM_IOWR(DRM_PSB_DPST_LEVEL + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_DPST_BL \
DRM_IOWR(DRM_PSB_DPST_BL + DRM_COMMAND_BASE, \
uint32_t)
#define DRM_IOCTL_PSB_GET_PIPE_FROM_CRTC_ID \
DRM_IOWR(DRM_PSB_GET_PIPE_FROM_CRTC_ID + DRM_COMMAND_BASE, \
struct drm_psb_get_pipe_from_crtc_id_arg)
/*pvr ioctls*/
#define PVR_DRM_SRVKM_IOCTL \
DRM_IOW(DRM_COMMAND_BASE + PVR_DRM_SRVKM_CMD, \
PVRSRV_BRIDGE_PACKAGE)
#define PVR_DRM_DISP_IOCTL \
DRM_IO(DRM_COMMAND_BASE + PVR_DRM_DISP_CMD)
#define PVR_DRM_IS_MASTER_IOCTL \
DRM_IO(DRM_COMMAND_BASE + PVR_DRM_IS_MASTER_CMD)
#define PVR_DRM_UNPRIV_IOCTL \
DRM_IOWR(DRM_COMMAND_BASE + PVR_DRM_UNPRIV_CMD, \
IMG_UINT32)
#if defined(PDUMP)
#define PVR_DRM_DBGDRV_IOCTL \
DRM_IOW(DRM_COMMAND_BASE + PVR_DRM_DBGDRV_CMD, IOCTL_PACKAGE)
#endif
/*DPU/DSR stuff*/
#define DRM_IOCRL_PSB_DPU_QUERY DRM_IOR(DRM_PSB_DPU_QUERY + DRM_COMMAND_BASE, IMG_UINT32)
#define DRM_IOCRL_PSB_DPU_DSR_ON DRM_IOW(DRM_PSB_DPU_DSR_ON + DRM_COMMAND_BASE, IMG_UINT32)
/* #define DRM_IOCRL_PSB_DPU_DSR_OFF DRM_IOW(DRM_PSB_DPU_DSR_OFF + DRM_COMMAND_BASE, IMG_UINT32) */
#define DRM_IOCRL_PSB_DPU_DSR_OFF DRM_IOW(DRM_PSB_DPU_DSR_OFF + DRM_COMMAND_BASE, struct drm_psb_drv_dsr_off_arg)
/*HDMI FB stuff*/
#define DRM_IOCTL_PSB_HDMI_FB_CMD DRM_IOWR(DRM_PSB_HDMI_FB_CMD + DRM_COMMAND_BASE, struct drm_psb_disp_ctrl)
/* HDCP IOCTLs */
#define DRM_IOCTL_PSB_QUERY_HDCP \
DRM_IOR(DRM_PSB_QUERY_HDCP + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_VALIDATE_HDCP_KSV \
DRM_IOWR(DRM_PSB_VALIDATE_HDCP_KSV + DRM_COMMAND_BASE, sqword_t)
#define DRM_IOCTL_PSB_GET_HDCP_STATUS \
DRM_IOR(DRM_PSB_GET_HDCP_STATUS + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_ENABLE_HDCP \
DRM_IO(DRM_PSB_ENABLE_HDCP + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_DISABLE_HDCP \
DRM_IO(DRM_PSB_DISABLE_HDCP + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_GET_HDCP_LINK_STATUS \
DRM_IOR(DRM_PSB_GET_HDCP_LINK_STATUS + DRM_COMMAND_BASE, uint32_t)
#define DRM_IOCTL_PSB_HDCP_DISPLAY_IED_OFF \
DRM_IO(DRM_PSB_HDCP_DISPLAY_IED_OFF + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_HDCP_DISPLAY_IED_ON \
DRM_IO(DRM_PSB_HDCP_DISPLAY_IED_ON + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_QUERY_HDCP_DISPLAY_IED_CAPS \
DRM_IOR(DRM_PSB_QUERY_HDCP_DISPLAY_IED_CAPS \
+ DRM_COMMAND_BASE, uint32_t)
/* CSC IOCTLS */
#define DRM_IOCTL_PSB_SET_CSC \
DRM_IOW(DRM_PSB_SET_CSC + DRM_COMMAND_BASE, struct drm_psb_csc_matrix)
/* VSYNC IOCTL */
#define DRM_IOCTL_PSB_VSYNC_SET \
DRM_IOWR(DRM_PSB_VSYNC_SET + DRM_COMMAND_BASE, \
struct drm_psb_vsync_set_arg)
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
#define DRM_IOCTL_PSB_PANEL_SWITCH \
DRM_IOWR(DRM_PSB_PANEL_SWITCH + DRM_COMMAND_BASE, \
uint32_t)
#endif
//ASUS_BSP: [DDS] ---
/* GET DC INFO IOCTL */
#define DRM_IOCTL_PSB_GET_DC_INFO \
DRM_IOR(DRM_PSB_GET_DC_INFO + DRM_COMMAND_BASE, \
struct drm_psb_dc_info)
/*CSC GAMMA Setting*/
#define DRM_IOCTL_PSB_CSC_GAMMA_SETTING \
DRM_IOWR(DRM_PSB_CSC_GAMMA_SETTING + DRM_COMMAND_BASE, struct drm_psb_csc_gamma_setting)
#define DRM_IOCTL_PSB_ENABLE_IED_SESSION \
DRM_IO(DRM_PSB_ENABLE_IED_SESSION + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_DISABLE_IED_SESSION \
DRM_IO(DRM_PSB_DISABLE_IED_SESSION + DRM_COMMAND_BASE)
/*
* TTM execbuf extension.
*/
#define DRM_IOCTL_PSB_CMDBUF \
DRM_IOW(DRM_PSB_CMDBUF + DRM_COMMAND_BASE, \
struct drm_psb_cmdbuf_arg)
#define DRM_IOCTL_PSB_SCENE_UNREF \
DRM_IOW(DRM_PSB_SCENE_UNREF + DRM_COMMAND_BASE, \
struct drm_psb_scene)
#define DRM_IOCTL_PSB_KMS_OFF DRM_IO(DRM_PSB_KMS_OFF + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_KMS_ON DRM_IO(DRM_PSB_KMS_ON + DRM_COMMAND_BASE)
#define DRM_IOCTL_PSB_EXTENSION \
DRM_IOWR(DRM_PSB_EXTENSION + DRM_COMMAND_BASE, \
union drm_psb_extension_arg)
/*
* TTM placement user extension.
*/
#define DRM_PSB_TTM_PL_CREATE (TTM_PL_CREATE + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_REFERENCE (TTM_PL_REFERENCE + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_UNREF (TTM_PL_UNREF + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_SYNCCPU (TTM_PL_SYNCCPU + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_WAITIDLE (TTM_PL_WAITIDLE + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_SETSTATUS (TTM_PL_SETSTATUS + DRM_PSB_PLACEMENT_OFFSET)
#define DRM_PSB_TTM_PL_CREATE_UB (TTM_PL_CREATE_UB + DRM_PSB_PLACEMENT_OFFSET)
/*
* TTM fence extension.
*/
#define DRM_PSB_FENCE_OFFSET (DRM_PSB_TTM_PL_CREATE_UB + 1)
#define DRM_PSB_TTM_FENCE_SIGNALED (TTM_FENCE_SIGNALED + DRM_PSB_FENCE_OFFSET)
#define DRM_PSB_TTM_FENCE_FINISH (TTM_FENCE_FINISH + DRM_PSB_FENCE_OFFSET)
#define DRM_PSB_TTM_FENCE_UNREF (TTM_FENCE_UNREF + DRM_PSB_FENCE_OFFSET)
#define DRM_PSB_FLIP (DRM_PSB_TTM_FENCE_UNREF + 1) /*20*/
/* PSB video extension */
#define DRM_PSB_VIDEO_GETPARAM (DRM_PSB_FLIP + 1)
#define DRM_IOCTL_PSB_TTM_PL_CREATE \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_CREATE,\
union ttm_pl_create_arg)
#define DRM_IOCTL_PSB_TTM_PL_REFERENCE \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_REFERENCE,\
union ttm_pl_reference_arg)
#define DRM_IOCTL_PSB_TTM_PL_UNREF \
DRM_IOW(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_UNREF,\
struct ttm_pl_reference_req)
#define DRM_IOCTL_PSB_TTM_PL_SYNCCPU \
DRM_IOW(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_SYNCCPU,\
struct ttm_pl_synccpu_arg)
#define DRM_IOCTL_PSB_TTM_PL_WAITIDLE \
DRM_IOW(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_WAITIDLE,\
struct ttm_pl_waitidle_arg)
#define DRM_IOCTL_PSB_TTM_PL_SETSTATUS \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_SETSTATUS,\
union ttm_pl_setstatus_arg)
#define DRM_IOCTL_PSB_TTM_PL_CREATE_UB \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_PL_CREATE_UB,\
union ttm_pl_create_ub_arg)
#define DRM_IOCTL_PSB_TTM_FENCE_SIGNALED \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_FENCE_SIGNALED, \
union ttm_fence_signaled_arg)
#define DRM_IOCTL_PSB_TTM_FENCE_FINISH \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_TTM_FENCE_FINISH, \
union ttm_fence_finish_arg)
#define DRM_IOCTL_PSB_TTM_FENCE_UNREF \
DRM_IOW(DRM_COMMAND_BASE + DRM_PSB_TTM_FENCE_UNREF, \
struct ttm_fence_unref_arg)
#define DRM_IOCTL_PSB_FLIP \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_FLIP, \
struct drm_psb_pageflip_arg)
#define DRM_IOCTL_PSB_VIDEO_GETPARAM \
DRM_IOWR(DRM_COMMAND_BASE + DRM_PSB_VIDEO_GETPARAM, \
struct drm_lnc_video_getparam_arg)
static int psb_vt_leave_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_vt_enter_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_sizes_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_fuse_reg_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_vbt_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_dc_state_ioctl(struct drm_device *dev, void * data,
struct drm_file *file_priv);
static int psb_adb_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_mode_operation_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_stolen_memory_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_vsync_set_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
static int psb_panel_switch_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#endif
//ASUS_BSP: [DDS] ---
static int psb_get_dc_info_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_register_rw_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#ifdef CONFIG_CTP_DPST
static int psb_hist_enable_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_hist_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_update_guard_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_init_comm_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_dpst_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_gamma_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_dpst_bl_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#endif
static int psb_dpu_query_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_dpu_dsr_on_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_dpu_dsr_off_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_disp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#ifdef CONFIG_SUPPORT_HDMI
static int psb_query_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_validate_hdcp_ksv_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_get_hdcp_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_enable_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_disable_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_get_hdcp_link_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_enable_display_ied_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_disable_display_ied_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_query_display_ied_caps_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#endif
static int psb_set_csc_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_csc_gamma_setting_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_enable_ied_session_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
static int psb_disable_ied_session_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#ifdef CONFIG_CTP_DPST
extern int psb_dpst_get_level_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv);
#endif
/* wrapper for PVR ioctl functions to avoid direct call */
int PVRDRM_Dummy_ioctl2(struct drm_device *dev, void *arg,
struct drm_file *pFile)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRDRM_Dummy_ioctl(dev, arg, pFile);
}
int PVRSRV_BridgeDispatchKM2(struct drm_device unref__ * dev,
void *arg, struct drm_file *pFile)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRSRV_BridgeDispatchKM(dev, arg, pFile);
}
int PVRDRMIsMaster2(struct drm_device *dev, void *arg,
struct drm_file *pFile)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRDRMIsMaster(dev, arg, pFile);
}
int PVRDRMUnprivCmd2(struct drm_device *dev, void *arg,
struct drm_file *pFile)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRDRMUnprivCmd(dev, arg, pFile);
}
void PVRSRVDrmPostClose2(struct drm_device *dev,
struct drm_file *file)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRSRVDrmPostClose(dev, file);
}
#if defined(PDUMP)
int SYSPVRDBGDrivIoctl2(struct drm_device *dev, IMG_VOID *arg,
struct drm_file *pFile)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->SYSPVRDBGDrivIoctl(dev, arg, pFile);
}
#endif
#define PSB_IOCTL_DEF(ioctl, func, flags) \
[DRM_IOCTL_NR(ioctl) - DRM_COMMAND_BASE] = {ioctl, flags, func, ioctl}
static struct drm_ioctl_desc psb_ioctls[] = {
PSB_IOCTL_DEF(DRM_IOCTL_PSB_KMS_OFF, psbfb_kms_off_ioctl,
DRM_ROOT_ONLY),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_KMS_ON,
psbfb_kms_on_ioctl,
DRM_ROOT_ONLY),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VT_LEAVE, psb_vt_leave_ioctl,
DRM_ROOT_ONLY),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VT_ENTER,
psb_vt_enter_ioctl,
DRM_ROOT_ONLY),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_EXTENSION, psb_extension_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_SIZES, psb_sizes_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_FUSE_REG, psb_fuse_reg_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VBT, psb_vbt_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_DC_STATE, psb_dc_state_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_ADB, psb_adb_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_MODE_OPERATION, psb_mode_operation_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_STOLEN_MEMORY, psb_stolen_memory_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_REGISTER_RW, psb_register_rw_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GTT_MAP,
psb_gtt_map_meminfo_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GTT_UNMAP,
psb_gtt_unmap_meminfo_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GETPAGEADDRS,
psb_getpageaddrs_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(PVR_DRM_SRVKM_IOCTL, PVRSRV_BridgeDispatchKM2,
DRM_UNLOCKED),
PSB_IOCTL_DEF(PVR_DRM_DISP_IOCTL, PVRDRM_Dummy_ioctl2, 0),
PSB_IOCTL_DEF(PVR_DRM_IS_MASTER_IOCTL, PVRDRMIsMaster2, DRM_MASTER),
PSB_IOCTL_DEF(PVR_DRM_UNPRIV_IOCTL, PVRDRMUnprivCmd2, DRM_UNLOCKED),
#ifdef CONFIG_CTP_DPST
PSB_IOCTL_DEF(DRM_IOCTL_PSB_HIST_ENABLE,
psb_hist_enable_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_HIST_STATUS,
psb_hist_status_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_UPDATE_GUARD, psb_update_guard_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_INIT_COMM, psb_init_comm_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_DPST, psb_dpst_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GAMMA, psb_gamma_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_DPST_BL, psb_dpst_bl_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_DPST_LEVEL, psb_dpst_get_level_ioctl, DRM_AUTH),
#endif
#if defined(PDUMP)
PSB_IOCTL_DEF(PVR_DRM_DBGDRV_IOCTL, SYSPVRDBGDrivIoctl2, 0),
#endif
#ifdef CONFIG_MDFD_VIDEO_DECODE
PSB_IOCTL_DEF(DRM_IOCTL_PSB_CMDBUF, psb_cmdbuf_ioctl,
DRM_AUTH | DRM_UNLOCKED),
/*to be removed later*/
/*PSB_IOCTL_DEF(DRM_IOCTL_PSB_SCENE_UNREF, drm_psb_scene_unref_ioctl,
DRM_AUTH),*/
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_CREATE, psb_pl_create_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_REFERENCE, psb_pl_reference_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_UNREF, psb_pl_unref_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_SYNCCPU, psb_pl_synccpu_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_WAITIDLE, psb_pl_waitidle_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_SETSTATUS, psb_pl_setstatus_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_PL_CREATE_UB, psb_pl_ub_create_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_FENCE_SIGNALED,
psb_fence_signaled_ioctl, DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_FENCE_FINISH, psb_fence_finish_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_TTM_FENCE_UNREF, psb_fence_unref_ioctl,
DRM_AUTH | DRM_UNLOCKED),
/*to be removed later */
/*PSB_IOCTL_DEF(DRM_IOCTL_PSB_FLIP, psb_page_flip, DRM_AUTH),*/
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VIDEO_GETPARAM,
psb_video_getparam, DRM_AUTH | DRM_UNLOCKED),
#endif
PSB_IOCTL_DEF(DRM_IOCRL_PSB_DPU_QUERY, psb_dpu_query_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCRL_PSB_DPU_DSR_ON, psb_dpu_dsr_on_ioctl,
DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCRL_PSB_DPU_DSR_OFF, psb_dpu_dsr_off_ioctl,
DRM_AUTH),
#ifdef CONFIG_SUPPORT_HDMI
PSB_IOCTL_DEF(DRM_IOCTL_PSB_HDMI_FB_CMD, psb_disp_ioctl, 0),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_QUERY_HDCP, psb_query_hdcp_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VALIDATE_HDCP_KSV, psb_validate_hdcp_ksv_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GET_HDCP_STATUS, psb_get_hdcp_status_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_ENABLE_HDCP, psb_enable_hdcp_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_DISABLE_HDCP, psb_disable_hdcp_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GET_HDCP_LINK_STATUS, psb_get_hdcp_link_status_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_HDCP_DISPLAY_IED_OFF,
psb_disable_display_ied_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_HDCP_DISPLAY_IED_ON,
psb_enable_display_ied_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_QUERY_HDCP_DISPLAY_IED_CAPS,
psb_query_display_ied_caps_ioctl, DRM_AUTH),
#endif
PSB_IOCTL_DEF(DRM_IOCTL_PSB_CSC_GAMMA_SETTING, psb_csc_gamma_setting_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_SET_CSC, psb_set_csc_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_VSYNC_SET, psb_vsync_set_ioctl,
DRM_AUTH | DRM_UNLOCKED),
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
PSB_IOCTL_DEF(DRM_IOCTL_PSB_PANEL_SWITCH, psb_panel_switch_ioctl,DRM_AUTH | DRM_UNLOCKED),
#endif
//ASUS_BSP: [DDS] ---
PSB_IOCTL_DEF(DRM_IOCTL_PSB_GET_DC_INFO, psb_get_dc_info_ioctl,
DRM_AUTH | DRM_UNLOCKED),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_ENABLE_IED_SESSION,
psb_enable_ied_session_ioctl, DRM_AUTH),
PSB_IOCTL_DEF(DRM_IOCTL_PSB_DISABLE_IED_SESSION,
psb_disable_ied_session_ioctl, DRM_AUTH),
};
static void get_imr_info(struct drm_psb_private *dev_priv)
{
u32 high, low, start, end;
int size = 0;
low = intel_mid_msgbus_read32(PNW_IMR_MSG_PORT,
PNW_IMR4L_MSG_REGADDR);
high = intel_mid_msgbus_read32(PNW_IMR_MSG_PORT,
PNW_IMR4H_MSG_REGADDR);
start = (low & PNW_IMR_ADDRESS_MASK) << PNW_IMR_ADDRESS_SHIFT;
end = (high & PNW_IMR_ADDRESS_MASK) << PNW_IMR_ADDRESS_SHIFT;
if (end > start)
size = end - start + 1;
if (size > 0) {
dev_priv->imr_region_start = start;
dev_priv->imr_region_size = size & PAGE_MASK;
} else {
dev_priv->imr_region_start = 0;
dev_priv->imr_region_size = 0;
}
DRM_INFO("IMR4 start=0x%08x, size=%dB (%d pages)\n",
dev_priv->imr_region_start,
dev_priv->imr_region_size,
dev_priv->imr_region_size >> PAGE_SHIFT);
return;
}
static void psb_set_uopt(struct drm_psb_uopt *uopt)
{
return;
}
static void psb_lastclose(struct drm_device *dev)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct msvdx_private *msvdx_priv = NULL;
if (!dev_priv)
return;
msvdx_priv = dev_priv->msvdx_private;
if (msvdx_priv) {
mutex_lock(&msvdx_priv->msvdx_mutex);
if (dev_priv->decode_context.buffers) {
vfree(dev_priv->decode_context.buffers);
dev_priv->decode_context.buffers = NULL;
}
mutex_unlock(&msvdx_priv->msvdx_mutex);
}
mutex_lock(&dev_priv->cmdbuf_mutex);
if (dev_priv->encode_context.buffers) {
vfree(dev_priv->encode_context.buffers);
dev_priv->encode_context.buffers = NULL;
}
mutex_unlock(&dev_priv->cmdbuf_mutex);
}
static void psb_do_takedown(struct drm_device *dev)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct ttm_bo_device *bdev = &dev_priv->bdev;
if (dev_priv->have_mem_mmu) {
ttm_bo_clean_mm(bdev, DRM_PSB_MEM_MMU);
dev_priv->have_mem_mmu = 0;
}
if (dev_priv->have_tt) {
ttm_bo_clean_mm(bdev, TTM_PL_TT);
dev_priv->have_tt = 0;
}
if (dev_priv->have_imr) {
ttm_bo_clean_mm(bdev, TTM_PL_IMR);
dev_priv->have_imr = 0;
}
#ifdef CONFIG_MDFD_VIDEO_DECODE
psb_msvdx_uninit(dev);
if (IS_MDFLD(dev))
pnw_topaz_uninit(dev);
#endif
}
#define FB_REG06_MRST 0xD08106F0
#define FB_REG06_MDFLD 0x108106F0
#define FB_TOPAZ_DISABLE BIT0
#define FB_MIPI_DISABLE BIT11
#define FB_REG09_MRST 0xD08109F0
#define FB_REG09_MDFLD 0x108109F0
#define FB_SKU_MASK (BIT12|BIT13|BIT14)
#define FB_SKU_SHIFT 12
#define FB_SKU_100 0
#define FB_SKU_100L 1
#define FB_SKU_83 2
#define FB_GFX_CLK_DIVIDE_MASK (BIT20|BIT21|BIT22)
#define FB_GFX_CLK_DIVIDE_SHIFT 20
#define FB_VED_CLK_DIVIDE_MASK (BIT23|BIT24)
#define FB_VED_CLK_DIVIDE_SHIFT 23
#define FB_VEC_CLK_DIVIDE_MASK (BIT25|BIT26)
#define FB_VEC_CLK_DIVIDE_SHIFT 25
void mrst_get_fuse_settings(struct drm_device *dev)
{
struct drm_psb_private *dev_priv = dev->dev_private;
uint32_t fuse_value = 0;
uint32_t fuse_value_tmp = 0;
fuse_value = intel_mid_msgbus_read32_raw(IS_MDFLD(dev) ?
FB_REG06_MDFLD : FB_REG06_MRST);
dev_priv->iLVDS_enable = fuse_value & FB_MIPI_DISABLE;
if (IS_MDFLD(dev)) {
dev_priv->iLVDS_enable = 0;
}
PSB_DEBUG_ENTRY("internal display is %s\n",
dev_priv->iLVDS_enable ? "LVDS display" : "MIPI display");
/*prevent Runtime suspend at start*/
if (dev_priv->iLVDS_enable) {
dev_priv->is_lvds_on = true;
dev_priv->is_mipi_on = false;
} else {
dev_priv->is_mipi_on = true;
dev_priv->is_lvds_on = false;
}
if (dev_priv->dev->pci_device == PCI_ID_TOPAZ_DISABLED)
dev_priv->topaz_disabled = 1;
else
dev_priv->topaz_disabled = 0;
dev_priv->video_device_fuse = fuse_value;
PSB_DEBUG_ENTRY("topaz is %s\n",
dev_priv->topaz_disabled ? "disabled" : "enabled");
fuse_value = intel_mid_msgbus_read32_raw(IS_MDFLD(dev) ?
FB_REG09_MDFLD : FB_REG09_MRST);
PSB_DEBUG_ENTRY("SKU values is 0x%x.\n", fuse_value);
fuse_value_tmp = (fuse_value & FB_SKU_MASK) >> FB_SKU_SHIFT;
dev_priv->fuse_reg_value = fuse_value;
switch (fuse_value_tmp) {
case FB_SKU_100:
dev_priv->core_freq = 200;
break;
case FB_SKU_100L:
dev_priv->core_freq = 100;
break;
case FB_SKU_83:
dev_priv->core_freq = 166;
break;
default:
DRM_ERROR("Invalid SKU values, SKU value = 0x%08x\n", fuse_value_tmp);
dev_priv->core_freq = 0;
}
PSB_DEBUG_ENTRY("LNC core clk is %dMHz.\n", dev_priv->core_freq);
#if 0 /* debug message */
fuse_value_tmp =
(fuse_value & FB_GFX_CLK_DIVIDE_MASK) >> FB_GFX_CLK_DIVIDE_SHIFT;
switch (fuse_value_tmp) {
case 0:
DRM_INFO("Gfx clk : core clk = 1:1. \n");
break;
case 1:
DRM_INFO("Gfx clk : core clk = 4:3. \n");
break;
case 2:
DRM_INFO("Gfx clk : core clk = 8:5. \n");
break;
case 3:
DRM_INFO("Gfx clk : core clk = 2:1. \n");
break;
case 4:
DRM_INFO("Gfx clk : core clk = 16:7. \n");
break;
case 5:
DRM_INFO("Gfx clk : core clk = 8:3. \n");
break;
case 6:
DRM_INFO("Gfx clk : core clk = 16:5. \n");
break;
case 7:
DRM_INFO("Gfx clk : core clk = 4:1. \n");
break;
default:
DRM_ERROR("Invalid GFX CLK DIVIDE values, value = 0x%08x\n",
fuse_value_tmp);
}
fuse_value_tmp =
(fuse_value & FB_VED_CLK_DIVIDE_MASK) >> FB_VED_CLK_DIVIDE_SHIFT;
switch (fuse_value_tmp) {
case 0:
DRM_INFO("Ved clk : core clk = 1:1. \n");
break;
case 1:
DRM_INFO("Ved clk : core clk = 4:3. \n");
break;
case 2:
DRM_INFO("Ved clk : core clk = 8:5. \n");
break;
case 3:
DRM_INFO("Ved clk : core clk = 2:1. \n");
break;
default:
DRM_ERROR("Invalid VED CLK DIVIDE values, value = 0x%08x\n",
fuse_value_tmp);
}
fuse_value_tmp =
(fuse_value & FB_VEC_CLK_DIVIDE_MASK) >> FB_VEC_CLK_DIVIDE_SHIFT;
switch (fuse_value_tmp) {
case 0:
DRM_INFO("Vec clk : core clk = 1:1. \n");
break;
case 1:
DRM_INFO("Vec clk : core clk = 4:3. \n");
break;
case 2:
DRM_INFO("Vec clk : core clk = 8:5. \n");
break;
case 3:
DRM_INFO("Vec clk : core clk = 2:1. \n");
break;
default:
DRM_ERROR("Invalid VEC CLK DIVIDE values, value = 0x%08x\n",
fuse_value_tmp);
}
#endif /* FIXME remove it after PO */
if (IS_MDFLD(dev)) {
#if KSEL_BYPASS_83_100_ENABLE
dev_priv->ksel = KSEL_BYPASS_83_100;
#endif /* KSEL_BYPASS_83_100_ENABLE */
#if KSEL_CRYSTAL_19_ENABLED
dev_priv->ksel = KSEL_CRYSTAL_19;
#endif /* KSEL_CRYSTAL_19_ENABLED */
#if KSEL_CRYSTAL_38_ENABLED
dev_priv->ksel = KSEL_CRYSTAL_38;
#endif /* KSEL_CRYSTAL_38_ENABLED */
}
return;
}
bool mid_get_pci_revID(struct drm_psb_private *dev_priv)
{
uint32_t platform_rev_id = 0;
struct pci_dev *pci_gfx_root = pci_get_bus_and_slot(0, PCI_DEVFN(2, 0));
if (!pci_gfx_root) {
DRM_ERROR("Invalid root\n");
return false;
}
/*get the revison ID, B0:D2:F0;0x08 */
pci_read_config_dword(pci_gfx_root, 0x08, &platform_rev_id);
dev_priv->platform_rev_id = (uint8_t) platform_rev_id;
pci_dev_put(pci_gfx_root);
PSB_DEBUG_ENTRY("platform_rev_id is %x\n", dev_priv->platform_rev_id);
return true;
}
static bool intel_mid_get_vbt_data(struct drm_psb_private *dev_priv)
{
u32 platform_config_address;
u8 *pVBT_virtual;
u8 primary_panel;
u8 number_desc = 0;
u8 panel_name[PANEL_NAME_MAX_LEN+1] = {0};
struct intel_mid_vbt *pVBT = &dev_priv->vbt_data;
void *panel_desc;
struct pci_dev *pci_gfx_root = pci_get_bus_and_slot(0, PCI_DEVFN(2, 0));
mdfld_dsi_encoder_t mipi_mode;
int ret = 0, len = 0;
struct platform_device *pdev;
PSB_DEBUG_ENTRY("\n");
if (!pci_gfx_root) {
DRM_ERROR("Invalid root\n");
return false;
}
/*get the address of the platform config vbt, B0:D2:F0;0xFC */
pci_read_config_dword(pci_gfx_root, 0xFC, &platform_config_address);
pci_dev_put(pci_gfx_root);
/**
* if platform_config_address is 0,
* that means FW doesn't support VBT
*/
if (platform_config_address == 0) {
pVBT->size = 0;
return false;
}
/*copy vbt data to local memory*/
pVBT_virtual = ioremap(platform_config_address, sizeof(*pVBT));
if (!pVBT_virtual) {
DRM_ERROR("fail to ioremap platform_config_address:0x%x\n",
platform_config_address);
return false;
}
memcpy(pVBT, pVBT_virtual, sizeof(*pVBT));
iounmap(pVBT_virtual); /* Free virtual address space */
if (strncmp(pVBT->signature, "$GCT", 4)) {
DRM_ERROR("wrong GCT signature\n");
return false;
}
PSB_DEBUG_ENTRY("GCT Revision is %#x\n", pVBT->revision);
number_desc = pVBT->num_of_panel_desc;
primary_panel = pVBT->primary_panel_idx;
dev_priv->gct_data.bpi = primary_panel; /*save boot panel id*/
/**
* current we just need parse revision 0x11 and 0x20
*/
switch (pVBT->revision) {
case 0x11:
/* number of descriptors defined in the GCT */
pVBT->panel_descs =
ioremap(platform_config_address + GCT_R11_HEADER_SIZE,
GCT_R11_DISPLAY_DESC_SIZE * number_desc);
panel_desc = (u8 *)pVBT->panel_descs +
(primary_panel * GCT_R11_DISPLAY_DESC_SIZE);
if (!panel_desc) {
DRM_ERROR("Invalid desc\n");
return false;
}
strncpy(panel_name, panel_desc, PANEL_NAME_MAX_LEN);
mipi_mode =
((struct gct_r11_panel_desc *)panel_desc)->display.mode ? \
MDFLD_DSI_ENCODER_DPI : MDFLD_DSI_ENCODER_DBI;
break;
case 0x20:
pVBT->panel_descs =
ioremap(platform_config_address + GCT_R20_HEADER_SIZE,
GCT_R20_DISPLAY_DESC_SIZE * number_desc);
panel_desc = (u8 *)pVBT->panel_descs +
(primary_panel * GCT_R20_DISPLAY_DESC_SIZE);
if (!panel_desc) {
DRM_ERROR("Invalid desc\n");
return false;
}
strncpy(panel_name, panel_desc, PANEL_NAME_MAX_LEN);
mipi_mode =
((struct gct_r20_panel_desc *)panel_desc)->panel_mode.mode ?\
MDFLD_DSI_ENCODER_DPI : MDFLD_DSI_ENCODER_DBI;
break;
default:
pr_err("unsupported GCT revision\n");
pVBT->size = 0;
return false;
}
len = strnlen(panel_name, PANEL_NAME_MAX_LEN);
if (len) {
strncpy(dev_priv->panel_info.name, panel_name, len);
dev_priv->panel_info.mode = mipi_mode;
} else {
DRM_ERROR("%s: detect panel info from gct error\n",
__func__);
return false;
}
pdev = platform_device_alloc(panel_name, -1);
if (!pdev) {
DRM_ERROR("%s: fail to alloc platform device\n", __func__);
return false;
}
ret = platform_device_add(pdev);
if (ret) {
DRM_ERROR("%s: fail to add platform device\n", __func__);
return false;
}
DRM_INFO("%s: panel name: %s, mipi_mode = %d\n", __func__,
panel_name, mipi_mode);
return true;
}
#ifdef CONFIG_SUPPORT_HDMI
void hdmi_do_audio_wq(struct work_struct *work)
{
struct drm_psb_private *dev_priv = container_of(work,
struct drm_psb_private,
hdmi_audio_wq);
bool hdmi_hpd_connected = false;
/* As in the hdmi_do_hotplug_wq() function above
* it seems we should not be running this section of
* code if we don't also have CONFIG_SUPPORT_HDMI set,
* some devices might not want/need support for HDMI
* early in the platform bring up and by having this
* available to run might produce unexpected results
* if HDMI connector is plugged in.
*/
DRM_INFO("hdmi_do_audio_wq: Checking for HDMI connection at boot\n");
hdmi_hpd_connected = android_hdmi_is_connected(dev_priv->dev);
if (hdmi_hpd_connected) {
DRM_INFO("hdmi_do_audio_wq: HDMI plugged in\n");
mid_hdmi_audio_signal_event(dev_priv->dev, HAD_EVENT_HOT_PLUG);
}
}
#endif
static int psb_do_init(struct drm_device *dev)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct ttm_bo_device *bdev = &dev_priv->bdev;
struct psb_gtt *pg = dev_priv->pg;
uint32_t tmp;
uint32_t stolen_gtt;
uint32_t tt_start;
uint32_t tt_pages;
int ret = -ENOMEM;
/*
* Initialize sequence numbers for the different command
* submission mechanisms.
*/
dev_priv->sequence[PSB_ENGINE_DECODE] = 1;
dev_priv->sequence[LNC_ENGINE_ENCODE] = 0;
if (pg->mmu_gatt_start & 0x0FFFFFFF) {
DRM_ERROR("Gatt must be 256M aligned. This is a bug.\n");
ret = -EINVAL;
goto out_err;
}
stolen_gtt = (pg->stolen_size >> PAGE_SHIFT) * 4;
stolen_gtt = (stolen_gtt + PAGE_SIZE - 1) >> PAGE_SHIFT;
stolen_gtt =
(stolen_gtt < pg->gtt_pages) ? stolen_gtt : pg->gtt_pages;
dev_priv->gatt_free_offset = pg->mmu_gatt_start +
(stolen_gtt << PAGE_SHIFT) * 1024;
if (1 || drm_debug) {
uint32_t core_id = PSB_RSGX32(PSB_CR_CORE_ID);
uint32_t core_rev = PSB_RSGX32(PSB_CR_CORE_REVISION);
DRM_INFO("SGX core id = 0x%08x\n", core_id);
DRM_INFO("SGX core rev major = 0x%02x, minor = 0x%02x\n",
(core_rev & _PSB_CC_REVISION_MAJOR_MASK) >>
_PSB_CC_REVISION_MAJOR_SHIFT,
(core_rev & _PSB_CC_REVISION_MINOR_MASK) >>
_PSB_CC_REVISION_MINOR_SHIFT);
DRM_INFO
("SGX core rev maintenance = 0x%02x, designer = 0x%02x\n",
(core_rev & _PSB_CC_REVISION_MAINTENANCE_MASK) >>
_PSB_CC_REVISION_MAINTENANCE_SHIFT,
(core_rev & _PSB_CC_REVISION_DESIGNER_MASK) >>
_PSB_CC_REVISION_DESIGNER_SHIFT);
}
spin_lock_init(&dev_priv->irqmask_lock);
spin_lock_init(&dev_priv->flip_lock);
tt_pages = (pg->gatt_pages < PSB_TT_PRIV0_PLIMIT) ?
pg->gatt_pages : PSB_TT_PRIV0_PLIMIT;
tt_start = dev_priv->gatt_free_offset - pg->mmu_gatt_start;
tt_pages -= tt_start >> PAGE_SHIFT;
dev_priv->sizes.ta_mem_size = 0;
/* IMR region managed by TTM */
tmp = dev_priv->imr_region_size >> PAGE_SHIFT; /* IMR region size */
if ((dev_priv->imr_region_size != 0) &&
!ttm_bo_init_mm(bdev, TTM_PL_IMR, tmp))
dev_priv->have_imr = 1;
/* TT region managed by TTM. */
tmp = pg->gatt_pages -
(pg->gtt_video_start >> PAGE_SHIFT) -
(dev_priv->ci_region_size >> PAGE_SHIFT); /* TT region size */
if (!ttm_bo_init_mm(bdev, TTM_PL_TT, tmp))
dev_priv->have_tt = 1;
/* MMU region managed by TTM */
tmp = PSB_MEM_IMR_START >> PAGE_SHIFT; /* MMU region size:MMU->IMR */
if (!ttm_bo_init_mm(bdev, DRM_PSB_MEM_MMU, tmp))
dev_priv->have_mem_mmu = 1;
if (IS_MSVDX_MEM_TILE(dev)) {
/* Create tiling MMU region managed by TTM */
tmp = (0x10000000) >> PAGE_SHIFT;
printk(KERN_INFO "init tiling heap, size is 0x%08x pages\n", tmp);
if (!ttm_bo_init_mm(bdev, DRM_PSB_MEM_MMU_TILING, tmp))
dev_priv->have_mem_mmu_tiling = 1;
}
PSB_DEBUG_INIT("Init MSVDX\n");
/*
* Customer will boot droidboot, then boot the MOS kernel.
* It is observed the video decode island is off during the
* MOS kernel boot.
* We need to power on it first, else will cause the fabric error.
*/
if (ospm_power_island_up(OSPM_VIDEO_DEC_ISLAND)) {
DRM_ERROR("ospm_video_dec_island_up failed.\n");
ret = -EINVAL;
goto out_err;
}
psb_msvdx_init(dev);
PSB_DEBUG_INIT("Init Topaz\n");
/*
* Customer will boot droidboot, then boot the MOS kernel.
* It is observed the video encode island is off during the
* MOS kernel boot while the panel is not connected.
* We need to power on it first, else will cause fabric error.
*/
if (ospm_power_island_up(OSPM_VIDEO_ENC_ISLAND)) {
DRM_ERROR("ospm_video_enc_island_up failed.\n");
ret = -EINVAL;
goto out_err;
}
if (IS_MDFLD(dev))
pnw_topaz_init(dev);
return 0;
out_err:
psb_do_takedown(dev);
return ret;
}
static void mdfld_free_overlay_backbuf(struct drm_device *dev)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
int i;
for (i = 0; i < OVERLAY_BACKBUF_NUM; i++) {
if (dev_priv->overlay_backbuf[i]) {
ttm_bo_kunmap(&dev_priv->overlay_kmap[i]);
ttm_bo_unref(&dev_priv->overlay_backbuf[i]);
}
}
}
static int mdfld_init_overlay_backbuf(struct drm_device *dev)
{
struct ttm_buffer_object *overlay_buf;
struct drm_psb_private *dev_priv = psb_priv(dev);
struct ttm_bo_device *bdev = &dev_priv->bdev;
int i, ret;
for (i = 0; i < OVERLAY_BACKBUF_NUM; i++) {
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 8, 0))
ret = ttm_buffer_object_create(bdev, 4096, ttm_bo_type_kernel,
TTM_PL_FLAG_TT | TTM_PL_FLAG_WC |
TTM_PL_FLAG_NO_EVICT,
16, 0, 0, NULL, &overlay_buf);
#else
ret = ttm_buffer_object_create(bdev, 4096, ttm_bo_type_kernel,
TTM_PL_FLAG_TT | TTM_PL_FLAG_WC |
TTM_PL_FLAG_NO_EVICT,
16, 0, NULL, &overlay_buf);
#endif
if (ret < 0) {
DRM_ERROR("failed to alloc back buf %d, ret:%d\n",
i, ret);
goto fail;
}
DRM_INFO("overlay backbuf:%d gpu offset:%#lx\n",
i, overlay_buf->offset);
ret = ttm_bo_kmap(overlay_buf, 0, overlay_buf->num_pages,
&dev_priv->overlay_kmap[i]);
if (ret) {
DRM_ERROR("fail to map overlay buf %d\n", i);
ttm_bo_unref(&overlay_buf);
goto fail;
}
dev_priv->overlay_backbuf[i] = overlay_buf;
}
mutex_init(&dev_priv->ov_ctrl_lock);
dev_priv->ov_ctrl_blk = kzalloc(sizeof(struct overlay_ctrl_blk) *
INTEL_OVERLAY_PLANE_NUM, GFP_KERNEL);
if (!dev_priv->ov_ctrl_blk)
goto fail;
return 0;
fail:
mdfld_free_overlay_backbuf(dev);
return ret;
}
static int psb_driver_unload(struct drm_device *dev)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
/*Fristly, unload pvr driver*/
if (!dev_priv || !dev_priv->pvr_ops)
return -EINVAL;
dev_priv->pvr_ops->PVRSRVDrmUnload(dev);
/*TODO: destroy DSR/DPU infos here*/
psb_backlight_exit(); /*writes minimum value to backlight HW reg */
if (drm_psb_no_fb == 0)
psb_modeset_cleanup(dev);
destroy_workqueue(te_wq);
destroy_workqueue(vsync_wq);
mdfld_free_overlay_backbuf(dev);
kfree(dev_priv->ov_ctrl_blk);
if (dev_priv) {
/* psb_watchdog_takedown(dev_priv); */
psb_do_takedown(dev);
#ifdef CONFIG_MDFD_VIDEO_DECODE
if (dev_priv->pf_pd) {
psb_mmu_free_pagedir(dev_priv->pf_pd);
dev_priv->pf_pd = NULL;
}
#endif
if (dev_priv->mmu) {
struct psb_gtt *pg = dev_priv->pg;
down_read(&pg->sem);
#ifdef CONFIG_MDFD_VIDEO_DECODE
psb_mmu_remove_pfn_sequence(
psb_mmu_get_default_pd
(dev_priv->mmu),
pg->mmu_gatt_start,
pg->vram_stolen_size >> PAGE_SHIFT);
if (pg->rar_stolen_size != 0)
psb_mmu_remove_pfn_sequence(
psb_mmu_get_default_pd
(dev_priv->mmu),
pg->rar_start,
pg->rar_stolen_size >> PAGE_SHIFT);
#endif
up_read(&pg->sem);
#ifdef CONFIG_MDFD_VIDEO_DECODE
psb_mmu_driver_takedown(dev_priv->mmu);
#endif
dev_priv->mmu = NULL;
}
psb_gtt_takedown(dev_priv->pg, 1);
if (dev_priv->scratch_page) {
__free_page(dev_priv->scratch_page);
dev_priv->scratch_page = NULL;
}
#ifdef CONFIG_MDFD_VIDEO_DECODE
if (dev_priv->has_bo_device) {
ttm_bo_device_release(&dev_priv->bdev);
dev_priv->has_bo_device = 0;
}
if (dev_priv->has_fence_device) {
ttm_fence_device_release(&dev_priv->fdev);
dev_priv->has_fence_device = 0;
}
#endif
if (dev_priv->vdc_reg) {
iounmap(dev_priv->vdc_reg);
dev_priv->vdc_reg = NULL;
}
if (dev_priv->sgx_reg) {
iounmap(dev_priv->sgx_reg);
dev_priv->sgx_reg = NULL;
}
#ifdef CONFIG_MDFD_GL3
if (IS_MDFLD(dev) && dev_priv->platform_rev_id != MDFLD_PNW_A0) {
iounmap(dev_priv->gl3_reg);
dev_priv->gl3_reg = NULL;
}
#endif
#ifdef CONFIG_MDFD_VIDEO_DECODE
if (dev_priv->msvdx_reg) {
iounmap(dev_priv->msvdx_reg);
dev_priv->msvdx_reg = NULL;
}
if (IS_TOPAZ(dev)) {
if (dev_priv->topaz_reg) {
iounmap(dev_priv->topaz_reg);
dev_priv->topaz_reg = NULL;
}
}
if (dev_priv->tdev)
ttm_object_device_release(&dev_priv->tdev);
if (dev_priv->has_global)
psb_ttm_global_release(dev_priv);
#endif
kfree(dev_priv->vblank_count);
kfree(dev_priv);
dev->dev_private = NULL;
}
ospm_power_uninit();
return 0;
}
static int psb_driver_load(struct drm_device *dev, unsigned long chipset)
{
struct drm_psb_private *dev_priv;
struct ttm_bo_device *bdev;
unsigned long resource_start;
struct psb_gtt *pg;
unsigned long irqflags;
int ret = -ENOMEM;
uint32_t tt_pages;
int i = 0;
DRM_INFO("psb - %s\n", PSB_PACKAGE_VERSION);
dev_priv = kzalloc(sizeof(*dev_priv), GFP_KERNEL);
if (dev_priv == NULL)
return -ENOMEM;
INIT_LIST_HEAD(&dev_priv->video_ctx);
spin_lock_init(&dev_priv->video_ctx_lock);
if (IS_CTP(dev)) {
dev_priv->num_pipe = 2;
} else if (IS_MDFLD(dev)) {
dev_priv->num_pipe = 1;
#ifdef CONFIG_MDFD_DUAL_MIPI
dev_priv->num_pipe++;
#endif
#ifdef CONFIG_SUPPORT_HDMI
dev_priv->num_pipe++;
#endif
} else if (IS_MRST(dev))
dev_priv->num_pipe = 1;
else
dev_priv->num_pipe = 2;
/*init DPST umcomm to NULL*/
dev_priv->psb_dpst_state = NULL;
dev_priv->um_start = false;
dev_priv->b_vblank_enable = false;
dev_priv->dev = dev;
bdev = &dev_priv->bdev;
hdmi_state = 0;
drm_hdmi_hpd_auto = 0;
dev_priv->ied_enabled = false;
dev_priv->ied_context = NULL;
dev_priv->bhdmiconnected = false;
dev_priv->bhdmi_enable = true;
dev_priv->dpms_on_off = false;
atomic_set(&dev_priv->mipi_flip_abnormal, 0);
dev_priv->brightness_adjusted = 0;
dev_priv->buf = kzalloc(PSB_REG_PRINT_SIZE * sizeof(char),
GFP_KERNEL);
if (dev_priv->buf == NULL)
return -ENOMEM;
#ifdef CONFIG_MDFD_VIDEO_DECODE
ret = psb_ttm_global_init(dev_priv);
if (unlikely(ret != 0))
goto out_err;
dev_priv->has_global = 1;
dev_priv->tdev = ttm_object_device_init
(dev_priv->mem_global_ref.object, PSB_OBJECT_HASH_ORDER);
if (unlikely(dev_priv->tdev == NULL))
goto out_err;
mutex_init(&dev_priv->cmdbuf_mutex);
INIT_LIST_HEAD(&dev_priv->decode_context.validate_list);
INIT_LIST_HEAD(&dev_priv->encode_context.validate_list);
/*
INIT_LIST_HEAD(&dev_priv->decode_context.kern_validate_list);
INIT_LIST_HEAD(&dev_priv->encode_context.kern_validate_list);
*/
#endif
mutex_init(&dev_priv->dpms_mutex);
mutex_init(&dev_priv->gamma_csc_lock);
mutex_init(&dev_priv->overlay_lock);
mutex_init(&dev_priv->vsync_lock);
spin_lock_init(&dev_priv->reloc_lock);
DRM_INIT_WAITQUEUE(&dev_priv->rel_mapped_queue);
dev->dev_private = (void *) dev_priv;
dev_priv->chipset = chipset;
psb_set_uopt(&dev_priv->uopt);
PSB_DEBUG_INIT("Mapping MMIO\n");
resource_start = pci_resource_start(dev->pdev, PSB_MMIO_RESOURCE);
#ifdef CONFIG_MDFD_VIDEO_DECODE
dev_priv->msvdx_reg =
ioremap(resource_start + MRST_MSVDX_OFFSET,
PSB_MSVDX_SIZE);
if (!dev_priv->msvdx_reg)
goto out_err;
if (IS_TOPAZ(dev)) {
if (IS_MDFLD(dev)) {
dev_priv->topaz_reg =
ioremap(resource_start + PNW_TOPAZ_OFFSET,
PNW_TOPAZ_SIZE);
} else
dev_priv->topaz_reg =
ioremap(resource_start + LNC_TOPAZ_OFFSET,
LNC_TOPAZ_SIZE);
if (!dev_priv->topaz_reg)
goto out_err;
}
#endif
dev_priv->vdc_reg =
ioremap(resource_start + PSB_VDC_OFFSET, PSB_VDC_SIZE);
if (!dev_priv->vdc_reg)
goto out_err;
if (IS_MID(dev))
dev_priv->sgx_reg =
ioremap(resource_start + MRST_SGX_OFFSET,
PSB_SGX_SIZE);
else
dev_priv->sgx_reg =
ioremap(resource_start + PSB_SGX_OFFSET, PSB_SGX_SIZE);
if (!dev_priv->sgx_reg)
goto out_err;
/*WA in PO: Program SGX544MP master clk gating earlier, or CPU stuck later
*before loading SGX driver, root cause is still unkown
*/
#if defined(SGX_FEATURE_MP)
if (IS_CTP_NEED_WA(dev)) {
if (SGX_FEATURE_MP_CORE_COUNT == 2) {
iowrite32(0x1, dev_priv->sgx_reg + 0x4000);
iowrite32(0x5, dev_priv->sgx_reg + 0x4004);
iowrite32(0xa, dev_priv->sgx_reg + 0x4004);
} else if (SGX_FEATURE_MP_CORE_COUNT == 1) {
iowrite32(0x0, dev_priv->sgx_reg + 0x4000);
iowrite32(0x1, dev_priv->sgx_reg + 0x4004);
iowrite32(0x2, dev_priv->sgx_reg + 0x4004);
}
iowrite32(0x2aa, dev_priv->sgx_reg + 0x4020);
}
#endif
/* setup hdmi driver */
android_hdmi_driver_setup(dev);
if (IS_MID(dev)) {
mrst_get_fuse_settings(dev);
intel_mid_get_vbt_data(dev_priv);
mid_get_pci_revID(dev_priv);
}
#ifdef CONFIG_MDFD_GL3
/* GL3 */
if (IS_MDFLD(dev) && dev_priv->platform_rev_id != MDFLD_PNW_A0) {
dev_priv->gl3_reg =
ioremap(resource_start + MDFLD_GL3_OFFSET, MDFLD_GL3_SIZE);
if (!dev_priv->gl3_reg)
goto out_err;
}
#endif
PSB_DEBUG_INIT("Init TTM fence and BO driver\n");
get_imr_info(dev_priv);
/* Init OSPM support */
ospm_power_init(dev);
#ifdef CONFIG_MDFD_VIDEO_DECODE
ret = psb_ttm_fence_device_init(&dev_priv->fdev);
if (unlikely(ret != 0))
goto out_err;
/* For VXD385 DE2.x firmware support 16bit fence value */
dev_priv->fdev.fence_class[PSB_ENGINE_DECODE].wrap_diff = (1 << 14);
dev_priv->fdev.fence_class[PSB_ENGINE_DECODE].flush_diff = (1 << 13);
dev_priv->fdev.fence_class[PSB_ENGINE_DECODE].sequence_mask =
0x0000ffff;
dev_priv->has_fence_device = 1;
ret = ttm_bo_device_init(bdev,
dev_priv->bo_global_ref.ref.object,
&psb_ttm_bo_driver,
DRM_PSB_FILE_PAGE_OFFSET, false);
if (unlikely(ret != 0))
goto out_err;
dev_priv->has_bo_device = 1;
ttm_lock_init(&dev_priv->ttm_lock);
#endif
ret = -ENOMEM;
dev_priv->scratch_page = alloc_page(GFP_DMA32 | __GFP_ZERO);
if (!dev_priv->scratch_page)
goto out_err;
set_pages_uc(dev_priv->scratch_page, 1);
dev_priv->pg = psb_gtt_alloc(dev);
if (!dev_priv->pg)
goto out_err;
ret = psb_gtt_init(dev_priv->pg, 0);
if (ret)
goto out_err;
ret = psb_gtt_mm_init(dev_priv->pg);
if (ret)
goto out_err;
#ifdef CONFIG_MDFD_VIDEO_DECODE
dev_priv->mmu = psb_mmu_driver_init((void *)0,
drm_psb_trap_pagefaults, 0,
dev_priv, IMG_MMU);
if (!dev_priv->mmu)
goto out_err;
#endif
pg = dev_priv->pg;
tt_pages = (pg->gatt_pages < PSB_TT_PRIV0_PLIMIT) ?
(pg->gatt_pages) : PSB_TT_PRIV0_PLIMIT;
/* CI/RAR use the lower half of TT. */
pg->gtt_video_start = (tt_pages / 2) << PAGE_SHIFT;
pg->rar_start = pg->gtt_video_start + pg->ci_stolen_size;
#ifdef CONFIG_MDFD_VIDEO_DECODE
/*
* Make MSVDX/TOPAZ MMU aware of the imr stolen memory area.
*/
if (dev_priv->pg->rar_stolen_size != 0) {
down_read(&pg->sem);
ret = psb_mmu_insert_pfn_sequence(
psb_mmu_get_default_pd(dev_priv->mmu),
dev_priv->imr_region_start >> PAGE_SHIFT,
PSB_MEM_IMR_START,
pg->rar_stolen_size >> PAGE_SHIFT, 0);
up_read(&pg->sem);
if (ret)
goto out_err;
}
dev_priv->pf_pd = psb_mmu_alloc_pd(dev_priv->mmu, 1, 0);
if (!dev_priv->pf_pd)
goto out_err;
psb_mmu_set_pd_context(psb_mmu_get_default_pd(dev_priv->mmu), 0);
psb_mmu_set_pd_context(dev_priv->pf_pd, 1);
spin_lock_init(&dev_priv->sequence_lock);
PSB_DEBUG_INIT("Begin to init MSVDX/Topaz\n");
#endif
ret = psb_do_init(dev);
if (ret)
return ret;
if (pci_enable_msi(dev->pdev)) {
DRM_ERROR("Enable MSI failed!\n");
} else {
PSB_DEBUG_INIT("Enabled MSI IRQ (%d)\n",
dev->pdev->irq);
/* pci_write_config_word(pdev, 0x04, 0x07); */
}
ret = drm_vblank_init(dev, dev_priv->num_pipe);
if (ret)
goto out_err;
/* disable vblank_disable_timer in drm_vblank_put
* because the vblank disable should be controlled
* by framework in Jellybean instead of in driver.
*/
drm_vblank_offdelay = 0;
DRM_INIT_WAITQUEUE(&dev_priv->vsync_queue);
dev_priv->vblank_count =
kmalloc(sizeof(atomic_t) * dev_priv->num_pipe, GFP_KERNEL);
if (!dev_priv->vblank_count)
goto out_err;
for (i = 0; i < dev_priv->num_pipe; i++)
atomic_set(&dev_priv->vblank_count[i], 0);
/*
* Install interrupt handlers prior to powering off SGX or else we will
* crash.
*/
dev_priv->vdc_irq_mask = 0;
dev_priv->pipestat[0] = 0;
dev_priv->pipestat[1] = 0;
dev_priv->pipestat[2] = 0;
spin_lock_irqsave(&dev_priv->irqmask_lock, irqflags);
PSB_WVDC32(0x00000000, PSB_INT_ENABLE_R);
PSB_WVDC32(0xFFFFFFFF, PSB_INT_MASK_R);
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
if (drm_core_check_feature(dev, DRIVER_MODESET)) {
drm_irq_install(dev);
if (IS_CTP(dev)) {
if (irq_set_affinity(drm_dev_to_irq(dev),
cpumask_of(0)))
pr_err("psb_drv: set irq affinity failed\n");
else
pr_info("psb_drv: set irq affnity to CPU0\n");
}
}
dev->vblank_disable_allowed = 1;
dev->max_vblank_count = 0xffffff; /* only 24 bits of frame count */
dev->driver->get_vblank_counter = psb_get_vblank_counter;
if (IS_MDFLD(dev)) {
#ifdef CONFIG_MDFLD_DSI_DPU
/*init dpu info*/
mdfld_dbi_dpu_init(dev);
#else
mdfld_dbi_dsr_init(dev);
#endif /*CONFIG_MDFLD_DSI_DPU*/
INIT_WORK(&dev_priv->te_work, mdfld_te_handler_work);
te_wq = alloc_workqueue("teworkq", WQ_UNBOUND, 1);
if (unlikely(!te_wq)) {
pr_err(": unable to create TE workqueue\n");
goto out_err;
}
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
INIT_WORK(&dev_priv->reset_panel_work,
mdfld_reset_same_dpi_panel_work);
#else
INIT_WORK(&dev_priv->reset_panel_work,
mdfld_reset_panel_handler_work);
#endif
//ASUS_BSP: [DDS] ---
INIT_WORK(&dev_priv->vsync_event_work, mdfld_vsync_event_work);
vsync_wq = alloc_workqueue("vsyncworkq", WQ_UNBOUND, 1);
if (unlikely(!vsync_wq)) {
pr_err(": unable to create Vsync workqueue\n");
destroy_workqueue(te_wq);
goto out_err;
}
}
if (drm_psb_no_fb == 0)
psb_modeset_init(dev);
/* GL3 */
#ifdef CONFIG_MDFD_GL3
if (drm_psb_gl3_enable)
gl3_enable();
#endif
#ifdef CONFIG_SUPPORT_HDMI
INIT_WORK(&dev_priv->hdmi_audio_wq, hdmi_do_audio_wq);
#endif
mdfld_init_overlay_backbuf(dev);
/*Intel drm driver load is done, continue doing pvr load*/
DRM_DEBUG("Pvr driver load\n");
dev_priv->pvr_ops = NULL;
/* Delay PVRSRVDrmLoad to PVR module init */
g_drm_dev = dev;
return 0;
out_err:
psb_driver_unload(dev);
return ret;
}
int psb_driver_device_is_agp(struct drm_device *dev)
{
return 0;
}
int psb_extension_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
union drm_psb_extension_arg *arg = data;
struct drm_psb_extension_rep *rep = &arg->rep;
if (strcmp(arg->extension, "psb_ttm_placement_alphadrop") == 0) {
rep->exists = 1;
rep->driver_ioctl_offset = DRM_PSB_PLACEMENT_OFFSET;
rep->sarea_offset = 0;
rep->major = 1;
rep->minor = 0;
rep->pl = 0;
return 0;
}
if (strcmp(arg->extension, "psb_ttm_fence_alphadrop") == 0) {
rep->exists = 1;
rep->driver_ioctl_offset = DRM_PSB_FENCE_OFFSET;
rep->sarea_offset = 0;
rep->major = 1;
rep->minor = 0;
rep->pl = 0;
return 0;
}
if (strcmp(arg->extension, "psb_ttm_execbuf_alphadrop") == 0) {
rep->exists = 1;
rep->driver_ioctl_offset = DRM_PSB_CMDBUF;
rep->sarea_offset = 0;
rep->major = 1;
rep->minor = 0;
rep->pl = 0;
return 0;
}
/*return the page flipping ioctl offset*/
if (strcmp(arg->extension, "psb_page_flipping_alphadrop") == 0) {
rep->exists = 1;
rep->driver_ioctl_offset = DRM_PSB_FLIP;
rep->sarea_offset = 0;
rep->major = 1;
rep->minor = 0;
rep->pl = 0;
return 0;
}
if (strcmp(arg->extension, "lnc_video_getparam") == 0) {
rep->exists = 1;
rep->driver_ioctl_offset = DRM_PSB_VIDEO_GETPARAM;
rep->sarea_offset = 0;
rep->major = 1;
rep->minor = 0;
rep->pl = 0;
return 0;
}
rep->exists = 0;
return 0;
}
static int psb_vt_leave_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct ttm_bo_device *bdev = &dev_priv->bdev;
struct ttm_mem_type_manager *man;
int clean;
int ret;
#ifdef CONFIG_MDFD_VIDEO_DECODE
ret = ttm_vt_lock(&dev_priv->ttm_lock, 1,
psb_fpriv(file_priv)->tfile);
if (unlikely(ret != 0))
return ret;
ret = ttm_bo_evict_mm(&dev_priv->bdev, TTM_PL_TT);
if (unlikely(ret != 0))
goto out_unlock;
man = &bdev->man[TTM_PL_TT];
/*spin_lock(&bdev->lru_lock);*//* lru_lock is removed from upstream TTM */
clean = drm_mm_clean((struct drm_mm *)man->priv);
/*spin_unlock(&bdev->lru_lock);*/
if (unlikely(!clean))
DRM_INFO("Warning: GATT was not clean after VT switch.\n");
ttm_bo_swapout_all(&dev_priv->bdev);
#endif
return 0;
out_unlock:
(void) ttm_vt_unlock(&dev_priv->ttm_lock);
return ret;
}
static int psb_vt_enter_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
return ttm_vt_unlock(&dev_priv->ttm_lock);
}
static int psb_sizes_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_sizes_arg *arg =
(struct drm_psb_sizes_arg *) data;
*arg = dev_priv->sizes;
return 0;
}
static int psb_fuse_reg_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
uint32_t *arg = data;
*arg = dev_priv->fuse_reg_value;
return 0;
}
static int psb_vbt_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct gct_ioctl_arg *pGCT = data;
memcpy(pGCT, &dev_priv->gct_data, sizeof(*pGCT));
return 0;
}
static int psb_flip_hdmi(struct drm_device *dev, uint32_t pipe)
{
int dspsurf;
unsigned long irqflags;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
if (dev_priv->head_fliped) {
printk(KERN_INFO "HDMI flipped already!");
return 0;
}
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, false)) {
switch (pipe) {
case 1:
dspsurf = DSPBSURF;
break;
default:
goto exit;
}
spin_lock_irqsave(&dev_priv->flip_lock, irqflags);
REG_WRITE(dspsurf, dev_priv->addr_array[dev_priv->flip_head]);
/* DRM_INFO("fliping:%d \n", dev_priv->flip_head); */
dev_priv->head_fliped = 1;
spin_unlock_irqrestore(&dev_priv->flip_lock, irqflags);
exit:
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
return 0;
}
static int psb_csc_gamma_setting_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_csc_gamma_setting *csc_gamma_setting = NULL;
int ret = 0;
csc_gamma_setting = (struct drm_psb_csc_gamma_setting *)
data;
printk("seting gamma ioctl!\n");
if (!csc_gamma_setting)
return -EINVAL;
if (csc_gamma_setting->type == GAMMA)
ret = mdfld_intel_crtc_set_gamma(dev,
&csc_gamma_setting->data.gamma_data);
else if (csc_gamma_setting->type == CSC)
ret = mdfld_intel_crtc_set_color_conversion(dev,
&csc_gamma_setting->data.csc_data);
return ret;
}
static int psb_enable_ied_session_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct mdfld_dsi_config *dsi_config = dev_priv->dsi_configs[0];
data = data;
DRM_INFO("Enabling IED session...\n");
if (file_priv == NULL) {
DRM_ERROR("%s: file_priv is NULL.\n", __func__);
return -1;
}
if (dev_priv->ied_enabled) {
DRM_ERROR("%s: ied_enabled has been set.\n", __func__);
return 0;
}
dev_priv->ied_enabled = true;
dev_priv->ied_context = file_priv->filp;
mdfld_dsi_dsr_forbid(dsi_config);
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
/* Set bit 31 to enable IED pipeline */
REG_WRITE(PSB_IED_DRM_CNTL_STATUS, 0x80000000);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return 0;
} else {
DRM_ERROR("%s: Failed to power on display island.\n", __func__);
return -1;
}
}
static int psb_disable_ied_session_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int ret = 0;
struct drm_psb_private *dev_priv = psb_priv(dev);
struct mdfld_dsi_config *dsi_config = dev_priv->dsi_configs[0];
data = data;
DRM_INFO("Disabling IED session...\n");
if (file_priv == NULL) {
DRM_ERROR("%s: file_priv is NULL.\n", __func__);
return -1;
}
if (dev_priv->ied_enabled == false) {
DRM_ERROR("%s: ied_enabled is not set.\n", __func__);
return 0;
}
if (dev_priv->ied_context != file_priv->filp) {
DRM_ERROR("%s: Wrong context.\n", __func__);
return -1;
}
dev_priv->ied_enabled = false;
dev_priv->ied_context = NULL;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
REG_WRITE(PSB_IED_DRM_CNTL_STATUS, 0);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
ret = 0;
} else {
DRM_ERROR("%s: Failed to power on display island.\n", __func__);
ret = -1;
}
mdfld_dsi_dsr_allow(dsi_config);
return ret;
}
void psb_cleanup_ied_session(struct drm_psb_private *dev_priv,
struct file *filp)
{
struct mdfld_dsi_config *dsi_config = NULL;
if (dev_priv == NULL || filp == NULL)
return;
if (dev_priv->ied_enabled && dev_priv->ied_context == filp) {
DRM_ERROR("%s: ied_enabled is not cleared.\n", __func__);
dev_priv->ied_enabled = false;
dev_priv->ied_context = NULL;
dsi_config = dev_priv->dsi_configs[0];
mdfld_dsi_dsr_allow(dsi_config);
}
}
static int psb_disp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_disp_ctrl *dp_ctrl = data;
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_flip_chain_data *flip_data = NULL;
struct drm_mode_object *obj;
struct drm_framebuffer *fb;
struct psb_framebuffer *psbfb;
unsigned long irqflags;
int ret = 0;
unsigned int i = 0;
int dspcntr_reg = DSPBCNTR;
int dspbase_reg = MRST_DSPBBASE;
u32 temp;
struct mdfld_dsi_config *dsi_config;
/*DRM_COPY_FROM_USER(&dp_ctrl, data, sizeof(struct drm_psb_disp_ctrl));*/
/*DRM_INFO("disp cmd:%d\n",dp_ctrl->cmd);*/
if (dp_ctrl->cmd == DRM_PSB_DISP_INIT_HDMI_FLIP_CHAIN) {
flip_data = &dp_ctrl->u.flip_chain_data;
if (flip_data->size > DRM_PSB_HDMI_FLIP_ARRAY_SIZE) {
ret = -EINVAL;
goto exit;
}
for (i = 0; i < flip_data->size; i++) {
dev_priv->flip_array[i] = flip_data->h_buffer_array[i];
obj = drm_mode_object_find(dev, (uint32_t)dev_priv->flip_array[i],
DRM_MODE_OBJECT_FB);
fb = obj_to_fb(obj);
psbfb = to_psb_fb(fb);
if (!psbfb) {
ret = -EINVAL;
goto exit;
}
dev_priv->addr_array[i] = psbfb->offset;
/* DRM_INFO("adding id:%d to psb flip chain \n",
(uint32_t)dev_priv->flip_array[i]); */
}
dev_priv->flip_valid_size = flip_data->size;
dev_priv->flip_inited = 1;
} else if (dp_ctrl->cmd == DRM_PSB_DISP_QUEUE_BUFFER) {
spin_lock_irqsave(&dev_priv->flip_lock, irqflags);
dev_priv->flip_head = (unsigned int)dp_ctrl->u.buf_data.h_buffer;
dev_priv->head_fliped = 0;
spin_unlock_irqrestore(&dev_priv->flip_lock, irqflags);
psb_flip_hdmi(dev, 1);
} else if (dp_ctrl->cmd == DRM_PSB_DISP_DEQUEUE_BUFFER) {
if (!dev_priv->flip_inited) {
ret = -EINVAL;
goto exit;
}
i = (dev_priv->flip_tail + 1) % dev_priv->flip_valid_size;
if (i != dev_priv->flip_head)
dev_priv->flip_tail = i;
dp_ctrl->u.buf_data.h_buffer = (void *)dev_priv->flip_tail;
} else if (dp_ctrl->cmd == DRM_PSB_DISP_PLANEB_DISABLE) {
dev_priv->bhdmi_enable = false;
if (DISP_PLANEB_STATUS == DISPLAY_PLANE_DISABLE)
ret = -1;
else {
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
/* If data=1, then force setting plane status. */
if (dp_ctrl->u.data == 1)
DISP_PLANEB_STATUS = DISPLAY_PLANE_DISABLE;
ret = -1;
goto exit;
}
/* Use Disable pipeB plane to turn off HDMI screen */
temp = REG_READ(dspcntr_reg);
if ((temp & DISPLAY_PLANE_ENABLE) != 0) {
REG_WRITE(dspcntr_reg,
temp & ~DISPLAY_PLANE_ENABLE);
/* Flush the plane changes */
REG_WRITE(dspbase_reg, REG_READ(dspbase_reg));
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
/* If data=1, then force setting plane status. */
if (dp_ctrl->u.data == 1)
DISP_PLANEB_STATUS = DISPLAY_PLANE_DISABLE;
} else if (dp_ctrl->cmd == DRM_PSB_DISP_PLANEB_ENABLE) {
/* If data=1, then force setting plane status. */
if (dp_ctrl->u.data == 1)
DISP_PLANEB_STATUS = DISPLAY_PLANE_ENABLE;
if (DISP_PLANEB_STATUS == DISPLAY_PLANE_DISABLE)
ret = -1;
else {
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
ret = -1;
goto exit;
}
/*Restore pipe B plane to turn on HDMI screen*/
temp = REG_READ(dspcntr_reg);
if ((temp & DISPLAY_PLANE_ENABLE) == 0) {
REG_WRITE(dspcntr_reg,
temp | DISPLAY_PLANE_ENABLE);
/* Flush the plane changes */
REG_WRITE(dspbase_reg, REG_READ(dspbase_reg));
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
dev_priv->bhdmi_enable = true;
} else if (dp_ctrl->cmd == DRM_PSB_HDMI_OSPM_ISLAND_DOWN) {
/* before turning off HDMI power island, re-check the
*HDMI hotplus status in case that there are plug-in
*signals agains.
*/
if (!hdmi_state) {
/* We need forbid DSR here to let DSR has a new
** chance to determine the hdmi status
** Or when HDMI unplug, the DSR has sometimes
** remained at Level 0
*/
dsi_config = dev_priv->dsi_configs[0];
mdfld_dsi_dsr_forbid(dsi_config);
/*Set power island down when hdmi disconnected*/
acquire_ospm_lock();
/*HDMI is considered totally disconected
*before power off its island */
dev_priv->bhdmiconnected = false;
if (pmu_nc_set_power_state(OSPM_DISPLAY_B_ISLAND,
OSPM_ISLAND_DOWN, OSPM_REG_TYPE))
BUG();
dev_priv->panel_desc &= ~DISPLAY_B;
DISP_PLANEB_STATUS = ~DISPLAY_PLANE_ENABLE;
release_ospm_lock();
mdfld_dsi_dsr_allow(dsi_config);
}
} else if (dp_ctrl->cmd == DRM_PSB_HDMI_NOTIFY_HOTPLUG_TO_AUDIO) {
if (dp_ctrl->u.data == 0) {
/* notify audio with HDMI unplug event */
if (dev_priv->hdmi_priv->monitor_type == MONITOR_TYPE_HDMI) {
DRM_INFO("HDMI plug out to audio driver\n");
mid_hdmi_audio_signal_event(dev, HAD_EVENT_HOT_UNPLUG);
}
} else {
/* notify audio with HDMI plug event */
if (dev_priv->hdmi_priv->monitor_type == MONITOR_TYPE_HDMI) {
DRM_INFO("HDMI plug in to audio driver\n");
mid_hdmi_audio_signal_event(dev, HAD_EVENT_HOT_PLUG);
}
}
}
exit:
return ret;
}
#ifdef CONFIG_SUPPORT_HDMI
static int psb_query_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t *arg = data;
uint8_t bksv[5];
/* Attempting to read the BKSV value from the HDMI device.
* A successful read of this data indicates that HDCP is
* supported. A value of zero would indicate that it's
* not.
*/
*arg = android_query_hdmi_hdcp_sink(dev, bksv);
return 0;
}
static int psb_validate_hdcp_ksv_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
sqword_t *arg = data;
sqword_t hw_bksv;
if (android_query_hdmi_hdcp_sink(dev, (uint8_t *)&hw_bksv)) {
*arg = hw_bksv;
return 0;
}
return -1;
}
static int psb_get_hdcp_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t *arg = data;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
*arg = android_check_hdmi_hdcp_enc_status(dev);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
return 0;
}
static int psb_enable_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int ret = 0;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
ret = android_enable_hdmi_hdcp(dev);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
if (ret)
return 0;
return -1;
}
static int psb_disable_hdcp_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int ret = 0;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
ret = android_disable_hdmi_hdcp(dev);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
if (ret)
return 0;
return -1;
}
static int psb_enable_display_ied_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int ret = 0;
int temp = 0;
struct drm_psb_private *dev_priv = psb_priv(dev);
if (HAS_DISPLAY_IED_CNTRL(dev)) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
temp = PSB_RVDC32(DSPCHICKENBIT);
temp &= ~(1 << 31);
PSB_WVDC32(temp, DSPCHICKENBIT);
temp = PSB_RVDC32(DSPCHICKENBIT);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else
ret = -1;
}
return ret;
}
static int psb_disable_display_ied_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
int ret = 0;
int temp = 0;
struct drm_psb_private *dev_priv = psb_priv(dev);
if (HAS_DISPLAY_IED_CNTRL(dev)) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
temp = PSB_RVDC32(DSPCHICKENBIT);
temp |= (1 << 31);
PSB_WVDC32(temp, DSPCHICKENBIT);
temp = PSB_RVDC32(DSPCHICKENBIT);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else
ret = -1;
}
return ret;
}
static int psb_query_display_ied_caps_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t *arg = data;
if (HAS_DISPLAY_IED_CNTRL(dev))
*arg = 1;
else
*arg = 0;
return 0;
}
static int psb_get_hdcp_link_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t *arg = data;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
*arg = android_check_hdmi_hdcp_link_status(dev);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
return 0;
}
#endif
static int psb_set_csc_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_csc_matrix *csc_matrix = data;
csc_program_DC(dev, csc_matrix->matrix, csc_matrix->pipe);
return 0;
}
static int psb_dc_state_ioctl(struct drm_device *dev, void * data,
struct drm_file *file_priv)
{
uint32_t flags;
uint32_t obj_id;
struct drm_mode_object *obj;
struct drm_connector *connector;
struct drm_crtc *crtc;
struct drm_psb_dc_state_arg *arg =
(struct drm_psb_dc_state_arg *)data;
if (IS_MID(dev))
return 0;
flags = arg->flags;
obj_id = arg->obj_id;
if (flags & PSB_DC_CRTC_MASK) {
obj = drm_mode_object_find(dev, obj_id,
DRM_MODE_OBJECT_CRTC);
if (!obj) {
DRM_DEBUG("Invalid CRTC object.\n");
return -EINVAL;
}
crtc = obj_to_crtc(obj);
mutex_lock(&dev->mode_config.mutex);
if (drm_helper_crtc_in_use(crtc)) {
if (flags & PSB_DC_CRTC_SAVE)
crtc->funcs->save(crtc);
else
crtc->funcs->restore(crtc);
}
mutex_unlock(&dev->mode_config.mutex);
return 0;
} else if (flags & PSB_DC_OUTPUT_MASK) {
obj = drm_mode_object_find(dev, obj_id,
DRM_MODE_OBJECT_CONNECTOR);
if (!obj) {
DRM_DEBUG("Invalid connector id.\n");
return -EINVAL;
}
connector = obj_to_connector(obj);
if (flags & PSB_DC_OUTPUT_SAVE)
connector->funcs->save(connector);
else
connector->funcs->restore(connector);
return 0;
}
DRM_DEBUG("Bad flags 0x%x\n", flags);
return -EINVAL;
}
#ifdef CONFIG_CTP_DPST
static int psb_dpst_bl_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
uint32_t *arg = data;
struct backlight_device bd;
dev_priv->blc_adj2 = *arg;
#ifdef CONFIG_BACKLIGHT_CLASS_DEVICE
bd.props.brightness = psb_get_brightness(&bd);
psb_set_brightness(&bd);
#endif
return 0;
}
#endif
static int psb_adb_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
uint32_t *arg = data;
struct backlight_device bd;
dev_priv->blc_adj1 = *arg;
#ifdef CONFIG_BACKLIGHT_CLASS_DEVICE
bd.props.brightness = psb_get_brightness(&bd);
psb_set_brightness(&bd);
#endif
return 0;
}
#ifdef CONFIG_CTP_DPST
static int psb_hist_enable_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
u32 irqCtrl = 0;
struct drm_psb_private *dev_priv = psb_priv(dev);
struct dpst_guardband guardband_reg;
struct dpst_ie_histogram_control ie_hist_cont_reg;
uint32_t *enable = data;
unsigned long irq_flags;
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, OSPM_UHB_ONLY_IF_ON)) {
return 0;
}
if (*enable == 1) {
ie_hist_cont_reg.data = PSB_RVDC32(HISTOGRAM_LOGIC_CONTROL);
ie_hist_cont_reg.ie_pipe_assignment = 0;
ie_hist_cont_reg.histogram_mode_select = DPST_YUV_LUMA_MODE;
ie_hist_cont_reg.ie_histogram_enable = 1;
PSB_WVDC32(ie_hist_cont_reg.data, HISTOGRAM_LOGIC_CONTROL);
guardband_reg.data = PSB_RVDC32(HISTOGRAM_INT_CONTROL);
guardband_reg.interrupt_enable = 1;
guardband_reg.interrupt_status = 1;
PSB_WVDC32(guardband_reg.data, HISTOGRAM_INT_CONTROL);
spin_lock_irqsave(&dev_priv->irqmask_lock, irq_flags);
irqCtrl = PSB_RVDC32(PIPEASTAT);
PSB_WVDC32(irqCtrl | PIPE_DPST_EVENT_ENABLE, PIPEASTAT);
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irq_flags);
/* Wait for two vblanks */
} else {
guardband_reg.data = PSB_RVDC32(HISTOGRAM_INT_CONTROL);
guardband_reg.interrupt_enable = 0;
guardband_reg.interrupt_status = 1;
PSB_WVDC32(guardband_reg.data, HISTOGRAM_INT_CONTROL);
ie_hist_cont_reg.data = PSB_RVDC32(HISTOGRAM_LOGIC_CONTROL);
ie_hist_cont_reg.ie_histogram_enable = 0;
PSB_WVDC32(ie_hist_cont_reg.data, HISTOGRAM_LOGIC_CONTROL);
spin_lock_irqsave(&dev_priv->irqmask_lock, irq_flags);
irqCtrl = PSB_RVDC32(PIPEASTAT);
irqCtrl &= ~PIPE_DPST_EVENT_ENABLE;
PSB_WVDC32(irqCtrl, PIPEASTAT);
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irq_flags);
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return 0;
}
static int psb_hist_status_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_hist_status_arg *hist_status = data;
uint32_t *arg = hist_status->buf;
u32 iedbr_reg_data = 0;
struct dpst_ie_histogram_control ie_hist_cont_reg;
u32 i;
int dpst3_bin_threshold_count = 0;
uint32_t blm_hist_ctl = HISTOGRAM_LOGIC_CONTROL;
uint32_t iebdr_reg = HISTOGRAM_BIN_DATA;
uint32_t segvalue_max_22_bit = 0x3fffff;
uint32_t iedbr_busy_bit = 0x80000000;
int dpst3_bin_count = 32;
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, OSPM_UHB_ONLY_IF_ON)) {
return 0;
}
ie_hist_cont_reg.data = PSB_RVDC32(blm_hist_ctl);
ie_hist_cont_reg.bin_reg_func_select = dpst3_bin_threshold_count;
ie_hist_cont_reg.bin_reg_index = 0;
PSB_WVDC32(ie_hist_cont_reg.data, blm_hist_ctl);
for (i = 0; i < dpst3_bin_count; i++) {
iedbr_reg_data = PSB_RVDC32(iebdr_reg);
if (!(iedbr_reg_data & iedbr_busy_bit)) {
arg[i] = iedbr_reg_data & segvalue_max_22_bit;
} else {
i = 0;
ie_hist_cont_reg.data = PSB_RVDC32(blm_hist_ctl);
ie_hist_cont_reg.bin_reg_index = 0;
PSB_WVDC32(ie_hist_cont_reg.data, blm_hist_ctl);
}
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return 0;
}
static int psb_init_comm_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct pci_dev *pdev = NULL;
struct device *ddev = NULL;
struct kobject *kobj = NULL;
uint32_t *arg = data;
if (*arg == 1) {
/*find handle to drm kboject*/
pdev = dev->pdev;
ddev = &pdev->dev;
kobj = &ddev->kobj;
if (dev_priv->psb_dpst_state == NULL) {
/*init dpst kmum comms*/
dev_priv->psb_dpst_state = psb_dpst_init(kobj);
} else {
PSB_DEBUG_ENTRY("DPST already initialized\n");
}
psb_irq_enable_dpst(dev);
psb_dpst_notify_change_um(DPST_EVENT_INIT_COMPLETE,
dev_priv->psb_dpst_state);
} else {
/*hotplug and dpst destroy examples*/
psb_irq_disable_dpst(dev);
psb_dpst_notify_change_um(DPST_EVENT_TERMINATE,
dev_priv->psb_dpst_state);
psb_dpst_device_pool_destroy(dev_priv->psb_dpst_state);
dev_priv->psb_dpst_state = NULL;
}
return 0;
}
/* return the current mode to the dpst module */
static int psb_dpst_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
uint32_t *arg = data;
uint32_t x;
uint32_t y;
uint32_t reg;
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, OSPM_UHB_ONLY_IF_ON)) {
return 0;
}
reg = PSB_RVDC32(PIPEASRC);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
/* horizontal is the left 16 bits */
x = reg >> 16;
/* vertical is the right 16 bits */
y = reg & 0x0000ffff;
/* the values are the image size minus one */
x += 1;
y += 1;
*arg = (x << 16) | y;
return 0;
}
static int psb_gamma_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_dpst_lut_arg *lut_arg = data;
struct drm_mode_object *obj;
struct drm_crtc *crtc;
struct drm_connector *connector;
struct psb_intel_crtc *psb_intel_crtc;
int i = 0;
int32_t obj_id;
obj_id = lut_arg->output_id;
obj = drm_mode_object_find(dev, obj_id, DRM_MODE_OBJECT_CONNECTOR);
if (!obj) {
DRM_DEBUG("Invalid Connector object.\n");
return -EINVAL;
}
connector = obj_to_connector(obj);
crtc = connector->encoder->crtc;
psb_intel_crtc = to_psb_intel_crtc(crtc);
for (i = 0; i < 256; i++)
psb_intel_crtc->lut_adj[i] = lut_arg->lut[i];
psb_intel_crtc_load_lut(crtc);
return 0;
}
static int psb_update_guard_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct dpst_guardband *input = (struct dpst_guardband *) data;
struct dpst_guardband reg_data;
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, OSPM_UHB_ONLY_IF_ON)) {
return 0;
}
reg_data.data = PSB_RVDC32(HISTOGRAM_INT_CONTROL);
reg_data.guardband = input->guardband;
reg_data.guardband_interrupt_delay = input->guardband_interrupt_delay;
/* PSB_DEBUG_ENTRY("guardband = %u\ninterrupt delay = %u\n",
reg_data.guardband, reg_data.guardband_interrupt_delay); */
PSB_WVDC32(reg_data.data, HISTOGRAM_INT_CONTROL);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return 0;
}
#endif
static int psb_mode_operation_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t obj_id;
uint16_t op;
struct drm_mode_modeinfo *umode;
struct drm_display_mode *mode = NULL;
struct drm_psb_mode_operation_arg *arg;
struct drm_mode_object *obj;
struct drm_connector *connector;
struct drm_framebuffer *drm_fb;
struct psb_framebuffer *psb_fb;
struct drm_connector_helper_funcs *connector_funcs;
int ret = 0;
int resp = MODE_OK;
struct drm_psb_private *dev_priv = psb_priv(dev);
arg = (struct drm_psb_mode_operation_arg *)data;
obj_id = arg->obj_id;
op = arg->operation;
switch (op) {
case PSB_MODE_OPERATION_SET_DC_BASE:
obj = drm_mode_object_find(dev, obj_id, DRM_MODE_OBJECT_FB);
if (!obj) {
DRM_ERROR("Invalid FB id %d\n", obj_id);
return -EINVAL;
}
drm_fb = obj_to_fb(obj);
psb_fb = to_psb_fb(drm_fb);
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_ONLY_IF_ON)) {
REG_WRITE(DSPASURF, psb_fb->offset);
REG_READ(DSPASURF);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else {
dev_priv->saveDSPASURF = psb_fb->offset;
}
return 0;
case PSB_MODE_OPERATION_MODE_VALID:
umode = &arg->mode;
mutex_lock(&dev->mode_config.mutex);
obj = drm_mode_object_find(dev, obj_id, DRM_MODE_OBJECT_CONNECTOR);
if (!obj) {
ret = -EINVAL;
goto mode_op_out;
}
connector = obj_to_connector(obj);
mode = drm_mode_create(dev);
if (!mode) {
ret = -ENOMEM;
goto mode_op_out;
}
/* drm_crtc_convert_umode(mode, umode); */
{
mode->clock = umode->clock;
mode->hdisplay = umode->hdisplay;
mode->hsync_start = umode->hsync_start;
mode->hsync_end = umode->hsync_end;
mode->htotal = umode->htotal;
mode->hskew = umode->hskew;
mode->vdisplay = umode->vdisplay;
mode->vsync_start = umode->vsync_start;
mode->vsync_end = umode->vsync_end;
mode->vtotal = umode->vtotal;
mode->vscan = umode->vscan;
mode->vrefresh = umode->vrefresh;
mode->flags = umode->flags;
mode->type = umode->type;
strncpy(mode->name, umode->name, DRM_DISPLAY_MODE_LEN);
mode->name[DRM_DISPLAY_MODE_LEN-1] = 0;
}
connector_funcs = (struct drm_connector_helper_funcs *)
connector->helper_private;
if (connector_funcs->mode_valid) {
resp = connector_funcs->mode_valid(connector, mode);
arg->data = (void *)resp;
}
/*do some clean up work*/
if (mode) {
drm_mode_destroy(dev, mode);
}
mode_op_out:
mutex_unlock(&dev->mode_config.mutex);
return ret;
default:
DRM_DEBUG("Unsupported psb mode operation");
return -EOPNOTSUPP;
}
return 0;
}
static int psb_stolen_memory_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_stolen_memory_arg *arg = data;
arg->base = dev_priv->pg->stolen_base;
arg->size = dev_priv->pg->vram_stolen_size;
return 0;
}
static int psb_dpu_query_ioctl(struct drm_device *dev, void *arg,
struct drm_file *file_priv)
{
IMG_INT *data = (IMG_INT *)arg;
DRM_DRIVER_PRIVATE_T *dev_priv = dev->dev_private;
int panel_type;
/*reject requests from non-mdfld platforms*/
if (!IS_MDFLD(dev)) {
DRM_INFO("Not Medfield platform! return.");
return -EOPNOTSUPP;
}
DRM_INFO("dsr query.\n");
dev_priv->um_start = true;
panel_type = get_panel_mode(dev);
if (panel_type == MDFLD_DSI_ENCODER_DPI) {
DRM_INFO("DSI panel is working in video mode\n");
dev_priv->b_dsr_enable = false;
*data = 0;
return 0;
}
#if defined(CONFIG_MDFLD_DSI_DSR)
dev_priv->b_dsr_enable = true;
*data = MDFLD_DSR_RR | MDFLD_DSR_FULLSCREEN;
#elif defined(CONFIG_MDFLD_DSI_DPU)
dev_priv->b_dsr_enable = true;
*data = MDFLD_DSR_RR | MDFLD_DPU_ENABLE;
#else /*DBI panel but DSR was not defined*/
DRM_INFO("DSR is disabled by kernel configuration.\n");
dev_priv->b_dsr_enable = false;
*data = 0;
#endif /*CONFIG_MDFLD_DSI_DSR*/
return 0;
}
static int psb_dpu_dsr_on_ioctl(struct drm_device *dev, void *arg,
struct drm_file *file_priv)
{
u32 * param = (u32 *)arg;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *)dev->dev_private;
int panel_type;
/*reject requests from non-mdfld platforms*/
if (!IS_MDFLD(dev)) {
DRM_INFO("Not Medfield platform! return.");
return -EOPNOTSUPP;
}
panel_type = get_panel_mode(dev);
if (panel_type == MDFLD_DSI_ENCODER_DPI) {
DRM_INFO("DSI panel is working in video mode\n");
dev_priv->b_dsr_enable = false;
return 0;
}
if (!param) {
DRM_ERROR("Invalid parameter\n");
return -EINVAL;
}
PSB_DEBUG_ENTRY("dsr kick in. param 0x%08x\n", *param);
if (*param == DRM_PSB_DSR_DISABLE) {
PSB_DEBUG_ENTRY("DSR is turned off\n");
dev_priv->b_dsr_enable = false;
#if defined(CONFIG_MDFLD_DSI_DPU)
mdfld_dbi_dpu_report_fullscreen_damage(dev);
#elif defined(CONFIG_MDFLD_DSI_DSR)
mdfld_dsi_dbi_exit_dsr(dev, MDFLD_DSR_2D_3D, 0, 0);
#endif
return 0;
} else if (*param == DRM_PSB_DSR_ENABLE) {
PSB_DEBUG_ENTRY("DSR is turned on\n");
#if defined(CONFIG_MDFLD_DSI_DPU) || defined(CONFIG_MDFLD_DSI_DSR)
dev_priv->b_dsr_enable = true;
#endif
return 0;
}
return -EINVAL;
}
static int psb_dpu_dsr_off_ioctl(struct drm_device *dev, void *arg,
struct drm_file *file_priv)
{
#if defined(CONFIG_MDFLD_DSI_DPU)
struct drm_psb_drv_dsr_off_arg *dsr_off_arg = (struct drm_psb_drv_dsr_off_arg *) arg;
struct psb_drm_dpu_rect rect = dsr_off_arg->damage_rect;
return mdfld_dsi_dbi_dsr_off(dev, &rect);
#elif defined(CONFIG_MDFLD_DSI_DSR)
struct drm_psb_private *dev_priv =
(struct drm_psb_private *)dev->dev_private;
if ((dev_priv->dsr_fb_update & MDFLD_DSR_2D_3D) != MDFLD_DSR_2D_3D)
mdfld_dsi_dbi_exit_dsr(dev, MDFLD_DSR_2D_3D, 0, 0);
#endif
return 0;
}
static void wait_for_pipeb_finish(struct drm_device *dev,
int pipenum)
{
static int prev_pipe;
int tmp, i;
if (prev_pipe == 1 && pipenum == 0) {
/* switch from Pipe B to Pipe A */
for (i = 0; i < 1000; i++) {
tmp = REG_READ(PIPEBCONF);
if ((tmp >> 30) != 0x01)
break;
/* Pipe is not fully disabled */
usleep_range(100, 200);
}
if (i == 1000)
DRM_ERROR("Fail to wait pipe B\n");
}
prev_pipe = pipenum;
}
/*wait for ovadd flip complete*/
static void overlay_wait_flip(struct drm_device *dev)
{
int retry = 60;
struct drm_psb_private *dev_priv = psb_priv(dev);
/**
* make sure overlay command buffer
* was copied before updating the system
* overlay command buffer.
* sleep long time first to avoid frequent wake up.
*/
if (BIT31 & PSB_RVDC32(OV_DOVASTA))
goto fliped;
usleep_range(6000, 12000);
while (--retry) {
if (BIT31 & PSB_RVDC32(OV_DOVASTA))
break;
usleep_range(500, 600);
}
fliped:
if (!retry)
DRM_DEBUG("OVADD flip timeout!\n");
}
/*wait for vblank*/
static void overlay_wait_vblank(struct drm_device *dev,
struct drm_file *file_priv,
uint32_t ovadd)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
union drm_wait_vblank vblwait;
uint32_t ovadd_pipe;
int pipe = 0;
ovadd_pipe = ((ovadd >> 6) & 0x3);
vblwait.request.type = (_DRM_VBLANK_RELATIVE | _DRM_VBLANK_NEXTONMISS);
vblwait.request.sequence = 1;
if (ovadd_pipe) {
pipe = 1;
vblwait.request.type |= _DRM_VBLANK_SECONDARY;
}
/*
* FIXME: don't enable vblank in this way.
* Current vblank usages didn't follow the DRM framework.
* drm_vblank_get()/drm_vblank_put() should be used to enable/disabe
* vblank interrupt. However, currently driver is using following way
* to enable vblank which may lead to some problems.
* Plus, drm_vblank_get() was called by PVR 3rd party display driver
* when creating a new swapchain, and drm_vblank_put() won't be called
* util destroy swapchain. this will make drm_vblank_get() in
* drm_wait_vblank useless since the refcount is not 0.
*/
if (!psb_enable_vblank(dev, pipe)) {
dev_priv->b_is_in_idle = false;
dev_priv->dsr_idle_count = 0;
DRM_DEBUG("%s: start drm_wait_vblank()\n", __func__);
drm_wait_vblank(dev, (void *)&vblwait, file_priv);
} else {
DRM_DEBUG("%s: psb_enable_vblank() failed\n", __func__);
}
}
static int validate_overlay_register_buffer(struct drm_file *file_priv,
uint32_t *OVADD, uint32_t buffer_handle)
{
#ifdef CONFIG_MDFD_VIDEO_DECODE
struct ttm_buffer_object *reg_buffer = NULL;
struct ttm_object_file *tfile = psb_fpriv(file_priv)->tfile;
struct ttm_placement placement;
uint32_t flags = TTM_PL_FLAG_TT | TTM_PL_FLAG_WC | TTM_PL_FLAG_UNCACHED;
int ret = -EINVAL;
reg_buffer = ttm_buffer_object_lookup(tfile, buffer_handle);
if (!reg_buffer)
goto out_err0;
placement.num_placement = 1;
placement.placement = &flags;
placement.num_busy_placement = 1;
placement.busy_placement = &flags;
placement.fpfn = 0;
placement.lpfn = 0;
ret = ttm_bo_reserve(reg_buffer, true, false, false, 0);
if (ret)
goto out_err1;
#if LINUX_VERSION_CODE < KERNEL_VERSION(3,8,0)
ret = ttm_bo_validate(reg_buffer, &placement, 1, 0, 0);
#else
ret = ttm_bo_validate(reg_buffer, &placement, 1, 0);
#endif
if (ret)
goto out_err2;
if ((reg_buffer->offset & 0x0fff0000) != (*OVADD & 0xffff0000)) {
DRM_DEBUG("%s: old value 0x%08x, buffer gpu address 0x%08x\n",
__func__, *OVADD, (unsigned int)reg_buffer->offset);
*OVADD = (*OVADD & 0xffff) + (reg_buffer->offset & 0x0fffffff);
DRM_DEBUG("patch ovadd value, new value 0x%08x\n", *OVADD);
}
out_err2:
ttm_bo_unreserve(reg_buffer);
out_err1:
ttm_bo_unref(&reg_buffer);
out_err0:
return ret;
#else
return 0;
#endif
}
/*
* use to dump display registers. and print to standard output.
*/
static int psb_register_dump(struct drm_device *dev, int start, int end)
{
int len = 0;
int Offset = 0;
int ret = 0;
PSB_DEBUG_ENTRY("start:0x%08x\n", start);
PSB_DEBUG_ENTRY("end: 0x%08x\n", end);
if ((start % 0x4) != 0) {
PSB_DEBUG_ENTRY("The address should be 4 byte aligned.\n");
ret = -EINVAL;
return ret;
}
if ((end % 0x4) != 0) {
PSB_DEBUG_ENTRY("The address should be 4 byte aligned.\n");
ret = -EINVAL;
return ret;
}
len = end - start + 1;
if (len <= 0)
PSB_DEBUG_ENTRY("The end should be greater than start.\n");
if (end < 0xa000 || end > 0x720ff) {
PSB_DEBUG_ENTRY("The end address is out of range.\n");
ret = -EINVAL;
return ret;
}
if (start < 0xa000 || start > 0x720ff) {
PSB_DEBUG_ENTRY("The start address is out of the range.\n");
ret = -EINVAL;
return ret;
}
for (Offset = start ; Offset < end; Offset = Offset + 0x10) {
printk(KERN_INFO
"[DISPLAY DUMP] 0x%08x: 0x%08x 0x%08x 0x%08x 0x%08x\n",
Offset,
REG_READ(Offset + 0x0),
REG_READ(Offset + 0x4),
REG_READ(Offset + 0x8),
REG_READ(Offset + 0xc));
}
return ret;
}
static int psb_display_reg_dump(struct drm_device *dev)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct mdfld_dsi_config *dsi_config;
dsi_config = dev_priv->dsi_configs[0];
mdfld_dsi_dsr_forbid(dsi_config);
/* DSI PLL */
printk(KERN_INFO "[DISPLAY REG DUMP] DSI PLL REG\n\n");
psb_register_dump(dev, 0xf010, 0xf020);
printk(KERN_INFO "\n");
/* MIPI A REGISTER */
printk(KERN_INFO "[DISPLAY REG DUMP] MIPI A\n\n");
psb_register_dump(dev, 0xb000, 0xb100);
printk(KERN_INFO "\n");
/* PIPE A */
printk(KERN_INFO "[DISPLAY REG DUMP] PIPE A\n\n");
psb_register_dump(dev, 0x60000, 0x60100);
printk(KERN_INFO "\n");
/* Plane A */
printk(KERN_INFO "[DISPLAY REG DUMP] PLANE A\n\n");
psb_register_dump(dev, 0x70000, 0x700FC);
psb_register_dump(dev, 0x70180, 0x701F4);
psb_register_dump(dev, 0x70400, 0x7044C);
psb_register_dump(dev, 0x70500, 0x70504);
printk(KERN_INFO "\n");
if (dev_priv->bhdmiconnected) {
/* PIPE B */
printk(KERN_INFO "[DISPLAY REG DUMP] PIPE B\n\n");
psb_register_dump(dev, 0x61000, 0x61100);
printk(KERN_INFO "\n");
/* Plane B */
printk(KERN_INFO "[DISPLAY REG DUMP] PLANE B\n\n");
psb_register_dump(dev, 0x71000, 0x710FC);
psb_register_dump(dev, 0x71180, 0x711F4);
psb_register_dump(dev, 0x71400, 0x7144C);
printk(KERN_INFO "\n");
}
/* OVERLAY */
printk(KERN_INFO "[DISPLAY REG DUMP] OVERLAY A\n\n");
psb_register_dump(dev, 0x30000, 0x30060);
psb_register_dump(dev, 0x30100, 0x301A4);
psb_register_dump(dev, 0x32000, 0x3201C);
psb_register_dump(dev, 0x33000, 0x33024);
printk(KERN_INFO "\n");
mdfld_dsi_dsr_allow(dsi_config);
return 0;
}
void psb_flip_abnormal_debug_info(struct drm_device *dev)
{
struct drm_psb_private *dev_priv = NULL;
struct mdfld_dsi_config *dsi_config = NULL;
int pipe = 0;
unsigned long long interval = 0;
unsigned long long second = 0;
unsigned long nanosec_rem;
if (!dev) {
DRM_INFO("%s dev is NUL\n", __func__);
return;
}
dev_priv =
(struct drm_psb_private *) dev->dev_private;
if (!dev_priv) {
DRM_INFO("%s dev_priv is NUL\n", __func__);
return;
}
dsi_config = dev_priv->dsi_configs[0];
if (!dsi_config) {
DRM_INFO("%s dsi_config is NUL\n", __func__);
return;
}
DRM_INFO("\n1.level1 interrupt status\n");
DRM_INFO("PSB_INT_MASK_R mask 0x%x\n", PSB_RVDC32(PSB_INT_MASK_R));
DRM_INFO("PSB_INT_ENABLE_R mask 0x%x\n", PSB_RVDC32(PSB_INT_ENABLE_R));
DRM_INFO("dev_priv->vdc_irq_mask = 0x%x\n\n", dev_priv->vdc_irq_mask);
DRM_INFO("2.level2 interrupt register\n");
DRM_INFO("pipe 0 config 0x%x status 0x%x\n",
REG_READ(0x70008), REG_READ(0x70024));
DRM_INFO("pipe 1 config 0x%x status 0x%x\n\n",
REG_READ(0x71008), REG_READ(0x71024));
DRM_INFO("3.check irq and workqueue relationship\n");
second = dev_priv->vsync_te_trouble_ts;
nanosec_rem = do_div(second, 1000000000);
DRM_INFO("vsync_te trouble: [%5lu.%06lu]\n",
(unsigned long) second,
nanosec_rem / 1000);
for (pipe = 0; pipe < PSB_NUM_PIPE; pipe++) {
if (pipe == 2)
continue;
second = dev_priv->vsync_te_irq_ts[pipe];
nanosec_rem = do_div(second, 1000000000);
DRM_INFO("pipe %d last vsync_te irq: [%5lu.%06lu]\n",
pipe, (unsigned long) second,
nanosec_rem / 1000);
second = dev_priv->vsync_te_worker_ts[pipe];
nanosec_rem = do_div(second, 1000000000);
DRM_INFO("pipe %d last vsync_te workqueue : [%5lu.%06lu]\n",
pipe, (unsigned long) second,
nanosec_rem / 1000);
if (dev_priv->vsync_te_irq_ts[pipe] <
dev_priv->vsync_te_worker_ts[pipe]) {
/*workqueue delay*/
interval = dev_priv->vsync_te_worker_ts[pipe] -
dev_priv->vsync_te_irq_ts[pipe];
nanosec_rem = do_div(interval, 1000000000);
DRM_INFO("pipe %d workqueue be delayed : [%5lu.%06lu]\n",
pipe, (unsigned long) interval,
nanosec_rem / 1000);
} else {
/*workqueue block*/
interval = cpu_clock(0) -
dev_priv->vsync_te_irq_ts[pipe];
nanosec_rem = do_div(interval, 1000000000);
DRM_INFO("pipe %d workqueue be blocked : [%5lu.%06lu]\n\n",
pipe, (unsigned long) interval,
nanosec_rem / 1000);
}
/*check whether real vsync te missing*/
interval = cpu_clock(0) -
dev_priv->vsync_te_irq_ts[pipe];
nanosec_rem = do_div(interval, 1000000000);
if (interval > 0 || nanosec_rem > 200000000) {
DRM_INFO("pipe %d vsync te missing %lldms !\n\n",
pipe, interval * 1000 + nanosec_rem/1000000);
dev_priv->vsync_te_working[pipe] = false;
if (pipe == 0)
atomic_set(&dev_priv->mipi_flip_abnormal, 1);
}
}
}
static int psb_get_dc_info_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_dc_info *dc = data;
if (IS_MDFLD_OLD(dev)) {
dc->pipe_count = INTEL_MDFLD_DISPLAY_PIPE_NUM;
dc->primary_plane_count = INTEL_MDFLD_SPRITE_PLANE_NUM;
dc->sprite_plane_count = 0;
dc->overlay_plane_count = INTEL_MDFLD_OVERLAY_PLANE_NUM;
dc->cursor_plane_count = INTEL_MDFLD_CURSOR_PLANE_NUM;
} else if (IS_CTP(dev)) {
dc->pipe_count = INTEL_CTP_DISPLAY_PIPE_NUM;
dc->primary_plane_count = INTEL_CTP_SPRITE_PLANE_NUM;
dc->sprite_plane_count = 0;
dc->overlay_plane_count = INTEL_CTP_OVERLAY_PLANE_NUM;
dc->cursor_plane_count = INTEL_CTP_CURSOR_PLANE_NUM;
} else {
DRM_INFO("unsupported platform in mrst driver now!");
return -1;
}
return 0;
}
//ASUS_BSP: [DDS] +++
#ifdef CONFIG_SUPPORT_DDS_MIPI_SWITCH
static int psb_panel_switch_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
uint32_t *connected = data;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
DRM_INFO("[DISPLAY] [DDS] %s: Enter, connected =%d\n", __func__, *connected);
//if(*connected){
// panel_id = 1; //800x1280
//}
//else{
// panel_id = 0; //480x800
//}
//printk("psb_panel_switch_ioctl : panel_id = %d\n",panel_id);
mdfld_reset_dpi_panel(dev_priv, *connected);
return 0;
}
#endif
//ASUS_BSP: [DDS] ---
static int psb_vsync_set_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_vsync_set_arg *arg = data;
struct mdfld_dsi_config *dsi_config;
unsigned long irq_flags;
struct timespec now;
uint32_t vsync_enable = 0;
uint32_t pipe;
u32 vbl_count = 0;
s64 nsecs = 0;
int ret = 0;
struct psb_fpriv *psb_fp = NULL;
mutex_lock(&dev_priv->vsync_lock);
if (arg->vsync_operation_mask) {
pipe = arg->vsync.pipe;
if (arg->vsync_operation_mask & GET_VSYNC_COUNT) {
vbl_count = intel_vblank_count(dev, pipe);
getrawmonotonic(&now);
nsecs = timespec_to_ns(&now);
arg->vsync.timestamp = (uint64_t)nsecs;
arg->vsync.vsync_count = (uint64_t)vbl_count;
}
if (arg->vsync_operation_mask & VSYNC_WAIT) {
vbl_count = intel_vblank_count(dev, pipe);
spin_lock_irqsave(&dev_priv->irqmask_lock, irq_flags);
vsync_enable = dev_priv->pipestat[pipe] &
(PIPE_TE_ENABLE | PIPE_VBLANK_INTERRUPT_ENABLE);
spin_unlock_irqrestore(&dev_priv->irqmask_lock,
irq_flags);
mutex_unlock(&dev_priv->vsync_lock);
if (vsync_enable) {
ret = wait_event_interruptible_timeout(
dev_priv->vsync_queue,
(intel_vblank_count(dev, pipe) !=
vbl_count),
3 * DRM_HZ);
if (!ret)
DRM_ERROR("Pipe %d vsync time out\n",
pipe);
}
getrawmonotonic(&now);
nsecs = timespec_to_ns(&now);
arg->vsync.timestamp = (uint64_t)nsecs;
return 0;
}
dsi_config = dev_priv->dsi_configs[0];
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, true)) {
mutex_unlock(&dev_priv->vsync_lock);
return -EINVAL;
}
if (arg->vsync_operation_mask & VSYNC_ENABLE) {
/*enable vblank/TE*/
/*drm_vblank_get(dev, pipe);*/
switch (pipe) {
case 0:
case 2:
psb_fp = psb_fpriv(file_priv);
mdfld_dsi_dsr_forbid(dsi_config);
psb_fp->dsr_blocked = true;
if (get_panel_mode(dev) ==
MDFLD_DSI_ENCODER_DPI) {
psb_enable_vblank(dev, pipe);
dev_priv->vblank_disable_cnt = 0;
dev_priv->vsync_enabled = true;
}
break;
case 1:
psb_enable_vblank(dev, pipe);
break;
}
}
if (arg->vsync_operation_mask & VSYNC_DISABLE) {
/*drm_vblank_put(dev, pipe);*/
switch (pipe) {
case 0:
case 2:
psb_fp = psb_fpriv(file_priv);
if (get_panel_mode(dev) ==
MDFLD_DSI_ENCODER_DPI) {
/* Do not disable vblank immediately,
* we may have pending flip item,
* and flip watchdog may be fired
* mistakenly.
*/
dev_priv->vsync_enabled = false;
}
mdfld_dsi_dsr_allow(dsi_config);
psb_fp->dsr_blocked = false;
break;
case 1:
psb_disable_vblank(dev, pipe);
break;
}
atomic_inc(&dev_priv->vblank_count[pipe]);
wake_up_interruptible(&dev_priv->vsync_queue);
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
mutex_unlock(&dev_priv->vsync_lock);
return 0;
}
static int psb_register_rw_ioctl(struct drm_device *dev, void *data,
struct drm_file *file_priv)
{
struct drm_psb_private *dev_priv = psb_priv(dev);
struct drm_psb_register_rw_arg *arg = data;
unsigned int iep_ble_status;
unsigned long iep_timeout;
UHBUsage usage =
arg->b_force_hw_on ? OSPM_UHB_FORCE_POWER_ON : OSPM_UHB_ONLY_IF_ON;
struct mdfld_dsi_config *dsi_config;
struct mdfld_dsi_hw_registers *regs;
struct mdfld_dsi_hw_context *ctx;
int ret = 0;
mutex_lock(&dev_priv->overlay_lock);
if (arg->display_write_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
if (arg->display_write_mask & REGRWBITS_PFIT_CONTROLS)
PSB_WVDC32(arg->display.pfit_controls,
PFIT_CONTROL);
if (arg->display_write_mask &
REGRWBITS_PFIT_AUTOSCALE_RATIOS)
PSB_WVDC32(arg->display.pfit_autoscale_ratios,
PFIT_AUTO_RATIOS);
if (arg->display_write_mask &
REGRWBITS_PFIT_PROGRAMMED_SCALE_RATIOS)
PSB_WVDC32(
arg->display.pfit_programmed_scale_ratios,
PFIT_PGM_RATIOS);
if (arg->display_write_mask & REGRWBITS_PIPEASRC)
PSB_WVDC32(arg->display.pipeasrc,
PIPEASRC);
if (arg->display_write_mask & REGRWBITS_PIPEBSRC)
PSB_WVDC32(arg->display.pipebsrc,
PIPEBSRC);
if (arg->display_write_mask & REGRWBITS_VTOTAL_A)
PSB_WVDC32(arg->display.vtotal_a,
VTOTAL_A);
if (arg->display_write_mask & REGRWBITS_VTOTAL_B)
PSB_WVDC32(arg->display.vtotal_b,
VTOTAL_B);
if (arg->display_write_mask & REGRWBITS_DSPACNTR)
PSB_WVDC32(arg->display.dspcntr_a,
DSPACNTR);
if (arg->display_write_mask & REGRWBITS_DSPBCNTR)
PSB_WVDC32(arg->display.dspcntr_b,
DSPBCNTR);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else {
if (arg->display_write_mask & REGRWBITS_PFIT_CONTROLS)
dev_priv->savePFIT_CONTROL =
arg->display.pfit_controls;
if (arg->display_write_mask &
REGRWBITS_PFIT_AUTOSCALE_RATIOS)
dev_priv->savePFIT_AUTO_RATIOS =
arg->display.pfit_autoscale_ratios;
if (arg->display_write_mask &
REGRWBITS_PFIT_PROGRAMMED_SCALE_RATIOS)
dev_priv->savePFIT_PGM_RATIOS =
arg->display.pfit_programmed_scale_ratios;
if (arg->display_write_mask & REGRWBITS_PIPEASRC)
dev_priv->savePIPEASRC = arg->display.pipeasrc;
if (arg->display_write_mask & REGRWBITS_PIPEBSRC)
dev_priv->savePIPEBSRC = arg->display.pipebsrc;
if (arg->display_write_mask & REGRWBITS_VTOTAL_A)
dev_priv->saveVTOTAL_A = arg->display.vtotal_a;
if (arg->display_write_mask & REGRWBITS_VTOTAL_B)
dev_priv->saveVTOTAL_B = arg->display.vtotal_b;
}
}
if (arg->display_read_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
if (arg->display_read_mask &
REGRWBITS_PFIT_CONTROLS)
arg->display.pfit_controls =
PSB_RVDC32(PFIT_CONTROL);
if (arg->display_read_mask &
REGRWBITS_PFIT_AUTOSCALE_RATIOS)
arg->display.pfit_autoscale_ratios =
PSB_RVDC32(PFIT_AUTO_RATIOS);
if (arg->display_read_mask &
REGRWBITS_PFIT_PROGRAMMED_SCALE_RATIOS)
arg->display.pfit_programmed_scale_ratios =
PSB_RVDC32(PFIT_PGM_RATIOS);
if (arg->display_read_mask & REGRWBITS_PIPEASRC)
arg->display.pipeasrc = PSB_RVDC32(PIPEASRC);
if (arg->display_read_mask & REGRWBITS_PIPEBSRC)
arg->display.pipebsrc = PSB_RVDC32(PIPEBSRC);
if (arg->display_read_mask & REGRWBITS_VTOTAL_A)
arg->display.vtotal_a = PSB_RVDC32(VTOTAL_A);
if (arg->display_read_mask & REGRWBITS_VTOTAL_B)
arg->display.vtotal_b = PSB_RVDC32(VTOTAL_B);
if (arg->display_read_mask & REGRWBITS_DSPACNTR)
arg->display.dspcntr_a = PSB_RVDC32(DSPACNTR);
if (arg->display_read_mask & REGRWBITS_DSPBCNTR)
arg->display.dspcntr_b = PSB_RVDC32(DSPBCNTR);
if (arg->display_read_mask & REGRWBITS_PIPEASTAT)
arg->display.pipestat_a = PSB_RVDC32(PIPEASTAT);
if (arg->display_read_mask & REGRWBITS_INT_MASK)
arg->display.int_mask =
PSB_RVDC32(PSB_INT_MASK_R);
if (arg->display_read_mask & REGRWBITS_INT_ENABLE)
arg->display.int_enable =
PSB_RVDC32(PSB_INT_ENABLE_R);
if (arg->display_read_mask & REGRWBITS_DISPLAY_ALL)
psb_display_reg_dump(dev);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else {
if (arg->display_read_mask &
REGRWBITS_PFIT_CONTROLS)
arg->display.pfit_controls =
dev_priv->savePFIT_CONTROL;
if (arg->display_read_mask &
REGRWBITS_PFIT_AUTOSCALE_RATIOS)
arg->display.pfit_autoscale_ratios =
dev_priv->savePFIT_AUTO_RATIOS;
if (arg->display_read_mask &
REGRWBITS_PFIT_PROGRAMMED_SCALE_RATIOS)
arg->display.pfit_programmed_scale_ratios =
dev_priv->savePFIT_PGM_RATIOS;
if (arg->display_read_mask & REGRWBITS_PIPEASRC)
arg->display.pipeasrc = dev_priv->savePIPEASRC;
if (arg->display_read_mask & REGRWBITS_PIPEBSRC)
arg->display.pipebsrc = dev_priv->savePIPEBSRC;
if (arg->display_read_mask & REGRWBITS_VTOTAL_A)
arg->display.vtotal_a = dev_priv->saveVTOTAL_A;
if (arg->display_read_mask & REGRWBITS_VTOTAL_B)
arg->display.vtotal_b = dev_priv->saveVTOTAL_B;
if (arg->display_read_mask & REGRWBITS_PIPEASTAT)
arg->display.pipestat_a = PSB_RVDC32(PIPEASTAT);
if (arg->display_read_mask & REGRWBITS_INT_MASK)
arg->display.int_mask =
PSB_RVDC32(PSB_INT_MASK_R);
if (arg->display_read_mask & REGRWBITS_INT_ENABLE)
arg->display.int_enable =
PSB_RVDC32(PSB_INT_ENABLE_R);
if (arg->display_read_mask & REGRWBITS_DISPLAY_ALL)
psb_display_reg_dump(dev);
}
}
if (arg->overlay_write_mask == OVSTATUS_REGRBIT_OVR_UPDT ) {
int index = arg->overlay.backbuf_index;
mutex_lock(&dev_priv->ov_ctrl_lock);
ret = copy_from_user(dev_priv->ov_ctrl_blk + index,
(void __user *)arg->overlay.backbuf_addr,
sizeof(struct overlay_ctrl_blk));
mutex_unlock(&dev_priv->ov_ctrl_lock);
mutex_unlock(&dev_priv->overlay_lock);
return ret;
}
if (arg->overlay_write_mask != 0) {
int pipenum;
uint32_t ovadd_pipe = (arg->overlay.OVADD >> 6) & 0x3;
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
dsi_config = dev_priv->dsi_configs[0];
regs = &dsi_config->regs;
ctx = &dsi_config->dsi_hw_context;
/*forbid dsr which will restore regs*/
if (ovadd_pipe == 0)
mdfld_dsi_dsr_forbid(dsi_config);
if (arg->overlay_write_mask & OV_REGRWBITS_OGAM_ALL) {
PSB_WVDC32(arg->overlay.OGAMC5, OV_OGAMC5);
PSB_WVDC32(arg->overlay.OGAMC4, OV_OGAMC4);
PSB_WVDC32(arg->overlay.OGAMC3, OV_OGAMC3);
PSB_WVDC32(arg->overlay.OGAMC2, OV_OGAMC2);
PSB_WVDC32(arg->overlay.OGAMC1, OV_OGAMC1);
PSB_WVDC32(arg->overlay.OGAMC0, OV_OGAMC0);
}
if (arg->overlay_write_mask & OVC_REGRWBITS_OGAM_ALL) {
PSB_WVDC32(arg->overlay.OGAMC5, OVC_OGAMC5);
PSB_WVDC32(arg->overlay.OGAMC4, OVC_OGAMC4);
PSB_WVDC32(arg->overlay.OGAMC3, OVC_OGAMC3);
PSB_WVDC32(arg->overlay.OGAMC2, OVC_OGAMC2);
PSB_WVDC32(arg->overlay.OGAMC1, OVC_OGAMC1);
PSB_WVDC32(arg->overlay.OGAMC0, OVC_OGAMC0);
}
if (arg->overlay_write_mask & OV_REGRWBITS_WAIT_FLIP)
overlay_wait_flip(dev);
if (arg->overlay_write_mask & OV_REGRWBITS_OVADD) {
if (arg->overlay.buffer_handle) {
ret = validate_overlay_register_buffer(
file_priv,
&arg->overlay.OVADD,
arg->overlay.buffer_handle);
if (ret) {
printk(KERN_ERR
"Invalid parameter\n");
mutex_unlock(&dev_priv->overlay_lock);
return -EINVAL;
}
}
if (ovadd_pipe == 0) {
/*lock*/
mutex_lock(&dsi_config->context_lock);
if (dev_priv->exit_idle &&
(dsi_config->type ==
MDFLD_DSI_ENCODER_DPI))
dev_priv->exit_idle(dev,
MDFLD_DSR_2D_3D,
NULL,
true);
/*flip overlay*/
PSB_WVDC32(arg->overlay.OVADD, OV_OVADD);
ctx->ovaadd = arg->overlay.OVADD;
/*update on-panel frame buffer*/
if (arg->overlay.b_wms)
mdfld_dsi_dsr_update_panel_fb(dsi_config);
mutex_unlock(&dsi_config->context_lock);
} else
PSB_WVDC32(arg->overlay.OVADD, OV_OVADD);
/* when switch back from HDMI to local
* this ensures the Pipe B is fully disabled */
pipenum = ((arg->overlay.OVADD >> 6) & 0x3) ? 1 : 0;
wait_for_pipeb_finish(dev, pipenum);
if (arg->overlay.b_wait_vblank)
overlay_wait_vblank(dev,
file_priv,
arg->overlay.OVADD);
if (IS_MDFLD(dev)) {
if (arg->overlay.IEP_ENABLED) {
/* VBLANK period */
iep_timeout = jiffies + HZ / 10;
do {
iep_ble_status = PSB_RVDC32(0x31800);
if (time_after_eq(jiffies, iep_timeout)) {
DRM_ERROR("IEP Lite timeout\n");
break;
}
cpu_relax();
} while ((iep_ble_status >> 1) != 1);
arg->overlay.IEP_BLE_MINMAX = PSB_RVDC32(0x31804);
arg->overlay.IEP_BSSCC_CONTROL = PSB_RVDC32(0x32000);
}
}
}
if (arg->overlay_write_mask & OVC_REGRWBITS_OVADD) {
if (arg->overlay.buffer_handle) {
ret = validate_overlay_register_buffer(
file_priv,
&arg->overlay.OVADD,
arg->overlay.buffer_handle);
if (ret) {
printk(KERN_ERR
"Invalid parameter\n");
mutex_unlock(&dev_priv->overlay_lock);
return -EINVAL;
}
}
PSB_WVDC32(arg->overlay.OVADD, OVC_OVADD);
if (arg->overlay.b_wait_vblank) {
/*Wait for 20ms.*/
unsigned long vblank_timeout = jiffies + HZ / 50;
uint32_t temp;
while (time_before_eq(jiffies, vblank_timeout)) {
temp = PSB_RVDC32(OVC_DOVCSTA);
if ((temp & (0x1 << 31)) != 0) {
break;
}
cpu_relax();
}
}
}
/*allow entering dsr*/
if (ovadd_pipe == 0)
mdfld_dsi_dsr_allow(dsi_config);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else {
if (arg->overlay_write_mask & OV_REGRWBITS_OGAM_ALL) {
dev_priv->saveOV_OGAMC5 = arg->overlay.OGAMC5;
dev_priv->saveOV_OGAMC4 = arg->overlay.OGAMC4;
dev_priv->saveOV_OGAMC3 = arg->overlay.OGAMC3;
dev_priv->saveOV_OGAMC2 = arg->overlay.OGAMC2;
dev_priv->saveOV_OGAMC1 = arg->overlay.OGAMC1;
dev_priv->saveOV_OGAMC0 = arg->overlay.OGAMC0;
}
if (arg->overlay_write_mask & OVC_REGRWBITS_OGAM_ALL) {
dev_priv->saveOVC_OGAMC5 = arg->overlay.OGAMC5;
dev_priv->saveOVC_OGAMC4 = arg->overlay.OGAMC4;
dev_priv->saveOVC_OGAMC3 = arg->overlay.OGAMC3;
dev_priv->saveOVC_OGAMC2 = arg->overlay.OGAMC2;
dev_priv->saveOVC_OGAMC1 = arg->overlay.OGAMC1;
dev_priv->saveOVC_OGAMC0 = arg->overlay.OGAMC0;
}
if (arg->overlay_write_mask & OV_REGRWBITS_OVADD)
dev_priv->saveOV_OVADD = arg->overlay.OVADD;
if (arg->overlay_write_mask & OVC_REGRWBITS_OVADD)
dev_priv->saveOVC_OVADD = arg->overlay.OVADD;
}
}
if (arg->overlay_read_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
if (arg->overlay_read_mask & OV_REGRWBITS_OGAM_ALL) {
arg->overlay.OGAMC5 = PSB_RVDC32(OV_OGAMC5);
arg->overlay.OGAMC4 = PSB_RVDC32(OV_OGAMC4);
arg->overlay.OGAMC3 = PSB_RVDC32(OV_OGAMC3);
arg->overlay.OGAMC2 = PSB_RVDC32(OV_OGAMC2);
arg->overlay.OGAMC1 = PSB_RVDC32(OV_OGAMC1);
arg->overlay.OGAMC0 = PSB_RVDC32(OV_OGAMC0);
}
if (arg->overlay_read_mask & OVC_REGRWBITS_OGAM_ALL) {
arg->overlay.OGAMC5 = PSB_RVDC32(OVC_OGAMC5);
arg->overlay.OGAMC4 = PSB_RVDC32(OVC_OGAMC4);
arg->overlay.OGAMC3 = PSB_RVDC32(OVC_OGAMC3);
arg->overlay.OGAMC2 = PSB_RVDC32(OVC_OGAMC2);
arg->overlay.OGAMC1 = PSB_RVDC32(OVC_OGAMC1);
arg->overlay.OGAMC0 = PSB_RVDC32(OVC_OGAMC0);
}
if (arg->overlay_read_mask & OV_REGRWBITS_OVADD)
arg->overlay.OVADD = PSB_RVDC32(OV_OVADD);
if (arg->overlay_read_mask & OVC_REGRWBITS_OVADD)
arg->overlay.OVADD = PSB_RVDC32(OVC_OVADD);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
} else {
if (arg->overlay_read_mask & OV_REGRWBITS_OGAM_ALL) {
arg->overlay.OGAMC5 = dev_priv->saveOV_OGAMC5;
arg->overlay.OGAMC4 = dev_priv->saveOV_OGAMC4;
arg->overlay.OGAMC3 = dev_priv->saveOV_OGAMC3;
arg->overlay.OGAMC2 = dev_priv->saveOV_OGAMC2;
arg->overlay.OGAMC1 = dev_priv->saveOV_OGAMC1;
arg->overlay.OGAMC0 = dev_priv->saveOV_OGAMC0;
}
if (arg->overlay_read_mask & OVC_REGRWBITS_OGAM_ALL) {
arg->overlay.OGAMC5 = dev_priv->saveOVC_OGAMC5;
arg->overlay.OGAMC4 = dev_priv->saveOVC_OGAMC4;
arg->overlay.OGAMC3 = dev_priv->saveOVC_OGAMC3;
arg->overlay.OGAMC2 = dev_priv->saveOVC_OGAMC2;
arg->overlay.OGAMC1 = dev_priv->saveOVC_OGAMC1;
arg->overlay.OGAMC0 = dev_priv->saveOVC_OGAMC0;
}
if (arg->overlay_read_mask & OV_REGRWBITS_OVADD)
arg->overlay.OVADD = dev_priv->saveOV_OVADD;
if (arg->overlay_read_mask & OVC_REGRWBITS_OVADD)
arg->overlay.OVADD = dev_priv->saveOVC_OVADD;
}
}
if (arg->sprite_enable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
PSB_WVDC32(0x1F3E, DSPARB);
PSB_WVDC32(arg->sprite.dspa_control | PSB_RVDC32(DSPACNTR), DSPACNTR);
PSB_WVDC32(arg->sprite.dspa_key_value, DSPAKEYVAL);
PSB_WVDC32(arg->sprite.dspa_key_mask, DSPAKEYMASK);
PSB_WVDC32(PSB_RVDC32(DSPASURF), DSPASURF);
PSB_RVDC32(DSPASURF);
PSB_WVDC32(arg->sprite.dspc_control, DSPCCNTR);
PSB_WVDC32(arg->sprite.dspc_stride, DSPCSTRIDE);
PSB_WVDC32(arg->sprite.dspc_position, DSPCPOS);
PSB_WVDC32(arg->sprite.dspc_linear_offset, DSPCLINOFF);
PSB_WVDC32(arg->sprite.dspc_size, DSPCSIZE);
PSB_WVDC32(arg->sprite.dspc_surface, DSPCSURF);
PSB_RVDC32(DSPCSURF);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
if (arg->sprite_disable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
PSB_WVDC32(0x3F3E, DSPARB);
PSB_WVDC32(0x0, DSPCCNTR);
PSB_WVDC32(arg->sprite.dspc_surface, DSPCSURF);
PSB_RVDC32(DSPCSURF);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
if (arg->subpicture_enable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
uint32_t temp;
if (arg->subpicture_enable_mask & REGRWBITS_DSPACNTR) {
temp = PSB_RVDC32(DSPACNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp &= ~DISPPLANE_BOTTOM;
temp |= DISPPLANE_32BPP;
PSB_WVDC32(temp, DSPACNTR);
temp = PSB_RVDC32(DSPABASE);
PSB_WVDC32(temp, DSPABASE);
PSB_RVDC32(DSPABASE);
temp = PSB_RVDC32(DSPASURF);
PSB_WVDC32(temp, DSPASURF);
PSB_RVDC32(DSPASURF);
}
if (arg->subpicture_enable_mask & REGRWBITS_DSPBCNTR) {
temp = PSB_RVDC32(DSPBCNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp &= ~DISPPLANE_BOTTOM;
temp |= DISPPLANE_32BPP;
PSB_WVDC32(temp, DSPBCNTR);
temp = PSB_RVDC32(DSPBBASE);
PSB_WVDC32(temp, DSPBBASE);
PSB_RVDC32(DSPBBASE);
temp = PSB_RVDC32(DSPBSURF);
PSB_WVDC32(temp, DSPBSURF);
PSB_RVDC32(DSPBSURF);
}
if (arg->subpicture_enable_mask & REGRWBITS_DSPCCNTR) {
temp = PSB_RVDC32(DSPCCNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp &= ~DISPPLANE_BOTTOM;
temp |= DISPPLANE_32BPP;
PSB_WVDC32(temp, DSPCCNTR);
temp = PSB_RVDC32(DSPCBASE);
PSB_WVDC32(temp, DSPCBASE);
PSB_RVDC32(DSPCBASE);
temp = PSB_RVDC32(DSPCSURF);
PSB_WVDC32(temp, DSPCSURF);
PSB_RVDC32(DSPCSURF);
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
if (arg->subpicture_disable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
uint32_t temp;
if (arg->subpicture_disable_mask & REGRWBITS_DSPACNTR) {
temp = PSB_RVDC32(DSPACNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp |= DISPPLANE_32BPP_NO_ALPHA;
PSB_WVDC32(temp, DSPACNTR);
temp = PSB_RVDC32(DSPABASE);
PSB_WVDC32(temp, DSPABASE);
PSB_RVDC32(DSPABASE);
temp = PSB_RVDC32(DSPASURF);
PSB_WVDC32(temp, DSPASURF);
PSB_RVDC32(DSPASURF);
}
if (arg->subpicture_disable_mask & REGRWBITS_DSPBCNTR) {
temp = PSB_RVDC32(DSPBCNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp |= DISPPLANE_32BPP_NO_ALPHA;
PSB_WVDC32(temp, DSPBCNTR);
temp = PSB_RVDC32(DSPBBASE);
PSB_WVDC32(temp, DSPBBASE);
PSB_RVDC32(DSPBBASE);
temp = PSB_RVDC32(DSPBSURF);
PSB_WVDC32(temp, DSPBSURF);
PSB_RVDC32(DSPBSURF);
}
if (arg->subpicture_disable_mask & REGRWBITS_DSPCCNTR) {
temp = PSB_RVDC32(DSPCCNTR);
temp &= ~DISPPLANE_PIXFORMAT_MASK;
temp |= DISPPLANE_32BPP_NO_ALPHA;
PSB_WVDC32(temp, DSPCCNTR);
temp = PSB_RVDC32(DSPCBASE);
PSB_WVDC32(temp, DSPCBASE);
PSB_RVDC32(DSPCBASE);
temp = PSB_RVDC32(DSPCSURF);
PSB_WVDC32(temp, DSPCSURF);
PSB_RVDC32(DSPCSURF);
}
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
if (arg->cursor_enable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
uint32_t temp;
temp = PSB_RVDC32(PIPEACONF);
temp &= ~PIPECONF_CURSOR_OFF;
PSB_WVDC32(temp, PIPEACONF);
PSB_WVDC32((arg->cursor.CursorSize == 1) ? 0x22 : 0x27,
CURACNTR);
PSB_WVDC32(arg->cursor.CursorADDR, CURABASE);
if ((arg->cursor.xPos > 0) && (arg->cursor.yPos > 0))
PSB_WVDC32(((arg->cursor.yPos << 16)
|(arg->cursor.xPos)), CURAPOS);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
if (arg->cursor_disable_mask != 0) {
if (ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND, usage)) {
uint32_t temp;
temp = PSB_RVDC32(PIPEACONF);
temp |= PIPECONF_CURSOR_OFF;
PSB_WVDC32(temp, PIPEACONF);
PSB_WVDC32(0x0, CURACNTR);
PSB_WVDC32(0x0, CURABASE);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
}
}
mutex_unlock(&dev_priv->overlay_lock);
return 0;
}
/* always available as we are SIGIO'd */
static unsigned int psb_poll(struct file *filp,
struct poll_table_struct *wait)
{
return POLLIN | POLLRDNORM;
}
static int psb_driver_open(struct drm_device *dev, struct drm_file *priv)
{
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
DRM_DEBUG("\n");
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRSRVOpen(dev, priv);
}
static long psb_unlocked_ioctl(struct file *filp, unsigned int cmd,
unsigned long arg)
{
unsigned int nr = DRM_IOCTL_NR(cmd);
long ret;
DRM_DEBUG("cmd = %x, nr = %x\n", cmd, nr);
/*
* The driver private ioctls and TTM ioctls should be
* thread-safe.
*/
if ((nr >= DRM_COMMAND_BASE) && (nr < DRM_COMMAND_END)
&& (nr < DRM_COMMAND_BASE + DRM_ARRAY_SIZE(psb_ioctls))) {
struct drm_ioctl_desc *ioctl =
&psb_ioctls[nr - DRM_COMMAND_BASE];
if (unlikely(ioctl->cmd != cmd)) {
DRM_ERROR(
"Invalid drm cmnd %d ioctl->cmd %x, cmd %x\n",
nr - DRM_COMMAND_BASE, ioctl->cmd, cmd);
return -EINVAL;
}
}
/*
* Not all old drm ioctls are thread-safe.
*/
/* lock_kernel(); */
ret = drm_ioctl(filp, cmd, arg);
/* unlock_kernel(); */
return ret;
}
#ifdef DISPLAY_DRIVER_DEBUG_INTERFACE
static int psb_blc_proc_show(struct seq_file *seq, void *v)
{
struct drm_minor *minor = (struct drm_minor *) seq->private;
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
int user_brightness = 0;
int final_brightness = 0;
user_brightness = psb_get_brightness(NULL);
final_brightness = (user_brightness * dev_priv->blc_adj1) / 100;
final_brightness = (final_brightness * dev_priv->blc_adj2) / 100;
DRM_INFO("%i\n", final_brightness);
seq_printf(seq, "%i\n", final_brightness);
return 0;
}
static int psb_blc_proc_open(struct inode *inode, struct file *file)
{
#if (LINUX_VERSION_CODE < KERNEL_VERSION(3, 8, 0))
return single_open(file, psb_blc_proc_show, PDE(inode));
#else
return single_open(file, psb_blc_proc_show, PDE_DATA(inode));
#endif
}
static const struct file_operations psb_blc_proc_fops = {
.owner = THIS_MODULE,
.open = psb_blc_proc_open,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release,
};
static int psb_rtpm_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
PSB_DEBUG_ENTRY("Current Runtime PM delay for GFX: %d (ms)\n", gfxrtdelay);
return 0;
}
static int psb_rtpm_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
char buf[2];
int temp = 0;
if (count != sizeof(buf)) {
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
temp = buf[0] - '0';
switch (temp) {
case 1:
gfxrtdelay = 10 * 1000;
break;
case 2:
gfxrtdelay = 20 * 1000;
break;
default:
gfxrtdelay = 30 * 1000;
break;
}
PSB_DEBUG_ENTRY("Runtime PM delay set for GFX: %d (ms)\n", gfxrtdelay);
}
return count;
}
static int psb_ospm_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
#ifdef OSPM_STAT
unsigned long on_time = 0;
unsigned long off_time = 0;
#endif
/*#ifdef SUPPORT_ACTIVE_POWER_MANAGEMENT
DRM_INFO("GFX D0i3: enabled ");
#else
DRM_INFO("GFX D0i3: disabled ");
#endif*/
if (drm_psb_ospm)
DRM_INFO("GFX D0i3: enabled ");
else
DRM_INFO("GFX D0i3: disabled ");
#ifdef OSPM_STAT
switch (dev_priv->graphics_state) {
case PSB_PWR_STATE_ON:
DRM_INFO("GFX state:%s\n", "on");
break;
case PSB_PWR_STATE_OFF:
DRM_INFO("GFX state:%s\n", "off");
break;
default:
DRM_INFO("GFX state:%s\n", "unknown");
}
on_time = dev_priv->gfx_on_time * 1000 / HZ;
off_time = dev_priv->gfx_off_time * 1000 / HZ;
switch (dev_priv->graphics_state) {
case PSB_PWR_STATE_ON:
on_time += (jiffies - dev_priv->gfx_last_mode_change) * \
1000 / HZ;
break;
case PSB_PWR_STATE_OFF:
off_time += (jiffies - dev_priv->gfx_last_mode_change) * \
1000 / HZ;
break;
}
DRM_INFO("GFX(count/ms):\n");
DRM_INFO("on:%lu/%lu, off:%lu/%lu \n",
dev_priv->gfx_on_cnt, on_time, dev_priv->gfx_off_cnt, off_time);
#endif
return nbytes;
}
static int psb_ospm_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
char buf[2];
if (count != sizeof(buf)) {
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
drm_psb_ospm = buf[0] - '0';
PSB_DEBUG_ENTRY(" SGX (D0i3) drm_psb_ospm: %d\n",
drm_psb_ospm);
}
return count;
}
static int psb_panel_register_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
char msg[PSB_REG_PRINT_SIZE];
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
/*do nothing*/
if (dev_priv->buf && dev_priv->count < PSB_REG_PRINT_SIZE)
memcpy(msg, dev_priv->buf, dev_priv->count);
return simple_read_from_buffer(buf, nbytes, ppos, msg, dev_priv->count);
}
/*
* use to read and write panel side register. and print to standard output.
*/
#define GENERIC_READ_FIFO_SIZE_MAX 0x40
static int psb_panel_register_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct mdfld_dsi_config *dsi_config = dev_priv->dsi_configs[0];
char buf[256];
char op = '0';
char type = '0';
int cmd = 0;
char par[256];
int pnum = 0;
int par_offset = 0;
int add_size = 0;
int ret = 0;
u8 *pdata = NULL;
int i = 0;
if (!dsi_config)
return -EINVAL;
dev_priv->count = 0;
memset(buf, '\0', sizeof(buf));
if (count > sizeof(buf)) {
PSB_DEBUG_ENTRY(
"The input is too bigger, kernel can not handle.\n");
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
PSB_DEBUG_ENTRY("input = %s", buf);
}
sscanf(buf, "%c%c%x%x", &op, &type, &cmd, &pnum);
par_offset = (sizeof("xx xx xx ") - 2);
memcpy(par, buf + par_offset, 256 - par_offset);
if (op != 'g' && op != 's') {
PSB_DEBUG_ENTRY("The input format is not right!\n");
PSB_DEBUG_ENTRY(
"sg: send generic. sm: send mcs. gg: get state\n");
PSB_DEBUG_ENTRY(
"gg cmd count (gg a 01 :get panel status.)\n");
PSB_DEBUG_ENTRY(
"sg cmd count par (sg 2c 00:set write_mem_start.)\n");
PSB_DEBUG_ENTRY(
"sm 00 count cmd+par(sm 00 01 28:set display on)\n");
return -EINVAL;
}
PSB_DEBUG_ENTRY("op= %c type= %c cmd=%x pnum=%x\n",
op, type, cmd, pnum);
PSB_DEBUG_ENTRY("par =%s", par);
if (op == 'g' && pnum == 0) {
PSB_DEBUG_ENTRY("get status must has parameter count!");
sprintf(dev_priv->buf,
"get status must has parameter count!\n");
return -EINVAL;
}
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
PSB_DEBUG_ENTRY("Display controller can not power on.!\n");
return -EPERM;
}
/*forbid dsr which will restore regs*/
mdfld_dsi_dsr_forbid(dsi_config);
if (op == 'g' && pnum <= GENERIC_READ_FIFO_SIZE_MAX) {
pdata = kmalloc(sizeof(u8)*pnum, GFP_KERNEL);
if (!pdata) {
DRM_ERROR("No memory for long_pkg data\n");
ret = -ENOMEM;
goto fun_exit;
}
ret = mdfld_dsi_get_panel_status(dsi_config, cmd,
pdata , MDFLD_DSI_LP_TRANSMISSION, pnum);
if (ret == pnum && ret != 0) {
PSB_DEBUG_ENTRY("read panel status\n");
PSB_DEBUG_ENTRY("cmd : 0x%02x\n", cmd);
add_size = sizeof("cmd : 0xFF\n");
if (dev_priv->buf && (dev_priv->count + add_size)
< PSB_REG_PRINT_SIZE)
dev_priv->count += sprintf(
dev_priv->buf + dev_priv->count,
"cmd : 0x%02x\n", cmd);
for (i = 0; i < pnum; i++) {
PSB_DEBUG_ENTRY("par%d= 0x%02x\n", i, pdata[i]);
add_size = sizeof("par1=0xFF 0xFF\n");
if (dev_priv->buf && (dev_priv->count + add_size)
< PSB_REG_PRINT_SIZE)
dev_priv->count += sprintf(
dev_priv->buf + dev_priv->count,
"par%d= 0x%02x\n", i, pdata[i]);
}
} else {
PSB_DEBUG_ENTRY("get panel status fail\n");
sprintf(dev_priv->buf, "get panel status fail\n");
}
kfree(pdata);
}
if (op == 's' && pnum <= GENERIC_READ_FIFO_SIZE_MAX) {
struct mdfld_dsi_pkg_sender *sender =
mdfld_dsi_get_pkg_sender(dsi_config);
if (!sender) {
DRM_ERROR("Invalid sender\n");
return -EINVAL;
}
pdata = kmalloc(sizeof(u8)*pnum, GFP_KERNEL);
if (!pdata) {
DRM_ERROR("No memory for long_pkg data\n");
ret = -ENOMEM;
goto fun_exit;
}
for (i = 0; i < pnum; i++)
sscanf(par + i * 3, "%x", &pdata[i]);
if (cmd == 0 && pnum != 0) {
if (type == 'g')
ret = mdfld_dsi_send_gen_long_hs(
sender, pdata, pnum, 0);
else if (type == 'm')
ret = mdfld_dsi_send_mcs_long_hs(
sender, pdata, pnum, 0);
}
else {
if (cmd == 0x2c)
atomic64_inc(&sender->te_seq);
ret = mdfld_dsi_send_dcs(sender,
cmd,
pdata,
pnum,
CMD_DATA_SRC_SYSTEM_MEM,
MDFLD_DSI_SEND_PACKAGE);
}
if (ret) {
PSB_DEBUG_ENTRY("set panel status fail!\n");
sprintf(dev_priv->buf, "set panel status fail!\n");
} else {
PSB_DEBUG_ENTRY("set panel status ok!\n");
sprintf(dev_priv->buf, "set panel status ok\n");
}
kfree(pdata);
}
fun_exit:
/*allow entering dsr*/
mdfld_dsi_dsr_allow(dsi_config);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return count;
}
static int psb_display_register_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
char msg[PSB_REG_PRINT_SIZE];
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
/*do nothing*/
if (dev_priv->buf && dev_priv->count < PSB_REG_PRINT_SIZE)
memcpy(msg, dev_priv->buf, dev_priv->count);
return simple_read_from_buffer(buf, nbytes, ppos, msg, dev_priv->count);
}
/*
* use to read and write display register. and print to standard output.
*/
static int psb_display_register_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct drm_psb_private *dev_priv =
(struct drm_psb_private *) dev->dev_private;
struct mdfld_dsi_config *dsi_config = dev_priv->dsi_configs[0];
int reg_val = 0;
char buf[256];
char op = '0';
int reg = 0, start = 0, end = 0;
unsigned int val = 0;
int len = 0;
int Offset = 0;
int add_size = 0;
int ret = 0;
if (!dsi_config)
return -EINVAL;
dev_priv->count = 0;
memset(buf, '\0', sizeof(buf));
if (count > sizeof(buf)) {
PSB_DEBUG_ENTRY("The input is too bigger, kernel can not handle.\n");
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
PSB_DEBUG_ENTRY("input = %s", buf);
}
sscanf(buf, "%c%x%x", &op, &reg, &val);
if (op != 'r' && op != 'w' && op != 'a') {
PSB_DEBUG_ENTRY("The input format is not right!\n");
PSB_DEBUG_ENTRY("for exampe: r 70184 (read register 70184.)\n");
PSB_DEBUG_ENTRY("for exampe: w 70184 123 (write register 70184 with value 123.)\n");
PSB_DEBUG_ENTRY("for exmape: a 60000 60010(read all registers start at 60000 and end at 60010.\n)");
return -EINVAL;
}
if ((reg < 0xa000 || reg > 0x720ff) && (reg < 0x40 || reg > 0x64)) {
PSB_DEBUG_ENTRY("the register is out of display controller registers rang.\n");
return -EINVAL;
}
if ((reg % 0x4) != 0) {
PSB_DEBUG_ENTRY("the register address should aligned to 4 byte.please refrence display controller specification.\n");
return -EINVAL;
}
if (!ospm_power_using_hw_begin(OSPM_DISPLAY_ISLAND,
OSPM_UHB_FORCE_POWER_ON)) {
PSB_DEBUG_ENTRY("Display controller can not power on.!\n");
return -EPERM;
}
/*forbid dsr which will restore regs*/
mdfld_dsi_dsr_forbid(dsi_config);
if (op == 'r') {
if (reg >= 0xa000) {
reg_val = REG_READ(reg);
PSB_DEBUG_ENTRY("Read :reg=0x%08x , val=0x%08x.\n", reg, reg_val);
} else {
reg_val = SGX_REG_READ(reg);
PSB_DEBUG_ENTRY("SGX Read :reg=0x%08x , val=0x%08x.\n", reg, reg_val);
}
add_size = sizeof("0xFFFFFFFF 0xFFFFFFFF\n");
if (dev_priv->buf &&
(dev_priv->count + add_size) < PSB_REG_PRINT_SIZE)
dev_priv->count = sprintf(dev_priv->buf, "%08x %08x\n", reg, reg_val);
}
if (op == 'w') {
if (reg >= 0xa000) {
reg_val = REG_READ(reg);
PSB_DEBUG_ENTRY("Before change:reg=0x%08x , val=0x%08x.\n", reg, reg_val);
REG_WRITE(reg, val);
reg_val = REG_READ(reg);
PSB_DEBUG_ENTRY("After change:reg=0x%08x , val=0x%08x.\n", reg, reg_val);
} else {
reg_val = SGX_REG_READ(reg);
PSB_DEBUG_ENTRY("Before change: sgx reg=0x%08x , val=0x%08x.\n", reg, reg_val);
SGX_REG_WRITE(reg, val);
reg_val = SGX_REG_READ(reg);
PSB_DEBUG_ENTRY("After change:sgx reg=0x%08x , val=0x%08x.\n", reg, reg_val);
}
}
if (op == 'a') {
start = reg;
end = val;
PSB_DEBUG_ENTRY("start:0x%08x\n", start);
PSB_DEBUG_ENTRY("end: 0x%08x\n", end);
if ((start % 0x4) != 0) {
PSB_DEBUG_ENTRY("The start address should be 4 byte aligned. Please reference the display controller specification.\n");
ret = -EINVAL;
goto fun_exit;
}
if ((end % 0x4) != 0) {
PSB_DEBUG_ENTRY("The end address should be 4 byte aligned. Please reference the display controller specification.\n");
ret = -EINVAL;
goto fun_exit;
}
len = end - start + 1;
if (len <= 0)
PSB_DEBUG_ENTRY("The end address should be greater than the start address.\n");
if (end < 0xa000 || end > 0x720ff) {
PSB_DEBUG_ENTRY("The end address is out of the display controller register range.\n");
ret = -EINVAL;
goto fun_exit;
}
if (start < 0xa000 || start > 0x720ff) {
PSB_DEBUG_ENTRY("The start address is out of the display controller register range.\n");
ret = -EINVAL;
goto fun_exit;
}
for (Offset = start ; Offset < end; Offset = Offset + 0x10) {
if (reg >= 0xa000) {
PSB_DEBUG_ENTRY("0x%08x: 0x%08x 0x%08x 0x%08x 0x%08x\n",
Offset,
REG_READ(Offset + 0x0),
REG_READ(Offset + 0x4),
REG_READ(Offset + 0x8),
REG_READ(Offset + 0xc));
add_size = 5 * sizeof("0xFFFFFFFF ");
if (dev_priv->buf &&
(dev_priv->count + add_size) < PSB_REG_PRINT_SIZE)
dev_priv->count += sprintf(dev_priv->buf + dev_priv->count,
"%08x %08x %08x %08x %08x\n",
Offset,
REG_READ(Offset + 0x0),
REG_READ(Offset + 0x4),
REG_READ(Offset + 0x8),
REG_READ(Offset + 0xc));
} else {
PSB_DEBUG_ENTRY("0x%08x: 0x%08x 0x%08x 0x%08x 0x%08x\n",
Offset,
SGX_REG_READ(Offset + 0x0),
SGX_REG_READ(Offset + 0x4),
SGX_REG_READ(Offset + 0x8),
SGX_REG_READ(Offset + 0xc));
add_size = 5 * sizeof("0xFFFFFFFF ");
if (dev_priv->buf &&
(dev_priv->count + add_size) < PSB_REG_PRINT_SIZE)
dev_priv->count += sprintf(dev_priv->buf + dev_priv->count,
"%08x %08x %08x %08x %08x\n",
Offset,
SGX_REG_READ(Offset + 0x0),
SGX_REG_READ(Offset + 0x4),
SGX_REG_READ(Offset + 0x8),
SGX_REG_READ(Offset + 0xc));
}
}
}
fun_exit:
/*allow entering dsr*/
mdfld_dsi_dsr_allow(dsi_config);
ospm_power_using_hw_end(OSPM_DISPLAY_ISLAND);
return count;
}
static int csc_control_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
return 0;
}
static int csc_control_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
char buf[2];
int csc_control;
struct drm_minor *minor = (struct drm_minor *)PDE_DATA(file_inode(file));
struct drm_device *dev = minor->dev;
struct csc_setting csc;
struct gamma_setting gamma;
if (count != sizeof(buf)) {
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
csc_control = buf[0] - '0';
PSB_DEBUG_ENTRY(" csc control: %d\n", csc_control);
switch (csc_control) {
case 0x0:
csc.pipe = 0;
csc.type = CSC_REG_SETTING;
csc.enable_state = true;
csc.data_len = CSC_REG_COUNT;
memcpy(csc.data.csc_reg_data, csc_setting, sizeof(csc.data.csc_reg_data));
mdfld_intel_crtc_set_color_conversion(dev, &csc);
break;
case 0x1:
gamma.pipe = 0;
gamma.type = GAMMA_REG_SETTING;
gamma.enable_state = true;
gamma.data_len = GAMMA_10_BIT_TABLE_COUNT;
memcpy(gamma.gamma_tableX100, gamma_setting, sizeof(gamma.gamma_tableX100));
mdfld_intel_crtc_set_gamma(dev, &gamma);
break;
default:
printk("invalied parameters\n");
}
}
return count;
}
#ifdef CONFIG_SUPPORT_HDMI
int gpio_control_read(struct file *file, char __user *buf,
size_t nbytes,loff_t *ppos)
{
unsigned int value = 0;
unsigned int pin_num = otm_hdmi_get_hpd_pin();
if (pin_num)
value = gpio_get_value(pin_num);
printk(KERN_ALERT "read pin_num: %8d value:%8d\n", pin_num, value);
return 0;
}
int gpio_control_write(struct file *file, const char *buffer,
size_t count, loff_t *ppos)
{
char buf[2];
int gpio_control;
int result = 0;
unsigned int pin_num = otm_hdmi_get_hpd_pin();
bool auto_state = drm_hdmi_hpd_auto;
if (!pin_num)
return -EINVAL;
if (count != sizeof(buf)) {
return -EINVAL;
} else {
if (copy_from_user(buf, buffer, count))
return -EINVAL;
if (buf[count-1] != '\n')
return -EINVAL;
gpio_control = buf[0] - '0';
printk(KERN_ALERT "GPIO set pin:%8d\n", pin_num);
printk(KERN_ALERT "value:%8d\n", gpio_control);
switch (gpio_control) {
case 0x0:
result = gpio_direction_output(pin_num, 0);
otm_hdmi_override_cable_status(false, auto_state);
if (result) {
printk(KERN_ALERT "Failed set GPIO as output\n");
return -EINVAL;
}
break;
case 0x1:
result = gpio_direction_output(pin_num, 0);
otm_hdmi_override_cable_status(true, auto_state);
if (result) {
printk(KERN_ALERT "Failed set GPIO as output\n");
return -EINVAL;
}
break;
default:
printk(KERN_ALERT "invalied parameters\n");
}
result = gpio_direction_input(pin_num);
if (result) {
printk(KERN_ALERT "Failed set GPIO as input\n");
return -EINVAL;
}
}
return count;
}
#endif
static const struct file_operations psb_gpio_proc_fops = {
.owner = THIS_MODULE,
.read = gpio_control_read,
.write = gpio_control_write,
};
static const struct file_operations psb_ospm_proc_fops = {
.owner = THIS_MODULE,
.read = psb_ospm_read,
.write = psb_ospm_write,
};
static const struct file_operations psb_rtpm_proc_fops = {
.owner = THIS_MODULE,
.read = psb_rtpm_read,
.write = psb_rtpm_write,
};
static const struct file_operations psb_display_proc_fops = {
.owner = THIS_MODULE,
.read = psb_display_register_read,
.write = psb_display_register_write,
};
static const struct file_operations psb_panel_proc_fops = {
.owner = THIS_MODULE,
.read = psb_panel_register_read,
.write = psb_panel_register_write,
};
static const struct file_operations psb_csc_proc_fops = {
.owner = THIS_MODULE,
.read = csc_control_read,
.write = csc_control_write,
};
#ifdef CONFIG_SUPPORT_HDMI
static int psb_hdmi_proc_init(struct drm_minor *minor)
{
struct proc_dir_entry *gpio_setting;
gpio_setting = proc_create_data(GPIO_PROC_ENTRY,
0644, minor->proc_root, &psb_gpio_proc_fops, minor);
if (!gpio_setting)
return -1;
return 0;
}
#endif
static int psb_proc_init(struct drm_minor *minor)
{
struct proc_dir_entry *ent;
struct proc_dir_entry *ent1;
struct proc_dir_entry *rtpm;
struct proc_dir_entry *ent_display_status;
struct proc_dir_entry *ent_panel_status;
struct proc_dir_entry *csc_setting;
ent = proc_create_data(OSPM_PROC_ENTRY, 0644, minor->proc_root, &psb_ospm_proc_fops, minor);
rtpm = proc_create_data(RTPM_PROC_ENTRY, 0644, minor->proc_root, &psb_rtpm_proc_fops, minor);
ent_display_status = proc_create_data(DISPLAY_PROC_ENTRY, 0644, minor->proc_root,
&psb_display_proc_fops, minor);
ent_panel_status = proc_create_data(PANEL_PROC_ENTRY, 0644, minor->proc_root,
&psb_panel_proc_fops, minor);
ent1 = proc_create_data(BLC_PROC_ENTRY, 0, minor->proc_root, &psb_blc_proc_fops, minor);
csc_setting = proc_create_data(CSC_PROC_ENTRY, 0644, minor->proc_root, &psb_csc_proc_fops, minor);
if (!ent || !ent1 || !rtpm || !ent_display_status || !ent_panel_status
|| !csc_setting)
return -1;
#ifdef CONFIG_SUPPORT_HDMI
psb_hdmi_proc_init(minor);
#endif
return 0;
}
static void psb_proc_cleanup(struct drm_minor *minor)
{
remove_proc_entry(OSPM_PROC_ENTRY, minor->proc_root);
remove_proc_entry(RTPM_PROC_ENTRY, minor->proc_root);
remove_proc_entry(BLC_PROC_ENTRY, minor->proc_root);
#ifdef CONFIG_SUPPORT_HDMI
remove_proc_entry(GPIO_PROC_ENTRY, minor->proc_root);
#endif
return;
}
#endif /* DISPLAY_DRIVER_DEBUG_INTERFACE */
/* When a client dies:
* - Check for and clean up flipped page state
*/
void psb_driver_preclose(struct drm_device *dev, struct drm_file *priv)
{
}
static void psb_remove(struct pci_dev *pdev)
{
struct drm_device *dev = pci_get_drvdata(pdev);
drm_put_dev(dev);
}
static const struct dev_pm_ops psb_pm_ops = {
.runtime_suspend = psb_runtime_suspend,
.runtime_resume = psb_runtime_resume,
.runtime_idle = psb_runtime_idle,
.suspend = psb_runtime_suspend,
.resume = psb_runtime_resume,
};
static struct vm_operations_struct psb_ttm_vm_ops;
/**
* NOTE: driver_private of drm_file is now a PVRSRV_FILE_PRIVATE_DATA struct
* pPriv in PVRSRV_FILE_PRIVATE_DATA contains the original psb_fpriv;
*/
int psb_open(struct inode *inode, struct file *filp)
{
struct drm_file *file_priv;
struct drm_psb_private *dev_priv;
struct psb_fpriv *psb_fp;
PVRSRV_FILE_PRIVATE_DATA *pvr_file_priv;
int ret;
DRM_DEBUG("\n");
ret = drm_open(inode, filp);
if (unlikely(ret))
return ret;
psb_fp = kzalloc(sizeof(*psb_fp), GFP_KERNEL);
if (unlikely(psb_fp == NULL))
goto out_err0;
file_priv = (struct drm_file *) filp->private_data;
/* In case that the local file priv has created a master,
* which has been referenced, even if it's not authenticated
* (non-root user). */
if ((file_priv->minor->master)
&& (file_priv->master == file_priv->minor->master)
&& (!file_priv->is_master))
file_priv->is_master = 1;
dev_priv = psb_priv(file_priv->minor->dev);
DRM_DEBUG("is_master %d\n", file_priv->is_master ? 1 : 0);
psb_fp->tfile = ttm_object_file_init(dev_priv->tdev,
PSB_FILE_OBJECT_HASH_ORDER);
if (unlikely(psb_fp->tfile == NULL))
goto out_err1;
pvr_file_priv = (PVRSRV_FILE_PRIVATE_DATA *)file_priv->driver_priv;
if (!pvr_file_priv) {
DRM_ERROR("drm file private is NULL\n");
goto out_err1;
}
pvr_file_priv->pPriv = psb_fp;
if (unlikely(dev_priv->bdev.dev_mapping == NULL))
dev_priv->bdev.dev_mapping = dev_priv->dev->dev_mapping;
return 0;
out_err1:
kfree(psb_fp);
out_err0:
(void) drm_release(inode, filp);
return ret;
}
int psb_release(struct inode *inode, struct file *filp)
{
struct psb_fpriv *psb_fp;
struct drm_psb_private *dev_priv;
struct msvdx_private *msvdx_priv;
int ret, island_is_on;
struct drm_file *file_priv = (struct drm_file *) filp->private_data;
#ifdef CONFIG_VIDEO_MRFLD
struct ttm_object_file *tfile = psb_fpriv(file_priv)->tfile;
int i;
struct psb_msvdx_ec_ctx *ec_ctx;
#endif
struct mdfld_dsi_config *dsi_config;
psb_fp = psb_fpriv(file_priv);
dev_priv = psb_priv(file_priv->minor->dev);
#ifdef CONFIG_MDFD_VIDEO_DECODE
msvdx_priv = (struct msvdx_private *)dev_priv->msvdx_private;
#ifdef CONFIG_VIDEO_MRFLD
/*cleanup for msvdx*/
#if 0
if (msvdx_priv->tfile == psb_fpriv(file_priv)->tfile) {
msvdx_priv->decoding_err = 0;
msvdx_priv->host_be_opp_enabled = 0;
memset(&msvdx_priv->frame_info, 0, sizeof(struct drm_psb_msvdx_frame_info) * MAX_DECODE_BUFFERS);
}
#endif
if (msvdx_priv->msvdx_ec_ctx[0] != NULL) {
for (i = 0; i < PSB_MAX_EC_INSTANCE; i++) {
if (msvdx_priv->msvdx_ec_ctx[i]->tfile == tfile)
break;
}
if (i < PSB_MAX_EC_INSTANCE) {
ec_ctx = msvdx_priv->msvdx_ec_ctx[i];
printk(KERN_DEBUG "remove ec ctx with tfile 0x%08x\n",
ec_ctx->tfile);
ec_ctx->tfile = NULL;
ec_ctx->fence = PSB_MSVDX_INVALID_FENCE;
}
}
#endif
ttm_object_file_release(&psb_fp->tfile);
#endif
if (psb_fp->dsr_blocked) {
dsi_config = dev_priv->dsi_configs[0];
mdfld_dsi_dsr_allow(dsi_config);
}
kfree(psb_fp);
#ifdef CONFIG_MDFD_VIDEO_DECODE
/* remove video context */
psb_remove_videoctx(dev_priv, filp);
#ifdef PSB_DRAM_SELF_REFRESH
/* FIXME: workaround for MRST HSD3469585
* re-enable DRAM Self Refresh Mode
* by setting DUNIT.DPMC0
*/
int ui32_reg_value = intel_mid_msgbus_read32_raw((0xD0 << 24) |
(0x1 << 16) | (0x4 << 8) | 0xF0);
intel_mid_msgbus_write32_raw((0xE0 << 24) | (0x1 << 16) |
(0x4 << 8) | 0xF0, ui32_reg_value | (0x1 << 7));
#endif
if (IS_MDFLD(dev_priv->dev)) {
struct pnw_topaz_private *topaz_priv =
(struct pnw_topaz_private *)dev_priv->topaz_private;
if (drm_topaz_pmpolicy == PSB_PMPOLICY_POWERDOWN)
schedule_delayed_work(&topaz_priv->topaz_suspend_wq,
msecs_to_jiffies(10));
}
island_is_on = ospm_power_is_hw_on(OSPM_VIDEO_DEC_ISLAND);
if ((drm_msvdx_pmpolicy == PSB_PMPOLICY_POWERDOWN) && island_is_on) {
PSB_DEBUG_PM("MSVDX: psb_release schedule msvdx suspend.\n");
schedule_delayed_work(&msvdx_priv->msvdx_suspend_wq,
msecs_to_jiffies(10));
}
#endif
ret = drm_release(inode, filp);
return ret;
}
/**
* if vm_pgoff < DRM_PSB_FILE_PAGE_OFFSET call directly to PVRMMap
*/
int psb_mmap(struct file *filp, struct vm_area_struct *vma)
{
struct drm_file *file_priv;
struct drm_psb_private *dev_priv;
int ret;
file_priv = (struct drm_file *) filp->private_data;
dev_priv = psb_priv(file_priv->minor->dev);
if (vma->vm_pgoff < DRM_PSB_FILE_PAGE_OFFSET ||
vma->vm_pgoff > 2 * DRM_PSB_FILE_PAGE_OFFSET) {
BUG_ON(!dev_priv->pvr_ops);
return dev_priv->pvr_ops->PVRMMap(filp, vma);
}
ret = ttm_bo_mmap(filp, vma, &dev_priv->bdev);
if (unlikely(ret != 0))
return ret;
if (unlikely(dev_priv->ttm_vm_ops == NULL)) {
dev_priv->ttm_vm_ops = (struct vm_operations_struct *)vma->vm_ops;
psb_ttm_vm_ops = *vma->vm_ops;
#ifdef CONFIG_MDFD_VIDEO_DECODE
psb_ttm_vm_ops.fault = &psb_ttm_fault;
#endif
}
vma->vm_ops = &psb_ttm_vm_ops;
return 0;
}
static const struct file_operations driver_psb_fops = {
.owner = THIS_MODULE,
.open = psb_open,
.release = psb_release,
.unlocked_ioctl = psb_unlocked_ioctl,
.mmap = psb_mmap,
.poll = psb_poll,
.fasync = drm_fasync,
.read = drm_read,
};
static struct drm_driver driver = {
.driver_features = DRIVER_HAVE_IRQ | DRIVER_IRQ_SHARED | \
DRIVER_IRQ_VBL | DRIVER_MODESET,
.load = psb_driver_load,
.unload = psb_driver_unload,
.ioctls = psb_ioctls,
.num_ioctls = DRM_ARRAY_SIZE(psb_ioctls),
.device_is_agp = psb_driver_device_is_agp,
.irq_preinstall = psb_irq_preinstall,
.irq_postinstall = psb_irq_postinstall,
.irq_uninstall = psb_irq_uninstall,
.irq_handler = psb_irq_handler,
.enable_vblank = psb_enable_vblank,
.disable_vblank = psb_disable_vblank,
.get_vblank_counter = psb_get_vblank_counter,
.firstopen = NULL,
.lastclose = psb_lastclose,
.open = psb_driver_open,
.postclose = PVRSRVDrmPostClose2,
#ifdef DISPLAY_DRIVER_DEBUG_INTERFACE
.debugfs_init = psb_proc_init,
.debugfs_cleanup = psb_proc_cleanup,
#else
.debugfs_init = NULL,
.debugfs_cleanup = NULL,
#endif
.preclose = psb_driver_preclose,
.fops = &driver_psb_fops,
.name = DRIVER_NAME,
.desc = DRIVER_DESC,
.date = PSB_DRM_DRIVER_DATE,
.major = PSB_DRM_DRIVER_MAJOR,
.minor = PSB_DRM_DRIVER_MINOR,
.patchlevel = PSB_DRM_DRIVER_PATCHLEVEL
};
static struct pci_driver psb_pci_driver = {
.name = DRIVER_NAME,
.id_table = pciidlist,
.probe = psb_probe,
.remove = psb_remove,
#ifdef CONFIG_PM
.driver.pm = &psb_pm_ops,
#endif
};
static int psb_probe(struct pci_dev *pdev, const struct pci_device_id *ent)
{
return drm_get_pci_dev(pdev, ent, &driver);
}
#ifndef MODULE
static __init int parse_panelid(char *arg)
{
/* panel ID can be passed in as a cmdline parameter */
/* to enable this feature add panelid=TMD to cmdline for TMD panel*/
if (!arg)
return -EINVAL;
return 0;
}
early_param("panelid", parse_panelid);
#endif
#ifndef MODULE
static __init int parse_hdmi_edid(char *arg)
{
/* HDMI EDID info can be passed in as a cmdline parameter,
* and need to remove it after we can get EDID info via MSIC.*/
if ((!arg) || (strlen(arg) >= HDMI_MONITOR_NAME_LENGTH))
return -EINVAL;
strncpy(HDMI_EDID, arg, HDMI_MONITOR_NAME_LENGTH - 1);
return 0;
}
early_param("hdmi_edid", parse_hdmi_edid);
#endif
static int __init psb_init(void)
{
int ret;
#if defined(MODULE) && defined(CONFIG_NET)
#ifdef CONFIG_SUPPORT_HDMI
psb_kobject_uevent_init();
#endif
#endif
#if 0
/* delay this until PVRSRVDrmLoad is to be loaded */
ret = SYSPVRInit();
if (ret != 0) {
return ret;
}
#endif
ret = drm_pci_init(&driver, &psb_pci_driver);
if (ret != 0) {
return ret;
}
#ifdef CONFIG_SUPPORT_HDMI
if (gpDrmDevice) {
struct drm_psb_private *dev_priv = NULL;
dev_priv = (struct drm_psb_private *)gpDrmDevice->dev_private;
if (dev_priv)
otm_hdmi_hpd_init();
}
#endif
#ifdef CONFIG_SUPPORT_MIPI_H8C7_CMD_DISPLAY
ret = platform_driver_register(&h8c7_lcd_driver);
if (ret != 0) {
return ret;
}
#endif
#ifdef CONFIG_SUPPORT_VB_MIPI_DISPLAY
ret = platform_driver_register(&vb_lcd_driver);
if (ret != 0) {
return ret;
}
#endif
#ifdef CONFIG_R63311_MIPI_VIDEO_MODE
ret = platform_driver_register(&jdi_r63311_lcd_driver);
if (ret != 0) {
return ret;
}
#endif
#ifdef CONFIG_SUPPORT_TMD_MIPI_600X1024_DISPLAY
ret = platform_driver_register(&tmd_lcd_driver);
#endif
#ifdef CONFIG_SUPPORT_OTM8018B_MIPI_480X854_DISPLAY
ret = platform_driver_register(&pf450cl_vid_lcd_driver);
#endif
return ret;
}
static void __exit psb_exit(void)
{
#ifdef CONFIG_SUPPORT_HDMI
if (gpDrmDevice) {
struct drm_psb_private *dev_priv = NULL;
dev_priv = (struct drm_psb_private *)gpDrmDevice->dev_private;
if (dev_priv)
otm_hdmi_hpd_deinit();
}
#endif
drm_pci_exit(&driver, &psb_pci_driver);
}
module_init(psb_init);
module_exit(psb_exit);
MODULE_AUTHOR(DRIVER_AUTHOR);
MODULE_DESCRIPTION(DRIVER_DESC);
MODULE_LICENSE("GPL");