/* SPDX-License-Identifier: GPL-2.0-only */ #include <stdlib.h> #include <tests/test.h> #include <edid.h> #include <assert.h> #include <string.h> #include <tests/lib/edid.h> struct test_state { int data_size; void *data; }; static uint8_t get_raw_edid_checksum(const unsigned char *x) { unsigned char sum = 0; int i; for (i = 0; i < 127; ++i) sum += x[i]; return 256 - sum; } static void edid_raw_calc_checksum(struct edid_raw *raw) { raw->checksum = get_raw_edid_checksum((const unsigned char *)raw); } static void test_decode_edid_no_edid(void **state) { assert_int_equal(EDID_ABSENT, decode_edid(NULL, 0, NULL)); } static void test_decode_edid_invalid_header(void **state) { struct edid_raw raw = {.header = EDID_HEADER_INVALID_RAW}; raw.checksum = get_raw_edid_checksum((const unsigned char *)&raw); assert_int_equal(EDID_ABSENT, decode_edid((unsigned char *)&raw, sizeof(raw), NULL)); } /* Frame is modified example of an LCD Desktop IT display from VESA E-EDID Standard Release A2. */ static int setup_decode_edid_basic_frame(void **state) { struct edid_raw raw = { EDID_RAW_DEFAULT_PARAMS, .video_input_type = EDID_ANALOG_VSI | EDID_SIGNAL_LEVEL_0 | EDID_VIDEO_SETUP_BLANK_EQ_BLACK | EDID_SEPARATE_SYNC_H_AND_V(1) | EDID_COMPOSITE_SYNC_H(1) | EDID_COMPOSITE_SYNC_ON_GREEN(1) | EDID_SERRATION_VSYNC(1), .horizontal_size = 43, /* [cm] */ .vertical_size = 32, /* [cm] */ .display_gamma = 120, /* 220% */ .supported_features = EDID_STANDBY_MODE(0) | EDID_SUSPEND_MODE(0) | EDID_ACTIVE_OFF(1) | EDID_COLOR_FORMAT_RGB444 | EDID_SRGB_SUPPORTED(0) | EDID_PREFERRED_TIMING_EXTENDED_INFO | EDID_DISPLAY_FREQUENCY_CONTINUOUS, .established_supported_timings = { [0] = EDID_ESTABLISHED_TIMINGS_1_720x400_70Hz | EDID_ESTABLISHED_TIMINGS_1_720x400_88Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_60Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_67Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_72Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_75Hz | EDID_ESTABLISHED_TIMINGS_1_800x600_56Hz | EDID_ESTABLISHED_TIMINGS_1_800x600_60Hz, [1] = EDID_ESTABLISHED_TIMINGS_2_800x600_72Hz | EDID_ESTABLISHED_TIMINGS_2_800x600_75Hz | EDID_ESTABLISHED_TIMINGS_2_832x624_75Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_80HzI | EDID_ESTABLISHED_TIMINGS_2_1024x768_60Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_70Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_75Hz | EDID_ESTABLISHED_TIMINGS_2_1280x1024_75Hz, }, .manufacturers_reserved_timing = EDID_MANUFACTURERS_TIMINGS_1152x870_75Hz, .standard_timings_supported = { [0] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1600), [1] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(85), [2] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1600), [3] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(75), [4] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1600), [5] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(70), [6] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1600), [7] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(65), [8] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [9] = EDID_ASPECT_RATIO_5_4 | EDID_FIELD_REFRESH_RATE(85), [10] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [11] = EDID_ASPECT_RATIO_5_4 | EDID_FIELD_REFRESH_RATE(60), [12] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1024), [13] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(85), [14] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(800), [15] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(85), }, .descriptor_block_1 = { [0] = EDID_PIXEL_CLOCK(162000000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(162000000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1600px, Horizontal Blanking is 560px. */ [2] = 0x40, [3] = 0x30, [4] = 0x62, /* Vertical Addressable Video is 1200 lines, Vertical Blanking is 50 lines. */ [5] = 0xB0, [6] = 0x32, [7] = 0x40, [8] = 64u, /* Horizontal Front Porch in pixels. */ [9] = 192u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 0x13, /* Vertical Front Porch is 1 line. */ [11] = 0x00, /* Vertical Sync Pulse Width is 3 lines. */ /* Horizontal Addressable Image Size is 427mm, Vertical Addressable Image Size is 320mm. */ [12] = 0xAB, [13] = 0x40, [14] = 0x11, [15] = 0x00, /* Horizontal Border Size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital separate syncs are requires. */ [17] = 0x1E, }, .descriptor_block_2 = { /* Display Range Limits Block Tag */ [0] = 0, [1] = 0, [2] = 0, [3] = 0xFD, [4] = 0, /* Horizontal and Vertical Rate Offsets are zero. */ [5] = 50u, /* Minimum Vertical Freq is 50Hz. */ [6] = 90u, /* Maximum Vertical Freq is 90Hz. */ [7] = 30u, /* Minimum Horizontal Freq is 30kHz. */ [8] = 110u, /* Maximum Horizontal Freq is 110kHz. */ [9] = 23u, /* Maximum Pixel Clock Freq is 230MHz. */ [10] = 0x4, /* Begin CVT Support Info */ [11] = 0x11, /* Compatible with CVT Version 1.1 */ [12] = 0, /* Maximum Pixel Clock Freq remains at 230MHz. */ [13] = 200, /* Maximum Active Pixels per Line is 1600. */ [14] = 0x90, /* Supported aspect ratios: 4:3, 5:4. */ /* Preferred Aspect Ratio is 4:3, Standard CVT Blanking is supported. */ [15] = 0, [16] = 0x50, /* H. & V. Stretch are supported and Shrinks are not. */ [17] = 60u, /* Preferred Refresh Rate is 60Hz. */ }, .descriptor_block_3 = { /* Established Timings III Block Tag */ [0] = 0, [1] = 0, [2] = 0, [3] = 0xF7, [4] = 0, [5] = 10u, /* VESA DMT Standard Version #10 */ /* * 640x350@85Hz, * 640x400@85Hz, * 720x400@85Hz, * 640x480@85Hz, * 800x600@85Hz, * 1024x768@85Hz, * 1152x864@75Hz are supported. */ [6] = 0x7F, /* * 1280x960@60Hz, * 1280x960@85Hz, * 1280x1024@60Hz, * 1280x1024@85Hz are supported. */ [7] = 0x0F, /* * 1400x1050@60Hz (Normal Blanking), * 1400x1050@75Hz are supported. */ [8] = 0x03, /* * 1400x1050@85Hz, * 1600x1200@60Hz, * 1600x1200@65Hz, * 1600x1200@70Hz are supported. */ [9] = 0x87, /* * 1600x1200@75Hz, * 1600x1200@85Hz are supported. */ [10] = 0xC0, /* 1920 timings not supported. */ [11] = 0x0, [12 ... 17] = 0, }, .descriptor_block_4 = { /* Display Product Name Block Tag */ [0] = 0, [1] = 0, [2] = 0, [3] = 0xFC, [4] = 0, /* Product name */ [5] = 'A', [6] = 'B', [7] = 'C', [8] = ' ', [9] = 'L', [10] = 'C', [11] = 'D', [12] = '2', [13] = '1', [14] = '\n', [15] = ' ', [16] = ' ', [17] = ' ', }, .extension_flag = 0x0, /* No extensions */ }; edid_raw_calc_checksum(&raw); *state = malloc(sizeof(struct test_state)); struct test_state ts = {.data_size = sizeof(struct edid_raw), .data = malloc(sizeof(struct edid_raw))}; memcpy(ts.data, &raw, sizeof(raw)); memcpy(*state, &ts, sizeof(ts)); return 0; } /* Test decoding of EDID frame without extensions. */ static void test_decode_edid_basic_frame(void **state) { struct edid out; struct test_state *ts = *state; /* In real-life situations frames often are not 100% conformant, but are at least correct when it comes to key data fields. */ assert_int_equal(EDID_CONFORMANT, decode_edid((unsigned char *)ts->data, ts->data_size, &out)); assert_int_equal(32, out.framebuffer_bits_per_pixel); assert_int_equal(0, out.panel_bits_per_color); assert_int_equal(0, out.panel_bits_per_pixel); assert_int_equal(0, out.link_clock); assert_int_equal(1600, out.x_resolution); assert_int_equal(1200, out.y_resolution); assert_int_equal(6400, out.bytes_per_line); assert_int_equal(0, out.hdmi_monitor_detected); assert_int_equal(0, strnlen(out.ascii_string, ARRAY_SIZE(out.ascii_string))); assert_string_equal(out.manufacturer_name, EDID_MANUFACTURER_NAME); /* Mode */ assert_null(out.mode.name); assert_int_equal(162000, out.mode.pixel_clock); assert_int_equal(1, out.mode.lvds_dual_channel); assert_int_equal(0, out.mode.refresh); assert_int_equal(1600, out.mode.ha); assert_int_equal(560, out.mode.hbl); assert_int_equal(64, out.mode.hso); assert_int_equal(192, out.mode.hspw); assert_int_equal(0, out.mode.hborder); assert_int_equal(1200, out.mode.va); assert_int_equal(50, out.mode.vbl); assert_int_equal(1, out.mode.vso); assert_int_equal(0, out.mode.vborder); assert_int_equal(43, out.mode.phsync); assert_int_equal(43, out.mode.pvsync); assert_int_equal(0, out.mode.x_mm); assert_int_equal(0, out.mode.y_mm); assert_int_equal(1, out.mode_is_supported[EDID_MODE_640x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_720x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1280x720_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1920x1080_60Hz]); } /* Frame is modified example of base EDID frame with CEA861 extension for DTV Display from VESA E-EDID Standard Release A2. */ static int setup_decode_edid_dtv_frame_with_extension(void **state) { struct edid_raw raw = { EDID_RAW_DEFAULT_PARAMS, .video_input_type = EDID_DIGITAL_VSI | EDID_INTERFACE_HDMI_A | EDID_COLOR_BIT_DEPTH_8B, .horizontal_size = 16, /* Aspect ratio 16:9 in landscape. */ .vertical_size = 0, /* Landscape flag */ .display_gamma = 120, /* 220% */ .supported_features = EDID_STANDBY_MODE(0) | EDID_SUSPEND_MODE(0) | EDID_ACTIVE_OFF(0) | EDID_COLOR_FORMAT_RGB444_YCRCB422_YCRCB422 | EDID_SRGB_SUPPORTED(1) | EDID_PREFERRED_TIMING_EXTENDED_INFO | EDID_DISPLAY_FREQUENCY_NON_CONTINUOUS, .established_supported_timings = { [0] = EDID_ESTABLISHED_TIMINGS_1_640x480_60Hz, [1] = 0, }, .manufacturers_reserved_timing = 0, .standard_timings_supported = { [0 ... 15] = 0, }, .descriptor_block_1 = { [0] = EDID_PIXEL_CLOCK(148500000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(148500000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1920px, Horizontal Blanking is 280px. */ [2] = 0x80, [3] = 0x18, [4] = 0x71, /* Vertical Addressable Video is 1080 lines, Vertical Blanking is 45 lines. */ [5] = 0x38, [6] = 0x2D, [7] = 0x40, [8] = 88u, /* Horizontal Front Porch in pixels. */ [9] = 44u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 4u, /* Vertical Front Porch is 4 lines. */ [11] = 5u, /* Vertical Sync Pulse Width is 5 lines. */ /* Horizontal Addressable Image Size is 1039mm, Vertical Addressable Image Size is 584mm. */ [12] = 0x0F, [13] = 0x48, [14] = 0x42, [15] = 0x00, /* Horizontal Border Size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital separate and syncs are requires. */ [17] = 0x1E, }, .descriptor_block_2 = { [0] = EDID_PIXEL_CLOCK(74250000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(74250000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1920px, Horizontal Blanking is 280px. */ [2] = 0x80, [3] = 0x18, [4] = 0x71, /* Vertical Addressable Video is 540 lines, Vertical Blanking is 22 lines. */ [5] = 0x1C, [6] = 0x16, [7] = 0x20, [8] = 88u, /* Horizontal Front Porch in pixels. */ [9] = 44u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 0x25, /* Vertical Front Porch is 2 lines. */ [11] = 0x00, /* Vertical Sync Pulse Width is 5 lines. */ /* Horizontal Addressable Image Size is 1039mm, Vertical Addressable Image Size is 584mm. */ [12] = 0x0F, [13] = 0x48, [14] = 0x42, [15] = 0x00, /* Horizontal Border Size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Interlaced Video, Stereo Video is not supported, Digital separate and syncs are requires. */ [17] = 0x9E, }, .descriptor_block_3 = { [0] = EDID_PIXEL_CLOCK(74250000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(74250000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1280px, Horizontal Blanking is 370px. */ [2] = 0x00, [3] = 0x72, [4] = 0x51, /* Vertical Addressable Video is 720 lines, Vertical Blanking is 30 lines. */ [5] = 0xD0, [6] = 0x1E, [7] = 0x20, [8] = 110u, /* Horizontal Front Porch in pixels. */ [9] = 40u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 0x55u, /* Vertical Front Porch is 5 lines. */ [11] = 0x00, /* Vertical Sync Pulse Width is 5 lines. */ /* Horizontal Addressable Image Size is 1039mm, Vertical Addressable Image Size is 584mm. */ [12] = 0x0F, [13] = 0x48, [14] = 0x42, [15] = 0x00, /* Horizontal Border Size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital separate syncs are requires. */ [17] = 0x1E, }, .descriptor_block_4 = { /* Display Product Name Block Tag */ [0] = 0, [1] = 0, [2] = 0, [3] = 0xFC, [4] = 0, /* Product name */ [5] = 'A', [6] = 'B', [7] = 'C', [8] = ' ', [9] = 'L', [10] = 'C', [11] = 'D', [12] = '4', [13] = '7', [14] = 'w', [15] = '\n', [16] = ' ', [17] = ' ', }, .extension_flag = 0x0, /* No extensions */ }; edid_raw_calc_checksum(&raw); unsigned char ext[128] = { [0] = 0x02, /* CEA 861 Extension Block Tag Code */ [1] = 0x03, /* CEA 861 Block Version */ [2] = 0x18, /* Detail Timing Descriptors start 0x18 bytes from here. */ /* Underscan is not supported, Basic Audio is supported, YCbCr 4:4:4 & YCbCr 4:2:2 are supported, Number of native formats: 2. */ [3] = 0x72, /* Video Data Block Tag Code is 2, Number of Short Video Descriptor Bytes is 7. */ [4] = 0x47, /* 1920x1080p 59.94/60 Hz 16 : 9 AR (CEA Format #16) is a supported Native Format. */ [5] = 0x90, /* 1920x1080i 59.94/60 Hz 16 : 9 AR (CEA Format #5) is a supported Native Format. */ [6] = 0x85, /* 1280x720p 59.94/60 Hz 16 : 9 AR (CEA Format #4) is a supported format. */ [7] = 0x04, /* 720x480p 59.94/60 Hz 16 : 9 AR (CEA Format #3) is a supported format. */ [8] = 0x03, /* 720x480p 59.94/60 Hz 4 : 3 AR (CEA Format #2) is a supported format. */ [9] = 0x02, /* 720x480i 59.94/60 Hz 16 : 9 AR (CEA Format #7) is a supported format. */ [10] = 0x07, /* 720x480i 59.94/60 Hz 4 : 3 AR (CEA Format #6) is a supported format. */ [11] = 0x06, /* Audio Data Block Tag Code is 1, Number of Short Audio Descriptor Bytes is 3. */ [12] = 0x23, /* Audio Format Tag Code is 1 --- LPCM is supported, Maximum number of audio channels is 2. */ [13] = 0x09, /* Supported Sampling Frequencies include: 48kHz; 44.1kHz & 32kHz. */ [14] = 0x07, /* Supported Sampling Bit Rates include: 24 bit; 20 bit & 16 bit. */ [15] = 0x07, /* Speaker Allocation Block Tag Code is 4, Number of Speaker Allocation Descriptor Bytes is 3. */ [16] = 0x83, /* Speaker Allocation is Front-Left & Front-Right. */ [17] = 0x01, /* Reserved */ [18 ... 19] = 0, /* Vendor Specific Data Block Tag Code is 3, Number of Vendor Specific Data Bytes is 5. */ [20] = 0x65, /* 24bit IEEE registration Identifier is 0x000C03. */ [21] = 0x03, [22] = 0x0C, [23] = 0x00, /* Vendor Specific Data is 0x10000. */ [24] = 0x01, [25] = 0x00, /* Descriptor Block 5 [18 Bytes] */ [26] = EDID_PIXEL_CLOCK(27027000u) & 0xFF, [27] = (EDID_PIXEL_CLOCK(27027000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 720px, Horizontal Blanking is 138 px. */ [28] = 0xD0, [29] = 0x8A, [30] = 0x20, /* Vertical Addressable Video is 480 lines, Vertical Blanking is 45 lines. */ [31] = 0xE0, [32] = 0x2D, [33] = 0x10, [34] = 16u, /* Horizontal Front Porch in pixels. */ [35] = 62u, /* Horizontal Sync Pulse Width in pixels. */ [36] = 0x96, /* Vertical Front Porch is 9 lines. */ [37] = 0x00, /* Vertical Sync Pulse Width is 6 lines. */ /* Displayed Image Aspect Ratio is 16:9 */ [38] = 16u, [39] = 9u, [40] = 0u, /* Horizontal and Vertical Border Size is 0 px */ [41] = 0u, [42] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [43] = 0x18, /* Descriptor Block 6 [18 Bytes] */ [44] = EDID_PIXEL_CLOCK(27027000u) & 0xFF, [45] = (EDID_PIXEL_CLOCK(27027000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 720px, Horizontal Blanking is 138 px. */ [46] = 0xD0, [47] = 0x8A, [48] = 0x20, /* Vertical Addressable Video is 480 lines, Vertical Blanking is 45 lines. */ [49] = 0xE0, [50] = 0x2D, [51] = 0x10, [52] = 16u, /* Horizontal Front Porch in pixels. */ [53] = 62u, /* Horizontal Sync Pulse Width in pixels. */ [54] = 0x96, /* Vertical Front Porch is 9 lines. */ [55] = 0x00, /* Vertical Sync Pulse Width is 6 lines. */ /* Displayed Image Aspect Ratio is 4:3. */ [56] = 4u, [57] = 3u, [58] = 0u, /* Horizontal and Vertical Border Size is 0px. */ [59] = 0u, [60] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [61] = 0x18, /* Descriptor Block 7 [18 Bytes] */ [62] = EDID_PIXEL_CLOCK(27027000u) & 0xFF, [63] = (EDID_PIXEL_CLOCK(27027000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1440px, Horizontal Blanking is 276 px. */ [64] = 0xA0, [65] = 0x14, [66] = 0x51, /* Vertical Addressable Video is 240 lines, Vertical Blanking is 23 lines. */ [67] = 0xF0, [68] = 0x16, [69] = 0x00, [70] = 38u, /* Horizontal Front Porch in pixels. */ [71] = 124u, /* Horizontal Sync Pulse Width in pixels. */ [72] = 0x43, /* Vertical Front Porch is 9 lines. */ [73] = 0x00, /* Vertical Sync Pulse Width is 6 lines. */ /* Displayed Image Aspect Ratio is 16:9 */ [74] = 16u, [75] = 9u, [76] = 0u, /* Horizontal and Vertical Border Size is 0px. */ [77] = 0u, [78] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [79] = 0x98, /* Descriptor Block 8 [18 Bytes] */ [80] = EDID_PIXEL_CLOCK(27027000u) & 0xFF, [81] = (EDID_PIXEL_CLOCK(27027000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1440px, Horizontal Blanking is 276 px. */ [82] = 0xA0, [83] = 0x14, [84] = 0x51, /* Vertical Addressable Video is 240 lines, Vertical Blanking is 23 lines. */ [85] = 0xF0, [86] = 0x16, [87] = 0x00, [88] = 38u, /* Horizontal Front Porch in pixels. */ [89] = 124u, /* Horizontal Sync Pulse Width in pixels. */ [90] = 0x43, /* Vertical Front Porch is 9 lines. */ [91] = 0x00, /* Vertical Sync Pulse Width is 6 lines. */ /* Displayed Image Aspect Ratio is 4:3. */ [92] = 4u, [93] = 3u, [94] = 0u, /* Horizontal and Vertical Border Size is 0px. */ [95] = 0u, [96] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [97] = 0x98, [99 ... 126] = 0}; ext[127] = get_raw_edid_checksum(ext); *state = malloc(sizeof(struct test_state)); struct test_state ts = {.data_size = sizeof(raw) + sizeof(ext), .data = malloc(sizeof(raw) + sizeof(ext))}; memcpy(ts.data, &raw, sizeof(raw)); memcpy(ts.data + sizeof(raw), &ext[0], sizeof(ext)); memcpy(*state, &ts, sizeof(ts)); return 0; } /* Test decoding of EDID frame with one extension. */ static void test_decode_edid_dtv_frame_with_extension(void **state) { struct edid out; struct test_state *ts = *state; /* In real-life situations frames often are not 100% conformant, but are at least correct when it comes to key data fields. */ assert_int_equal(EDID_CONFORMANT, decode_edid((unsigned char *)ts->data, ts->data_size, &out)); assert_int_equal(32, out.framebuffer_bits_per_pixel); assert_int_equal(8, out.panel_bits_per_color); assert_int_equal(24, out.panel_bits_per_pixel); assert_int_equal(0, out.link_clock); assert_int_equal(1920, out.x_resolution); assert_int_equal(1080, out.y_resolution); assert_int_equal(7680, out.bytes_per_line); assert_int_equal(1, out.hdmi_monitor_detected); assert_int_equal(0, strnlen(out.ascii_string, ARRAY_SIZE(out.ascii_string))); assert_string_equal(out.manufacturer_name, EDID_MANUFACTURER_NAME); /* Mode */ assert_null(out.mode.name); assert_int_equal(148500, out.mode.pixel_clock); assert_int_equal(1, out.mode.lvds_dual_channel); assert_int_equal(0, out.mode.refresh); assert_int_equal(1920, out.mode.ha); assert_int_equal(280, out.mode.hbl); assert_int_equal(88, out.mode.hso); assert_int_equal(44, out.mode.hspw); assert_int_equal(0, out.mode.hborder); assert_int_equal(1080, out.mode.va); assert_int_equal(45, out.mode.vbl); assert_int_equal(16, out.mode.vso); assert_int_equal(0, out.mode.vborder); assert_int_equal(43, out.mode.phsync); assert_int_equal(43, out.mode.pvsync); assert_int_equal(0, out.mode.x_mm); assert_int_equal(0, out.mode.y_mm); assert_int_equal(1, out.mode_is_supported[EDID_MODE_640x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_720x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1280x720_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1920x1080_60Hz]); } /* Test decoding of EDID frame with one extension. Tested frame is modified example of base EDID frame with CEA861 extension for IT/DTV Display from VESA E-EDID Standard Release A2. */ static int setup_decode_edid_it_dtv_frame_with_extension(void **state) { struct edid_raw raw = { EDID_RAW_DEFAULT_PARAMS, .video_input_type = EDID_DIGITAL_VSI | EDID_INTERFACE_HDMI_A | EDID_COLOR_BIT_DEPTH_8B, .horizontal_size = 121, /* Aspect ratio 16:9 in landscape */ .vertical_size = 68, /* Landscape flag */ .display_gamma = 120, /* 220% */ .supported_features = EDID_STANDBY_MODE(0) | EDID_SUSPEND_MODE(0) | EDID_ACTIVE_OFF(0) | EDID_COLOR_FORMAT_RGB444_YCRCB422_YCRCB422 | EDID_SRGB_SUPPORTED(1) | EDID_PREFERRED_TIMING_EXTENDED_INFO | EDID_DISPLAY_FREQUENCY_NON_CONTINUOUS, .established_supported_timings = { [0] = EDID_ESTABLISHED_TIMINGS_1_800x600_60Hz | EDID_ESTABLISHED_TIMINGS_1_800x600_56Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_75Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_72Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_67Hz | EDID_ESTABLISHED_TIMINGS_1_640x480_60Hz | EDID_ESTABLISHED_TIMINGS_1_720x400_88Hz | EDID_ESTABLISHED_TIMINGS_1_720x400_70Hz, [1] = EDID_ESTABLISHED_TIMINGS_2_1280x1024_75Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_75Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_70Hz | EDID_ESTABLISHED_TIMINGS_2_1024x768_60Hz | EDID_ESTABLISHED_TIMINGS_2_832x624_75Hz | EDID_ESTABLISHED_TIMINGS_2_800x600_75Hz | EDID_ESTABLISHED_TIMINGS_2_800x600_72Hz, }, .manufacturers_reserved_timing = EDID_MANUFACTURERS_TIMINGS_1152x870_75Hz, .standard_timings_supported = { [0] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [1] = EDID_ASPECT_RATIO_5_4 | EDID_FIELD_REFRESH_RATE(85), [2] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [3] = EDID_ASPECT_RATIO_5_4 | EDID_FIELD_REFRESH_RATE(60), [4] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [5] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(85), [6] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1280), [7] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(60), [8] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(1024), [9] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(85), [10] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(848), [11] = EDID_ASPECT_RATIO_16_9 | EDID_FIELD_REFRESH_RATE(60), [12] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(800), [13] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(60), [14] = EDID_HORIZONTAL_ACCESSIBLE_PIXELS(640), [15] = EDID_ASPECT_RATIO_4_3 | EDID_FIELD_REFRESH_RATE(60), }, .descriptor_block_1 = { [0] = EDID_PIXEL_CLOCK(85500000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(85500000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1360px, Horizontal Blanking is 432px. */ [2] = 0x50, [3] = 0xB0, [4] = 0x51, /* Vertical Addressable Video is 768 lines, Vertical Blanking is 27 lines. */ [5] = 0x00, [6] = 0x1B, [7] = 0x30, [8] = 64u, /* Horizontal Front Porch in pixels. */ [9] = 112u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 0x36, /* Vertical Front Porch is 3 lines. */ [11] = 0u, /* Vertical Sync Pulse Width is 6 lines. */ /* Horizontal Addressable Image Size is 1214mm, Vertical Addressable Image Size is 683mm. */ [12] = 0xBE, [13] = 0xAB, [14] = 0x42, [15] = 0x00, /* Horizontal border size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital separate and syncs are requires. */ [17] = 0x1E, }, .descriptor_block_2 = { [0] = EDID_PIXEL_CLOCK(74250000u) & 0xFF, [1] = (EDID_PIXEL_CLOCK(74250000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1280px, Horizontal Blanking is 370px. */ [2] = 0x00, [3] = 0x72, [4] = 0x51, /* Vertical Addressable Video is 720 lines, Vertical Blanking is 30 lines. */ [5] = 0xD0, [6] = 0x1E, [7] = 0x20, [8] = 110u, /* Horizontal Front Porch in pixels. */ [9] = 40u, /* Horizontal Pulse Sync Width in pixels. */ [10] = 0x55, /* Vertical Front Porch is 5 lines. */ [11] = 0x00, /* Vertical Sync Pulse Width is 5 lines. */ /* Horizontal Addressable Image Size is 1214mm, Vertical Addressable Image Size is 683mm. */ [12] = 0xBE, [13] = 0xAB, [14] = 0x42, [15] = 0x00, /* Horizontal border size is 0px. */ [16] = 0x00, /* Vertical Border Size is 0px. */ /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital separate and syncs are required. */ [17] = 0x1E, }, .descriptor_block_3 = { /* Established timings III Block Tag */ [0 ... 2] = 0u, [3] = 0xF7, [4] = 0u, /* VESA DMT Standard Version #10 */ [5] = 10u, /* * 640x350@85Hz, * 640x400@85Hz, * 720x400@85Hz, * 640x480@85Hz, * 800x600@85Hz, * 1024x768@85Hz, * 1152x864@75Hz are supported. */ [6] = 0x7F, /* * 1280x960@60Hz, * 1280x960@85Hz, * 1280x1024@60Hz, * 1280x1024@85Hz */ [7] = 0x0F, /* * 1400x1050@60Hz (Normal Blanking), * 1400x1050@75Hz are supported. */ [8] = 0x03, /* * 1400x1050@85Hz, * 1600x1200@60Hz, * 1600x1200@65Hz, * 1600x1200@70Hz are supported. */ [9] = 0x87, /* * 1600x1200@75Hz, * 1600x1200@85Hz are supported. */ [10] = 0xC0, /* 1920 PC Timings are not supported. */ [11] = 0u, /* Reserved */ [12 ... 17] = 0, }, .descriptor_block_4 = { /* Display Product Name Block Tag */ [0] = 0, [1] = 0, [2] = 0, [3] = 0xFC, [4] = 0, /* Product name */ [5] = 'A', [6] = 'B', [7] = 'C', [8] = ' ', [9] = 'P', [10] = 'L', [11] = 'A', [12] = '5', [13] = '5', [14] = '\n', [15] = ' ', [16] = ' ', [17] = ' ', }, .extension_flag = 0x0, /* No extensions */ }; edid_raw_calc_checksum(&raw); unsigned char ext[128] = { [0] = 0x02, /* CEA 861 Extension Block Tag Code */ [1] = 0x03, /* CEA 861 Block Version */ [2] = 0x17, /* Detail Timing Descriptors start 0x17 bytesfrom here */ /* Underscan is supported, Basic Audio is supported, YCbCr 4:4:4 & YCbCr 4:2:2 are supported, Number of native formats: 0. */ [3] = 0xF0, /* Video Data Block Tag Code is 2. Number of Short Video Descriptor Bytes is 6. */ [4] = 0x46, /* 1920x1080i 59.94/60 Hz 16 : 9 AR (CEA Format #5) is a supported format. */ [5] = 0x05, /* 1280x720p 59.94/60 Hz 16 : 9 AR (CEA Format #4) is a supported format. */ [6] = 0x04, /* 720x480p 59.94/60 Hz 16 : 9 AR (CEA Format #3) is a supported format. */ [7] = 0x03, /* 720x480p 59.94/60 Hz 4 : 3 AR (CEA Format #2) is a supported format. */ [8] = 0x02, /* 720x480i 59.94/60 Hz 16 : 9 AR (CEA Format #7) is a supported format. */ [9] = 0x07, /* 720x480i 59.94/60 Hz 4 : 3 AR (CEA Format #6) is a supported format. */ [10] = 0x06, /* Audio Data Block Tag Code is 1, Number of Short Audio Descriptor Bytes is 3. */ [11] = 0x23, /* Audio Format Tag Code is 1 --- LPCM is supported, Maximum number of audio channels is 2. */ [12] = 0x09, /* Supported Sampling Frequencies include: 48kHz; 44.1kHz & 32kHz. */ [13] = 0x07, /* Supported Sampling Bit Rates include: 24 bit; 20 bit & 16 bit. */ [14] = 0x07, /* Speaker Allocation Block Tag Code is 4, Number of Speaker Allocation Descriptor Bytes is 3. */ [15] = 0x83, /* Speaker Allocation is Front-Left & Front-Right */ [16] = 0x01, /* Reserved */ [17 ... 18] = 0, /* Vendor Specific Data Block Tag Code is 3, Number of Vendor Specific Data Bytes is 5. */ [19] = 0x65, /* 24bit IEEE registration Identifier is 0x000C03. */ [20] = 0x03, [21] = 0x0C, [22] = 0x00, /* Vendor Specific Data is 0x10000. */ [23] = 0x01, [24] = 0x00, /* Descriptor Block 5 [18 Bytes] */ [25] = EDID_PIXEL_CLOCK(74250000u) & 0xFF, [26] = (EDID_PIXEL_CLOCK(74250000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1920px, Horizontal Blanking is 280px. */ [27] = 0x80, [28] = 0x18, [29] = 0x71, /* Vertical Addressable Video is 540 lines, Vertical Blanking is 22 lines. */ [30] = 0x1C, [31] = 0x16, [32] = 0x20, [33] = 88u, /* Horizontal Front Porch in pixels. */ [34] = 44u, /* Horizontal Sync Pulse Width in pixels. */ [35] = 0x25, /* Vertical Front Porch is 2 lines. */ [36] = 0x00, /* Vertical Sync Pulse Width is 5 lines. */ /* Image size: 1039mm x 584mm */ [37] = 0x0F, [38] = 0x48, [39] = 0x42, /* Horizontal and Vertical Border Size is 0px. */ [40] = 0u, [41] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [42] = 0x9E, /* Descriptor Block 6 [18 Bytes] */ [43] = EDID_PIXEL_CLOCK(74250000u) & 0xFF, [44] = (EDID_PIXEL_CLOCK(74250000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1280px, Horizontal Blanking is 370 px. */ [45] = 0x00, [46] = 0x72, [47] = 0x51, /* Vertical Addressable Video is 720 lines, Vertical Blanking is 30 lines. */ [48] = 0xD0, [49] = 0x1E, [50] = 0x20, [51] = 110u, /* Horizontal Front Porch in pixels. */ [52] = 40u, /* Horizontal Sync Pulse Width in pixels. */ [53] = 0x55, /* Vertical Front Porch is 5 lines. */ [54] = 0x00, /* Vertical Sync Pulse Width is 5 lines. */ /* Image size: 1039mm x 584mm */ [55] = 0x0F, [56] = 0x48, [57] = 0x42, /* Horizontal and Vertical Border Size is 0px. */ [58] = 0u, [59] = 0u, /* Timing is Non-Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [60] = 0x1E, /* Descriptor Block 7 [18 Bytes] */ [61] = EDID_PIXEL_CLOCK(27000000u) & 0xFF, [62] = (EDID_PIXEL_CLOCK(27000000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1440px, Horizontal Blanking is 276 px. */ [63] = 0xA0, [64] = 0x14, [65] = 0x51, /* Vertical Addressable Video is 240 lines, Vertical Blanking is 23 lines. */ [66] = 0xF0, [67] = 0x16, [68] = 0x00, [69] = 38u, /* Horizontal Front Porch in pixels. */ [70] = 124u, /* Horizontal Sync Pulse Width in pixels. */ [71] = 0x43, /* Vertical Front Porch is 4 lines. */ [72] = 0x00, /* Vertical Sync Pulse Width is 3 lines. */ /* Image size: 1039mm x 584mm */ [73] = 0x0F, [74] = 0x48, [75] = 0x42, /* Horizontal and Vertical Border Size is 0px. */ [76] = 0u, [77] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [78] = 0x18, /* Descriptor Block 8 [18 Bytes] */ [79] = EDID_PIXEL_CLOCK(27027000u) & 0xFF, [80] = (EDID_PIXEL_CLOCK(27027000u) >> 8) & 0xFF, /* Horizontal Addressable Video is 1440px, Horizontal Blanking is 276 px. */ [81] = 0xA0, [82] = 0x14, [83] = 0x51, /* Vertical Addressable Video is 240 lines, Vertical Blanking is 23 lines. */ [84] = 0xF0, [85] = 0x16, [86] = 0x00, [87] = 38u, /* Horizontal Front Porch in pixels. */ [88] = 124u, /* Horizontal Sync Pulse Width in pixels. */ [89] = 0x43, /* Vertical Front Porch is 4 lines. */ [90] = 0x00, /* Vertical Sync Pulse Width is 3 lines. */ /* Image size: 1039mm x 584mm */ [91] = 0x0F, [92] = 0x48, [93] = 0x42, /* Horizontal and Vertical Border Size is 0px. */ [94] = 0u, [95] = 0u, /* Timing is Interlaced Video, Stereo Video is not supported, Digital Separate Syncs are required. */ [96] = 0x98, [97 ... 126] = 0, }; ext[127] = get_raw_edid_checksum(ext); *state = malloc(sizeof(struct test_state)); struct test_state ts = {.data_size = sizeof(raw) + sizeof(ext), .data = malloc(sizeof(raw) + sizeof(ext))}; memcpy(ts.data, &raw, sizeof(raw)); memcpy(ts.data + sizeof(raw), &ext[0], sizeof(ext)); memcpy(*state, &ts, sizeof(ts)); return 0; } static void test_decode_edid_it_dtv_frame_with_extension(void **state) { struct edid out; struct test_state *ts = *state; /* In real-life situations frames often are not 100% conformant, but are at least correct when it comes to key data fields. */ assert_int_equal(EDID_CONFORMANT, decode_edid((unsigned char *)ts->data, ts->data_size, &out)); assert_int_equal(32, out.framebuffer_bits_per_pixel); assert_int_equal(8, out.panel_bits_per_color); assert_int_equal(24, out.panel_bits_per_pixel); assert_int_equal(0, out.link_clock); assert_int_equal(1360, out.x_resolution); assert_int_equal(768, out.y_resolution); assert_int_equal(5440, out.bytes_per_line); assert_int_equal(1, out.hdmi_monitor_detected); assert_int_equal(0, strnlen(out.ascii_string, ARRAY_SIZE(out.ascii_string))); assert_string_equal(out.manufacturer_name, EDID_MANUFACTURER_NAME); /* Mode */ assert_null(out.mode.name); assert_int_equal(85500, out.mode.pixel_clock); assert_int_equal(0, out.mode.lvds_dual_channel); assert_int_equal(0, out.mode.refresh); assert_int_equal(1360, out.mode.ha); assert_int_equal(432, out.mode.hbl); assert_int_equal(64, out.mode.hso); assert_int_equal(112, out.mode.hspw); assert_int_equal(0, out.mode.hborder); assert_int_equal(768, out.mode.va); assert_int_equal(27, out.mode.vbl); assert_int_equal(3, out.mode.vso); assert_int_equal(0, out.mode.vborder); assert_int_equal(43, out.mode.phsync); assert_int_equal(43, out.mode.pvsync); assert_int_equal(0, out.mode.x_mm); assert_int_equal(0, out.mode.y_mm); assert_int_equal(1, out.mode_is_supported[EDID_MODE_640x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_720x480_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1280x720_60Hz]); assert_int_equal(0, out.mode_is_supported[EDID_MODE_1920x1080_60Hz]); } static int teardown_edid_test(void **state) { struct test_state *ts; if (*state == NULL) return 0; ts = (struct test_state *)*state; free(ts->data); free(ts); return 0; } static void test_edid_set_framebuffer_bits_per_pixel(void **state) { struct edid out; struct test_state *ts = *state; decode_edid((unsigned char *)ts->data, ts->data_size, &out); edid_set_framebuffer_bits_per_pixel(&out, 16, 2); assert_int_equal(16, out.framebuffer_bits_per_pixel); assert_int_equal(out.mode.ha * 2, out.bytes_per_line); assert_int_equal(out.bytes_per_line / (16 / 8), out.x_resolution); assert_int_equal(out.mode.va, out.y_resolution); edid_set_framebuffer_bits_per_pixel(&out, 24, 4); assert_int_equal(24, out.framebuffer_bits_per_pixel); assert_int_equal(out.mode.ha * 3, out.bytes_per_line); assert_int_equal(out.bytes_per_line / (24 / 8), out.x_resolution); assert_int_equal(out.mode.va, out.y_resolution); edid_set_framebuffer_bits_per_pixel(&out, 32, 4); assert_int_equal(32, out.framebuffer_bits_per_pixel); assert_int_equal(out.mode.ha * 4, out.bytes_per_line); assert_int_equal(out.bytes_per_line / (32 / 8), out.x_resolution); assert_int_equal(out.mode.va, out.y_resolution); } int main(void) { const struct CMUnitTest tests[] = { cmocka_unit_test(test_decode_edid_no_edid), cmocka_unit_test(test_decode_edid_invalid_header), cmocka_unit_test_setup_teardown(test_decode_edid_basic_frame, setup_decode_edid_basic_frame, teardown_edid_test), cmocka_unit_test_setup_teardown(test_decode_edid_dtv_frame_with_extension, setup_decode_edid_dtv_frame_with_extension, teardown_edid_test), cmocka_unit_test_setup_teardown(test_decode_edid_it_dtv_frame_with_extension, setup_decode_edid_it_dtv_frame_with_extension, teardown_edid_test), cmocka_unit_test_setup_teardown(test_edid_set_framebuffer_bits_per_pixel, setup_decode_edid_basic_frame, teardown_edid_test), }; return cb_run_group_tests(tests, NULL, NULL); }