164b70da0SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-or-later
2f7018c21STomi Valkeinen /*
3f7018c21STomi Valkeinen  * Copyright 2011 Florian Tobias Schandinat <FlorianSchandinat@gmx.de>
4f7018c21STomi Valkeinen  */
5f7018c21STomi Valkeinen /*
6f7018c21STomi Valkeinen  * generic EDID driver
7f7018c21STomi Valkeinen  */
8f7018c21STomi Valkeinen 
9f7018c21STomi Valkeinen #include <linux/slab.h>
10f7018c21STomi Valkeinen #include <linux/fb.h>
11f7018c21STomi Valkeinen #include "via_aux.h"
12f7018c21STomi Valkeinen #include "../edid.h"
13f7018c21STomi Valkeinen 
14f7018c21STomi Valkeinen 
15f7018c21STomi Valkeinen static const char *name = "EDID";
16f7018c21STomi Valkeinen 
17f7018c21STomi Valkeinen 
query_edid(struct via_aux_drv * drv)18f7018c21STomi Valkeinen static void query_edid(struct via_aux_drv *drv)
19f7018c21STomi Valkeinen {
20f7018c21STomi Valkeinen 	struct fb_monspecs *spec = drv->data;
21f7018c21STomi Valkeinen 	unsigned char edid[EDID_LENGTH];
22f7018c21STomi Valkeinen 	bool valid = false;
23f7018c21STomi Valkeinen 
24f7018c21STomi Valkeinen 	if (spec) {
25f7018c21STomi Valkeinen 		fb_destroy_modedb(spec->modedb);
26f7018c21STomi Valkeinen 	} else {
27f7018c21STomi Valkeinen 		spec = kmalloc(sizeof(*spec), GFP_KERNEL);
28f7018c21STomi Valkeinen 		if (!spec)
29f7018c21STomi Valkeinen 			return;
30f7018c21STomi Valkeinen 	}
31f7018c21STomi Valkeinen 
32f7018c21STomi Valkeinen 	spec->version = spec->revision = 0;
33f7018c21STomi Valkeinen 	if (via_aux_read(drv, 0x00, edid, EDID_LENGTH)) {
34f7018c21STomi Valkeinen 		fb_edid_to_monspecs(edid, spec);
35f7018c21STomi Valkeinen 		valid = spec->version || spec->revision;
36f7018c21STomi Valkeinen 	}
37f7018c21STomi Valkeinen 
38f7018c21STomi Valkeinen 	if (!valid) {
39f7018c21STomi Valkeinen 		kfree(spec);
40f7018c21STomi Valkeinen 		spec = NULL;
41f7018c21STomi Valkeinen 	} else
42f7018c21STomi Valkeinen 		printk(KERN_DEBUG "EDID: %s %s\n", spec->manufacturer, spec->monitor);
43f7018c21STomi Valkeinen 
44f7018c21STomi Valkeinen 	drv->data = spec;
45f7018c21STomi Valkeinen }
46f7018c21STomi Valkeinen 
get_preferred_mode(struct via_aux_drv * drv)47f7018c21STomi Valkeinen static const struct fb_videomode *get_preferred_mode(struct via_aux_drv *drv)
48f7018c21STomi Valkeinen {
49f7018c21STomi Valkeinen 	struct fb_monspecs *spec = drv->data;
50f7018c21STomi Valkeinen 	int i;
51f7018c21STomi Valkeinen 
52f7018c21STomi Valkeinen 	if (!spec || !spec->modedb || !(spec->misc & FB_MISC_1ST_DETAIL))
53f7018c21STomi Valkeinen 		return NULL;
54f7018c21STomi Valkeinen 
55f7018c21STomi Valkeinen 	for (i = 0; i < spec->modedb_len; i++) {
56f7018c21STomi Valkeinen 		if (spec->modedb[i].flag & FB_MODE_IS_FIRST &&
57f7018c21STomi Valkeinen 			spec->modedb[i].flag & FB_MODE_IS_DETAILED)
58f7018c21STomi Valkeinen 			return &spec->modedb[i];
59f7018c21STomi Valkeinen 	}
60f7018c21STomi Valkeinen 
61f7018c21STomi Valkeinen 	return NULL;
62f7018c21STomi Valkeinen }
63f7018c21STomi Valkeinen 
cleanup(struct via_aux_drv * drv)64f7018c21STomi Valkeinen static void cleanup(struct via_aux_drv *drv)
65f7018c21STomi Valkeinen {
66f7018c21STomi Valkeinen 	struct fb_monspecs *spec = drv->data;
67f7018c21STomi Valkeinen 
68f7018c21STomi Valkeinen 	if (spec)
69f7018c21STomi Valkeinen 		fb_destroy_modedb(spec->modedb);
70f7018c21STomi Valkeinen }
71f7018c21STomi Valkeinen 
via_aux_edid_probe(struct via_aux_bus * bus)72f7018c21STomi Valkeinen void via_aux_edid_probe(struct via_aux_bus *bus)
73f7018c21STomi Valkeinen {
74f7018c21STomi Valkeinen 	struct via_aux_drv drv = {
75f7018c21STomi Valkeinen 		.bus	=	bus,
76f7018c21STomi Valkeinen 		.addr	=	0x50,
77f7018c21STomi Valkeinen 		.name	=	name,
78f7018c21STomi Valkeinen 		.cleanup	=	cleanup,
79f7018c21STomi Valkeinen 		.get_preferred_mode	=	get_preferred_mode};
80f7018c21STomi Valkeinen 
81f7018c21STomi Valkeinen 	query_edid(&drv);
82f7018c21STomi Valkeinen 
83f7018c21STomi Valkeinen 	/* as EDID devices can be connected/disconnected just add the driver */
84f7018c21STomi Valkeinen 	via_aux_add(&drv);
85f7018c21STomi Valkeinen }
86