162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright (C) STMicroelectronics SA 2014 462306a36Sopenharmony_ci * Author: Fabien Dessenne <fabien.dessenne@st.com> for STMicroelectronics. 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/clk.h> 862306a36Sopenharmony_ci#include <linux/component.h> 962306a36Sopenharmony_ci#include <linux/io.h> 1062306a36Sopenharmony_ci#include <linux/mod_devicetable.h> 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/platform_device.h> 1362306a36Sopenharmony_ci#include <linux/seq_file.h> 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#include <drm/drm_atomic_helper.h> 1662306a36Sopenharmony_ci#include <drm/drm_bridge.h> 1762306a36Sopenharmony_ci#include <drm/drm_debugfs.h> 1862306a36Sopenharmony_ci#include <drm/drm_device.h> 1962306a36Sopenharmony_ci#include <drm/drm_file.h> 2062306a36Sopenharmony_ci#include <drm/drm_print.h> 2162306a36Sopenharmony_ci#include <drm/drm_probe_helper.h> 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci/* HDformatter registers */ 2462306a36Sopenharmony_ci#define HDA_ANA_CFG 0x0000 2562306a36Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_Y 0x0004 2662306a36Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_CB 0x0008 2762306a36Sopenharmony_ci#define HDA_ANA_SCALE_CTRL_CR 0x000C 2862306a36Sopenharmony_ci#define HDA_ANA_ANC_CTRL 0x0010 2962306a36Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG 0x0014 3062306a36Sopenharmony_ci#define HDA_COEFF_Y_PH1_TAP123 0x0018 3162306a36Sopenharmony_ci#define HDA_COEFF_Y_PH1_TAP456 0x001C 3262306a36Sopenharmony_ci#define HDA_COEFF_Y_PH2_TAP123 0x0020 3362306a36Sopenharmony_ci#define HDA_COEFF_Y_PH2_TAP456 0x0024 3462306a36Sopenharmony_ci#define HDA_COEFF_Y_PH3_TAP123 0x0028 3562306a36Sopenharmony_ci#define HDA_COEFF_Y_PH3_TAP456 0x002C 3662306a36Sopenharmony_ci#define HDA_COEFF_Y_PH4_TAP123 0x0030 3762306a36Sopenharmony_ci#define HDA_COEFF_Y_PH4_TAP456 0x0034 3862306a36Sopenharmony_ci#define HDA_ANA_SRC_C_CFG 0x0040 3962306a36Sopenharmony_ci#define HDA_COEFF_C_PH1_TAP123 0x0044 4062306a36Sopenharmony_ci#define HDA_COEFF_C_PH1_TAP456 0x0048 4162306a36Sopenharmony_ci#define HDA_COEFF_C_PH2_TAP123 0x004C 4262306a36Sopenharmony_ci#define HDA_COEFF_C_PH2_TAP456 0x0050 4362306a36Sopenharmony_ci#define HDA_COEFF_C_PH3_TAP123 0x0054 4462306a36Sopenharmony_ci#define HDA_COEFF_C_PH3_TAP456 0x0058 4562306a36Sopenharmony_ci#define HDA_COEFF_C_PH4_TAP123 0x005C 4662306a36Sopenharmony_ci#define HDA_COEFF_C_PH4_TAP456 0x0060 4762306a36Sopenharmony_ci#define HDA_SYNC_AWGI 0x0300 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci/* HDA_ANA_CFG */ 5062306a36Sopenharmony_ci#define CFG_AWG_ASYNC_EN BIT(0) 5162306a36Sopenharmony_ci#define CFG_AWG_ASYNC_HSYNC_MTD BIT(1) 5262306a36Sopenharmony_ci#define CFG_AWG_ASYNC_VSYNC_MTD BIT(2) 5362306a36Sopenharmony_ci#define CFG_AWG_SYNC_DEL BIT(3) 5462306a36Sopenharmony_ci#define CFG_AWG_FLTR_MODE_SHIFT 4 5562306a36Sopenharmony_ci#define CFG_AWG_FLTR_MODE_MASK (0xF << CFG_AWG_FLTR_MODE_SHIFT) 5662306a36Sopenharmony_ci#define CFG_AWG_FLTR_MODE_SD (0 << CFG_AWG_FLTR_MODE_SHIFT) 5762306a36Sopenharmony_ci#define CFG_AWG_FLTR_MODE_ED (1 << CFG_AWG_FLTR_MODE_SHIFT) 5862306a36Sopenharmony_ci#define CFG_AWG_FLTR_MODE_HD (2 << CFG_AWG_FLTR_MODE_SHIFT) 5962306a36Sopenharmony_ci#define CFG_SYNC_ON_PBPR_MASK BIT(8) 6062306a36Sopenharmony_ci#define CFG_PREFILTER_EN_MASK BIT(9) 6162306a36Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_SHIFT 16 6262306a36Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_MASK (0x7FF << CFG_PBPR_SYNC_OFF_SHIFT) 6362306a36Sopenharmony_ci#define CFG_PBPR_SYNC_OFF_VAL 0x117 /* Voltage dependent. stiH416 */ 6462306a36Sopenharmony_ci 6562306a36Sopenharmony_ci/* Default scaling values */ 6662306a36Sopenharmony_ci#define SCALE_CTRL_Y_DFLT 0x00C50256 6762306a36Sopenharmony_ci#define SCALE_CTRL_CB_DFLT 0x00DB0249 6862306a36Sopenharmony_ci#define SCALE_CTRL_CR_DFLT 0x00DB0249 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci/* Video DACs control */ 7162306a36Sopenharmony_ci#define DAC_CFG_HD_HZUVW_OFF_MASK BIT(1) 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci/* Upsampler values for the alternative 2X Filter */ 7462306a36Sopenharmony_ci#define SAMPLER_COEF_NB 8 7562306a36Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG_ALT_2X 0x01130000 7662306a36Sopenharmony_cistatic u32 coef_y_alt_2x[] = { 7762306a36Sopenharmony_ci 0x00FE83FB, 0x1F900401, 0x00000000, 0x00000000, 7862306a36Sopenharmony_ci 0x00F408F9, 0x055F7C25, 0x00000000, 0x00000000 7962306a36Sopenharmony_ci}; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci#define HDA_ANA_SRC_C_CFG_ALT_2X 0x01750004 8262306a36Sopenharmony_cistatic u32 coef_c_alt_2x[] = { 8362306a36Sopenharmony_ci 0x001305F7, 0x05274BD0, 0x00000000, 0x00000000, 8462306a36Sopenharmony_ci 0x0004907C, 0x09C80B9D, 0x00000000, 0x00000000 8562306a36Sopenharmony_ci}; 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_ci/* Upsampler values for the 4X Filter */ 8862306a36Sopenharmony_ci#define HDA_ANA_SRC_Y_CFG_4X 0x01ED0005 8962306a36Sopenharmony_ci#define HDA_ANA_SRC_C_CFG_4X 0x01ED0004 9062306a36Sopenharmony_cistatic u32 coef_yc_4x[] = { 9162306a36Sopenharmony_ci 0x00FC827F, 0x008FE20B, 0x00F684FC, 0x050F7C24, 9262306a36Sopenharmony_ci 0x00F4857C, 0x0A1F402E, 0x00FA027F, 0x0E076E1D 9362306a36Sopenharmony_ci}; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci/* AWG instructions for some video modes */ 9662306a36Sopenharmony_ci#define AWG_MAX_INST 64 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ci/* 720p@50 */ 9962306a36Sopenharmony_cistatic u32 AWGi_720p_50[] = { 10062306a36Sopenharmony_ci 0x00000971, 0x00000C26, 0x0000013B, 0x00000CDA, 10162306a36Sopenharmony_ci 0x00000104, 0x00000E7E, 0x00000E7F, 0x0000013B, 10262306a36Sopenharmony_ci 0x00000D8E, 0x00000104, 0x00001804, 0x00000971, 10362306a36Sopenharmony_ci 0x00000C26, 0x0000003B, 0x00000FB4, 0x00000FB5, 10462306a36Sopenharmony_ci 0x00000104, 0x00001AE8 10562306a36Sopenharmony_ci}; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci#define NN_720p_50 ARRAY_SIZE(AWGi_720p_50) 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_ci/* 720p@60 */ 11062306a36Sopenharmony_cistatic u32 AWGi_720p_60[] = { 11162306a36Sopenharmony_ci 0x00000971, 0x00000C26, 0x0000013B, 0x00000CDA, 11262306a36Sopenharmony_ci 0x00000104, 0x00000E7E, 0x00000E7F, 0x0000013B, 11362306a36Sopenharmony_ci 0x00000C44, 0x00000104, 0x00001804, 0x00000971, 11462306a36Sopenharmony_ci 0x00000C26, 0x0000003B, 0x00000F0F, 0x00000F10, 11562306a36Sopenharmony_ci 0x00000104, 0x00001AE8 11662306a36Sopenharmony_ci}; 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci#define NN_720p_60 ARRAY_SIZE(AWGi_720p_60) 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci/* 1080p@30 */ 12162306a36Sopenharmony_cistatic u32 AWGi_1080p_30[] = { 12262306a36Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 12362306a36Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 12462306a36Sopenharmony_ci 0x00000C2A, 0x00000104, 0x00001804, 0x00000971, 12562306a36Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000EBE, 0x00000EBF, 12662306a36Sopenharmony_ci 0x00000EBF, 0x00000104, 0x00001A2F, 0x00001C4B, 12762306a36Sopenharmony_ci 0x00001C52 12862306a36Sopenharmony_ci}; 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci#define NN_1080p_30 ARRAY_SIZE(AWGi_1080p_30) 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci/* 1080p@25 */ 13362306a36Sopenharmony_cistatic u32 AWGi_1080p_25[] = { 13462306a36Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 13562306a36Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 13662306a36Sopenharmony_ci 0x00000DE2, 0x00000104, 0x00001804, 0x00000971, 13762306a36Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000F51, 0x00000F51, 13862306a36Sopenharmony_ci 0x00000F52, 0x00000104, 0x00001A2F, 0x00001C4B, 13962306a36Sopenharmony_ci 0x00001C52 14062306a36Sopenharmony_ci}; 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci#define NN_1080p_25 ARRAY_SIZE(AWGi_1080p_25) 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci/* 1080p@24 */ 14562306a36Sopenharmony_cistatic u32 AWGi_1080p_24[] = { 14662306a36Sopenharmony_ci 0x00000971, 0x00000C2A, 0x0000013B, 0x00000C56, 14762306a36Sopenharmony_ci 0x00000104, 0x00000FDC, 0x00000FDD, 0x0000013B, 14862306a36Sopenharmony_ci 0x00000E50, 0x00000104, 0x00001804, 0x00000971, 14962306a36Sopenharmony_ci 0x00000C2A, 0x0000003B, 0x00000F76, 0x00000F76, 15062306a36Sopenharmony_ci 0x00000F76, 0x00000104, 0x00001A2F, 0x00001C4B, 15162306a36Sopenharmony_ci 0x00001C52 15262306a36Sopenharmony_ci}; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci#define NN_1080p_24 ARRAY_SIZE(AWGi_1080p_24) 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci/* 720x480p@60 */ 15762306a36Sopenharmony_cistatic u32 AWGi_720x480p_60[] = { 15862306a36Sopenharmony_ci 0x00000904, 0x00000F18, 0x0000013B, 0x00001805, 15962306a36Sopenharmony_ci 0x00000904, 0x00000C3D, 0x0000003B, 0x00001A06 16062306a36Sopenharmony_ci}; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci#define NN_720x480p_60 ARRAY_SIZE(AWGi_720x480p_60) 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci/* Video mode category */ 16562306a36Sopenharmony_cienum sti_hda_vid_cat { 16662306a36Sopenharmony_ci VID_SD, 16762306a36Sopenharmony_ci VID_ED, 16862306a36Sopenharmony_ci VID_HD_74M, 16962306a36Sopenharmony_ci VID_HD_148M 17062306a36Sopenharmony_ci}; 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_cistruct sti_hda_video_config { 17362306a36Sopenharmony_ci struct drm_display_mode mode; 17462306a36Sopenharmony_ci u32 *awg_instr; 17562306a36Sopenharmony_ci int nb_instr; 17662306a36Sopenharmony_ci enum sti_hda_vid_cat vid_cat; 17762306a36Sopenharmony_ci}; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci/* HD analog supported modes 18062306a36Sopenharmony_ci * Interlaced modes may be added when supported by the whole display chain 18162306a36Sopenharmony_ci */ 18262306a36Sopenharmony_cistatic const struct sti_hda_video_config hda_supported_modes[] = { 18362306a36Sopenharmony_ci /* 1080p30 74.250Mhz */ 18462306a36Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2008, 18562306a36Sopenharmony_ci 2052, 2200, 0, 1080, 1084, 1089, 1125, 0, 18662306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 18762306a36Sopenharmony_ci AWGi_1080p_30, NN_1080p_30, VID_HD_74M}, 18862306a36Sopenharmony_ci /* 1080p30 74.176Mhz */ 18962306a36Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74176, 1920, 2008, 19062306a36Sopenharmony_ci 2052, 2200, 0, 1080, 1084, 1089, 1125, 0, 19162306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 19262306a36Sopenharmony_ci AWGi_1080p_30, NN_1080p_30, VID_HD_74M}, 19362306a36Sopenharmony_ci /* 1080p24 74.250Mhz */ 19462306a36Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2558, 19562306a36Sopenharmony_ci 2602, 2750, 0, 1080, 1084, 1089, 1125, 0, 19662306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 19762306a36Sopenharmony_ci AWGi_1080p_24, NN_1080p_24, VID_HD_74M}, 19862306a36Sopenharmony_ci /* 1080p24 74.176Mhz */ 19962306a36Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74176, 1920, 2558, 20062306a36Sopenharmony_ci 2602, 2750, 0, 1080, 1084, 1089, 1125, 0, 20162306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 20262306a36Sopenharmony_ci AWGi_1080p_24, NN_1080p_24, VID_HD_74M}, 20362306a36Sopenharmony_ci /* 1080p25 74.250Mhz */ 20462306a36Sopenharmony_ci {{DRM_MODE("1920x1080", DRM_MODE_TYPE_DRIVER, 74250, 1920, 2448, 20562306a36Sopenharmony_ci 2492, 2640, 0, 1080, 1084, 1089, 1125, 0, 20662306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 20762306a36Sopenharmony_ci AWGi_1080p_25, NN_1080p_25, VID_HD_74M}, 20862306a36Sopenharmony_ci /* 720p60 74.250Mhz */ 20962306a36Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74250, 1280, 1390, 21062306a36Sopenharmony_ci 1430, 1650, 0, 720, 725, 730, 750, 0, 21162306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 21262306a36Sopenharmony_ci AWGi_720p_60, NN_720p_60, VID_HD_74M}, 21362306a36Sopenharmony_ci /* 720p60 74.176Mhz */ 21462306a36Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74176, 1280, 1390, 21562306a36Sopenharmony_ci 1430, 1650, 0, 720, 725, 730, 750, 0, 21662306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 21762306a36Sopenharmony_ci AWGi_720p_60, NN_720p_60, VID_HD_74M}, 21862306a36Sopenharmony_ci /* 720p50 74.250Mhz */ 21962306a36Sopenharmony_ci {{DRM_MODE("1280x720", DRM_MODE_TYPE_DRIVER, 74250, 1280, 1720, 22062306a36Sopenharmony_ci 1760, 1980, 0, 720, 725, 730, 750, 0, 22162306a36Sopenharmony_ci DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_PVSYNC)}, 22262306a36Sopenharmony_ci AWGi_720p_50, NN_720p_50, VID_HD_74M}, 22362306a36Sopenharmony_ci /* 720x480p60 27.027Mhz */ 22462306a36Sopenharmony_ci {{DRM_MODE("720x480", DRM_MODE_TYPE_DRIVER, 27027, 720, 736, 22562306a36Sopenharmony_ci 798, 858, 0, 480, 489, 495, 525, 0, 22662306a36Sopenharmony_ci DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC)}, 22762306a36Sopenharmony_ci AWGi_720x480p_60, NN_720x480p_60, VID_ED}, 22862306a36Sopenharmony_ci /* 720x480p60 27.000Mhz */ 22962306a36Sopenharmony_ci {{DRM_MODE("720x480", DRM_MODE_TYPE_DRIVER, 27000, 720, 736, 23062306a36Sopenharmony_ci 798, 858, 0, 480, 489, 495, 525, 0, 23162306a36Sopenharmony_ci DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC)}, 23262306a36Sopenharmony_ci AWGi_720x480p_60, NN_720x480p_60, VID_ED} 23362306a36Sopenharmony_ci}; 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_ci/* 23662306a36Sopenharmony_ci * STI hd analog structure 23762306a36Sopenharmony_ci * 23862306a36Sopenharmony_ci * @dev: driver device 23962306a36Sopenharmony_ci * @drm_dev: pointer to drm device 24062306a36Sopenharmony_ci * @mode: current display mode selected 24162306a36Sopenharmony_ci * @regs: HD analog register 24262306a36Sopenharmony_ci * @video_dacs_ctrl: video DACS control register 24362306a36Sopenharmony_ci * @enabled: true if HD analog is enabled else false 24462306a36Sopenharmony_ci */ 24562306a36Sopenharmony_cistruct sti_hda { 24662306a36Sopenharmony_ci struct device dev; 24762306a36Sopenharmony_ci struct drm_device *drm_dev; 24862306a36Sopenharmony_ci struct drm_display_mode mode; 24962306a36Sopenharmony_ci void __iomem *regs; 25062306a36Sopenharmony_ci void __iomem *video_dacs_ctrl; 25162306a36Sopenharmony_ci struct clk *clk_pix; 25262306a36Sopenharmony_ci struct clk *clk_hddac; 25362306a36Sopenharmony_ci bool enabled; 25462306a36Sopenharmony_ci}; 25562306a36Sopenharmony_ci 25662306a36Sopenharmony_cistruct sti_hda_connector { 25762306a36Sopenharmony_ci struct drm_connector drm_connector; 25862306a36Sopenharmony_ci struct drm_encoder *encoder; 25962306a36Sopenharmony_ci struct sti_hda *hda; 26062306a36Sopenharmony_ci}; 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci#define to_sti_hda_connector(x) \ 26362306a36Sopenharmony_ci container_of(x, struct sti_hda_connector, drm_connector) 26462306a36Sopenharmony_ci 26562306a36Sopenharmony_cistatic u32 hda_read(struct sti_hda *hda, int offset) 26662306a36Sopenharmony_ci{ 26762306a36Sopenharmony_ci return readl(hda->regs + offset); 26862306a36Sopenharmony_ci} 26962306a36Sopenharmony_ci 27062306a36Sopenharmony_cistatic void hda_write(struct sti_hda *hda, u32 val, int offset) 27162306a36Sopenharmony_ci{ 27262306a36Sopenharmony_ci writel(val, hda->regs + offset); 27362306a36Sopenharmony_ci} 27462306a36Sopenharmony_ci 27562306a36Sopenharmony_ci/** 27662306a36Sopenharmony_ci * hda_get_mode_idx - Search for a video mode in the supported modes table 27762306a36Sopenharmony_ci * 27862306a36Sopenharmony_ci * @mode: mode being searched 27962306a36Sopenharmony_ci * @idx: index of the found mode 28062306a36Sopenharmony_ci * 28162306a36Sopenharmony_ci * Return true if mode is found 28262306a36Sopenharmony_ci */ 28362306a36Sopenharmony_cistatic bool hda_get_mode_idx(struct drm_display_mode mode, int *idx) 28462306a36Sopenharmony_ci{ 28562306a36Sopenharmony_ci unsigned int i; 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_supported_modes); i++) 28862306a36Sopenharmony_ci if (drm_mode_equal(&hda_supported_modes[i].mode, &mode)) { 28962306a36Sopenharmony_ci *idx = i; 29062306a36Sopenharmony_ci return true; 29162306a36Sopenharmony_ci } 29262306a36Sopenharmony_ci return false; 29362306a36Sopenharmony_ci} 29462306a36Sopenharmony_ci 29562306a36Sopenharmony_ci/** 29662306a36Sopenharmony_ci * hda_enable_hd_dacs - Enable the HD DACS 29762306a36Sopenharmony_ci * 29862306a36Sopenharmony_ci * @hda: pointer to HD analog structure 29962306a36Sopenharmony_ci * @enable: true if HD DACS need to be enabled, else false 30062306a36Sopenharmony_ci */ 30162306a36Sopenharmony_cistatic void hda_enable_hd_dacs(struct sti_hda *hda, bool enable) 30262306a36Sopenharmony_ci{ 30362306a36Sopenharmony_ci if (hda->video_dacs_ctrl) { 30462306a36Sopenharmony_ci u32 val; 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_ci val = readl(hda->video_dacs_ctrl); 30762306a36Sopenharmony_ci if (enable) 30862306a36Sopenharmony_ci val &= ~DAC_CFG_HD_HZUVW_OFF_MASK; 30962306a36Sopenharmony_ci else 31062306a36Sopenharmony_ci val |= DAC_CFG_HD_HZUVW_OFF_MASK; 31162306a36Sopenharmony_ci 31262306a36Sopenharmony_ci writel(val, hda->video_dacs_ctrl); 31362306a36Sopenharmony_ci } 31462306a36Sopenharmony_ci} 31562306a36Sopenharmony_ci 31662306a36Sopenharmony_ci#define DBGFS_DUMP(reg) seq_printf(s, "\n %-25s 0x%08X", #reg, \ 31762306a36Sopenharmony_ci readl(hda->regs + reg)) 31862306a36Sopenharmony_ci 31962306a36Sopenharmony_cistatic void hda_dbg_cfg(struct seq_file *s, int val) 32062306a36Sopenharmony_ci{ 32162306a36Sopenharmony_ci seq_puts(s, "\tAWG "); 32262306a36Sopenharmony_ci seq_puts(s, val & CFG_AWG_ASYNC_EN ? "enabled" : "disabled"); 32362306a36Sopenharmony_ci} 32462306a36Sopenharmony_ci 32562306a36Sopenharmony_cistatic void hda_dbg_awg_microcode(struct seq_file *s, void __iomem *reg) 32662306a36Sopenharmony_ci{ 32762306a36Sopenharmony_ci unsigned int i; 32862306a36Sopenharmony_ci 32962306a36Sopenharmony_ci seq_puts(s, "\n\n HDA AWG microcode:"); 33062306a36Sopenharmony_ci for (i = 0; i < AWG_MAX_INST; i++) { 33162306a36Sopenharmony_ci if (i % 8 == 0) 33262306a36Sopenharmony_ci seq_printf(s, "\n %04X:", i); 33362306a36Sopenharmony_ci seq_printf(s, " %04X", readl(reg + i * 4)); 33462306a36Sopenharmony_ci } 33562306a36Sopenharmony_ci} 33662306a36Sopenharmony_ci 33762306a36Sopenharmony_cistatic void hda_dbg_video_dacs_ctrl(struct seq_file *s, void __iomem *reg) 33862306a36Sopenharmony_ci{ 33962306a36Sopenharmony_ci u32 val = readl(reg); 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci seq_printf(s, "\n\n %-25s 0x%08X", "VIDEO_DACS_CONTROL", val); 34262306a36Sopenharmony_ci seq_puts(s, "\tHD DACs "); 34362306a36Sopenharmony_ci seq_puts(s, val & DAC_CFG_HD_HZUVW_OFF_MASK ? "disabled" : "enabled"); 34462306a36Sopenharmony_ci} 34562306a36Sopenharmony_ci 34662306a36Sopenharmony_cistatic int hda_dbg_show(struct seq_file *s, void *data) 34762306a36Sopenharmony_ci{ 34862306a36Sopenharmony_ci struct drm_info_node *node = s->private; 34962306a36Sopenharmony_ci struct sti_hda *hda = (struct sti_hda *)node->info_ent->data; 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci seq_printf(s, "HD Analog: (vaddr = 0x%p)", hda->regs); 35262306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_CFG); 35362306a36Sopenharmony_ci hda_dbg_cfg(s, readl(hda->regs + HDA_ANA_CFG)); 35462306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_Y); 35562306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_CB); 35662306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SCALE_CTRL_CR); 35762306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_ANC_CTRL); 35862306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SRC_Y_CFG); 35962306a36Sopenharmony_ci DBGFS_DUMP(HDA_ANA_SRC_C_CFG); 36062306a36Sopenharmony_ci hda_dbg_awg_microcode(s, hda->regs + HDA_SYNC_AWGI); 36162306a36Sopenharmony_ci if (hda->video_dacs_ctrl) 36262306a36Sopenharmony_ci hda_dbg_video_dacs_ctrl(s, hda->video_dacs_ctrl); 36362306a36Sopenharmony_ci seq_putc(s, '\n'); 36462306a36Sopenharmony_ci return 0; 36562306a36Sopenharmony_ci} 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_cistatic struct drm_info_list hda_debugfs_files[] = { 36862306a36Sopenharmony_ci { "hda", hda_dbg_show, 0, NULL }, 36962306a36Sopenharmony_ci}; 37062306a36Sopenharmony_ci 37162306a36Sopenharmony_cistatic void hda_debugfs_init(struct sti_hda *hda, struct drm_minor *minor) 37262306a36Sopenharmony_ci{ 37362306a36Sopenharmony_ci unsigned int i; 37462306a36Sopenharmony_ci 37562306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_debugfs_files); i++) 37662306a36Sopenharmony_ci hda_debugfs_files[i].data = hda; 37762306a36Sopenharmony_ci 37862306a36Sopenharmony_ci drm_debugfs_create_files(hda_debugfs_files, 37962306a36Sopenharmony_ci ARRAY_SIZE(hda_debugfs_files), 38062306a36Sopenharmony_ci minor->debugfs_root, minor); 38162306a36Sopenharmony_ci} 38262306a36Sopenharmony_ci 38362306a36Sopenharmony_ci/** 38462306a36Sopenharmony_ci * sti_hda_configure_awg - Configure AWG, writing instructions 38562306a36Sopenharmony_ci * 38662306a36Sopenharmony_ci * @hda: pointer to HD analog structure 38762306a36Sopenharmony_ci * @awg_instr: pointer to AWG instructions table 38862306a36Sopenharmony_ci * @nb: nb of AWG instructions 38962306a36Sopenharmony_ci */ 39062306a36Sopenharmony_cistatic void sti_hda_configure_awg(struct sti_hda *hda, u32 *awg_instr, int nb) 39162306a36Sopenharmony_ci{ 39262306a36Sopenharmony_ci unsigned int i; 39362306a36Sopenharmony_ci 39462306a36Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 39562306a36Sopenharmony_ci 39662306a36Sopenharmony_ci for (i = 0; i < nb; i++) 39762306a36Sopenharmony_ci hda_write(hda, awg_instr[i], HDA_SYNC_AWGI + i * 4); 39862306a36Sopenharmony_ci for (i = nb; i < AWG_MAX_INST; i++) 39962306a36Sopenharmony_ci hda_write(hda, 0, HDA_SYNC_AWGI + i * 4); 40062306a36Sopenharmony_ci} 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_cistatic void sti_hda_disable(struct drm_bridge *bridge) 40362306a36Sopenharmony_ci{ 40462306a36Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 40562306a36Sopenharmony_ci u32 val; 40662306a36Sopenharmony_ci 40762306a36Sopenharmony_ci if (!hda->enabled) 40862306a36Sopenharmony_ci return; 40962306a36Sopenharmony_ci 41062306a36Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 41162306a36Sopenharmony_ci 41262306a36Sopenharmony_ci /* Disable HD DAC and AWG */ 41362306a36Sopenharmony_ci val = hda_read(hda, HDA_ANA_CFG); 41462306a36Sopenharmony_ci val &= ~CFG_AWG_ASYNC_EN; 41562306a36Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 41662306a36Sopenharmony_ci hda_write(hda, 0, HDA_ANA_ANC_CTRL); 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci hda_enable_hd_dacs(hda, false); 41962306a36Sopenharmony_ci 42062306a36Sopenharmony_ci /* Disable/unprepare hda clock */ 42162306a36Sopenharmony_ci clk_disable_unprepare(hda->clk_hddac); 42262306a36Sopenharmony_ci clk_disable_unprepare(hda->clk_pix); 42362306a36Sopenharmony_ci 42462306a36Sopenharmony_ci hda->enabled = false; 42562306a36Sopenharmony_ci} 42662306a36Sopenharmony_ci 42762306a36Sopenharmony_cistatic void sti_hda_pre_enable(struct drm_bridge *bridge) 42862306a36Sopenharmony_ci{ 42962306a36Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 43062306a36Sopenharmony_ci u32 val, i, mode_idx; 43162306a36Sopenharmony_ci u32 src_filter_y, src_filter_c; 43262306a36Sopenharmony_ci u32 *coef_y, *coef_c; 43362306a36Sopenharmony_ci u32 filter_mode; 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 43662306a36Sopenharmony_ci 43762306a36Sopenharmony_ci if (hda->enabled) 43862306a36Sopenharmony_ci return; 43962306a36Sopenharmony_ci 44062306a36Sopenharmony_ci /* Prepare/enable clocks */ 44162306a36Sopenharmony_ci if (clk_prepare_enable(hda->clk_pix)) 44262306a36Sopenharmony_ci DRM_ERROR("Failed to prepare/enable hda_pix clk\n"); 44362306a36Sopenharmony_ci if (clk_prepare_enable(hda->clk_hddac)) 44462306a36Sopenharmony_ci DRM_ERROR("Failed to prepare/enable hda_hddac clk\n"); 44562306a36Sopenharmony_ci 44662306a36Sopenharmony_ci if (!hda_get_mode_idx(hda->mode, &mode_idx)) { 44762306a36Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 44862306a36Sopenharmony_ci return; 44962306a36Sopenharmony_ci } 45062306a36Sopenharmony_ci 45162306a36Sopenharmony_ci switch (hda_supported_modes[mode_idx].vid_cat) { 45262306a36Sopenharmony_ci case VID_HD_148M: 45362306a36Sopenharmony_ci DRM_ERROR("Beyond HD analog capabilities\n"); 45462306a36Sopenharmony_ci return; 45562306a36Sopenharmony_ci case VID_HD_74M: 45662306a36Sopenharmony_ci /* HD use alternate 2x filter */ 45762306a36Sopenharmony_ci filter_mode = CFG_AWG_FLTR_MODE_HD; 45862306a36Sopenharmony_ci src_filter_y = HDA_ANA_SRC_Y_CFG_ALT_2X; 45962306a36Sopenharmony_ci src_filter_c = HDA_ANA_SRC_C_CFG_ALT_2X; 46062306a36Sopenharmony_ci coef_y = coef_y_alt_2x; 46162306a36Sopenharmony_ci coef_c = coef_c_alt_2x; 46262306a36Sopenharmony_ci break; 46362306a36Sopenharmony_ci case VID_ED: 46462306a36Sopenharmony_ci /* ED uses 4x filter */ 46562306a36Sopenharmony_ci filter_mode = CFG_AWG_FLTR_MODE_ED; 46662306a36Sopenharmony_ci src_filter_y = HDA_ANA_SRC_Y_CFG_4X; 46762306a36Sopenharmony_ci src_filter_c = HDA_ANA_SRC_C_CFG_4X; 46862306a36Sopenharmony_ci coef_y = coef_yc_4x; 46962306a36Sopenharmony_ci coef_c = coef_yc_4x; 47062306a36Sopenharmony_ci break; 47162306a36Sopenharmony_ci case VID_SD: 47262306a36Sopenharmony_ci DRM_ERROR("Not supported\n"); 47362306a36Sopenharmony_ci return; 47462306a36Sopenharmony_ci default: 47562306a36Sopenharmony_ci DRM_ERROR("Undefined resolution\n"); 47662306a36Sopenharmony_ci return; 47762306a36Sopenharmony_ci } 47862306a36Sopenharmony_ci DRM_DEBUG_DRIVER("Using HDA mode #%d\n", mode_idx); 47962306a36Sopenharmony_ci 48062306a36Sopenharmony_ci /* Enable HD Video DACs */ 48162306a36Sopenharmony_ci hda_enable_hd_dacs(hda, true); 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci /* Configure scaler */ 48462306a36Sopenharmony_ci hda_write(hda, SCALE_CTRL_Y_DFLT, HDA_ANA_SCALE_CTRL_Y); 48562306a36Sopenharmony_ci hda_write(hda, SCALE_CTRL_CB_DFLT, HDA_ANA_SCALE_CTRL_CB); 48662306a36Sopenharmony_ci hda_write(hda, SCALE_CTRL_CR_DFLT, HDA_ANA_SCALE_CTRL_CR); 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci /* Configure sampler */ 48962306a36Sopenharmony_ci hda_write(hda , src_filter_y, HDA_ANA_SRC_Y_CFG); 49062306a36Sopenharmony_ci hda_write(hda, src_filter_c, HDA_ANA_SRC_C_CFG); 49162306a36Sopenharmony_ci for (i = 0; i < SAMPLER_COEF_NB; i++) { 49262306a36Sopenharmony_ci hda_write(hda, coef_y[i], HDA_COEFF_Y_PH1_TAP123 + i * 4); 49362306a36Sopenharmony_ci hda_write(hda, coef_c[i], HDA_COEFF_C_PH1_TAP123 + i * 4); 49462306a36Sopenharmony_ci } 49562306a36Sopenharmony_ci 49662306a36Sopenharmony_ci /* Configure main HDFormatter */ 49762306a36Sopenharmony_ci val = 0; 49862306a36Sopenharmony_ci val |= (hda->mode.flags & DRM_MODE_FLAG_INTERLACE) ? 49962306a36Sopenharmony_ci 0 : CFG_AWG_ASYNC_VSYNC_MTD; 50062306a36Sopenharmony_ci val |= (CFG_PBPR_SYNC_OFF_VAL << CFG_PBPR_SYNC_OFF_SHIFT); 50162306a36Sopenharmony_ci val |= filter_mode; 50262306a36Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 50362306a36Sopenharmony_ci 50462306a36Sopenharmony_ci /* Configure AWG */ 50562306a36Sopenharmony_ci sti_hda_configure_awg(hda, hda_supported_modes[mode_idx].awg_instr, 50662306a36Sopenharmony_ci hda_supported_modes[mode_idx].nb_instr); 50762306a36Sopenharmony_ci 50862306a36Sopenharmony_ci /* Enable AWG */ 50962306a36Sopenharmony_ci val = hda_read(hda, HDA_ANA_CFG); 51062306a36Sopenharmony_ci val |= CFG_AWG_ASYNC_EN; 51162306a36Sopenharmony_ci hda_write(hda, val, HDA_ANA_CFG); 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_ci hda->enabled = true; 51462306a36Sopenharmony_ci} 51562306a36Sopenharmony_ci 51662306a36Sopenharmony_cistatic void sti_hda_set_mode(struct drm_bridge *bridge, 51762306a36Sopenharmony_ci const struct drm_display_mode *mode, 51862306a36Sopenharmony_ci const struct drm_display_mode *adjusted_mode) 51962306a36Sopenharmony_ci{ 52062306a36Sopenharmony_ci struct sti_hda *hda = bridge->driver_private; 52162306a36Sopenharmony_ci u32 mode_idx; 52262306a36Sopenharmony_ci int hddac_rate; 52362306a36Sopenharmony_ci int ret; 52462306a36Sopenharmony_ci 52562306a36Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 52662306a36Sopenharmony_ci 52762306a36Sopenharmony_ci drm_mode_copy(&hda->mode, mode); 52862306a36Sopenharmony_ci 52962306a36Sopenharmony_ci if (!hda_get_mode_idx(hda->mode, &mode_idx)) { 53062306a36Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 53162306a36Sopenharmony_ci return; 53262306a36Sopenharmony_ci } 53362306a36Sopenharmony_ci 53462306a36Sopenharmony_ci switch (hda_supported_modes[mode_idx].vid_cat) { 53562306a36Sopenharmony_ci case VID_HD_74M: 53662306a36Sopenharmony_ci /* HD use alternate 2x filter */ 53762306a36Sopenharmony_ci hddac_rate = mode->clock * 1000 * 2; 53862306a36Sopenharmony_ci break; 53962306a36Sopenharmony_ci case VID_ED: 54062306a36Sopenharmony_ci /* ED uses 4x filter */ 54162306a36Sopenharmony_ci hddac_rate = mode->clock * 1000 * 4; 54262306a36Sopenharmony_ci break; 54362306a36Sopenharmony_ci default: 54462306a36Sopenharmony_ci DRM_ERROR("Undefined mode\n"); 54562306a36Sopenharmony_ci return; 54662306a36Sopenharmony_ci } 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ci /* HD DAC = 148.5Mhz or 108 Mhz */ 54962306a36Sopenharmony_ci ret = clk_set_rate(hda->clk_hddac, hddac_rate); 55062306a36Sopenharmony_ci if (ret < 0) 55162306a36Sopenharmony_ci DRM_ERROR("Cannot set rate (%dHz) for hda_hddac clk\n", 55262306a36Sopenharmony_ci hddac_rate); 55362306a36Sopenharmony_ci 55462306a36Sopenharmony_ci /* HDformatter clock = compositor clock */ 55562306a36Sopenharmony_ci ret = clk_set_rate(hda->clk_pix, mode->clock * 1000); 55662306a36Sopenharmony_ci if (ret < 0) 55762306a36Sopenharmony_ci DRM_ERROR("Cannot set rate (%dHz) for hda_pix clk\n", 55862306a36Sopenharmony_ci mode->clock * 1000); 55962306a36Sopenharmony_ci} 56062306a36Sopenharmony_ci 56162306a36Sopenharmony_cistatic void sti_hda_bridge_nope(struct drm_bridge *bridge) 56262306a36Sopenharmony_ci{ 56362306a36Sopenharmony_ci /* do nothing */ 56462306a36Sopenharmony_ci} 56562306a36Sopenharmony_ci 56662306a36Sopenharmony_cistatic const struct drm_bridge_funcs sti_hda_bridge_funcs = { 56762306a36Sopenharmony_ci .pre_enable = sti_hda_pre_enable, 56862306a36Sopenharmony_ci .enable = sti_hda_bridge_nope, 56962306a36Sopenharmony_ci .disable = sti_hda_disable, 57062306a36Sopenharmony_ci .post_disable = sti_hda_bridge_nope, 57162306a36Sopenharmony_ci .mode_set = sti_hda_set_mode, 57262306a36Sopenharmony_ci}; 57362306a36Sopenharmony_ci 57462306a36Sopenharmony_cistatic int sti_hda_connector_get_modes(struct drm_connector *connector) 57562306a36Sopenharmony_ci{ 57662306a36Sopenharmony_ci unsigned int i; 57762306a36Sopenharmony_ci int count = 0; 57862306a36Sopenharmony_ci struct sti_hda_connector *hda_connector 57962306a36Sopenharmony_ci = to_sti_hda_connector(connector); 58062306a36Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 58162306a36Sopenharmony_ci 58262306a36Sopenharmony_ci DRM_DEBUG_DRIVER("\n"); 58362306a36Sopenharmony_ci 58462306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(hda_supported_modes); i++) { 58562306a36Sopenharmony_ci struct drm_display_mode *mode = 58662306a36Sopenharmony_ci drm_mode_duplicate(hda->drm_dev, 58762306a36Sopenharmony_ci &hda_supported_modes[i].mode); 58862306a36Sopenharmony_ci if (!mode) 58962306a36Sopenharmony_ci continue; 59062306a36Sopenharmony_ci 59162306a36Sopenharmony_ci /* the first mode is the preferred mode */ 59262306a36Sopenharmony_ci if (i == 0) 59362306a36Sopenharmony_ci mode->type |= DRM_MODE_TYPE_PREFERRED; 59462306a36Sopenharmony_ci 59562306a36Sopenharmony_ci drm_mode_probed_add(connector, mode); 59662306a36Sopenharmony_ci count++; 59762306a36Sopenharmony_ci } 59862306a36Sopenharmony_ci 59962306a36Sopenharmony_ci return count; 60062306a36Sopenharmony_ci} 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci#define CLK_TOLERANCE_HZ 50 60362306a36Sopenharmony_ci 60462306a36Sopenharmony_cistatic enum drm_mode_status 60562306a36Sopenharmony_cisti_hda_connector_mode_valid(struct drm_connector *connector, 60662306a36Sopenharmony_ci struct drm_display_mode *mode) 60762306a36Sopenharmony_ci{ 60862306a36Sopenharmony_ci int target = mode->clock * 1000; 60962306a36Sopenharmony_ci int target_min = target - CLK_TOLERANCE_HZ; 61062306a36Sopenharmony_ci int target_max = target + CLK_TOLERANCE_HZ; 61162306a36Sopenharmony_ci int result; 61262306a36Sopenharmony_ci int idx; 61362306a36Sopenharmony_ci struct sti_hda_connector *hda_connector 61462306a36Sopenharmony_ci = to_sti_hda_connector(connector); 61562306a36Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 61662306a36Sopenharmony_ci 61762306a36Sopenharmony_ci if (!hda_get_mode_idx(*mode, &idx)) { 61862306a36Sopenharmony_ci return MODE_BAD; 61962306a36Sopenharmony_ci } else { 62062306a36Sopenharmony_ci result = clk_round_rate(hda->clk_pix, target); 62162306a36Sopenharmony_ci 62262306a36Sopenharmony_ci DRM_DEBUG_DRIVER("target rate = %d => available rate = %d\n", 62362306a36Sopenharmony_ci target, result); 62462306a36Sopenharmony_ci 62562306a36Sopenharmony_ci if ((result < target_min) || (result > target_max)) { 62662306a36Sopenharmony_ci DRM_DEBUG_DRIVER("hda pixclk=%d not supported\n", 62762306a36Sopenharmony_ci target); 62862306a36Sopenharmony_ci return MODE_BAD; 62962306a36Sopenharmony_ci } 63062306a36Sopenharmony_ci } 63162306a36Sopenharmony_ci 63262306a36Sopenharmony_ci return MODE_OK; 63362306a36Sopenharmony_ci} 63462306a36Sopenharmony_ci 63562306a36Sopenharmony_cistatic const 63662306a36Sopenharmony_cistruct drm_connector_helper_funcs sti_hda_connector_helper_funcs = { 63762306a36Sopenharmony_ci .get_modes = sti_hda_connector_get_modes, 63862306a36Sopenharmony_ci .mode_valid = sti_hda_connector_mode_valid, 63962306a36Sopenharmony_ci}; 64062306a36Sopenharmony_ci 64162306a36Sopenharmony_cistatic int sti_hda_late_register(struct drm_connector *connector) 64262306a36Sopenharmony_ci{ 64362306a36Sopenharmony_ci struct sti_hda_connector *hda_connector 64462306a36Sopenharmony_ci = to_sti_hda_connector(connector); 64562306a36Sopenharmony_ci struct sti_hda *hda = hda_connector->hda; 64662306a36Sopenharmony_ci 64762306a36Sopenharmony_ci hda_debugfs_init(hda, hda->drm_dev->primary); 64862306a36Sopenharmony_ci 64962306a36Sopenharmony_ci return 0; 65062306a36Sopenharmony_ci} 65162306a36Sopenharmony_ci 65262306a36Sopenharmony_cistatic const struct drm_connector_funcs sti_hda_connector_funcs = { 65362306a36Sopenharmony_ci .fill_modes = drm_helper_probe_single_connector_modes, 65462306a36Sopenharmony_ci .destroy = drm_connector_cleanup, 65562306a36Sopenharmony_ci .reset = drm_atomic_helper_connector_reset, 65662306a36Sopenharmony_ci .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state, 65762306a36Sopenharmony_ci .atomic_destroy_state = drm_atomic_helper_connector_destroy_state, 65862306a36Sopenharmony_ci .late_register = sti_hda_late_register, 65962306a36Sopenharmony_ci}; 66062306a36Sopenharmony_ci 66162306a36Sopenharmony_cistatic struct drm_encoder *sti_hda_find_encoder(struct drm_device *dev) 66262306a36Sopenharmony_ci{ 66362306a36Sopenharmony_ci struct drm_encoder *encoder; 66462306a36Sopenharmony_ci 66562306a36Sopenharmony_ci list_for_each_entry(encoder, &dev->mode_config.encoder_list, head) { 66662306a36Sopenharmony_ci if (encoder->encoder_type == DRM_MODE_ENCODER_DAC) 66762306a36Sopenharmony_ci return encoder; 66862306a36Sopenharmony_ci } 66962306a36Sopenharmony_ci 67062306a36Sopenharmony_ci return NULL; 67162306a36Sopenharmony_ci} 67262306a36Sopenharmony_ci 67362306a36Sopenharmony_cistatic int sti_hda_bind(struct device *dev, struct device *master, void *data) 67462306a36Sopenharmony_ci{ 67562306a36Sopenharmony_ci struct sti_hda *hda = dev_get_drvdata(dev); 67662306a36Sopenharmony_ci struct drm_device *drm_dev = data; 67762306a36Sopenharmony_ci struct drm_encoder *encoder; 67862306a36Sopenharmony_ci struct sti_hda_connector *connector; 67962306a36Sopenharmony_ci struct drm_connector *drm_connector; 68062306a36Sopenharmony_ci struct drm_bridge *bridge; 68162306a36Sopenharmony_ci int err; 68262306a36Sopenharmony_ci 68362306a36Sopenharmony_ci /* Set the drm device handle */ 68462306a36Sopenharmony_ci hda->drm_dev = drm_dev; 68562306a36Sopenharmony_ci 68662306a36Sopenharmony_ci encoder = sti_hda_find_encoder(drm_dev); 68762306a36Sopenharmony_ci if (!encoder) 68862306a36Sopenharmony_ci return -ENOMEM; 68962306a36Sopenharmony_ci 69062306a36Sopenharmony_ci connector = devm_kzalloc(dev, sizeof(*connector), GFP_KERNEL); 69162306a36Sopenharmony_ci if (!connector) 69262306a36Sopenharmony_ci return -ENOMEM; 69362306a36Sopenharmony_ci 69462306a36Sopenharmony_ci connector->hda = hda; 69562306a36Sopenharmony_ci 69662306a36Sopenharmony_ci bridge = devm_kzalloc(dev, sizeof(*bridge), GFP_KERNEL); 69762306a36Sopenharmony_ci if (!bridge) 69862306a36Sopenharmony_ci return -ENOMEM; 69962306a36Sopenharmony_ci 70062306a36Sopenharmony_ci bridge->driver_private = hda; 70162306a36Sopenharmony_ci bridge->funcs = &sti_hda_bridge_funcs; 70262306a36Sopenharmony_ci drm_bridge_attach(encoder, bridge, NULL, 0); 70362306a36Sopenharmony_ci 70462306a36Sopenharmony_ci connector->encoder = encoder; 70562306a36Sopenharmony_ci 70662306a36Sopenharmony_ci drm_connector = (struct drm_connector *)connector; 70762306a36Sopenharmony_ci 70862306a36Sopenharmony_ci drm_connector->polled = DRM_CONNECTOR_POLL_HPD; 70962306a36Sopenharmony_ci 71062306a36Sopenharmony_ci drm_connector_init(drm_dev, drm_connector, 71162306a36Sopenharmony_ci &sti_hda_connector_funcs, DRM_MODE_CONNECTOR_Component); 71262306a36Sopenharmony_ci drm_connector_helper_add(drm_connector, 71362306a36Sopenharmony_ci &sti_hda_connector_helper_funcs); 71462306a36Sopenharmony_ci 71562306a36Sopenharmony_ci err = drm_connector_attach_encoder(drm_connector, encoder); 71662306a36Sopenharmony_ci if (err) { 71762306a36Sopenharmony_ci DRM_ERROR("Failed to attach a connector to a encoder\n"); 71862306a36Sopenharmony_ci goto err_sysfs; 71962306a36Sopenharmony_ci } 72062306a36Sopenharmony_ci 72162306a36Sopenharmony_ci /* force to disable hd dacs at startup */ 72262306a36Sopenharmony_ci hda_enable_hd_dacs(hda, false); 72362306a36Sopenharmony_ci 72462306a36Sopenharmony_ci return 0; 72562306a36Sopenharmony_ci 72662306a36Sopenharmony_cierr_sysfs: 72762306a36Sopenharmony_ci return -EINVAL; 72862306a36Sopenharmony_ci} 72962306a36Sopenharmony_ci 73062306a36Sopenharmony_cistatic void sti_hda_unbind(struct device *dev, 73162306a36Sopenharmony_ci struct device *master, void *data) 73262306a36Sopenharmony_ci{ 73362306a36Sopenharmony_ci} 73462306a36Sopenharmony_ci 73562306a36Sopenharmony_cistatic const struct component_ops sti_hda_ops = { 73662306a36Sopenharmony_ci .bind = sti_hda_bind, 73762306a36Sopenharmony_ci .unbind = sti_hda_unbind, 73862306a36Sopenharmony_ci}; 73962306a36Sopenharmony_ci 74062306a36Sopenharmony_cistatic int sti_hda_probe(struct platform_device *pdev) 74162306a36Sopenharmony_ci{ 74262306a36Sopenharmony_ci struct device *dev = &pdev->dev; 74362306a36Sopenharmony_ci struct sti_hda *hda; 74462306a36Sopenharmony_ci struct resource *res; 74562306a36Sopenharmony_ci 74662306a36Sopenharmony_ci DRM_INFO("%s\n", __func__); 74762306a36Sopenharmony_ci 74862306a36Sopenharmony_ci hda = devm_kzalloc(dev, sizeof(*hda), GFP_KERNEL); 74962306a36Sopenharmony_ci if (!hda) 75062306a36Sopenharmony_ci return -ENOMEM; 75162306a36Sopenharmony_ci 75262306a36Sopenharmony_ci hda->dev = pdev->dev; 75362306a36Sopenharmony_ci 75462306a36Sopenharmony_ci /* Get resources */ 75562306a36Sopenharmony_ci res = platform_get_resource_byname(pdev, IORESOURCE_MEM, "hda-reg"); 75662306a36Sopenharmony_ci if (!res) { 75762306a36Sopenharmony_ci DRM_ERROR("Invalid hda resource\n"); 75862306a36Sopenharmony_ci return -ENOMEM; 75962306a36Sopenharmony_ci } 76062306a36Sopenharmony_ci hda->regs = devm_ioremap(dev, res->start, resource_size(res)); 76162306a36Sopenharmony_ci if (!hda->regs) 76262306a36Sopenharmony_ci return -ENOMEM; 76362306a36Sopenharmony_ci 76462306a36Sopenharmony_ci res = platform_get_resource_byname(pdev, IORESOURCE_MEM, 76562306a36Sopenharmony_ci "video-dacs-ctrl"); 76662306a36Sopenharmony_ci if (res) { 76762306a36Sopenharmony_ci hda->video_dacs_ctrl = devm_ioremap(dev, res->start, 76862306a36Sopenharmony_ci resource_size(res)); 76962306a36Sopenharmony_ci if (!hda->video_dacs_ctrl) 77062306a36Sopenharmony_ci return -ENOMEM; 77162306a36Sopenharmony_ci } else { 77262306a36Sopenharmony_ci /* If no existing video-dacs-ctrl resource continue the probe */ 77362306a36Sopenharmony_ci DRM_DEBUG_DRIVER("No video-dacs-ctrl resource\n"); 77462306a36Sopenharmony_ci hda->video_dacs_ctrl = NULL; 77562306a36Sopenharmony_ci } 77662306a36Sopenharmony_ci 77762306a36Sopenharmony_ci /* Get clock resources */ 77862306a36Sopenharmony_ci hda->clk_pix = devm_clk_get(dev, "pix"); 77962306a36Sopenharmony_ci if (IS_ERR(hda->clk_pix)) { 78062306a36Sopenharmony_ci DRM_ERROR("Cannot get hda_pix clock\n"); 78162306a36Sopenharmony_ci return PTR_ERR(hda->clk_pix); 78262306a36Sopenharmony_ci } 78362306a36Sopenharmony_ci 78462306a36Sopenharmony_ci hda->clk_hddac = devm_clk_get(dev, "hddac"); 78562306a36Sopenharmony_ci if (IS_ERR(hda->clk_hddac)) { 78662306a36Sopenharmony_ci DRM_ERROR("Cannot get hda_hddac clock\n"); 78762306a36Sopenharmony_ci return PTR_ERR(hda->clk_hddac); 78862306a36Sopenharmony_ci } 78962306a36Sopenharmony_ci 79062306a36Sopenharmony_ci platform_set_drvdata(pdev, hda); 79162306a36Sopenharmony_ci 79262306a36Sopenharmony_ci return component_add(&pdev->dev, &sti_hda_ops); 79362306a36Sopenharmony_ci} 79462306a36Sopenharmony_ci 79562306a36Sopenharmony_cistatic void sti_hda_remove(struct platform_device *pdev) 79662306a36Sopenharmony_ci{ 79762306a36Sopenharmony_ci component_del(&pdev->dev, &sti_hda_ops); 79862306a36Sopenharmony_ci} 79962306a36Sopenharmony_ci 80062306a36Sopenharmony_cistatic const struct of_device_id hda_of_match[] = { 80162306a36Sopenharmony_ci { .compatible = "st,stih416-hda", }, 80262306a36Sopenharmony_ci { .compatible = "st,stih407-hda", }, 80362306a36Sopenharmony_ci { /* end node */ } 80462306a36Sopenharmony_ci}; 80562306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, hda_of_match); 80662306a36Sopenharmony_ci 80762306a36Sopenharmony_cistruct platform_driver sti_hda_driver = { 80862306a36Sopenharmony_ci .driver = { 80962306a36Sopenharmony_ci .name = "sti-hda", 81062306a36Sopenharmony_ci .owner = THIS_MODULE, 81162306a36Sopenharmony_ci .of_match_table = hda_of_match, 81262306a36Sopenharmony_ci }, 81362306a36Sopenharmony_ci .probe = sti_hda_probe, 81462306a36Sopenharmony_ci .remove_new = sti_hda_remove, 81562306a36Sopenharmony_ci}; 81662306a36Sopenharmony_ci 81762306a36Sopenharmony_ciMODULE_AUTHOR("Benjamin Gaignard <benjamin.gaignard@st.com>"); 81862306a36Sopenharmony_ciMODULE_DESCRIPTION("STMicroelectronics SoC DRM driver"); 81962306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 820