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