mirror of
https://gitlab.freedesktop.org/emersion/libdisplay-info.git
synced 2025-01-13 20:01:23 +01:00
97a8f16d1f
di-edid-decode.c is getting bit and messy. Use one file per spec. Signed-off-by: Simon Ser <contact@emersion.fr>
276 lines
9.1 KiB
C
276 lines
9.1 KiB
C
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <libdisplay-info/displayid.h>
|
|
|
|
#include "di-edid-decode.h"
|
|
|
|
static bool is_displayid_base_block = true;
|
|
|
|
static void
|
|
print_displayid_display_params(const struct di_displayid_display_params *params)
|
|
{
|
|
printf(" Image size: %.1f mm x %.1f mm\n",
|
|
params->horiz_image_mm, params->vert_image_mm);
|
|
printf(" Display native pixel format: %dx%d\n",
|
|
params->horiz_pixels, params->vert_pixels);
|
|
|
|
printf(" Feature support flags:\n");
|
|
if (params->features->audio)
|
|
printf(" Audio support on video interface\n");
|
|
if (params->features->separate_audio_inputs)
|
|
printf(" Separate audio inputs provided\n");
|
|
if (params->features->audio_input_override)
|
|
printf(" Audio input override\n");
|
|
if (params->features->power_management)
|
|
printf(" Power management (DPM)\n");
|
|
if (params->features->fixed_timing)
|
|
printf(" Fixed timing\n");
|
|
if (params->features->fixed_pixel_format)
|
|
printf(" Fixed pixel format\n");
|
|
if (params->features->ai)
|
|
printf(" Support ACP, ISRC1, or ISRC2packets\n");
|
|
if (params->features->deinterlacing)
|
|
printf(" De-interlacing\n");
|
|
|
|
if (params->gamma != 0)
|
|
printf(" Gamma: %.2f\n", params->gamma);
|
|
printf(" Aspect ratio: %.2f\n", params->aspect_ratio);
|
|
printf(" Dynamic bpc native: %d\n", params->bits_per_color_native);
|
|
printf(" Dynamic bpc overall: %d\n", params->bits_per_color_overall);
|
|
}
|
|
|
|
static void
|
|
get_displayid_type_i_timing_aspect_ratio(enum di_displayid_type_i_timing_aspect_ratio ratio,
|
|
int *horiz, int *vert)
|
|
{
|
|
switch (ratio) {
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_1_1:
|
|
*horiz = *vert = 1;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_5_4:
|
|
*horiz = 5;
|
|
*vert = 4;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_4_3:
|
|
*horiz = 4;
|
|
*vert = 3;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_15_9:
|
|
*horiz = 15;
|
|
*vert = 9;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_16_9:
|
|
*horiz = 16;
|
|
*vert = 9;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_16_10:
|
|
*horiz = 16;
|
|
*vert = 10;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_64_27:
|
|
*horiz = 64;
|
|
*vert = 27;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_256_135:
|
|
*horiz = 256;
|
|
*vert = 135;
|
|
return;
|
|
case DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_UNDEFINED:
|
|
*horiz = *vert = 0;
|
|
return;
|
|
}
|
|
abort(); /* Unreachable */
|
|
}
|
|
|
|
static const char *
|
|
displayid_type_i_timing_stereo_3d_name(enum di_displayid_type_i_timing_stereo_3d stereo_3d)
|
|
{
|
|
switch (stereo_3d) {
|
|
case DI_DISPLAYID_TYPE_I_TIMING_STEREO_3D_NEVER:
|
|
return "no 3D stereo";
|
|
case DI_DISPLAYID_TYPE_I_TIMING_STEREO_3D_ALWAYS:
|
|
return "3D stereo";
|
|
case DI_DISPLAYID_TYPE_I_TIMING_STEREO_3D_USER:
|
|
return "3D stereo depends on user action";
|
|
}
|
|
abort(); /* Unreachable */
|
|
}
|
|
|
|
static const char *
|
|
displayid_type_i_timing_sync_polarity_name(enum di_displayid_type_i_timing_sync_polarity pol)
|
|
{
|
|
switch (pol) {
|
|
case DI_DISPLAYID_TYPE_I_TIMING_SYNC_NEGATIVE:
|
|
return "N";
|
|
case DI_DISPLAYID_TYPE_I_TIMING_SYNC_POSITIVE:
|
|
return "P";
|
|
}
|
|
abort(); /* Unreachable */
|
|
}
|
|
|
|
static void
|
|
print_displayid_type_i_timing(const struct di_displayid_type_i_timing *t)
|
|
{
|
|
int horiz_total, vert_total;
|
|
int horiz_back_porch, vert_back_porch;
|
|
int horiz_ratio, vert_ratio;
|
|
double pixel_clock_hz, refresh, horiz_freq_hz;
|
|
|
|
get_displayid_type_i_timing_aspect_ratio(t->aspect_ratio,
|
|
&horiz_ratio, &vert_ratio);
|
|
|
|
horiz_total = t->horiz_active + t->horiz_blank;
|
|
vert_total = t->vert_active + t->vert_blank;
|
|
pixel_clock_hz = t->pixel_clock_mhz * 1000 * 1000;
|
|
refresh = pixel_clock_hz / (horiz_total * vert_total);
|
|
horiz_freq_hz = pixel_clock_hz / horiz_total;
|
|
|
|
printf(" DTD:");
|
|
printf(" %5dx%-5d", t->horiz_active, t->vert_active);
|
|
if (t->interlaced) {
|
|
printf("i");
|
|
}
|
|
printf(" %10.6f Hz", refresh);
|
|
printf(" %3d:%-3d", horiz_ratio, vert_ratio);
|
|
printf(" %8.3f kHz %13.6f MHz", horiz_freq_hz / 1000,
|
|
t->pixel_clock_mhz);
|
|
printf(" (aspect ");
|
|
if (t->aspect_ratio == DI_DISPLAYID_TYPE_I_TIMING_ASPECT_RATIO_UNDEFINED)
|
|
printf("undefined");
|
|
else
|
|
printf("%d:%d", horiz_ratio, vert_ratio);
|
|
printf(", %s", displayid_type_i_timing_stereo_3d_name(t->stereo_3d));
|
|
if (t->preferred)
|
|
printf(", preferred");
|
|
printf(")\n");
|
|
|
|
horiz_back_porch = t->horiz_blank - t->horiz_sync_width - t->horiz_offset;
|
|
printf(" Hfront %4d Hsync %3d Hback %4d Hpol %s",
|
|
t->horiz_offset, t->horiz_sync_width, horiz_back_porch,
|
|
displayid_type_i_timing_sync_polarity_name(t->horiz_sync_polarity));
|
|
printf("\n");
|
|
|
|
vert_back_porch = t->vert_blank - t->vert_sync_width - t->vert_offset;
|
|
printf(" Vfront %4d Vsync %3d Vback %4d Vpol %s",
|
|
t->vert_offset, t->vert_sync_width, vert_back_porch,
|
|
displayid_type_i_timing_sync_polarity_name(t->vert_sync_polarity));
|
|
printf("\n");
|
|
}
|
|
|
|
static void
|
|
print_displayid_type_i_timing_block(const struct di_displayid_data_block *data_block)
|
|
{
|
|
size_t i;
|
|
const struct di_displayid_type_i_timing *const *timings;
|
|
|
|
timings = di_displayid_data_block_get_type_i_timings(data_block);
|
|
for (i = 0; timings[i] != NULL; i++)
|
|
print_displayid_type_i_timing(timings[i]);
|
|
}
|
|
|
|
static const char *
|
|
displayid_product_type_name(enum di_displayid_product_type type)
|
|
{
|
|
switch (type) {
|
|
case DI_DISPLAYID_PRODUCT_TYPE_EXTENSION:
|
|
return "Extension Section";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_TEST:
|
|
return "Test Structure; test equipment only";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_DISPLAY_PANEL:
|
|
return "Display panel or other transducer, LCD or PDP module, etc.";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_STANDALONE_DISPLAY:
|
|
return "Standalone display device";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_TV_RECEIVER:
|
|
return "Television receiver";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_REPEATER:
|
|
return "Repeater/translator";
|
|
case DI_DISPLAYID_PRODUCT_TYPE_DIRECT_DRIVE:
|
|
return "DIRECT DRIVE monitor";
|
|
}
|
|
abort();
|
|
}
|
|
|
|
static const char *
|
|
displayid_data_block_tag_name(enum di_displayid_data_block_tag tag)
|
|
{
|
|
switch (tag) {
|
|
case DI_DISPLAYID_DATA_BLOCK_PRODUCT_ID:
|
|
return "Product Identification Data Block (0x00)";
|
|
case DI_DISPLAYID_DATA_BLOCK_DISPLAY_PARAMS:
|
|
return "Display Parameters Data Block (0x01)";
|
|
case DI_DISPLAYID_DATA_BLOCK_COLOR_CHARACT:
|
|
return "Color Characteristics Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_I_TIMING:
|
|
return "Video Timing Modes Type 1 - Detailed Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_II_TIMING:
|
|
return "Video Timing Modes Type 2 - Detailed Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_III_TIMING:
|
|
return "Video Timing Modes Type 3 - Short Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_IV_TIMING:
|
|
return "Video Timing Modes Type 4 - DMT Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_VESA_TIMING:
|
|
return "Supported Timing Modes Type 1 - VESA DMT Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_CEA_TIMING:
|
|
return "Supported Timing Modes Type 2 - CTA-861 Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TIMING_RANGE_LIMITS:
|
|
return "Video Timing Range Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_PRODUCT_SERIAL:
|
|
return "Product Serial Number Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_ASCII_STRING:
|
|
return "GP ASCII String Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_DISPLAY_DEVICE_DATA:
|
|
return "Display Device Data Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_INTERFACE_POWER_SEQ:
|
|
return "Interface Power Sequencing Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TRANSFER_CHARACT:
|
|
return "Transfer Characteristics Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_DISPLAY_INTERFACE:
|
|
return "Display Interface Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_STEREO_DISPLAY_INTERFACE:
|
|
return "Stereo Display Interface Data Block (0x10)";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_V_TIMING:
|
|
return "Video Timing Modes Type 5 - Short Timings Data Block";
|
|
case DI_DISPLAYID_DATA_BLOCK_TILED_DISPLAY_TOPO:
|
|
return "Tiled Display Topology Data Block (0x12)";
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_VI_TIMING:
|
|
return "Video Timing Modes Type 6 - Detailed Timings Data Block";
|
|
}
|
|
abort();
|
|
}
|
|
|
|
void
|
|
print_displayid(const struct di_displayid *displayid)
|
|
{
|
|
const struct di_displayid_data_block *const *data_blocks;
|
|
const struct di_displayid_data_block *data_block;
|
|
enum di_displayid_data_block_tag tag;
|
|
size_t i;
|
|
const struct di_displayid_display_params *display_params;
|
|
|
|
printf(" Version: %d.%d\n", di_displayid_get_version(displayid),
|
|
di_displayid_get_revision(displayid));
|
|
|
|
if (is_displayid_base_block)
|
|
printf(" Display Product Type: %s\n",
|
|
displayid_product_type_name(di_displayid_get_product_type(displayid)));
|
|
is_displayid_base_block = false;
|
|
|
|
data_blocks = di_displayid_get_data_blocks(displayid);
|
|
for (i = 0; data_blocks[i] != NULL; i++) {
|
|
data_block = data_blocks[i];
|
|
tag = di_displayid_data_block_get_tag(data_block);
|
|
printf(" %s:\n", displayid_data_block_tag_name(tag));
|
|
switch (tag) {
|
|
case DI_DISPLAYID_DATA_BLOCK_DISPLAY_PARAMS:
|
|
display_params = di_displayid_data_block_get_display_params(data_block);
|
|
print_displayid_display_params(display_params);
|
|
break;
|
|
case DI_DISPLAYID_DATA_BLOCK_TYPE_I_TIMING:
|
|
print_displayid_type_i_timing_block(data_block);
|
|
break;
|
|
default:
|
|
break; /* Ignore */
|
|
}
|
|
}
|
|
}
|