1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /*
3 drm_edid_load.c: use a built-in EDID data set or load it via the firmware
4 interface
5
6 Copyright (C) 2012 Carsten Emde <C.Emde@osadl.org>
7
8 */
9
10 #include <linux/firmware.h>
11 #include <linux/module.h>
12 #include <linux/platform_device.h>
13
14 #include <drm/drm_connector.h>
15 #include <drm/drm_drv.h>
16 #include <drm/drm_edid.h>
17 #include <drm/drm_print.h>
18
19 #include "drm_crtc_internal.h"
20
21 static char edid_firmware[PATH_MAX];
22 module_param_string(edid_firmware, edid_firmware, sizeof(edid_firmware), 0644);
23 MODULE_PARM_DESC(edid_firmware,
24 "Do not probe monitor, use specified EDID blob from /lib/firmware instead.");
25
edid_load(struct drm_connector * connector,const char * name)26 static const struct drm_edid *edid_load(struct drm_connector *connector, const char *name)
27 {
28 const struct firmware *fw = NULL;
29 const struct drm_edid *drm_edid;
30 int err;
31
32 err = request_firmware(&fw, name, connector->dev->dev);
33 if (err) {
34 drm_err(connector->dev,
35 "[CONNECTOR:%d:%s] Requesting EDID firmware \"%s\" failed (err=%d)\n",
36 connector->base.id, connector->name,
37 name, err);
38 return ERR_PTR(err);
39 }
40
41 drm_dbg_kms(connector->dev, "[CONNECTOR:%d:%s] Loaded external firmware EDID \"%s\"\n",
42 connector->base.id, connector->name, name);
43
44 drm_edid = drm_edid_alloc(fw->data, fw->size);
45 if (!drm_edid_valid(drm_edid)) {
46 drm_err(connector->dev, "Invalid firmware EDID \"%s\"\n", name);
47 drm_edid_free(drm_edid);
48 drm_edid = ERR_PTR(-EINVAL);
49 }
50
51 release_firmware(fw);
52
53 return drm_edid;
54 }
55
drm_edid_load_firmware(struct drm_connector * connector)56 const struct drm_edid *drm_edid_load_firmware(struct drm_connector *connector)
57 {
58 char *edidname, *last, *colon, *fwstr, *edidstr, *fallback = NULL;
59 const struct drm_edid *drm_edid;
60
61 if (edid_firmware[0] == '\0')
62 return ERR_PTR(-ENOENT);
63
64 /*
65 * If there are multiple edid files specified and separated
66 * by commas, search through the list looking for one that
67 * matches the connector.
68 *
69 * If there's one or more that doesn't specify a connector, keep
70 * the last one found one as a fallback.
71 */
72 fwstr = kstrdup(edid_firmware, GFP_KERNEL);
73 if (!fwstr)
74 return ERR_PTR(-ENOMEM);
75 edidstr = fwstr;
76
77 while ((edidname = strsep(&edidstr, ","))) {
78 colon = strchr(edidname, ':');
79 if (colon != NULL) {
80 if (strncmp(connector->name, edidname, colon - edidname))
81 continue;
82 edidname = colon + 1;
83 break;
84 }
85
86 if (*edidname != '\0') /* corner case: multiple ',' */
87 fallback = edidname;
88 }
89
90 if (!edidname) {
91 if (!fallback) {
92 kfree(fwstr);
93 return ERR_PTR(-ENOENT);
94 }
95 edidname = fallback;
96 }
97
98 last = edidname + strlen(edidname) - 1;
99 if (*last == '\n')
100 *last = '\0';
101
102 drm_edid = edid_load(connector, edidname);
103
104 kfree(fwstr);
105
106 return drm_edid;
107 }
108