4
0
mirror of https://github.com/RT-Thread/rt-thread.git synced 2025-01-25 10:47:23 +08:00

140 lines
4.2 KiB
C
Raw Normal View History

/*
* Copyright (c) 2023 HPMicro
*
* SPDX-License-Identifier: BSD-3-Clause
*
*/
#include "hpm_panel.h"
#include "hpm_pixelmux_drv.h"
#include "hpm_lvb_drv.h"
static void lvds_panel_lvb_init(hpm_panel_t *panel)
{
LVB_Type *lvb_base = panel->hw_if.video.lvds.lvb_base;
lvb_config_t lvb_config;
lvb_get_default_config(&lvb_config);
lvb_config.split_mode_en = false;
lvb_config.txclk_shift = lvb_txclk_shift_1100011;
lvb_init(lvb_base, &lvb_config);
lvb_ch_config_t lvb_ch_cfg;
lvb_ch_cfg.map = lvb_ch_mapping_vesa;
if (panel->hw_if.video.lvds.channel_di_index == 0)
lvb_ch_cfg.data_src = lvb_ch_data_source_di0;
else
lvb_ch_cfg.data_src = lvb_ch_data_source_di1;
if (panel->hw_if.video.lvds.channel_index == 0) {
lvb_ch_config(lvb_base, lvb_ch_num_0, &lvb_ch_cfg);
lvb_ch_enable(lvb_base, lvb_ch_num_0);
} else {
lvb_ch_config(lvb_base, lvb_ch_num_1, &lvb_ch_cfg);
lvb_ch_enable(lvb_base, lvb_ch_num_1);
}
}
static void lvds_panel_phy_init(hpm_panel_t *panel)
{
LVB_Type *lvb_base = panel->hw_if.video.lvds.lvb_base;
lvds_phy_clk_param_t param;
uint32_t pixel_clk = panel->hw_if.lcdc_pixel_clk_khz * 1000;
pixelmux_lvds_phy_calc_pll_cfg(pixel_clk, false, &param);
lvb_lvds_phy_lane_config_t lvds_lane_cfg;
lvb_lvds_phy_lane_get_default_config(&lvds_lane_cfg);
lvds_lane_cfg.fvco_div4 = param.reg.data_rate_div4;
if (panel->hw_if.video.lvds.channel_index == 0) {
pixelmux_config_lvds_tx_phy0_clk(&param.reg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds0_tx0, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds0_tx1, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds0_tx2, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds0_tx3, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds0_txck, &lvds_lane_cfg);
lvb_lvds_phy0_poweron(lvb_base);
while (lvb_lvds_phy0_pll_is_lock(lvb_base) == false) {
}
} else {
pixelmux_config_lvds_tx_phy1_clk(&param.reg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds1_tx0, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds1_tx1, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds1_tx2, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds1_tx3, &lvds_lane_cfg);
lvb_lvds_phy_lane_init(lvb_base, lvb_lvds_lane_idx_lvds1_txck, &lvds_lane_cfg);
lvb_lvds_phy1_poweron(lvb_base);
while (lvb_lvds_phy1_pll_is_lock(lvb_base) == false) {
}
}
}
static void reset(hpm_panel_t *panel)
{
if (!panel->hw_if.set_reset_pin_level)
return;
panel->hw_if.set_reset_pin_level(0);
hpm_panel_delay_us(2000);
panel->hw_if.set_reset_pin_level(1);
hpm_panel_delay_us(2000);
}
static void init(hpm_panel_t *panel)
{
if (panel->hw_if.set_video_router)
panel->hw_if.set_video_router();
lvds_panel_lvb_init(panel);
lvds_panel_phy_init(panel);
}
static void power_on(hpm_panel_t *panel)
{
if (panel->state.power_state != HPM_PANEL_STATE_POWER_ON &&
panel->hw_if.set_backlight) {
if (panel->state.backlight_percent == 0)
panel->state.backlight_percent = 100;
panel->hw_if.set_backlight(panel->state.backlight_percent);
panel->state.power_state = HPM_PANEL_STATE_POWER_ON;
}
}
static void power_off(hpm_panel_t *panel)
{
if (panel->state.power_state != HPM_PANEL_STATE_POWER_OFF &&
panel->hw_if.set_backlight) {
panel->hw_if.set_backlight(0);
panel->state.power_state = HPM_PANEL_STATE_POWER_OFF;
}
}
hpm_panel_t panel_cc10128007 = {
.name = "cc10128007",
.if_type = HPM_PANEL_IF_TYPE_LVDS_SINGLE,
.timing = {
.pixel_clock_khz = 74250,
.hactive = 800,
.hfront_porch = 60,
.hback_porch = 60,
.hsync_len = 40,
.vactive = 1280,
.vfront_porch = 18,
.vback_porch = 18,
.vsync_len = 6,
},
.funcs = {
.reset = reset,
.init = init,
.power_on = power_on,
.power_off = power_off,
},
};