[igt-dev] [PATCH i-g-t v2] lib/igt_edid: new library for generating EDIDs

Ser, Simon simon.ser at intel.com
Tue Apr 16 08:46:53 UTC 2019


Thanks for your review!

On Tue, 2019-04-16 at 11:11 +0300, Arkadiusz Hiler wrote:
> On Mon, Apr 15, 2019 at 05:20:43PM +0300, Simon Ser wrote:
> > For the purposes of testing different EDID features, we need to generate more
> > and more complex EDID blobs (e.g. with audio support). However currently IGT
> > uses a macro-based system to generate EDIDs. This doesn't scale well and is
> > pretty inflexible.
> > 
> > This commit introduces a new little library to generate EDIDs. For now it can't
> > do more than the old macro. Future commits will extend the API.
> > 
> > The structures are mostly based on the Linux kernel code (drm_edid.h). Setters
> > have been added for convenience.
> > 
> > Signed-off-by: Simon Ser <simon.ser at intel.com>
> > ---
> > 
> > Changes in v2: fix include guard name
> > 
> > Adding Ville because he wrote the current EDID code, and Arek because he's
> > interested in this patch.
> > 
> >  lib/igt_edid.c          | 268 ++++++++++++++++++++++++++++++++++++++++
> >  lib/igt_edid.h          | 196 +++++++++++++++++++++++++++++
> >  lib/igt_edid_template.h |  74 -----------
> >  lib/igt_kms.c           | 113 +++++++++--------
> >  lib/meson.build         |   3 +-
> >  5 files changed, 521 insertions(+), 133 deletions(-)
> >  create mode 100644 lib/igt_edid.c
> >  create mode 100644 lib/igt_edid.h
> >  delete mode 100644 lib/igt_edid_template.h
> > 
> > diff --git a/lib/igt_edid.c b/lib/igt_edid.c
> > new file mode 100644
> > index 00000000..155ebebe
> > --- /dev/null
> > +++ b/lib/igt_edid.c
> > @@ -0,0 +1,268 @@
> > +/*
> > + * Copyright © 2019 Intel Corporation
> > + *
> > + * Permission is hereby granted, free of charge, to any person obtaining a
> > + * copy of this software and associated documentation files (the "Software"),
> > + * to deal in the Software without restriction, including without limitation
> > + * the rights to use, copy, modify, merge, publish, distribute, sublicense,
> > + * and/or sell copies of the Software, and to permit persons to whom the
> > + * Software is furnished to do so, subject to the following conditions:
> > + *
> > + * The above copyright notice and this permission notice (including the next
> > + * paragraph) shall be included in all copies or substantial portions of the
> > + * Software.
> > + *
> > + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
> > + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
> > + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
> > + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
> > + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
> > + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
> > + * IN THE SOFTWARE.
> > + *
> > + * Authors: Simon Ser <simon.ser at intel.com>
> > + */
> > +
> > +#include "config.h"
> > +
> > +#include <assert.h>
> > +#include <string.h>
> > +#include <stdint.h>
> > +#include <time.h>
> > +#include <xf86drmMode.h>
> > +
> > +#include "igt_core.h"
> > +#include "igt_edid.h"
> > +
> > +static const char edid_header[] = {
> > +	0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x00
> > +};
> > +
> > +/* vfreq is in Hz */
> > +static void std_timing_set(struct std_timing *st, int hsize, int vfreq,
> > +			   enum std_timing_aspect aspect)
> > +{
> > +	assert(hsize >= 256 && hsize <= 2288);
> > +	st->hsize = hsize / 8 - 31;
> > +	st->vfreq_aspect = aspect << 6 | (vfreq - 60);
> > +}
> > +
> > +static void std_timing_unset(struct std_timing *st)
> > +{
> > +	memset(st, 0x01, sizeof(struct std_timing));
> > +}
> > +
> > +/**
> > + * detailed_timing_set_mode: fill a detailed timing based on a mode
> > + */
> > +void detailed_timing_set_mode(struct detailed_timing *dt, drmModeModeInfo *mode,
> > +			      int width_mm, int height_mm)
> > +{
> > +	int hactive, hblank, vactive, vblank, hsync_offset, hsync_pulse_width,
> > +	    vsync_offset, vsync_pulse_width;
> > +	struct detailed_pixel_timing *pt = &dt->data.pixel_data;
> > +
> > +	hactive = mode->hdisplay;
> > +	hsync_offset = mode->hsync_start - mode->hdisplay;
> > +	hsync_pulse_width = mode->hsync_end - mode->hsync_start;
> > +	hblank = mode->htotal - mode->hdisplay;
> > +
> > +	vactive = mode->vdisplay;
> > +	vsync_offset = mode->vsync_start - mode->vdisplay;
> > +	vsync_pulse_width = mode->vsync_end - mode->vsync_start;
> > +	vblank = mode->vtotal - mode->vdisplay;
> > +
> > +	dt->pixel_clock[0] = (mode->clock / 10) & 0x00FF;
> > +	dt->pixel_clock[1] = ((mode->clock / 10) & 0xFF00) >> 8;
> > +
> > +	assert(hactive <= 0xFFF);
> > +	assert(hblank <= 0xFFF);
> > +	pt->hactive_lo = hactive & 0x0FF;
> > +	pt->hblank_lo = hblank & 0x0FF;
> > +	pt->hactive_hblank_hi = (hactive & 0xF00) >> 4 | (hblank & 0xF00) >> 8;
> > +
> > +	assert(vactive <= 0xFFF);
> > +	assert(vblank <= 0xFFF);
> > +	pt->vactive_lo = vactive & 0x0FF;
> > +	pt->vblank_lo = vblank & 0x0FF;
> > +	pt->vactive_vblank_hi = (vactive & 0xF00) >> 4 | (vblank & 0xF00) >> 8;
> > +
> > +	assert(hsync_offset <= 0x3FF);
> > +	assert(hsync_pulse_width <= 0x3FF);
> > +	assert(vsync_offset <= 0x3F);
> > +	assert(vsync_pulse_width <= 0x3F);
> > +	pt->hsync_offset_lo = hsync_offset & 0x0FF;
> > +	pt->hsync_pulse_width_lo = hsync_pulse_width & 0x0FF;
> > +	pt->vsync_offset_pulse_width_lo = (vsync_offset & 0xF) << 4
> > +					  | (vsync_pulse_width & 0xF);
> > +	pt->hsync_vsync_offset_pulse_width_hi = 
> 
>                                                ^ tailing whitespace
> 
> > +		((hsync_offset & 0x300) >> 2) | ((hsync_pulse_width & 0x300) >> 4)
> > +		| ((vsync_offset & 0x30) >> 2) | ((vsync_pulse_width & 0x30) >> 4);
> > +
> > +	assert(width_mm <= 0xFFF);
> > +	assert(height_mm <= 0xFFF);
> > +	pt->width_mm_lo = width_mm & 0x0FF;
> > +	pt->height_mm_lo = height_mm & 0x0FF;
> > +	pt->width_height_mm_hi = (width_mm & 0xF00) >> 4
> > +				 | (height_mm & 0xF00) >> 8;
> > +	
> 
>    ^^^^^ tab
> 
> autocmd BufWritePre * %s/\s\+$//e
> 
> may help you

Ah, thanks for the tip!

Also pulled my vim dotfiles so that whitespace characters are visible.

> > +	pt->misc = 0;
> > +	if (mode->flags & DRM_MODE_FLAG_PHSYNC)
> > +		pt->misc |= EDID_PT_HSYNC_POSITIVE;
> > +	if (mode->flags & DRM_MODE_FLAG_PVSYNC)
> > +		pt->misc |= EDID_PT_VSYNC_POSITIVE;
> > +}
> > +
> > +/**
> > + * detailed_timing_set_monitor_range_mode: set a detailed timing to be a
> > + * monitor range based on a mode
> > + */
> > +void detailed_timing_set_monitor_range_mode(struct detailed_timing *dt,
> > +					    drmModeModeInfo *mode)
> > +{
> > +	struct detailed_non_pixel *np = &dt->data.other_data;
> > +	struct detailed_data_monitor_range *mr = &np->data.range;
> > +
> > +	dt->pixel_clock[0] = dt->pixel_clock[1] = 0;
> > +
> > +	np->type = EDID_DETAIL_MONITOR_RANGE;
> > +
> > +	mr->min_vfreq = mode->vrefresh - 1;
> > +	mr->max_vfreq = mode->vrefresh + 1;
> > +	mr->min_hfreq_khz = (mode->clock / mode->htotal) - 1;
> > +	mr->max_hfreq_khz = (mode->clock / mode->htotal) + 1;
> > +	mr->pixel_clock_mhz = (mode->clock / 10000) + 1;
> > +	mr->flags = 0;
> > +
> > +	static const char padding[] = {
> > +		0x0a, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20
> > +	};
> > +	memcpy(mr->formula.pad, padding, sizeof(padding));
> > +}
> > +
> > +/**
> > + * detailed_timing_set_string: set a detailed timing to be a string
> > + */
> > +void detailed_timing_set_string(struct detailed_timing *dt,
> > +				enum detailed_non_pixel_type type,
> > +				const char *str)
> > +{
> > +	struct detailed_non_pixel *np = &dt->data.other_data;
> > +	struct detailed_data_string *ds = &np->data.string;
> > +	size_t len;
> > +
> > +	switch (type) {
> > +	case EDID_DETAIL_MONITOR_NAME:
> > +	case EDID_DETAIL_MONITOR_STRING:
> > +	case EDID_DETAIL_MONITOR_SERIAL:
> > +		break;
> > +	default:
> > +		assert(0); /* not a string type */
> > +	}
> > +
> > +	dt->pixel_clock[0] = dt->pixel_clock[1] = 0;
> > +
> > +	np->type = type;
> > +
> > +	strncpy(ds->str, str, sizeof(ds->str));
> > +	len = strlen(str);
> > +	if (len < sizeof(ds->str))
> > +		ds->str[len] = '\n';
> > +}
> > +
> > +static void edid_set_mfg(struct edid *edid, char mfg[static 3])
> > +{
> > +	edid->mfg_id[0] = (mfg[0] - '@') << 2 | (mfg[1] - '@') >> 3;
> > +	edid->mfg_id[1] = (mfg[1] - '@') << 5 | (mfg[2] - '@');
> > +}
> > +
> > +static void edid_set_gamma(struct edid *edid, float gamma)
> > +{
> > +	edid->gamma = (gamma * 100) - 100;
> > +}
> > +
> > +/**
> > + * edid_init: initialize an EDID
> > + *
> > + * The EDID will be pre-filled with established and standard timings:
> > + *
> > + *  - 1920x1080 60Hz
> > + *  - 1280x720 60Hz
> > + *  - 1024x768 60Hz
> > + *  - 800x600 60Hz
> > + *  - 640x480 60Hz
> > + */
> > +void edid_init(struct edid *edid)
> > +{
> > +	size_t i;
> > +	time_t t;
> > +	struct tm *tm;
> > +
> > +	memset(edid, 0, sizeof(struct edid));
> > +
> > +	memcpy(edid->header, edid_header, sizeof(edid_header));
> > +	edid_set_mfg(edid, "IGT");
> > +	edid->version = 1;
> > +	edid->revision = 3;
> > +	edid->input = 0x80;
> > +	edid->width_cm = 52;
> > +	edid->height_cm = 30;
> > +	edid_set_gamma(edid, 2.20);
> > +	edid->features = 0x02;
> > +
> > +	/* Year of manufacture */
> > +	t = time(NULL);
> > +	tm = localtime(&t);
> > +	edid->mfg_year = tm->tm_year - 90;
> > +
> > +	/* Established timings: 640x480 60Hz, 800x600 60Hz, 1024x768 60Hz */
> > +	edid->established_timings.t1 = 0x21;
> > +	edid->established_timings.t2 = 0x08;
> > +
> > +	/* Standard timings */
> > +	/* 1920x1080 60Hz */
> > +	std_timing_set(&edid->standard_timings[0], 1920, 60, STD_TIMING_16_9);
> > +	/* 1280x720 60Hz */
> > +	std_timing_set(&edid->standard_timings[1], 1280, 60, STD_TIMING_16_9);
> > +	/* 1024x768 60Hz */
> > +	std_timing_set(&edid->standard_timings[2], 1024, 60, STD_TIMING_4_3);
> > +	/* 800x600 60Hz */
> > +	std_timing_set(&edid->standard_timings[3], 800, 60, STD_TIMING_4_3);
> > +	/* 640x480 60Hz */
> > +	std_timing_set(&edid->standard_timings[4], 640, 60, STD_TIMING_4_3);
> > +	for (i = 5; i < STD_TIMINGS_LEN; i++)
> > +		std_timing_unset(&edid->standard_timings[i]);
> > +}
> > +
> > +/**
> > + * edid_init_with_mode: initialize an EDID and sets its preferred mode
> > + */
> > +void edid_init_with_mode(struct edid *edid, drmModeModeInfo *mode)
> > +{
> > +	edid_init(edid);
> > +
> > +	/* Preferred timing */
> > +	detailed_timing_set_mode(&edid->detailed_timings[0], mode,
> > +				 edid->width_cm * 10, edid->height_cm * 10);
> > +	detailed_timing_set_monitor_range_mode(&edid->detailed_timings[1],
> > +					       mode);
> > +	detailed_timing_set_string(&edid->detailed_timings[2],
> > +				   EDID_DETAIL_MONITOR_NAME, "IGT");
> > +}
> > +
> > +/**
> > + * edid_finalize: finalize an EDID by computing its checksum
> > + */
> > +void edid_finalize(struct edid *edid)
> > +{
> > +	size_t i;
> > +	const uint8_t *buf = (const uint8_t *) edid;
> > +	uint8_t sum = 0;
> > +
> > +	/* calculate checksum */
> > +	for (i = 0; i < sizeof(struct edid) - 1; i++) {
> > +		sum = sum + buf[i];
> > +	}
> > +
> > +	edid->checksum = 256 - sum;
> > +}
> 
> I get why using `finalize` here may seems tempting, but fini and
> finalize tend to be used for destructors (or analogues in GCed
> languages), so this may be a bit confusing. I think we should go with
> edid_update_checksum() or similar.

Agreed, a more explicit name is pobably not a bad idea.

> Other than that, it's a solid improvement over what we had, which will
> come handy the more we fiddle with EDIDs. Thanks!

:)

Will send a v3 with your comments fixed.

> Do you already have some thoughts on how we want to solve handling the
> extensions blocks?

I've already started to work on this. I think callers should allocate a
large enough buffer, e.g.:

  char buf[sizeof(struct edid) + 2*sizeof(struct edid_ext)];

(Extension blocks have a fixed size)

And then should map a struct edid to it:

  struct edid *edid = (struct edid *) buf;

The annoying thing is that the number of extensions at the end of the
EDID block isn't fixed. We could add a new field at the end of struct
edid:

  struct edid {
    …
    struct edid_ext extensions[];
  };

So that callers can do this:

  struct edid_ext *ext1 = edid->extensions[0];
  struct edid_ext *ext2 = edid->extensions[1];

Another annoying thing is that extensions can themselves contain
multiple sub-blocks of different sizes… For now I've opted for a char
buffer that can be casted into sub-blocks. This isn't the best solution
ever, but I haven't found a better design yet. Let's discuss about this
in the next patch. :)


More information about the igt-dev mailing list