18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Copyright (C) STMicroelectronics SA 2014 48c2ecf20Sopenharmony_ci * Author: Fabien Dessenne <fabien.dessenne@st.com> for STMicroelectronics. 58c2ecf20Sopenharmony_ci */ 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/clk.h> 88c2ecf20Sopenharmony_ci#include <linux/component.h> 98c2ecf20Sopenharmony_ci#include <linux/io.h> 108c2ecf20Sopenharmony_ci#include <linux/module.h> 118c2ecf20Sopenharmony_ci#include <linux/platform_device.h> 128c2ecf20Sopenharmony_ci#include <linux/seq_file.h> 138c2ecf20Sopenharmony_ci 148c2ecf20Sopenharmony_ci#include <drm/drm_atomic_helper.h> 158c2ecf20Sopenharmony_ci#include <drm/drm_bridge.h> 168c2ecf20Sopenharmony_ci#include <drm/drm_debugfs.h> 178c2ecf20Sopenharmony_ci#include <drm/drm_device.h> 188c2ecf20Sopenharmony_ci#include <drm/drm_file.h> 198c2ecf20Sopenharmony_ci#include <drm/drm_print.h> 208c2ecf20Sopenharmony_ci#include <drm/drm_probe_helper.h> 218c2ecf20Sopenharmony_ci 228c2ecf20Sopenharmony_ci/* HDformatter registers */ 238c2ecf20Sopenharmony_ci#define HDA_ANA_CFG 0x0000 248c2ecf20Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_Y 0x0004 258c2ecf20Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_CB 0x0008 268c2ecf20Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_CR 0x000C 278c2ecf20Sopenharmony_ci#define HDA_ANA_ANC_CTRL 0x0010 288c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG 0x0014 298c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH1_TAP123 0x0018 308c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH1_TAP456 0x001C 318c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH2_TAP123 0x0020 328c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH2_TAP456 0x0024 338c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH3_TAP123 0x0028 348c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH3_TAP456 0x002C 358c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH4_TAP123 0x0030 368c2ecf20Sopenharmony_ci#define HDA_COEFF_Y_PH4_TAP456 0x0034 378c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_C_CFG 0x0040 388c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH1_TAP123 0x0044 398c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH1_TAP456 0x0048 408c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH2_TAP123 0x004C 418c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH2_TAP456 0x0050 428c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH3_TAP123 0x0054 438c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH3_TAP456 0x0058 448c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH4_TAP123 0x005C 458c2ecf20Sopenharmony_ci#define HDA_COEFF_C_PH4_TAP456 0x0060 468c2ecf20Sopenharmony_ci#define HDA_SYNC_AWGI 0x0300 478c2ecf20Sopenharmony_ci 488c2ecf20Sopenharmony_ci/* HDA_ANA_CFG */ 498c2ecf20Sopenharmony_ci#define CFG_AWG_ASYNC_EN BIT(0) 508c2ecf20Sopenharmony_ci#define CFG_AWG_ASYNC_HSYNC_MTD BIT(1) 518c2ecf20Sopenharmony_ci#define CFG_AWG_ASYNC_VSYNC_MTD BIT(2) 528c2ecf20Sopenharmony_ci#define CFG_AWG_SYNC_DEL BIT(3) 538c2ecf20Sopenharmony_ci#define CFG_AWG_FLTR_MODE_SHIFT 4 548c2ecf20Sopenharmony_ci#define CFG_AWG_FLTR_MODE_MASK (0xF << CFG_AWG_FLTR_MODE_SHIFT) 558c2ecf20Sopenharmony_ci#define CFG_AWG_FLTR_MODE_SD (0 << CFG_AWG_FLTR_MODE_SHIFT) 568c2ecf20Sopenharmony_ci#define CFG_AWG_FLTR_MODE_ED (1 << CFG_AWG_FLTR_MODE_SHIFT) 578c2ecf20Sopenharmony_ci#define CFG_AWG_FLTR_MODE_HD (2 << CFG_AWG_FLTR_MODE_SHIFT) 588c2ecf20Sopenharmony_ci#define CFG_SYNC_ON_PBPR_MASK BIT(8) 598c2ecf20Sopenharmony_ci#define CFG_PREFILTER_EN_MASK BIT(9) 608c2ecf20Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_SHIFT 16 618c2ecf20Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_MASK (0x7FF << CFG_PBPR_SYNC_OFF_SHIFT) 628c2ecf20Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_VAL 0x117 /* Voltage dependent. stiH416 */ 638c2ecf20Sopenharmony_ci 648c2ecf20Sopenharmony_ci/* Default scaling values */ 658c2ecf20Sopenharmony_ci#define SCALE_CTRL_Y_DFLT 0x00C50256 668c2ecf20Sopenharmony_ci#define SCALE_CTRL_CB_DFLT 0x00DB0249 678c2ecf20Sopenharmony_ci#define SCALE_CTRL_CR_DFLT 0x00DB0249 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_ci/* Video DACs control */ 708c2ecf20Sopenharmony_ci#define DAC_CFG_HD_HZUVW_OFF_MASK BIT(1) 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_ci/* Upsampler values for the alternative 2X Filter */ 738c2ecf20Sopenharmony_ci#define SAMPLER_COEF_NB 8 748c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG_ALT_2X 0x01130000 758c2ecf20Sopenharmony_cistatic u32 coef_y_alt_2x[] = { 768c2ecf20Sopenharmony_ci 0x00FE83FB, 0x1F900401, 0x00000000, 0x00000000, 778c2ecf20Sopenharmony_ci 0x00F408F9, 0x055F7C25, 0x00000000, 0x00000000 788c2ecf20Sopenharmony_ci}; 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_C_CFG_ALT_2X 0x01750004 818c2ecf20Sopenharmony_cistatic u32 coef_c_alt_2x[] = { 828c2ecf20Sopenharmony_ci 0x001305F7, 0x05274BD0, 0x00000000, 0x00000000, 838c2ecf20Sopenharmony_ci 0x0004907C, 0x09C80B9D, 0x00000000, 0x00000000 848c2ecf20Sopenharmony_ci}; 858c2ecf20Sopenharmony_ci 868c2ecf20Sopenharmony_ci/* Upsampler values for the 4X Filter */ 878c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG_4X 0x01ED0005 888c2ecf20Sopenharmony_ci#define HDA_ANA_SRC_C_CFG_4X 0x01ED0004 898c2ecf20Sopenharmony_cistatic u32 coef_yc_4x[] = { 908c2ecf20Sopenharmony_ci 0x00FC827F, 0x008FE20B, 0x00F684FC, 0x050F7C24, 918c2ecf20Sopenharmony_ci 0x00F4857C, 0x0A1F402E, 0x00FA027F, 0x0E076E1D 928c2ecf20Sopenharmony_ci}; 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci/* AWG instructions for some video modes */ 958c2ecf20Sopenharmony_ci#define AWG_MAX_INST 64 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci/* 720p@50 */ 988c2ecf20Sopenharmony_cistatic u32 AWGi_720p_50[] = { 998c2ecf20Sopenharmony_ci 0x00000971, 0x00000C26, 0x0000013B, 0x00000CDA, 1008c2ecf20Sopenharmony_ci 0x00000104, 0x00000E7E, 0x00000E7F, 0x0000013B, 1018c2ecf20Sopenharmony_ci 0x00000D8E, 0x00000104, 0x00001804, 0x00000971, 1028c2ecf20Sopenharmony_ci 0x00000C26, 0x0000003B, 0x00000FB4, 0x00000FB5, 1038c2ecf20Sopenharmony_ci 0x00000104, 0x00001AE8 1048c2ecf20Sopenharmony_ci}; 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci#define NN_720p_50 ARRAY_SIZE(AWGi_720p_50) 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci/* 720p@60 */ 1098c2ecf20Sopenharmony_cistatic u32 AWGi_720p_60[] = { 1108c2ecf20Sopenharmony_ci 0x00000971, 0x00000C26, 0x0000013B, 0x00000CDA, 1118c2ecf20Sopenharmony_ci 0x00000104, 0x00000E7E, 0x00000E7F, 0x0000013B, 1128c2ecf20Sopenharmony_ci 0x00000C44, 0x00000104, 0x00001804, 0x00000971, 1138c2ecf20Sopenharmony_ci 0x00000C26, 0x0000003B, 0x00000F0F, 0x00000F10, 1148c2ecf20Sopenharmony_ci 0x00000104, 0x00001AE8 1158c2ecf20Sopenharmony_ci}; 1168c2ecf20Sopenharmony_ci 1178c2ecf20Sopenharmony_ci#define NN_720p_60 ARRAY_SIZE(AWGi_720p_60) 1188c2ecf20Sopenharmony_ci 1198c2ecf20Sopenharmony_ci/* 1080p@30 */ 1208c2ecf20Sopenharmony_cistatic u32 AWGi_1080p_30[] = { 1218c2ecf20Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 1228c2ecf20Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 1238c2ecf20Sopenharmony_ci 0x00000C2A, 0x00000104, 0x00001804, 0x00000971, 1248c2ecf20Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000EBE, 0x00000EBF, 1258c2ecf20Sopenharmony_ci 0x00000EBF, 0x00000104, 0x00001A2F, 0x00001C4B, 1268c2ecf20Sopenharmony_ci 0x00001C52 1278c2ecf20Sopenharmony_ci}; 1288c2ecf20Sopenharmony_ci 1298c2ecf20Sopenharmony_ci#define NN_1080p_30 ARRAY_SIZE(AWGi_1080p_30) 1308c2ecf20Sopenharmony_ci 1318c2ecf20Sopenharmony_ci/* 1080p@25 */ 1328c2ecf20Sopenharmony_cistatic u32 AWGi_1080p_25[] = { 1338c2ecf20Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 1348c2ecf20Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 1358c2ecf20Sopenharmony_ci 0x00000DE2, 0x00000104, 0x00001804, 0x00000971, 1368c2ecf20Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000F51, 0x00000F51, 1378c2ecf20Sopenharmony_ci 0x00000F52, 0x00000104, 0x00001A2F, 0x00001C4B, 1388c2ecf20Sopenharmony_ci 0x00001C52 1398c2ecf20Sopenharmony_ci}; 1408c2ecf20Sopenharmony_ci 1418c2ecf20Sopenharmony_ci#define NN_1080p_25 ARRAY_SIZE(AWGi_1080p_25) 1428c2ecf20Sopenharmony_ci 1438c2ecf20Sopenharmony_ci/* 1080p@24 */ 1448c2ecf20Sopenharmony_cistatic u32 AWGi_1080p_24[] = { 1458c2ecf20Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 1468c2ecf20Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 1478c2ecf20Sopenharmony_ci 0x00000E50, 0x00000104, 0x00001804, 0x00000971, 1488c2ecf20Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000F76, 0x00000F76, 1498c2ecf20Sopenharmony_ci 0x00000F76, 0x00000104, 0x00001A2F, 0x00001C4B, 1508c2ecf20Sopenharmony_ci 0x00001C52 1518c2ecf20Sopenharmony_ci}; 1528c2ecf20Sopenharmony_ci 1538c2ecf20Sopenharmony_ci#define NN_1080p_24 ARRAY_SIZE(AWGi_1080p_24) 1548c2ecf20Sopenharmony_ci 1558c2ecf20Sopenharmony_ci/* 720x480p@60 */ 1568c2ecf20Sopenharmony_cistatic u32 AWGi_720x480p_60[] = { 1578c2ecf20Sopenharmony_ci 0x00000904, 0x00000F18, 0x0000013B, 0x00001805, 1588c2ecf20Sopenharmony_ci 0x00000904, 0x00000C3D, 0x0000003B, 0x00001A06 1598c2ecf20Sopenharmony_ci}; 1608c2ecf20Sopenharmony_ci 1618c2ecf20Sopenharmony_ci#define NN_720x480p_60 ARRAY_SIZE(AWGi_720x480p_60) 1628c2ecf20Sopenharmony_ci 1638c2ecf20Sopenharmony_ci/* Video mode category */ 1648c2ecf20Sopenharmony_cienum sti_hda_vid_cat { 1658c2ecf20Sopenharmony_ci VID_SD, 1668c2ecf20Sopenharmony_ci VID_ED, 1678c2ecf20Sopenharmony_ci VID_HD_74M, 1688c2ecf20Sopenharmony_ci VID_HD_148M 1698c2ecf20Sopenharmony_ci}; 1708c2ecf20Sopenharmony_ci 1718c2ecf20Sopenharmony_cistruct sti_hda_video_config { 1728c2ecf20Sopenharmony_ci struct drm_display_mode mode; 1738c2ecf20Sopenharmony_ci u32 *awg_instr; 1748c2ecf20Sopenharmony_ci int nb_instr; 1758c2ecf20Sopenharmony_ci enum sti_hda_vid_cat vid_cat; 1768c2ecf20Sopenharmony_ci}; 1778c2ecf20Sopenharmony_ci 1788c2ecf20Sopenharmony_ci/* HD analog supported modes 1798c2ecf20Sopenharmony_ci * Interlaced modes may be added when supported by the whole display chain 1808c2ecf20Sopenharmony_ci */ 1818c2ecf20Sopenharmony_cistatic const struct sti_hda_video_config hda_supported_modes[] = { 1828c2ecf20Sopenharmony_ci /* 1080p30 74.250Mhz */ 1838c2ecf20Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2008, 1848c2ecf20Sopenharmony_ci 2052, 2200, 0, 1080, 1084, 1089, 1125, 0, 1858c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 1868c2ecf20Sopenharmony_ci AWGi_1080p_30, NN_1080p_30, VID_HD_74M}, 1878c2ecf20Sopenharmony_ci /* 1080p30 74.176Mhz */ 1888c2ecf20Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74176, 1920, 2008, 1898c2ecf20Sopenharmony_ci 2052, 2200, 0, 1080, 1084, 1089, 1125, 0, 1908c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 1918c2ecf20Sopenharmony_ci AWGi_1080p_30, NN_1080p_30, VID_HD_74M}, 1928c2ecf20Sopenharmony_ci /* 1080p24 74.250Mhz */ 1938c2ecf20Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2558, 1948c2ecf20Sopenharmony_ci 2602, 2750, 0, 1080, 1084, 1089, 1125, 0, 1958c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 1968c2ecf20Sopenharmony_ci AWGi_1080p_24, NN_1080p_24, VID_HD_74M}, 1978c2ecf20Sopenharmony_ci /* 1080p24 74.176Mhz */ 1988c2ecf20Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74176, 1920, 2558, 1998c2ecf20Sopenharmony_ci 2602, 2750, 0, 1080, 1084, 1089, 1125, 0, 2008c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 2018c2ecf20Sopenharmony_ci AWGi_1080p_24, NN_1080p_24, VID_HD_74M}, 2028c2ecf20Sopenharmony_ci /* 1080p25 74.250Mhz */ 2038c2ecf20Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2448, 2048c2ecf20Sopenharmony_ci 2492, 2640, 0, 1080, 1084, 1089, 1125, 0, 2058c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 2068c2ecf20Sopenharmony_ci AWGi_1080p_25, NN_1080p_25, VID_HD_74M}, 2078c2ecf20Sopenharmony_ci /* 720p60 74.250Mhz */ 2088c2ecf20Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74250, 1280, 1390, 2098c2ecf20Sopenharmony_ci 1430, 1650, 0, 720, 725, 730, 750, 0, 2108c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 2118c2ecf20Sopenharmony_ci AWGi_720p_60, NN_720p_60, VID_HD_74M}, 2128c2ecf20Sopenharmony_ci /* 720p60 74.176Mhz */ 2138c2ecf20Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74176, 1280, 1390, 2148c2ecf20Sopenharmony_ci 1430, 1650, 0, 720, 725, 730, 750, 0, 2158c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 2168c2ecf20Sopenharmony_ci AWGi_720p_60, NN_720p_60, VID_HD_74M}, 2178c2ecf20Sopenharmony_ci /* 720p50 74.250Mhz */ 2188c2ecf20Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74250, 1280, 1720, 2198c2ecf20Sopenharmony_ci 1760, 1980, 0, 720, 725, 730, 750, 0, 2208c2ecf20Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 2218c2ecf20Sopenharmony_ci AWGi_720p_50, NN_720p_50, VID_HD_74M}, 2228c2ecf20Sopenharmony_ci /* 720x480p60 27.027Mhz */ 2238c2ecf20Sopenharmony_ci {{DRM_MODE("720x480", DRM_MODE_TYPE_DRIVER, 27027, 720, 736, 2248c2ecf20Sopenharmony_ci 798, 858, 0, 480, 489, 495, 525, 0, 2258c2ecf20Sopenharmony_ci DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC)}, 2268c2ecf20Sopenharmony_ci AWGi_720x480p_60, NN_720x480p_60, VID_ED}, 2278c2ecf20Sopenharmony_ci /* 720x480p60 27.000Mhz */ 2288c2ecf20Sopenharmony_ci {{DRM_MODE("720x480", DRM_MODE_TYPE_DRIVER, 27000, 720, 736, 2298c2ecf20Sopenharmony_ci 798, 858, 0, 480, 489, 495, 525, 0, 2308c2ecf20Sopenharmony_ci DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC)}, 2318c2ecf20Sopenharmony_ci AWGi_720x480p_60, NN_720x480p_60, VID_ED} 2328c2ecf20Sopenharmony_ci}; 2338c2ecf20Sopenharmony_ci 2348c2ecf20Sopenharmony_ci/* 2358c2ecf20Sopenharmony_ci * STI hd analog structure 2368c2ecf20Sopenharmony_ci * 2378c2ecf20Sopenharmony_ci * @dev: driver device 2388c2ecf20Sopenharmony_ci * @drm_dev: pointer to drm device 2398c2ecf20Sopenharmony_ci * @mode: current display mode selected 2408c2ecf20Sopenharmony_ci * @regs: HD analog register 2418c2ecf20Sopenharmony_ci * @video_dacs_ctrl: video DACS control register 2428c2ecf20Sopenharmony_ci * @enabled: true if HD analog is enabled else false 2438c2ecf20Sopenharmony_ci */ 2448c2ecf20Sopenharmony_cistruct sti_hda { 2458c2ecf20Sopenharmony_ci struct device dev; 2468c2ecf20Sopenharmony_ci struct drm_device *drm_dev; 2478c2ecf20Sopenharmony_ci struct drm_display_mode mode; 2488c2ecf20Sopenharmony_ci void __iomem *regs; 2498c2ecf20Sopenharmony_ci void __iomem *video_dacs_ctrl; 2508c2ecf20Sopenharmony_ci struct clk *clk_pix; 2518c2ecf20Sopenharmony_ci struct clk *clk_hddac; 2528c2ecf20Sopenharmony_ci bool enabled; 2538c2ecf20Sopenharmony_ci}; 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_cistruct sti_hda_connector { 2568c2ecf20Sopenharmony_ci struct drm_connector drm_connector; 2578c2ecf20Sopenharmony_ci struct drm_encoder *encoder; 2588c2ecf20Sopenharmony_ci struct sti_hda *hda; 2598c2ecf20Sopenharmony_ci}; 2608c2ecf20Sopenharmony_ci 2618c2ecf20Sopenharmony_ci#define to_sti_hda_connector(x) \ 2628c2ecf20Sopenharmony_ci container_of(x, struct sti_hda_connector, drm_connector) 2638c2ecf20Sopenharmony_ci 2648c2ecf20Sopenharmony_cistatic u32 hda_read(struct sti_hda *hda, int offset) 2658c2ecf20Sopenharmony_ci{ 2668c2ecf20Sopenharmony_ci return readl(hda->regs + offset); 2678c2ecf20Sopenharmony_ci} 2688c2ecf20Sopenharmony_ci 2698c2ecf20Sopenharmony_cistatic void hda_write(struct sti_hda *hda, u32 val, int offset) 2708c2ecf20Sopenharmony_ci{ 2718c2ecf20Sopenharmony_ci writel(val, hda->regs + offset); 2728c2ecf20Sopenharmony_ci} 2738c2ecf20Sopenharmony_ci 2748c2ecf20Sopenharmony_ci/** 2758c2ecf20Sopenharmony_ci * Search for a video mode in the supported modes table 2768c2ecf20Sopenharmony_ci * 2778c2ecf20Sopenharmony_ci * @mode: mode being searched 2788c2ecf20Sopenharmony_ci * @idx: index of the found mode 2798c2ecf20Sopenharmony_ci * 2808c2ecf20Sopenharmony_ci * Return true if mode is found 2818c2ecf20Sopenharmony_ci */ 2828c2ecf20Sopenharmony_cistatic bool hda_get_mode_idx(struct drm_display_mode mode, int *idx) 2838c2ecf20Sopenharmony_ci{ 2848c2ecf20Sopenharmony_ci unsigned int i; 2858c2ecf20Sopenharmony_ci 2868c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_supported_modes); i++) 2878c2ecf20Sopenharmony_ci if (drm_mode_equal(&hda_supported_modes[i].mode, &mode)) { 2888c2ecf20Sopenharmony_ci *idx = i; 2898c2ecf20Sopenharmony_ci return true; 2908c2ecf20Sopenharmony_ci } 2918c2ecf20Sopenharmony_ci return false; 2928c2ecf20Sopenharmony_ci} 2938c2ecf20Sopenharmony_ci 2948c2ecf20Sopenharmony_ci/** 2958c2ecf20Sopenharmony_ci * Enable the HD DACS 2968c2ecf20Sopenharmony_ci * 2978c2ecf20Sopenharmony_ci * @hda: pointer to HD analog structure 2988c2ecf20Sopenharmony_ci * @enable: true if HD DACS need to be enabled, else false 2998c2ecf20Sopenharmony_ci */ 3008c2ecf20Sopenharmony_cistatic void hda_enable_hd_dacs(struct sti_hda *hda, bool enable) 3018c2ecf20Sopenharmony_ci{ 3028c2ecf20Sopenharmony_ci if (hda->video_dacs_ctrl) { 3038c2ecf20Sopenharmony_ci u32 val; 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_ci val = readl(hda->video_dacs_ctrl); 3068c2ecf20Sopenharmony_ci if (enable) 3078c2ecf20Sopenharmony_ci val &= ~DAC_CFG_HD_HZUVW_OFF_MASK; 3088c2ecf20Sopenharmony_ci else 3098c2ecf20Sopenharmony_ci val |= DAC_CFG_HD_HZUVW_OFF_MASK; 3108c2ecf20Sopenharmony_ci 3118c2ecf20Sopenharmony_ci writel(val, hda->video_dacs_ctrl); 3128c2ecf20Sopenharmony_ci } 3138c2ecf20Sopenharmony_ci} 3148c2ecf20Sopenharmony_ci 3158c2ecf20Sopenharmony_ci#define DBGFS_DUMP(reg) seq_printf(s, "\n %-25s 0x%08X", #reg, \ 3168c2ecf20Sopenharmony_ci readl(hda->regs + reg)) 3178c2ecf20Sopenharmony_ci 3188c2ecf20Sopenharmony_cistatic void hda_dbg_cfg(struct seq_file *s, int val) 3198c2ecf20Sopenharmony_ci{ 3208c2ecf20Sopenharmony_ci seq_puts(s, "\tAWG "); 3218c2ecf20Sopenharmony_ci seq_puts(s, val & CFG_AWG_ASYNC_EN ? "enabled" : "disabled"); 3228c2ecf20Sopenharmony_ci} 3238c2ecf20Sopenharmony_ci 3248c2ecf20Sopenharmony_cistatic void hda_dbg_awg_microcode(struct seq_file *s, void __iomem *reg) 3258c2ecf20Sopenharmony_ci{ 3268c2ecf20Sopenharmony_ci unsigned int i; 3278c2ecf20Sopenharmony_ci 3288c2ecf20Sopenharmony_ci seq_puts(s, "\n\n HDA AWG microcode:"); 3298c2ecf20Sopenharmony_ci for (i = 0; i < AWG_MAX_INST; i++) { 3308c2ecf20Sopenharmony_ci if (i % 8 == 0) 3318c2ecf20Sopenharmony_ci seq_printf(s, "\n %04X:", i); 3328c2ecf20Sopenharmony_ci seq_printf(s, " %04X", readl(reg + i * 4)); 3338c2ecf20Sopenharmony_ci } 3348c2ecf20Sopenharmony_ci} 3358c2ecf20Sopenharmony_ci 3368c2ecf20Sopenharmony_cistatic void hda_dbg_video_dacs_ctrl(struct seq_file *s, void __iomem *reg) 3378c2ecf20Sopenharmony_ci{ 3388c2ecf20Sopenharmony_ci u32 val = readl(reg); 3398c2ecf20Sopenharmony_ci 3408c2ecf20Sopenharmony_ci seq_printf(s, "\n\n %-25s 0x%08X", "VIDEO_DACS_CONTROL", val); 3418c2ecf20Sopenharmony_ci seq_puts(s, "\tHD DACs "); 3428c2ecf20Sopenharmony_ci seq_puts(s, val & DAC_CFG_HD_HZUVW_OFF_MASK ? "disabled" : "enabled"); 3438c2ecf20Sopenharmony_ci} 3448c2ecf20Sopenharmony_ci 3458c2ecf20Sopenharmony_cistatic int hda_dbg_show(struct seq_file *s, void *data) 3468c2ecf20Sopenharmony_ci{ 3478c2ecf20Sopenharmony_ci struct drm_info_node *node = s->private; 3488c2ecf20Sopenharmony_ci struct sti_hda *hda = (struct sti_hda *)node->info_ent->data; 3498c2ecf20Sopenharmony_ci 3508c2ecf20Sopenharmony_ci seq_printf(s, "HD Analog: (vaddr = 0x%p)", hda->regs); 3518c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_CFG); 3528c2ecf20Sopenharmony_ci hda_dbg_cfg(s, readl(hda->regs + HDA_ANA_CFG)); 3538c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_Y); 3548c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_CB); 3558c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_CR); 3568c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_ANC_CTRL); 3578c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SRC_Y_CFG); 3588c2ecf20Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SRC_C_CFG); 3598c2ecf20Sopenharmony_ci hda_dbg_awg_microcode(s, hda->regs + HDA_SYNC_AWGI); 3608c2ecf20Sopenharmony_ci if (hda->video_dacs_ctrl) 3618c2ecf20Sopenharmony_ci hda_dbg_video_dacs_ctrl(s, hda->video_dacs_ctrl); 3628c2ecf20Sopenharmony_ci seq_putc(s, '\n'); 3638c2ecf20Sopenharmony_ci return 0; 3648c2ecf20Sopenharmony_ci} 3658c2ecf20Sopenharmony_ci 3668c2ecf20Sopenharmony_cistatic struct drm_info_list hda_debugfs_files[] = { 3678c2ecf20Sopenharmony_ci { "hda", hda_dbg_show, 0, NULL }, 3688c2ecf20Sopenharmony_ci}; 3698c2ecf20Sopenharmony_ci 3708c2ecf20Sopenharmony_cistatic void hda_debugfs_init(struct sti_hda *hda, struct drm_minor *minor) 3718c2ecf20Sopenharmony_ci{ 3728c2ecf20Sopenharmony_ci unsigned int i; 3738c2ecf20Sopenharmony_ci 3748c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_debugfs_files); i++) 3758c2ecf20Sopenharmony_ci hda_debugfs_files[i].data = hda; 3768c2ecf20Sopenharmony_ci 3778c2ecf20Sopenharmony_ci drm_debugfs_create_files(hda_debugfs_files, 3788c2ecf20Sopenharmony_ci ARRAY_SIZE(hda_debugfs_files), 3798c2ecf20Sopenharmony_ci minor->debugfs_root, minor); 3808c2ecf20Sopenharmony_ci} 3818c2ecf20Sopenharmony_ci 3828c2ecf20Sopenharmony_ci/** 3838c2ecf20Sopenharmony_ci * Configure AWG, writing instructions 3848c2ecf20Sopenharmony_ci * 3858c2ecf20Sopenharmony_ci * @hda: pointer to HD analog structure 3868c2ecf20Sopenharmony_ci * @awg_instr: pointer to AWG instructions table 3878c2ecf20Sopenharmony_ci * @nb: nb of AWG instructions 3888c2ecf20Sopenharmony_ci */ 3898c2ecf20Sopenharmony_cistatic void sti_hda_configure_awg(struct sti_hda *hda, u32 *awg_instr, int nb) 3908c2ecf20Sopenharmony_ci{ 3918c2ecf20Sopenharmony_ci unsigned int i; 3928c2ecf20Sopenharmony_ci 3938c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 3948c2ecf20Sopenharmony_ci 3958c2ecf20Sopenharmony_ci for (i = 0; i < nb; i++) 3968c2ecf20Sopenharmony_ci hda_write(hda, awg_instr[i], HDA_SYNC_AWGI + i * 4); 3978c2ecf20Sopenharmony_ci for (i = nb; i < AWG_MAX_INST; i++) 3988c2ecf20Sopenharmony_ci hda_write(hda, 0, HDA_SYNC_AWGI + i * 4); 3998c2ecf20Sopenharmony_ci} 4008c2ecf20Sopenharmony_ci 4018c2ecf20Sopenharmony_cistatic void sti_hda_disable(struct drm_bridge *bridge) 4028c2ecf20Sopenharmony_ci{ 4038c2ecf20Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 4048c2ecf20Sopenharmony_ci u32 val; 4058c2ecf20Sopenharmony_ci 4068c2ecf20Sopenharmony_ci if (!hda->enabled) 4078c2ecf20Sopenharmony_ci return; 4088c2ecf20Sopenharmony_ci 4098c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 4108c2ecf20Sopenharmony_ci 4118c2ecf20Sopenharmony_ci /* Disable HD DAC and AWG */ 4128c2ecf20Sopenharmony_ci val = hda_read(hda, HDA_ANA_CFG); 4138c2ecf20Sopenharmony_ci val &= ~CFG_AWG_ASYNC_EN; 4148c2ecf20Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 4158c2ecf20Sopenharmony_ci hda_write(hda, 0, HDA_ANA_ANC_CTRL); 4168c2ecf20Sopenharmony_ci 4178c2ecf20Sopenharmony_ci hda_enable_hd_dacs(hda, false); 4188c2ecf20Sopenharmony_ci 4198c2ecf20Sopenharmony_ci /* Disable/unprepare hda clock */ 4208c2ecf20Sopenharmony_ci clk_disable_unprepare(hda->clk_hddac); 4218c2ecf20Sopenharmony_ci clk_disable_unprepare(hda->clk_pix); 4228c2ecf20Sopenharmony_ci 4238c2ecf20Sopenharmony_ci hda->enabled = false; 4248c2ecf20Sopenharmony_ci} 4258c2ecf20Sopenharmony_ci 4268c2ecf20Sopenharmony_cistatic void sti_hda_pre_enable(struct drm_bridge *bridge) 4278c2ecf20Sopenharmony_ci{ 4288c2ecf20Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 4298c2ecf20Sopenharmony_ci u32 val, i, mode_idx; 4308c2ecf20Sopenharmony_ci u32 src_filter_y, src_filter_c; 4318c2ecf20Sopenharmony_ci u32 *coef_y, *coef_c; 4328c2ecf20Sopenharmony_ci u32 filter_mode; 4338c2ecf20Sopenharmony_ci 4348c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 4358c2ecf20Sopenharmony_ci 4368c2ecf20Sopenharmony_ci if (hda->enabled) 4378c2ecf20Sopenharmony_ci return; 4388c2ecf20Sopenharmony_ci 4398c2ecf20Sopenharmony_ci /* Prepare/enable clocks */ 4408c2ecf20Sopenharmony_ci if (clk_prepare_enable(hda->clk_pix)) 4418c2ecf20Sopenharmony_ci DRM_ERROR("Failed to prepare/enable hda_pix clk\n"); 4428c2ecf20Sopenharmony_ci if (clk_prepare_enable(hda->clk_hddac)) 4438c2ecf20Sopenharmony_ci DRM_ERROR("Failed to prepare/enable hda_hddac clk\n"); 4448c2ecf20Sopenharmony_ci 4458c2ecf20Sopenharmony_ci if (!hda_get_mode_idx(hda->mode, &mode_idx)) { 4468c2ecf20Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 4478c2ecf20Sopenharmony_ci return; 4488c2ecf20Sopenharmony_ci } 4498c2ecf20Sopenharmony_ci 4508c2ecf20Sopenharmony_ci switch (hda_supported_modes[mode_idx].vid_cat) { 4518c2ecf20Sopenharmony_ci case VID_HD_148M: 4528c2ecf20Sopenharmony_ci DRM_ERROR("Beyond HD analog capabilities\n"); 4538c2ecf20Sopenharmony_ci return; 4548c2ecf20Sopenharmony_ci case VID_HD_74M: 4558c2ecf20Sopenharmony_ci /* HD use alternate 2x filter */ 4568c2ecf20Sopenharmony_ci filter_mode = CFG_AWG_FLTR_MODE_HD; 4578c2ecf20Sopenharmony_ci src_filter_y = HDA_ANA_SRC_Y_CFG_ALT_2X; 4588c2ecf20Sopenharmony_ci src_filter_c = HDA_ANA_SRC_C_CFG_ALT_2X; 4598c2ecf20Sopenharmony_ci coef_y = coef_y_alt_2x; 4608c2ecf20Sopenharmony_ci coef_c = coef_c_alt_2x; 4618c2ecf20Sopenharmony_ci break; 4628c2ecf20Sopenharmony_ci case VID_ED: 4638c2ecf20Sopenharmony_ci /* ED uses 4x filter */ 4648c2ecf20Sopenharmony_ci filter_mode = CFG_AWG_FLTR_MODE_ED; 4658c2ecf20Sopenharmony_ci src_filter_y = HDA_ANA_SRC_Y_CFG_4X; 4668c2ecf20Sopenharmony_ci src_filter_c = HDA_ANA_SRC_C_CFG_4X; 4678c2ecf20Sopenharmony_ci coef_y = coef_yc_4x; 4688c2ecf20Sopenharmony_ci coef_c = coef_yc_4x; 4698c2ecf20Sopenharmony_ci break; 4708c2ecf20Sopenharmony_ci case VID_SD: 4718c2ecf20Sopenharmony_ci DRM_ERROR("Not supported\n"); 4728c2ecf20Sopenharmony_ci return; 4738c2ecf20Sopenharmony_ci default: 4748c2ecf20Sopenharmony_ci DRM_ERROR("Undefined resolution\n"); 4758c2ecf20Sopenharmony_ci return; 4768c2ecf20Sopenharmony_ci } 4778c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("Using HDA mode #%d\n", mode_idx); 4788c2ecf20Sopenharmony_ci 4798c2ecf20Sopenharmony_ci /* Enable HD Video DACs */ 4808c2ecf20Sopenharmony_ci hda_enable_hd_dacs(hda, true); 4818c2ecf20Sopenharmony_ci 4828c2ecf20Sopenharmony_ci /* Configure scaler */ 4838c2ecf20Sopenharmony_ci hda_write(hda, SCALE_CTRL_Y_DFLT, HDA_ANA_SCALE_CTRL_Y); 4848c2ecf20Sopenharmony_ci hda_write(hda, SCALE_CTRL_CB_DFLT, HDA_ANA_SCALE_CTRL_CB); 4858c2ecf20Sopenharmony_ci hda_write(hda, SCALE_CTRL_CR_DFLT, HDA_ANA_SCALE_CTRL_CR); 4868c2ecf20Sopenharmony_ci 4878c2ecf20Sopenharmony_ci /* Configure sampler */ 4888c2ecf20Sopenharmony_ci hda_write(hda , src_filter_y, HDA_ANA_SRC_Y_CFG); 4898c2ecf20Sopenharmony_ci hda_write(hda, src_filter_c, HDA_ANA_SRC_C_CFG); 4908c2ecf20Sopenharmony_ci for (i = 0; i < SAMPLER_COEF_NB; i++) { 4918c2ecf20Sopenharmony_ci hda_write(hda, coef_y[i], HDA_COEFF_Y_PH1_TAP123 + i * 4); 4928c2ecf20Sopenharmony_ci hda_write(hda, coef_c[i], HDA_COEFF_C_PH1_TAP123 + i * 4); 4938c2ecf20Sopenharmony_ci } 4948c2ecf20Sopenharmony_ci 4958c2ecf20Sopenharmony_ci /* Configure main HDFormatter */ 4968c2ecf20Sopenharmony_ci val = 0; 4978c2ecf20Sopenharmony_ci val |= (hda->mode.flags & DRM_MODE_FLAG_INTERLACE) ? 4988c2ecf20Sopenharmony_ci 0 : CFG_AWG_ASYNC_VSYNC_MTD; 4998c2ecf20Sopenharmony_ci val |= (CFG_PBPR_SYNC_OFF_VAL << CFG_PBPR_SYNC_OFF_SHIFT); 5008c2ecf20Sopenharmony_ci val |= filter_mode; 5018c2ecf20Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 5028c2ecf20Sopenharmony_ci 5038c2ecf20Sopenharmony_ci /* Configure AWG */ 5048c2ecf20Sopenharmony_ci sti_hda_configure_awg(hda, hda_supported_modes[mode_idx].awg_instr, 5058c2ecf20Sopenharmony_ci hda_supported_modes[mode_idx].nb_instr); 5068c2ecf20Sopenharmony_ci 5078c2ecf20Sopenharmony_ci /* Enable AWG */ 5088c2ecf20Sopenharmony_ci val = hda_read(hda, HDA_ANA_CFG); 5098c2ecf20Sopenharmony_ci val |= CFG_AWG_ASYNC_EN; 5108c2ecf20Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 5118c2ecf20Sopenharmony_ci 5128c2ecf20Sopenharmony_ci hda->enabled = true; 5138c2ecf20Sopenharmony_ci} 5148c2ecf20Sopenharmony_ci 5158c2ecf20Sopenharmony_cistatic void sti_hda_set_mode(struct drm_bridge *bridge, 5168c2ecf20Sopenharmony_ci const struct drm_display_mode *mode, 5178c2ecf20Sopenharmony_ci const struct drm_display_mode *adjusted_mode) 5188c2ecf20Sopenharmony_ci{ 5198c2ecf20Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 5208c2ecf20Sopenharmony_ci u32 mode_idx; 5218c2ecf20Sopenharmony_ci int hddac_rate; 5228c2ecf20Sopenharmony_ci int ret; 5238c2ecf20Sopenharmony_ci 5248c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 5258c2ecf20Sopenharmony_ci 5268c2ecf20Sopenharmony_ci drm_mode_copy(&hda->mode, mode); 5278c2ecf20Sopenharmony_ci 5288c2ecf20Sopenharmony_ci if (!hda_get_mode_idx(hda->mode, &mode_idx)) { 5298c2ecf20Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 5308c2ecf20Sopenharmony_ci return; 5318c2ecf20Sopenharmony_ci } 5328c2ecf20Sopenharmony_ci 5338c2ecf20Sopenharmony_ci switch (hda_supported_modes[mode_idx].vid_cat) { 5348c2ecf20Sopenharmony_ci case VID_HD_74M: 5358c2ecf20Sopenharmony_ci /* HD use alternate 2x filter */ 5368c2ecf20Sopenharmony_ci hddac_rate = mode->clock * 1000 * 2; 5378c2ecf20Sopenharmony_ci break; 5388c2ecf20Sopenharmony_ci case VID_ED: 5398c2ecf20Sopenharmony_ci /* ED uses 4x filter */ 5408c2ecf20Sopenharmony_ci hddac_rate = mode->clock * 1000 * 4; 5418c2ecf20Sopenharmony_ci break; 5428c2ecf20Sopenharmony_ci default: 5438c2ecf20Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 5448c2ecf20Sopenharmony_ci return; 5458c2ecf20Sopenharmony_ci } 5468c2ecf20Sopenharmony_ci 5478c2ecf20Sopenharmony_ci /* HD DAC = 148.5Mhz or 108 Mhz */ 5488c2ecf20Sopenharmony_ci ret = clk_set_rate(hda->clk_hddac, hddac_rate); 5498c2ecf20Sopenharmony_ci if (ret < 0) 5508c2ecf20Sopenharmony_ci DRM_ERROR("Cannot set rate (%dHz) for hda_hddac clk\n", 5518c2ecf20Sopenharmony_ci hddac_rate); 5528c2ecf20Sopenharmony_ci 5538c2ecf20Sopenharmony_ci /* HDformatter clock = compositor clock */ 5548c2ecf20Sopenharmony_ci ret = clk_set_rate(hda->clk_pix, mode->clock * 1000); 5558c2ecf20Sopenharmony_ci if (ret < 0) 5568c2ecf20Sopenharmony_ci DRM_ERROR("Cannot set rate (%dHz) for hda_pix clk\n", 5578c2ecf20Sopenharmony_ci mode->clock * 1000); 5588c2ecf20Sopenharmony_ci} 5598c2ecf20Sopenharmony_ci 5608c2ecf20Sopenharmony_cistatic void sti_hda_bridge_nope(struct drm_bridge *bridge) 5618c2ecf20Sopenharmony_ci{ 5628c2ecf20Sopenharmony_ci /* do nothing */ 5638c2ecf20Sopenharmony_ci} 5648c2ecf20Sopenharmony_ci 5658c2ecf20Sopenharmony_cistatic const struct drm_bridge_funcs sti_hda_bridge_funcs = { 5668c2ecf20Sopenharmony_ci .pre_enable = sti_hda_pre_enable, 5678c2ecf20Sopenharmony_ci .enable = sti_hda_bridge_nope, 5688c2ecf20Sopenharmony_ci .disable = sti_hda_disable, 5698c2ecf20Sopenharmony_ci .post_disable = sti_hda_bridge_nope, 5708c2ecf20Sopenharmony_ci .mode_set = sti_hda_set_mode, 5718c2ecf20Sopenharmony_ci}; 5728c2ecf20Sopenharmony_ci 5738c2ecf20Sopenharmony_cistatic int sti_hda_connector_get_modes(struct drm_connector *connector) 5748c2ecf20Sopenharmony_ci{ 5758c2ecf20Sopenharmony_ci unsigned int i; 5768c2ecf20Sopenharmony_ci int count = 0; 5778c2ecf20Sopenharmony_ci struct sti_hda_connector *hda_connector 5788c2ecf20Sopenharmony_ci = to_sti_hda_connector(connector); 5798c2ecf20Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 5808c2ecf20Sopenharmony_ci 5818c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 5828c2ecf20Sopenharmony_ci 5838c2ecf20Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_supported_modes); i++) { 5848c2ecf20Sopenharmony_ci struct drm_display_mode *mode = 5858c2ecf20Sopenharmony_ci drm_mode_duplicate(hda->drm_dev, 5868c2ecf20Sopenharmony_ci &hda_supported_modes[i].mode); 5878c2ecf20Sopenharmony_ci if (!mode) 5888c2ecf20Sopenharmony_ci continue; 5898c2ecf20Sopenharmony_ci 5908c2ecf20Sopenharmony_ci /* the first mode is the preferred mode */ 5918c2ecf20Sopenharmony_ci if (i == 0) 5928c2ecf20Sopenharmony_ci mode->type |= DRM_MODE_TYPE_PREFERRED; 5938c2ecf20Sopenharmony_ci 5948c2ecf20Sopenharmony_ci drm_mode_probed_add(connector, mode); 5958c2ecf20Sopenharmony_ci count++; 5968c2ecf20Sopenharmony_ci } 5978c2ecf20Sopenharmony_ci 5988c2ecf20Sopenharmony_ci return count; 5998c2ecf20Sopenharmony_ci} 6008c2ecf20Sopenharmony_ci 6018c2ecf20Sopenharmony_ci#define CLK_TOLERANCE_HZ 50 6028c2ecf20Sopenharmony_ci 6038c2ecf20Sopenharmony_cistatic enum drm_mode_status 6048c2ecf20Sopenharmony_cisti_hda_connector_mode_valid(struct drm_connector *connector, 6058c2ecf20Sopenharmony_ci struct drm_display_mode *mode) 6068c2ecf20Sopenharmony_ci{ 6078c2ecf20Sopenharmony_ci int target = mode->clock * 1000; 6088c2ecf20Sopenharmony_ci int target_min = target - CLK_TOLERANCE_HZ; 6098c2ecf20Sopenharmony_ci int target_max = target + CLK_TOLERANCE_HZ; 6108c2ecf20Sopenharmony_ci int result; 6118c2ecf20Sopenharmony_ci int idx; 6128c2ecf20Sopenharmony_ci struct sti_hda_connector *hda_connector 6138c2ecf20Sopenharmony_ci = to_sti_hda_connector(connector); 6148c2ecf20Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 6158c2ecf20Sopenharmony_ci 6168c2ecf20Sopenharmony_ci if (!hda_get_mode_idx(*mode, &idx)) { 6178c2ecf20Sopenharmony_ci return MODE_BAD; 6188c2ecf20Sopenharmony_ci } else { 6198c2ecf20Sopenharmony_ci result = clk_round_rate(hda->clk_pix, target); 6208c2ecf20Sopenharmony_ci 6218c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("target rate = %d => available rate = %d\n", 6228c2ecf20Sopenharmony_ci target, result); 6238c2ecf20Sopenharmony_ci 6248c2ecf20Sopenharmony_ci if ((result < target_min) || (result > target_max)) { 6258c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("hda pixclk=%d not supported\n", 6268c2ecf20Sopenharmony_ci target); 6278c2ecf20Sopenharmony_ci return MODE_BAD; 6288c2ecf20Sopenharmony_ci } 6298c2ecf20Sopenharmony_ci } 6308c2ecf20Sopenharmony_ci 6318c2ecf20Sopenharmony_ci return MODE_OK; 6328c2ecf20Sopenharmony_ci} 6338c2ecf20Sopenharmony_ci 6348c2ecf20Sopenharmony_cistatic const 6358c2ecf20Sopenharmony_cistruct drm_connector_helper_funcs sti_hda_connector_helper_funcs = { 6368c2ecf20Sopenharmony_ci .get_modes = sti_hda_connector_get_modes, 6378c2ecf20Sopenharmony_ci .mode_valid = sti_hda_connector_mode_valid, 6388c2ecf20Sopenharmony_ci}; 6398c2ecf20Sopenharmony_ci 6408c2ecf20Sopenharmony_cistatic int sti_hda_late_register(struct drm_connector *connector) 6418c2ecf20Sopenharmony_ci{ 6428c2ecf20Sopenharmony_ci struct sti_hda_connector *hda_connector 6438c2ecf20Sopenharmony_ci = to_sti_hda_connector(connector); 6448c2ecf20Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 6458c2ecf20Sopenharmony_ci 6468c2ecf20Sopenharmony_ci hda_debugfs_init(hda, hda->drm_dev->primary); 6478c2ecf20Sopenharmony_ci 6488c2ecf20Sopenharmony_ci return 0; 6498c2ecf20Sopenharmony_ci} 6508c2ecf20Sopenharmony_ci 6518c2ecf20Sopenharmony_cistatic const struct drm_connector_funcs sti_hda_connector_funcs = { 6528c2ecf20Sopenharmony_ci .fill_modes = drm_helper_probe_single_connector_modes, 6538c2ecf20Sopenharmony_ci .destroy = drm_connector_cleanup, 6548c2ecf20Sopenharmony_ci .reset = drm_atomic_helper_connector_reset, 6558c2ecf20Sopenharmony_ci .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, 6568c2ecf20Sopenharmony_ci .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, 6578c2ecf20Sopenharmony_ci .late_register = sti_hda_late_register, 6588c2ecf20Sopenharmony_ci}; 6598c2ecf20Sopenharmony_ci 6608c2ecf20Sopenharmony_cistatic struct drm_encoder *sti_hda_find_encoder(struct drm_device *dev) 6618c2ecf20Sopenharmony_ci{ 6628c2ecf20Sopenharmony_ci struct drm_encoder *encoder; 6638c2ecf20Sopenharmony_ci 6648c2ecf20Sopenharmony_ci list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) { 6658c2ecf20Sopenharmony_ci if (encoder->encoder_type == DRM_MODE_ENCODER_DAC) 6668c2ecf20Sopenharmony_ci return encoder; 6678c2ecf20Sopenharmony_ci } 6688c2ecf20Sopenharmony_ci 6698c2ecf20Sopenharmony_ci return NULL; 6708c2ecf20Sopenharmony_ci} 6718c2ecf20Sopenharmony_ci 6728c2ecf20Sopenharmony_cistatic int sti_hda_bind(struct device *dev, struct device *master, void *data) 6738c2ecf20Sopenharmony_ci{ 6748c2ecf20Sopenharmony_ci struct sti_hda *hda = dev_get_drvdata(dev); 6758c2ecf20Sopenharmony_ci struct drm_device *drm_dev = data; 6768c2ecf20Sopenharmony_ci struct drm_encoder *encoder; 6778c2ecf20Sopenharmony_ci struct sti_hda_connector *connector; 6788c2ecf20Sopenharmony_ci struct drm_connector *drm_connector; 6798c2ecf20Sopenharmony_ci struct drm_bridge *bridge; 6808c2ecf20Sopenharmony_ci int err; 6818c2ecf20Sopenharmony_ci 6828c2ecf20Sopenharmony_ci /* Set the drm device handle */ 6838c2ecf20Sopenharmony_ci hda->drm_dev = drm_dev; 6848c2ecf20Sopenharmony_ci 6858c2ecf20Sopenharmony_ci encoder = sti_hda_find_encoder(drm_dev); 6868c2ecf20Sopenharmony_ci if (!encoder) 6878c2ecf20Sopenharmony_ci return -ENOMEM; 6888c2ecf20Sopenharmony_ci 6898c2ecf20Sopenharmony_ci connector = devm_kzalloc(dev, sizeof(*connector), GFP_KERNEL); 6908c2ecf20Sopenharmony_ci if (!connector) 6918c2ecf20Sopenharmony_ci return -ENOMEM; 6928c2ecf20Sopenharmony_ci 6938c2ecf20Sopenharmony_ci connector->hda = hda; 6948c2ecf20Sopenharmony_ci 6958c2ecf20Sopenharmony_ci bridge = devm_kzalloc(dev, sizeof(*bridge), GFP_KERNEL); 6968c2ecf20Sopenharmony_ci if (!bridge) 6978c2ecf20Sopenharmony_ci return -ENOMEM; 6988c2ecf20Sopenharmony_ci 6998c2ecf20Sopenharmony_ci bridge->driver_private = hda; 7008c2ecf20Sopenharmony_ci bridge->funcs = &sti_hda_bridge_funcs; 7018c2ecf20Sopenharmony_ci drm_bridge_attach(encoder, bridge, NULL, 0); 7028c2ecf20Sopenharmony_ci 7038c2ecf20Sopenharmony_ci connector->encoder = encoder; 7048c2ecf20Sopenharmony_ci 7058c2ecf20Sopenharmony_ci drm_connector = (struct drm_connector *)connector; 7068c2ecf20Sopenharmony_ci 7078c2ecf20Sopenharmony_ci drm_connector->polled = DRM_CONNECTOR_POLL_HPD; 7088c2ecf20Sopenharmony_ci 7098c2ecf20Sopenharmony_ci drm_connector_init(drm_dev, drm_connector, 7108c2ecf20Sopenharmony_ci &sti_hda_connector_funcs, DRM_MODE_CONNECTOR_Component); 7118c2ecf20Sopenharmony_ci drm_connector_helper_add(drm_connector, 7128c2ecf20Sopenharmony_ci &sti_hda_connector_helper_funcs); 7138c2ecf20Sopenharmony_ci 7148c2ecf20Sopenharmony_ci err = drm_connector_attach_encoder(drm_connector, encoder); 7158c2ecf20Sopenharmony_ci if (err) { 7168c2ecf20Sopenharmony_ci DRM_ERROR("Failed to attach a connector to a encoder\n"); 7178c2ecf20Sopenharmony_ci goto err_sysfs; 7188c2ecf20Sopenharmony_ci } 7198c2ecf20Sopenharmony_ci 7208c2ecf20Sopenharmony_ci /* force to disable hd dacs at startup */ 7218c2ecf20Sopenharmony_ci hda_enable_hd_dacs(hda, false); 7228c2ecf20Sopenharmony_ci 7238c2ecf20Sopenharmony_ci return 0; 7248c2ecf20Sopenharmony_ci 7258c2ecf20Sopenharmony_cierr_sysfs: 7268c2ecf20Sopenharmony_ci return -EINVAL; 7278c2ecf20Sopenharmony_ci} 7288c2ecf20Sopenharmony_ci 7298c2ecf20Sopenharmony_cistatic void sti_hda_unbind(struct device *dev, 7308c2ecf20Sopenharmony_ci struct device *master, void *data) 7318c2ecf20Sopenharmony_ci{ 7328c2ecf20Sopenharmony_ci} 7338c2ecf20Sopenharmony_ci 7348c2ecf20Sopenharmony_cistatic const struct component_ops sti_hda_ops = { 7358c2ecf20Sopenharmony_ci .bind = sti_hda_bind, 7368c2ecf20Sopenharmony_ci .unbind = sti_hda_unbind, 7378c2ecf20Sopenharmony_ci}; 7388c2ecf20Sopenharmony_ci 7398c2ecf20Sopenharmony_cistatic int sti_hda_probe(struct platform_device *pdev) 7408c2ecf20Sopenharmony_ci{ 7418c2ecf20Sopenharmony_ci struct device *dev = &pdev->dev; 7428c2ecf20Sopenharmony_ci struct sti_hda *hda; 7438c2ecf20Sopenharmony_ci struct resource *res; 7448c2ecf20Sopenharmony_ci 7458c2ecf20Sopenharmony_ci DRM_INFO("%s\n", __func__); 7468c2ecf20Sopenharmony_ci 7478c2ecf20Sopenharmony_ci hda = devm_kzalloc(dev, sizeof(*hda), GFP_KERNEL); 7488c2ecf20Sopenharmony_ci if (!hda) 7498c2ecf20Sopenharmony_ci return -ENOMEM; 7508c2ecf20Sopenharmony_ci 7518c2ecf20Sopenharmony_ci hda->dev = pdev->dev; 7528c2ecf20Sopenharmony_ci 7538c2ecf20Sopenharmony_ci /* Get resources */ 7548c2ecf20Sopenharmony_ci res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "hda-reg"); 7558c2ecf20Sopenharmony_ci if (!res) { 7568c2ecf20Sopenharmony_ci DRM_ERROR("Invalid hda resource\n"); 7578c2ecf20Sopenharmony_ci return -ENOMEM; 7588c2ecf20Sopenharmony_ci } 7598c2ecf20Sopenharmony_ci hda->regs = devm_ioremap(dev, res->start, resource_size(res)); 7608c2ecf20Sopenharmony_ci if (!hda->regs) 7618c2ecf20Sopenharmony_ci return -ENOMEM; 7628c2ecf20Sopenharmony_ci 7638c2ecf20Sopenharmony_ci res = platform_get_resource_byname(pdev, IORESOURCE_MEM, 7648c2ecf20Sopenharmony_ci "video-dacs-ctrl"); 7658c2ecf20Sopenharmony_ci if (res) { 7668c2ecf20Sopenharmony_ci hda->video_dacs_ctrl = devm_ioremap(dev, res->start, 7678c2ecf20Sopenharmony_ci resource_size(res)); 7688c2ecf20Sopenharmony_ci if (!hda->video_dacs_ctrl) 7698c2ecf20Sopenharmony_ci return -ENOMEM; 7708c2ecf20Sopenharmony_ci } else { 7718c2ecf20Sopenharmony_ci /* If no existing video-dacs-ctrl resource continue the probe */ 7728c2ecf20Sopenharmony_ci DRM_DEBUG_DRIVER("No video-dacs-ctrl resource\n"); 7738c2ecf20Sopenharmony_ci hda->video_dacs_ctrl = NULL; 7748c2ecf20Sopenharmony_ci } 7758c2ecf20Sopenharmony_ci 7768c2ecf20Sopenharmony_ci /* Get clock resources */ 7778c2ecf20Sopenharmony_ci hda->clk_pix = devm_clk_get(dev, "pix"); 7788c2ecf20Sopenharmony_ci if (IS_ERR(hda->clk_pix)) { 7798c2ecf20Sopenharmony_ci DRM_ERROR("Cannot get hda_pix clock\n"); 7808c2ecf20Sopenharmony_ci return PTR_ERR(hda->clk_pix); 7818c2ecf20Sopenharmony_ci } 7828c2ecf20Sopenharmony_ci 7838c2ecf20Sopenharmony_ci hda->clk_hddac = devm_clk_get(dev, "hddac"); 7848c2ecf20Sopenharmony_ci if (IS_ERR(hda->clk_hddac)) { 7858c2ecf20Sopenharmony_ci DRM_ERROR("Cannot get hda_hddac clock\n"); 7868c2ecf20Sopenharmony_ci return PTR_ERR(hda->clk_hddac); 7878c2ecf20Sopenharmony_ci } 7888c2ecf20Sopenharmony_ci 7898c2ecf20Sopenharmony_ci platform_set_drvdata(pdev, hda); 7908c2ecf20Sopenharmony_ci 7918c2ecf20Sopenharmony_ci return component_add(&pdev->dev, &sti_hda_ops); 7928c2ecf20Sopenharmony_ci} 7938c2ecf20Sopenharmony_ci 7948c2ecf20Sopenharmony_cistatic int sti_hda_remove(struct platform_device *pdev) 7958c2ecf20Sopenharmony_ci{ 7968c2ecf20Sopenharmony_ci component_del(&pdev->dev, &sti_hda_ops); 7978c2ecf20Sopenharmony_ci return 0; 7988c2ecf20Sopenharmony_ci} 7998c2ecf20Sopenharmony_ci 8008c2ecf20Sopenharmony_cistatic const struct of_device_id hda_of_match[] = { 8018c2ecf20Sopenharmony_ci { .compatible = "st,stih416-hda", }, 8028c2ecf20Sopenharmony_ci { .compatible = "st,stih407-hda", }, 8038c2ecf20Sopenharmony_ci { /* end node */ } 8048c2ecf20Sopenharmony_ci}; 8058c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, hda_of_match); 8068c2ecf20Sopenharmony_ci 8078c2ecf20Sopenharmony_cistruct platform_driver sti_hda_driver = { 8088c2ecf20Sopenharmony_ci .driver = { 8098c2ecf20Sopenharmony_ci .name = "sti-hda", 8108c2ecf20Sopenharmony_ci .owner = THIS_MODULE, 8118c2ecf20Sopenharmony_ci .of_match_table = hda_of_match, 8128c2ecf20Sopenharmony_ci }, 8138c2ecf20Sopenharmony_ci .probe = sti_hda_probe, 8148c2ecf20Sopenharmony_ci .remove = sti_hda_remove, 8158c2ecf20Sopenharmony_ci}; 8168c2ecf20Sopenharmony_ci 8178c2ecf20Sopenharmony_ciMODULE_AUTHOR("Benjamin Gaignard <benjamin.gaignard@st.com>"); 8188c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("STMicroelectronics SoC DRM driver"); 8198c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 820