Amlogic android7.1 s905x hdmi相关内容浅析

amlogic android7.1 s905x hdmi相关内容浅析

文件节点目录:./drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_main.c
1、读取edid原始数据:
stvs9:/ # cat ./sys/devices/virtual/amhdmitx/amhdmitx0/rawedid
00ffffffffffff0025e4032301010101011d0103805022782a96a1a355539f250a5054210800818081c0a9c0d1c001010101010101014eab70b8d1a042509060640820
4f3100001a000000fd00174c1f703c000a202020202020000000ff0053657269616c4e756d6265720a000000fc004c43442d474357513334315844015802033d704b10
0102030405202256585a23090707830100006a030c002000383c20000067d85dc401788001e30f0004e200cae60607018b6011e3056000e77c70a0d0a0295030203a00
204f3100001a565e00a0a0a0295030203500204f3100001a295900a0a038274030203a00204f3100001a000000000000000000000000ea

2、通过以下节点得到RX所支持的HDMI显示模式(分辨率)支持的情况
130|:/ # cat /sys/class/amhdmitx/amhdmitx0/disp_cap
480p60hz
720p60hz
1080i60hz
1080p60hz
1080p30hz
1080p24hz
2560x1080p60hz
2560x1080p60hz

3、设置分辨率
stvs9:/ # echo 1080p60hz > sys/class/display/mode //设置分辨率为1080p 3440x1440p60hz
stvs9:/ # cat sys/class/display/mode //查看当前设备分辨率
1080p60hz

4、HDMI在Spec 1.4中已经定义了的颜色空间有:
YCbCr 4:4:4、YCbCr 4:2:2、RGB 4:4:4
定义了的色深模式有:
8bit X 3,即24 bits
10bit X 3,即30 bits
12bit X 3,即36 bits
16bit X 3,即48 bits
到了HDMI2.0时,颜色空间增加了YCbCr 4:2:0

通过如下方式得到RX所支持的色彩空间/色深模式支持的情况
/ # cat /sys/class/amhdmitx/amhdmitx0/dc_cap
420,10bit
420,8bit
444,10bit
444,8bit
422,10bit
422,8bit
rgb,10bit
rgb,8bit

5、查看或者设置当前rx的color format参数
# cat sys/class/amhdmitx/amhdmitx0/attr
rgb,8bit
#
# echo rgb,8bit > sys/class/amhdmitx/amhdmitx0/attr

6、例如设定720p50hz Y444 10bit模式,那么通过
echo 444,10bit > /sys/class/amhdmitx/amhdmitx0/attr
echo 720p50hz > /sys/class/display/mode
便可以实现这种模式输出。
测试脚本,test.sh(见附件,具体参考脚本内容),使用举例如下:
#Examples:
system/bin/test.sh 2160p60hz420 420,8bit 0 1
#Limits:
#1) For YCbCr420 modes, only 2160p50hz420/2160p60hz420 support
#2) 444,12bit 444,10bit rgb,12bit rgb,10bit is not valid for 2160p60hz, 2160p50hz

echo rgb,8bit > sys/class/amhdmitx/amhdmitx0/attr
echo 3440x1440p60hz > sys/class/display/mode

脚本操作:
system/bin/test.sh 3440x1440p60hz rgb,8bit 0 0
system/bin/test.sh 3440x1440p75hz rgb,8bit 0 0

130|stvs9:/ # cat sys/class/display/axis
0 0 1920 1080 0 0 18 18

输出BIST COLORBAR模式:
echo bist1920 > /sys/class/amhdmitx/amhdmitx0/debug
输出BIST LINE模式:
echo bistline > /sys/class/amhdmitx/amhdmitx0/debug
输出BIST DOT模式:
echo bistdot > /sys/class/amhdmitx/amhdmitx0/debug
调整BIST起始位置:
echo biststart100 > /sys/class/amhdmitx/amhdmitx0/debug
关闭BIST:
echo bistoff > /sys/class/amhdmitx/amhdmitx0/debug

uboot状态下的hdmi输出:

hdmi output输出3440x1440p60hz
gxl_stvs9_v1#hdmitx output 3440x1440p60hz
set hdmitx VIC = 793
Not find VIC = 793 for hpll setting
hdmitx: not find VIC: 793
rx version is 2.0 div=10
hdmtix: set audio
hdmitx phy setting done

gxl_stvs9_v1#hdmitx output 3440x1440p60hz
set hdmitx VIC = 793
Not find VIC = 793 for hpll setting
hdmitx: not find VIC: 793
rx version is 2.0 div=10
hdmtix: set audio
hdmitx phy setting done
gxl_stvs9_v1#
gxl_stvs9_v1#
gxl_stvs9_v1#
gxl_stvs9_v1#
gxl_stvs9_v1#hdmitx output bist 1
gxl_stvs9_v1#
gxl_stvs9_v1#hdmitx info
3440x1440p60hz 793
cd4 cs1 cr1
frac_rate: 0
gxl_stvs9_v1#

设置分辨率的代码流程跟踪:
从节点开始:

/* ./drivers/amlogic/display/vout/vout_serve.c */
mode_store-->
-->set_vout_mode
	-->if (strcmp(name, local_name)) //判断分辨率是否跟当前分辨率一致

			/* 文件目录:./drivers/amlogic/display/vout/tv_vout.c */
		-->mode = validate_vmode(name) --> tv_validate_vmode;
			-->*info = get_valid_vinfo(mode);// 判断mode是否是有效、能使用的mode
				-->ret = set_current_vmode(mode); //设置当前video mode

common目录即(kernel部分):
驱动目录:
drivers/amlogic/hdmi/hdmi_tx_20/

文件:drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_main.c

		->amhdmitx_probe //从跑probe开始分析
			->INIT_DELAYED_WORK(&hdmi_delay_event_work, hdmi_delay_event_post); //初始化延时工作队列,
			->hdmitx_init_parameters(&hdmitx_device.hdmi_info); //hdmitx协议级接口; 在文件:./drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_video.c
				->
			
			->HDMITX_Meson_Init(&hdmitx_device); //HDMI TX初始化
				->hdmi_hwp_init(hdev);
					->hdmitx_uboot_already_display() // 文件:./drivers/amlogic/hdmi/hdmi_tx_20/hw/hdmi_tx_hw.c
						->get_hdmitx_format(); //获取记录HDMI TX当前hdmitx format参数,与uboot匹配(通过下面的读取寄存器值的方式)
							->return hd_read_reg(P_ISA_DEBUG_REG0); //读取寄存器 P_ISA_DEBUG_REG0 的值,这个寄存器值在uboot中设置
							  /* ISA_DEBUG_REG0 0x2600				//VIC值在./include/linux/amlogic/hdmi_tx/hdmi_common.h有枚举定义
							   * bit[11]: Y420
							   * bit[10:8]: HDMI VIC  //这里 #######获取VIC值#########,
							   * bit[7:0]: CEA VIC
							   */							   
			->hdmitx_init_fmt_attr(&hdmitx_device, fmt_attr);// ############ 设置当前color format ###########
				->memcpy(fmt_attr, "422,", 4); //例如
				->strcat(fmt_attr, "10bit"); //例如

				
			->hdmitx_device.task = kthread_run(hdmi_task_handle, &hdmitx_device, "kthread_hdmi"); //kthread_run创建和运行内核线程,入口函数是 hdmi_task_handle
				->hdmi_task_handle
					->switch_set_state(&hdmi_power, hdmitx_device->hpd_state);//设置hdmi拔插状态
					->INIT_WORK(&hdmitx_device->work_hdr, hdr_work_func); //初始化hdmi状态检测工作队列hdr_work_func
						->hdr_work_func
							->hdmitx_sdr_hdr_uevent(hdev); //hdmi拔插检测的uevent事件
								->switch_set_state(&hdmi_hdr, 1); //插入hdmi?
								->switch_set_state(&hdmi_hdr, 0); //拔出hdmi?
					 
				->hdmitx_edid_ram_buffer_clear(hdmitx_device); //初始化hdmi时,清除hdmitx模块edid ram和edid缓冲区
				->INIT_DELAYED_WORK(&hdmitx_device->work_hpd_plugin, hdmitx_hpd_plugin_handler); //初始化延时工作队列hdmitx_hpd_plugin_handler,检测hdmi插入
				->INIT_DELAYED_WORK(&hdmitx_device->work_hpd_plugout, hdmitx_hpd_plugout_handler); //初始化延时工作队列hdmitx_hpd_plugout_handler,检测hdmi拔出
					->void hdmitx_hpd_plugin_handler(struct work_struct *work); //延时工作队列函数
						->if (hdev->repeater_tx) //判断hdmi是否插入,设置hpd状态
							rx_repeat_hpd_state(1); //设置hpd状态为插入hdmi
						->hdmitx_get_edid(hdev); //获取EDID:通过i2c通讯的方式读取edid,读取128个字节,这里读取两次,
							||	->gpio_read_edid(hdev->EDID_buf);
									->edid_rx_data(regaddr & 0xff, rx_data, 128); //读取128个字节的edid
										->err = i2c_transfer(i2c_edid_client->adapter, msgs, ARRAY_SIZE(msgs));
							||调用
							\/
							------------------------------------------------------------
							文件./drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_edid.c
							->hdmitx_edid_clear(hdev); //清除HDMI Sink的EDID的解析结果,原函数在文件./drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_edid.c中
							->hdmitx_edid_parse(hdev); //解析读到的edid的内容--【128个字节】,
								->ret_val = Edid_DecodeHeader(&hdmitx_device->hdmi_info, &EDID_buf[0]); //判断edid的引导码是否正确,一般是:00ffffffffffff00开头的引导码
								->如果识别到的edid块数量为0,则  #################识别读取不到显示器edid####################
									->if (BlockCount == 0) {
										->hdmitx_edid_set_default_vic(hdmitx_device); //设置默认的vic值
										->
									}
								->hdmitx_edid_block_parse(hdmitx_device, &(EDID_buf[i*128])); //
									->Edid_DTD_parsing(struct rx_cap *pRXCap, unsigned char *data); //EDID DTD 详细时序解析
										->para = hdmi_match_dtd_paras(t); //调用 hdmi_match_dtd_paras()检查t时序参数是否与VIC对应时序参数匹配
											||
											\/
											文件:./drivers/amlogic/hdmi/hdmi_common/hdmi_parameters.c
											->*hdmi_match_dtd_paras(struct dtd *t) //这里hdmi_match_dtd_paras函数的t参数会和all_fmt_paras对应的参数比对,如果一致就返回对应参数对应结构体
												->return all_fmt_paras[i];																		\/
													->struct hdmi_format_para *all_fmt_paras[] = { //hdmi的显示参数结构体,分辨率、clk和前后阶参数等都在这里
														...
														&fmt_para_3840x2160p60_16x9,
														...
														}
											                     ||
																 \/
																	static struct hdmi_format_para fmt_para_3840x2160p60_16x9 = {
																				.vic = HDMI_3840x2160p60_16x9,
																				.name = "3840x2160p60hz",
																				.sname = "2160p60hz",
																				.pixel_repetition_factor = 0,
																				.progress_mode = 1,
																				.scrambler_en = 1,
																				.tmds_clk_div40 = 1,
																				.tmds_clk = 594000,
																				.timing = {
																						.pixel_freq = 594000,
																						.frac_freq = 593407,
																						.h_freq = 135000,
																						.v_freq = 60000,
																						.vsync_polarity = 1,
																						.hsync_polarity = 1,
																						.h_active = 3840,
																						.h_total = 4400,
																						.h_blank = 560,
																						.h_front = 176,
																						.h_sync = 88,
																						.h_back = 296,
																						.v_active = 2160,
																						.v_total = 2250,
																						.v_blank = 90,
																						.v_front = 8,
																						.v_sync = 10,
																						.v_back = 72,
																						.v_sync_ln = 1,
																						},
																	};
																
									   ->dump_dtd_info(t); //dump出hdmi的显示参数
							------------------------------------------------------------
						->set_disp_mode_auto(); //###########设置显示分辨率##############
							->hdmi_get_current_vinfo
								->info = get_current_vinfo();
									->vinfo_s *get_current_vinfo(void)
										->return &vinfo_1080p60hz; //这里分辨率设置为1080p60hz
											-> ||
											   \/
											   -----------------------------
											    104 const struct vinfo_s vinfo_1080p60hz = {
												105         .name = "1080p60hz",
												106         .mode = VMODE_1080P,
												107         .width = 1920,
												108         .height = 1080,
												109         .field_height = 1080,
												110         .aspect_ratio_num = 16,
												111         .aspect_ratio_den  = 9,
												112         .sync_duration_num = 60,
												113         .sync_duration_den = 1,
												114         .video_clk = 148500000,
												115 };
											   -----------------------------
										
						->hdmitx_edid_buf_compare_print(hdev); //打印edid buf内容到log输出
								||
								\/
								---------------------------------------
								文件:./drivers/amlogic/hdmi/hdmi_tx_20/hdmi_tx_edid.c
								->hdmitx_edid_buf_compare_print(struct hdmitx_dev *hdmitx_device) //假设DDC成功读取EDID两次,然后比较EDID_buf和EDID_buf1。如果相同,
																								  //只打印出EDID buf原始数据,否则打印出2个缓冲区数据
									->for (blk_idx = 0; blk_idx < valid_blk_no; blk_idx++)
										hdmitx_edid_blk_print(&buf0[blk_idx*128], blk_idx);									
									->valid_blk_no = hdmitx_edid_check_valid_blocks(buf1);
										for (blk_idx = 0; blk_idx < valid_blk_no; blk_idx++)
											hdmitx_edid_blk_print(&buf1[blk_idx*128], blk_idx);
								---------------------------------------

							||
							\/
						->set_disp_mode_auto(); //
							->hdmi_get_current_vinfo();//获取当前显示信息
								->para = hdmi_get_fmt_name(mode, fmt_attr); //
									->vic = hdmitx_edid_get_VIC(hdev, mode, 1); //
									->ret = hdmitx_set_display(hdev, vic); //
									->recalc_vinfo_sync_duration(info, hdev->frac_rate_policy);
									->hdmitx_set_audio(hdev, &(hdev->cur_audio_param), hdmi_ch); //打开hdmi音频

你可能感兴趣的:(Amlogic系列驱动开发,aml,amlogic,hdmi,hdmiout)