mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-17 07:34:03 +00:00
e1e96ba6a2
Changes in relation to FriendlyARM's U-Boot nanopi2-v2016.01: - nexell_display.c: Changed to DM, CONFIG_FB_ADDR can not be used anymore because framebuffer is allocated by video_reserve() in video-uclass.c. Therefore code changed appropriately. - '#ifdef CONFIG...' changed to 'if (IS_ENABLED(CONFIG...))' where possible (and similar). - livetree API (dev_read_...) is used instead of fdt one (fdt...). Signed-off-by: Stefan Bosch <stefan_b@posteo.net>
341 lines
10 KiB
C
341 lines
10 KiB
C
// SPDX-License-Identifier: GPL-2.0+
|
|
/*
|
|
* Copyright (C) 2016 Nexell Co., Ltd.
|
|
*
|
|
* Author: junghyun, kim <jhkim@nexell.co.kr>
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <common.h>
|
|
#include <errno.h>
|
|
#include <log.h>
|
|
#include <asm/arch/reset.h>
|
|
#include <asm/arch/nexell.h>
|
|
#include <asm/arch/display.h>
|
|
|
|
#include "soc/s5pxx18_soc_disptop.h"
|
|
#include "soc/s5pxx18_soc_dpc.h"
|
|
#include "soc/s5pxx18_soc_mlc.h"
|
|
|
|
#define MLC_LAYER_RGB_0 0 /* number of RGB layer 0 */
|
|
#define MLC_LAYER_RGB_1 1 /* number of RGB layer 1 */
|
|
#define MLC_LAYER_VIDEO 3 /* number of Video layer: 3 = VIDEO */
|
|
|
|
#define __io_address(a) (void *)(uintptr_t)(a)
|
|
|
|
void dp_control_init(int module)
|
|
{
|
|
void *base;
|
|
|
|
/* top */
|
|
base = __io_address(nx_disp_top_get_physical_address());
|
|
nx_disp_top_set_base_address(base);
|
|
|
|
/* control */
|
|
base = __io_address(nx_dpc_get_physical_address(module));
|
|
nx_dpc_set_base_address(module, base);
|
|
|
|
/* top controller */
|
|
nx_rstcon_setrst(RESET_ID_DISP_TOP, RSTCON_ASSERT);
|
|
nx_rstcon_setrst(RESET_ID_DISP_TOP, RSTCON_NEGATE);
|
|
|
|
/* display controller */
|
|
nx_rstcon_setrst(RESET_ID_DISPLAY, RSTCON_ASSERT);
|
|
nx_rstcon_setrst(RESET_ID_DISPLAY, RSTCON_NEGATE);
|
|
|
|
nx_dpc_set_clock_pclk_mode(module, nx_pclkmode_always);
|
|
}
|
|
|
|
int dp_control_setup(int module,
|
|
struct dp_sync_info *sync, struct dp_ctrl_info *ctrl)
|
|
{
|
|
unsigned int out_format;
|
|
unsigned int delay_mask;
|
|
int rgb_pvd = 0, hsync_cp1 = 7, vsync_fram = 7, de_cp2 = 7;
|
|
int v_vso = 1, v_veo = 1, e_vso = 1, e_veo = 1;
|
|
|
|
int interlace = 0;
|
|
int invert_field;
|
|
int swap_rb;
|
|
unsigned int yc_order;
|
|
int vck_select;
|
|
int vclk_invert;
|
|
int emb_sync;
|
|
|
|
enum nx_dpc_dither r_dither, g_dither, b_dither;
|
|
int rgb_mode = 0;
|
|
|
|
if (NULL == sync || NULL == ctrl) {
|
|
debug("error, dp.%d not set sync or pad clock info !!!\n",
|
|
module);
|
|
return -EINVAL;
|
|
}
|
|
|
|
out_format = ctrl->out_format;
|
|
delay_mask = ctrl->delay_mask;
|
|
interlace = sync->interlace;
|
|
invert_field = ctrl->invert_field;
|
|
swap_rb = ctrl->swap_RB;
|
|
yc_order = ctrl->yc_order;
|
|
vck_select = ctrl->vck_select;
|
|
vclk_invert = ctrl->clk_inv_lv0 | ctrl->clk_inv_lv1;
|
|
emb_sync = (out_format == DPC_FORMAT_CCIR656 ? 1 : 0);
|
|
|
|
/* set delay mask */
|
|
if (delay_mask & DP_SYNC_DELAY_RGB_PVD)
|
|
rgb_pvd = ctrl->d_rgb_pvd;
|
|
if (delay_mask & DP_SYNC_DELAY_HSYNC_CP1)
|
|
hsync_cp1 = ctrl->d_hsync_cp1;
|
|
if (delay_mask & DP_SYNC_DELAY_VSYNC_FRAM)
|
|
vsync_fram = ctrl->d_vsync_fram;
|
|
if (delay_mask & DP_SYNC_DELAY_DE_CP)
|
|
de_cp2 = ctrl->d_de_cp2;
|
|
|
|
if (ctrl->vs_start_offset != 0 ||
|
|
ctrl->vs_end_offset != 0 ||
|
|
ctrl->ev_start_offset != 0 || ctrl->ev_end_offset != 0) {
|
|
v_vso = ctrl->vs_start_offset;
|
|
v_veo = ctrl->vs_end_offset;
|
|
e_vso = ctrl->ev_start_offset;
|
|
e_veo = ctrl->ev_end_offset;
|
|
}
|
|
|
|
if (nx_dpc_format_rgb555 == out_format ||
|
|
nx_dpc_format_mrgb555a == out_format ||
|
|
nx_dpc_format_mrgb555b == out_format) {
|
|
r_dither = nx_dpc_dither_5bit;
|
|
g_dither = nx_dpc_dither_5bit;
|
|
b_dither = nx_dpc_dither_5bit;
|
|
rgb_mode = 1;
|
|
} else if (nx_dpc_format_rgb565 == out_format ||
|
|
nx_dpc_format_mrgb565 == out_format) {
|
|
r_dither = nx_dpc_dither_5bit;
|
|
b_dither = nx_dpc_dither_5bit;
|
|
g_dither = nx_dpc_dither_6bit, rgb_mode = 1;
|
|
} else if ((nx_dpc_format_rgb666 == out_format) ||
|
|
(nx_dpc_format_mrgb666 == out_format)) {
|
|
r_dither = nx_dpc_dither_6bit;
|
|
g_dither = nx_dpc_dither_6bit;
|
|
b_dither = nx_dpc_dither_6bit;
|
|
rgb_mode = 1;
|
|
} else {
|
|
r_dither = nx_dpc_dither_bypass;
|
|
g_dither = nx_dpc_dither_bypass;
|
|
b_dither = nx_dpc_dither_bypass;
|
|
rgb_mode = 1;
|
|
}
|
|
|
|
/* CLKGEN0/1 */
|
|
nx_dpc_set_clock_source(module, 0, ctrl->clk_src_lv0 == 3 ?
|
|
6 : ctrl->clk_src_lv0);
|
|
nx_dpc_set_clock_divisor(module, 0, ctrl->clk_div_lv0);
|
|
nx_dpc_set_clock_source(module, 1, ctrl->clk_src_lv1);
|
|
nx_dpc_set_clock_divisor(module, 1, ctrl->clk_div_lv1);
|
|
nx_dpc_set_clock_out_delay(module, 0, ctrl->clk_delay_lv0);
|
|
nx_dpc_set_clock_out_delay(module, 1, ctrl->clk_delay_lv1);
|
|
|
|
/* LCD out */
|
|
nx_dpc_set_mode(module, out_format, interlace, invert_field,
|
|
rgb_mode, swap_rb, yc_order, emb_sync, emb_sync,
|
|
vck_select, vclk_invert, 0);
|
|
nx_dpc_set_hsync(module, sync->h_active_len, sync->h_sync_width,
|
|
sync->h_front_porch, sync->h_back_porch,
|
|
sync->h_sync_invert);
|
|
nx_dpc_set_vsync(module, sync->v_active_len, sync->v_sync_width,
|
|
sync->v_front_porch, sync->v_back_porch,
|
|
sync->v_sync_invert, sync->v_active_len,
|
|
sync->v_sync_width, sync->v_front_porch,
|
|
sync->v_back_porch);
|
|
nx_dpc_set_vsync_offset(module, v_vso, v_veo, e_vso, e_veo);
|
|
nx_dpc_set_delay(module, rgb_pvd, hsync_cp1, vsync_fram, de_cp2);
|
|
nx_dpc_set_dither(module, r_dither, g_dither, b_dither);
|
|
|
|
if (IS_ENABLED(CONFIG_MACH_S5P6818)) {
|
|
/* Set TFT_CLKCTRL (offset : 1030h)
|
|
* Field name : DPC0_CLKCTRL, DPC1_CLKCRL
|
|
* Default value : clk_inv_lv0/1 = 0 : PADCLK_InvCLK
|
|
* Invert case : clk_inv_lv0/1 = 1 : PADCLK_CLK
|
|
*/
|
|
if (module == 0 && ctrl->clk_inv_lv0)
|
|
nx_disp_top_set_padclock(padmux_primary_mlc,
|
|
padclk_clk);
|
|
if (module == 1 && ctrl->clk_inv_lv1)
|
|
nx_disp_top_set_padclock(padmux_secondary_mlc,
|
|
padclk_clk);
|
|
}
|
|
|
|
debug("%s: dp.%d x:%4d, hf:%3d, hb:%3d, hs:%3d, hi=%d\n",
|
|
__func__, module, sync->h_active_len, sync->h_front_porch,
|
|
sync->h_back_porch, sync->h_sync_width, sync->h_sync_invert);
|
|
debug("%s: dp.%d y:%4d, vf:%3d, vb:%3d, vs:%3d, vi=%d\n",
|
|
__func__, module, sync->v_active_len, sync->v_front_porch,
|
|
sync->v_back_porch, sync->v_sync_width, sync->h_sync_invert);
|
|
debug("%s: dp.%d ck.0:%d:%d:%d, ck.1:%d:%d:%d\n",
|
|
__func__, module,
|
|
ctrl->clk_src_lv0, ctrl->clk_div_lv0, ctrl->clk_inv_lv0,
|
|
ctrl->clk_src_lv1, ctrl->clk_div_lv1, ctrl->clk_inv_lv1);
|
|
debug("%s: dp.%d vs:%d, ve:%d, es:%d, ee:%d\n",
|
|
__func__, module, v_vso, v_veo, e_vso, e_veo);
|
|
debug("%s: dp.%d delay RGB:%d, hs:%d, vs:%d, de:%d, fmt:0x%x\n",
|
|
__func__, module, rgb_pvd, hsync_cp1, vsync_fram, de_cp2,
|
|
out_format);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void dp_control_enable(int module, int on)
|
|
{
|
|
debug("%s: dp.%d top %s\n", __func__, module, on ? "ON" : "OFF");
|
|
|
|
nx_dpc_set_dpc_enable(module, on);
|
|
nx_dpc_set_clock_divisor_enable(module, on);
|
|
}
|
|
|
|
void dp_plane_init(int module)
|
|
{
|
|
void *base = __io_address(nx_mlc_get_physical_address(module));
|
|
|
|
nx_mlc_set_base_address(module, base);
|
|
nx_mlc_set_clock_pclk_mode(module, nx_pclkmode_always);
|
|
nx_mlc_set_clock_bclk_mode(module, nx_bclkmode_always);
|
|
}
|
|
|
|
int dp_plane_screen_setup(int module, struct dp_plane_top *top)
|
|
{
|
|
int width = top->screen_width;
|
|
int height = top->screen_height;
|
|
int interlace = top->interlace;
|
|
int video_prior = top->video_prior;
|
|
unsigned int bg_color = top->back_color;
|
|
|
|
/* MLC TOP layer */
|
|
nx_mlc_set_screen_size(module, width, height);
|
|
nx_mlc_set_layer_priority(module, video_prior);
|
|
nx_mlc_set_background(module, bg_color);
|
|
nx_mlc_set_field_enable(module, interlace);
|
|
nx_mlc_set_rgblayer_gama_table_power_mode(module, 0, 0, 0);
|
|
nx_mlc_set_rgblayer_gama_table_sleep_mode(module, 1, 1, 1);
|
|
nx_mlc_set_rgblayer_gamma_enable(module, 0);
|
|
nx_mlc_set_dither_enable_when_using_gamma(module, 0);
|
|
nx_mlc_set_gamma_priority(module, 0);
|
|
nx_mlc_set_top_power_mode(module, 1);
|
|
nx_mlc_set_top_sleep_mode(module, 0);
|
|
|
|
debug("%s: dp.%d screen %dx%d, %s, priority:%d, bg:0x%x\n",
|
|
__func__, module, width, height,
|
|
interlace ? "Interlace" : "Progressive",
|
|
video_prior, bg_color);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void dp_plane_screen_enable(int module, int on)
|
|
{
|
|
/* enable top screen */
|
|
nx_mlc_set_mlc_enable(module, on);
|
|
nx_mlc_set_top_dirty_flag(module);
|
|
debug("%s: dp.%d top %s\n", __func__, module, on ? "ON" : "OFF");
|
|
}
|
|
|
|
int dp_plane_layer_setup(int module, struct dp_plane_info *plane)
|
|
{
|
|
int sx = plane->left;
|
|
int sy = plane->top;
|
|
int ex = sx + plane->width - 1;
|
|
int ey = sy + plane->height - 1;
|
|
int pixel_byte = plane->pixel_byte;
|
|
int mem_lock_size = 16; /* fix mem lock size */
|
|
int layer = plane->layer;
|
|
unsigned int format = plane->format;
|
|
|
|
if (!plane->enable)
|
|
return -EINVAL;
|
|
|
|
/* MLC layer */
|
|
nx_mlc_set_lock_size(module, layer, mem_lock_size);
|
|
nx_mlc_set_alpha_blending(module, layer, 0, 15);
|
|
nx_mlc_set_transparency(module, layer, 0, 0);
|
|
nx_mlc_set_color_inversion(module, layer, 0, 0);
|
|
nx_mlc_set_rgblayer_invalid_position(module, layer, 0, 0, 0, 0, 0, 0);
|
|
nx_mlc_set_rgblayer_invalid_position(module, layer, 1, 0, 0, 0, 0, 0);
|
|
nx_mlc_set_format_rgb(module, layer, format);
|
|
nx_mlc_set_position(module, layer, sx, sy, ex, ey);
|
|
nx_mlc_set_rgblayer_stride(module, layer, pixel_byte,
|
|
plane->width * pixel_byte);
|
|
nx_mlc_set_rgblayer_address(module, layer, plane->fb_base);
|
|
|
|
debug("%s: dp.%d.%d %d * %d, %dbpp, fmt:0x%x\n",
|
|
__func__, module, layer, plane->width, plane->height,
|
|
pixel_byte * 8, format);
|
|
debug("%s: b:0x%x, l:%d, t:%d, r:%d, b:%d, hs:%d, vs:%d\n",
|
|
__func__, plane->fb_base, sx, sy, ex, ey,
|
|
plane->width * pixel_byte, pixel_byte);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dp_plane_set_enable(int module, int layer, int on)
|
|
{
|
|
int hl, hc;
|
|
int vl, vc;
|
|
|
|
debug("%s: dp.%d.%d %s:%s\n",
|
|
__func__, module, layer,
|
|
layer == MLC_LAYER_VIDEO ? "Video" : "RGB",
|
|
on ? "ON" : "OFF");
|
|
|
|
if (layer != MLC_LAYER_VIDEO) {
|
|
nx_mlc_set_layer_enable(module, layer, on);
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
return 0;
|
|
}
|
|
|
|
/* video layer */
|
|
if (on) {
|
|
nx_mlc_set_video_layer_line_buffer_power_mode(module, 1);
|
|
nx_mlc_set_video_layer_line_buffer_sleep_mode(module, 0);
|
|
nx_mlc_set_layer_enable(module, layer, 1);
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
} else {
|
|
nx_mlc_set_layer_enable(module, layer, 0);
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
nx_mlc_get_video_layer_scale_filter(module,
|
|
&hl, &hc, &vl, &vc);
|
|
if (hl || hc || vl || vc)
|
|
nx_mlc_set_video_layer_scale_filter(module, 0, 0, 0, 0);
|
|
nx_mlc_set_video_layer_line_buffer_power_mode(module, 0);
|
|
nx_mlc_set_video_layer_line_buffer_sleep_mode(module, 1);
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void dp_plane_layer_enable(int module,
|
|
struct dp_plane_info *plane, int on)
|
|
{
|
|
dp_plane_set_enable(module, plane->layer, on);
|
|
}
|
|
|
|
int dp_plane_set_address(int module, int layer, unsigned int address)
|
|
{
|
|
nx_mlc_set_rgblayer_address(module, layer, address);
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int dp_plane_wait_vsync(int module, int layer, int fps)
|
|
{
|
|
int cnt = 0;
|
|
|
|
if (fps == 0)
|
|
return (int)nx_mlc_get_dirty_flag(module, layer);
|
|
|
|
while (fps > cnt++) {
|
|
while (nx_mlc_get_dirty_flag(module, layer))
|
|
;
|
|
nx_mlc_set_dirty_flag(module, layer);
|
|
}
|
|
return 0;
|
|
}
|