xref: /openbmc/linux/drivers/firmware/sysfb.c (revision 0db00e5d86dc793aab9722ad3728d99166eb7d96)
1d391c582SJavier Martinez Canillas // SPDX-License-Identifier: GPL-2.0-or-later
2d391c582SJavier Martinez Canillas /*
38633ef82SJavier Martinez Canillas  * Generic System Framebuffers
4d391c582SJavier Martinez Canillas  * Copyright (c) 2012-2013 David Herrmann <dh.herrmann@gmail.com>
5d391c582SJavier Martinez Canillas  */
6d391c582SJavier Martinez Canillas 
7d391c582SJavier Martinez Canillas /*
88633ef82SJavier Martinez Canillas  * Simple-Framebuffer support
9d391c582SJavier Martinez Canillas  * Create a platform-device for any available boot framebuffer. The
10d391c582SJavier Martinez Canillas  * simple-framebuffer platform device is already available on DT systems, so
11d391c582SJavier Martinez Canillas  * this module parses the global "screen_info" object and creates a suitable
12d391c582SJavier Martinez Canillas  * platform device compatible with the "simple-framebuffer" DT object. If
13d391c582SJavier Martinez Canillas  * the framebuffer is incompatible, we instead create a legacy
14d391c582SJavier Martinez Canillas  * "vesa-framebuffer", "efi-framebuffer" or "platform-framebuffer" device and
15d391c582SJavier Martinez Canillas  * pass the screen_info as platform_data. This allows legacy drivers
16d391c582SJavier Martinez Canillas  * to pick these devices up without messing with simple-framebuffer drivers.
17d391c582SJavier Martinez Canillas  * The global "screen_info" is still valid at all times.
18d391c582SJavier Martinez Canillas  *
198633ef82SJavier Martinez Canillas  * If CONFIG_SYSFB_SIMPLEFB is not selected, never register "simple-framebuffer"
20d391c582SJavier Martinez Canillas  * platform devices, but only use legacy framebuffer devices for
21d391c582SJavier Martinez Canillas  * backwards compatibility.
22d391c582SJavier Martinez Canillas  *
23d391c582SJavier Martinez Canillas  * TODO: We set the dev_id field of all platform-devices to 0. This allows
248633ef82SJavier Martinez Canillas  * other OF/DT parsers to create such devices, too. However, they must
25d391c582SJavier Martinez Canillas  * start at offset 1 for this to work.
26d391c582SJavier Martinez Canillas  */
27d391c582SJavier Martinez Canillas 
28d391c582SJavier Martinez Canillas #include <linux/err.h>
29d391c582SJavier Martinez Canillas #include <linux/init.h>
30d391c582SJavier Martinez Canillas #include <linux/kernel.h>
31d391c582SJavier Martinez Canillas #include <linux/mm.h>
32d391c582SJavier Martinez Canillas #include <linux/platform_data/simplefb.h>
33d391c582SJavier Martinez Canillas #include <linux/platform_device.h>
34d391c582SJavier Martinez Canillas #include <linux/screen_info.h>
35d391c582SJavier Martinez Canillas #include <linux/sysfb.h>
36d391c582SJavier Martinez Canillas 
37bc824922SJavier Martinez Canillas static struct platform_device *pd;
38bc824922SJavier Martinez Canillas static DEFINE_MUTEX(disable_lock);
39bc824922SJavier Martinez Canillas static bool disabled;
40bc824922SJavier Martinez Canillas 
sysfb_unregister(void)41bc824922SJavier Martinez Canillas static bool sysfb_unregister(void)
42bc824922SJavier Martinez Canillas {
43bc824922SJavier Martinez Canillas 	if (IS_ERR_OR_NULL(pd))
44bc824922SJavier Martinez Canillas 		return false;
45bc824922SJavier Martinez Canillas 
46bc824922SJavier Martinez Canillas 	platform_device_unregister(pd);
47bc824922SJavier Martinez Canillas 	pd = NULL;
48bc824922SJavier Martinez Canillas 
49bc824922SJavier Martinez Canillas 	return true;
50bc824922SJavier Martinez Canillas }
51bc824922SJavier Martinez Canillas 
52bc824922SJavier Martinez Canillas /**
53bc824922SJavier Martinez Canillas  * sysfb_disable() - disable the Generic System Framebuffers support
54bc824922SJavier Martinez Canillas  *
55bc824922SJavier Martinez Canillas  * This disables the registration of system framebuffer devices that match the
56bc824922SJavier Martinez Canillas  * generic drivers that make use of the system framebuffer set up by firmware.
57bc824922SJavier Martinez Canillas  *
58bc824922SJavier Martinez Canillas  * It also unregisters a device if this was already registered by sysfb_init().
59bc824922SJavier Martinez Canillas  *
60bc824922SJavier Martinez Canillas  * Context: The function can sleep. A @disable_lock mutex is acquired to serialize
61bc824922SJavier Martinez Canillas  *          against sysfb_init(), that registers a system framebuffer device.
62bc824922SJavier Martinez Canillas  */
sysfb_disable(void)63bc824922SJavier Martinez Canillas void sysfb_disable(void)
64bc824922SJavier Martinez Canillas {
65bc824922SJavier Martinez Canillas 	mutex_lock(&disable_lock);
66bc824922SJavier Martinez Canillas 	sysfb_unregister();
67bc824922SJavier Martinez Canillas 	disabled = true;
68bc824922SJavier Martinez Canillas 	mutex_unlock(&disable_lock);
69bc824922SJavier Martinez Canillas }
70bc824922SJavier Martinez Canillas EXPORT_SYMBOL_GPL(sysfb_disable);
71bc824922SJavier Martinez Canillas 
sysfb_init(void)72d391c582SJavier Martinez Canillas static __init int sysfb_init(void)
73d391c582SJavier Martinez Canillas {
74d391c582SJavier Martinez Canillas 	struct screen_info *si = &screen_info;
75d391c582SJavier Martinez Canillas 	struct simplefb_platform_data mode;
76d391c582SJavier Martinez Canillas 	const char *name;
77d391c582SJavier Martinez Canillas 	bool compatible;
78bc824922SJavier Martinez Canillas 	int ret = 0;
79bc824922SJavier Martinez Canillas 
80*a168da31SThomas Zimmermann 	screen_info_apply_fixups();
81*a168da31SThomas Zimmermann 
82bc824922SJavier Martinez Canillas 	mutex_lock(&disable_lock);
83bc824922SJavier Martinez Canillas 	if (disabled)
84bc824922SJavier Martinez Canillas 		goto unlock_mutex;
85d391c582SJavier Martinez Canillas 
863615c786SHans de Goede 	sysfb_apply_efi_quirks();
873615c786SHans de Goede 
88d391c582SJavier Martinez Canillas 	/* try to create a simple-framebuffer device */
898633ef82SJavier Martinez Canillas 	compatible = sysfb_parse_mode(si, &mode);
90d391c582SJavier Martinez Canillas 	if (compatible) {
910949ee75SJavier Martinez Canillas 		pd = sysfb_create_simplefb(si, &mode);
920949ee75SJavier Martinez Canillas 		if (!IS_ERR(pd))
93bc824922SJavier Martinez Canillas 			goto unlock_mutex;
94d391c582SJavier Martinez Canillas 	}
95d391c582SJavier Martinez Canillas 
96d391c582SJavier Martinez Canillas 	/* if the FB is incompatible, create a legacy framebuffer device */
97d391c582SJavier Martinez Canillas 	if (si->orig_video_isVGA == VIDEO_TYPE_EFI)
98d391c582SJavier Martinez Canillas 		name = "efi-framebuffer";
99d391c582SJavier Martinez Canillas 	else if (si->orig_video_isVGA == VIDEO_TYPE_VLFB)
100d391c582SJavier Martinez Canillas 		name = "vesa-framebuffer";
1010db5b61eSThomas Zimmermann 	else if (si->orig_video_isVGA == VIDEO_TYPE_VGAC)
1020db5b61eSThomas Zimmermann 		name = "vga-framebuffer";
1030db5b61eSThomas Zimmermann 	else if (si->orig_video_isVGA == VIDEO_TYPE_EGAC)
1040db5b61eSThomas Zimmermann 		name = "ega-framebuffer";
105d391c582SJavier Martinez Canillas 	else
106d391c582SJavier Martinez Canillas 		name = "platform-framebuffer";
107d391c582SJavier Martinez Canillas 
1088633ef82SJavier Martinez Canillas 	pd = platform_device_alloc(name, 0);
109bc824922SJavier Martinez Canillas 	if (!pd) {
110bc824922SJavier Martinez Canillas 		ret = -ENOMEM;
111bc824922SJavier Martinez Canillas 		goto unlock_mutex;
112bc824922SJavier Martinez Canillas 	}
1138633ef82SJavier Martinez Canillas 
1143615c786SHans de Goede 	sysfb_set_efifb_fwnode(pd);
1158633ef82SJavier Martinez Canillas 
1168633ef82SJavier Martinez Canillas 	ret = platform_device_add_data(pd, si, sizeof(*si));
1178633ef82SJavier Martinez Canillas 	if (ret)
1188633ef82SJavier Martinez Canillas 		goto err;
1198633ef82SJavier Martinez Canillas 
1208633ef82SJavier Martinez Canillas 	ret = platform_device_add(pd);
1218633ef82SJavier Martinez Canillas 	if (ret)
1228633ef82SJavier Martinez Canillas 		goto err;
1238633ef82SJavier Martinez Canillas 
124bc824922SJavier Martinez Canillas 	goto unlock_mutex;
1258633ef82SJavier Martinez Canillas err:
1268633ef82SJavier Martinez Canillas 	platform_device_put(pd);
127bc824922SJavier Martinez Canillas unlock_mutex:
128bc824922SJavier Martinez Canillas 	mutex_unlock(&disable_lock);
1298633ef82SJavier Martinez Canillas 	return ret;
130d391c582SJavier Martinez Canillas }
131d391c582SJavier Martinez Canillas 
132d391c582SJavier Martinez Canillas /* must execute after PCI subsystem for EFI quirks */
133e256f6d3SThomas Zimmermann device_initcall(sysfb_init);
134