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

Jani Nikula jani.nikula at linux.intel.com
Tue Apr 16 11:05:44 UTC 2019


On Tue, 16 Apr 2019, "Ser, Simon" <simon.ser at intel.com> wrote:
> 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. :)

While it's easy to write something that's better than what we currently
have, as you see this gets complicated more quickly than you'd like,
especially working in C. We're not the only ones needing EDID
generation, so I'd suggest looking at what others are doing before
embarking on writing a full blown generator. See e.g. [1].

BR,
Jani.


[1] http://mid.mail-archive.com/20181121133624.i4vxgt7atnv3ads6@sirius.home.kraxel.org


-- 
Jani Nikula, Intel Open Source Graphics Center


More information about the igt-dev mailing list