本文转载自:http://blog.csdn.net/dearsq/article/details/55049182 Platform: RK3399 OS: Android 6.0 Version: v2016.08 本文分为两部分。 《[RK3399] 双屏异显代码实现流程分析(一)》为分析 RK video 部分标准的代码(base on 2017.2.13 updated) 《[RK3399] 双屏异显代码实现流程分析(二)》为打上双屏异显 patch 后的代码流程分析(eDP + mipi) 代码流程参考 KrisFei 大神总结的 3288 display 模块加载流程。 http://blog.csdn.net/kris_fei/article/details/52584903 KrisFei 归纳的代码流程如下: mipi dsi 接口信息初始化 fb相关信息读取 timing参数初始化 mipi dsi controller初始化 lcdc控制器注册
代码详解在 RK3399 上代码没有太大的变化。下面为 display 部分的标准流程。 在 make menuconfig 配置的时候 Location: | -> Device Drivers | -> Graphics support | -> Rockchip Misc Video driver | -> LCD Panel Select ( [=y]) drivers/video/rockchip/screen/Kconfig choice 包括 General lcd panel 和 rk mipi dsi lcd 差别是 < # CONFIG_LCD_GENERAL is not set
< CONFIG_LCD_MIPI=y
---
> CONFIG_LCD_GENERAL=y
> # CONFIG_LCD_MIPI is not set
分别对应的驱动文件是 lcd_general.c lcd_mipi.c 现在的版本(2017.2.13)中,lcd_general.c 还未实现代码。 所以我们从分析默认的 lcd_mipi.c 开始。 mipi dsi 接口信息初始化lcd_mipi.c rk_mipi_screen_init -> ——platform_driver_probe -> //name是rk_mipi_screen ————rk_mipi_screen_probe -> ——————rk_mipi_screen_init_dt ——————//读取mipi信息(包括 screen_init),dsi_lane,dsi_hs_clk,mipi_dsi_num, power, rst, gpio, 屏幕的 timing 信息(包括 sceen on cmds, cmd_type, cmd_delay, cmd_debug) fb相关信息读取rk_fb_init -> rk_fb.c ——platform_driver_register -> //name: “rockchip,rk-fb” ————rk_fb_probe -> //获取disp-mode, u-boot-logo-on等参数。 ——————rockchip_ion_client_create //创建ion client。 /* ION与PMEM类似,管理一或多个内存池,其中有一些会在boot time的时候预先分配,以备给特殊的硬件使用(GPU,显示控制器等)。它通过ION heaps来管理这些pool。它可以被userspace的process之间或者内核中的模块之间进行内存共享。*/ timing 参数初始化//不管是那种接口类型的lcd,lcd的时序参数都是要读取的。 rk_screen_init -> rk_screen.c ——platform_driver_register -> //name: “rk-screen” ————rk_screen_probe -> ——————rk_fb_prase_timing_dt -> rk_fb.c //读取来的配置存在结构体 rk_screen 变量中 ————————of_get_display_timing //获取时序参数,dts中可以配置多组,这里会循环读取。 ————————display_timings_get //根据当前native-mode来选取当前使用哪组时序参数。 ————————rk_fb_video_mode_from_timing //把 timing转换到fb video mode中去供后续使用。 mipi dsi controller 初始化//如果是另外的接口那就调用相应的接口控制器驱动来初始化. rk32_mipi_dsi_init -> rk32_mipi_dsi.c ——platform_driver_register -> //name: “rk32-mipi” ————rk32_mipi_dsi_probe -> //初始化struct dsi结构,包括clock, dsi ops, rk_screen 传递过来的参数, ——————rk_fb_get_prmry_screen -> rk_screen.c //获取在之前 rk_screen_probe() 中初始化的rk_screen变量. ————————//rk_mipi_dsi_probe -> //这个在 3399 代码中没有了 ————————register_dsi_ops //dsi->ops给dsi_ops ————————//dsi_probe_current_chip //检测dsi chip是否存在,这个在 3399 的代码中没有了 ————————rk_fb_trsm_ops_register //注册trsm_mipi_ops为trsm_dsi_ops 这里 3288 中的 rk_mipi_dsi_probe 在 3399 中被删掉了 直接在 rk_fb_get_prmry_screen 中 register_dsi_ops,也省略掉了 dsi_probe_current_chip lcdc控制器注册rk3368_lcdc_module_init -> rk3368_lcdc.c ——platform_driver_register -> //.name = “rk3368-lcdc”, ————rk3368_lcdc_probe -> ——————of_property_read_u32(np, “rockchip,prop”, &prop); ——————//判断屏幕是 primary 还是 extend,如果是 extend 会延后 register ——————rk3368_lcdc_parse_dt //读取lcdc控制器的参数 ——————dev_drv->ops = &lcdc_drv_ops; //lcdc对应ops ——————devm_request_irq //lcdc对应irq是rk3368_lcdc_isr() ——————rk_fb_register -> //对应ops是lcdc_drv_ops ————————rk_fb->lcdc_dev_drv[i] = dev_drv; //根据 RK30_MAX_LCDC_SUPPORT,循环注册两组 lcdc_dev_drv ————————init_lcdc_device_driver -> //初始化 lcdc_device_driver ——————————init_lcdc_win //一个lcdc能支持4层win. ——————————rk_disp_pwr_ctr_parse_dt //解析lcdc power ctrl相关内容。 ——————————rk_fb_set_prmry_screen ——————————rk_fb_trsm_ops_get //根据不同的屏幕类型选择对应的ops. ————————framebuffer_alloc //系统根据 win 的多少来创建相应数量的 fb ————————fb_videomode_to_var //将 fb_videomode 转化为 fb_var_screeninfo ————————dsp_mode == ONE_VOP_DUAL_MIPI_VER_SCAN ————————//判断双屏同显的刷新方式,这里如果是垂直刷新的话 ————————//设置 fbi->var.xres /= 2;fbi->var.yres = 2; fbi->var.xres_virtual /= 2; fbi->var.yres_virtual = 2; ————————fbi->fbops = &fb_ops; //fb ops ————————rkfb_create_sysfs //生成到/dev/graphics/fbx/下 ————————register_framebuffer ————————rkfb_create_sysfs ————————//以下 code 只跑一次 ————————kthread_run //创建rk_fb_wait_for_vsync_thread ————————dev_drv->ops->post_dspbuf //show logo for primary display device
|