xref: /openbmc/linux/drivers/video/cmdline.c (revision 1ac731c529cd4d6adbce134754b51ff7d822b145)
1*93604a5aSThomas Zimmermann // SPDX-License-Identifier: GPL-2.0
2*93604a5aSThomas Zimmermann /*
3*93604a5aSThomas Zimmermann  * Based on the fbdev code in drivers/video/fbdev/core/fb_cmdline:
4*93604a5aSThomas Zimmermann  *
5*93604a5aSThomas Zimmermann  *  Copyright (C) 2014 Intel Corp
6*93604a5aSThomas Zimmermann  *  Copyright (C) 1994 Martin Schaller
7*93604a5aSThomas Zimmermann  *
8*93604a5aSThomas Zimmermann  *	2001 - Documented with DocBook
9*93604a5aSThomas Zimmermann  *	- Brad Douglas <brad@neruo.com>
10*93604a5aSThomas Zimmermann  *
11*93604a5aSThomas Zimmermann  * This file is subject to the terms and conditions of the GNU General Public
12*93604a5aSThomas Zimmermann  * License.  See the file COPYING in the main directory of this archive
13*93604a5aSThomas Zimmermann  * for more details.
14*93604a5aSThomas Zimmermann  *
15*93604a5aSThomas Zimmermann  * Authors:
16*93604a5aSThomas Zimmermann  *    Daniel Vetter <daniel.vetter@ffwll.ch>
17*93604a5aSThomas Zimmermann  */
18*93604a5aSThomas Zimmermann 
19*93604a5aSThomas Zimmermann #include <linux/fb.h> /* for FB_MAX */
20*93604a5aSThomas Zimmermann #include <linux/init.h>
21*93604a5aSThomas Zimmermann 
22*93604a5aSThomas Zimmermann #include <video/cmdline.h>
23*93604a5aSThomas Zimmermann 
24*93604a5aSThomas Zimmermann /*
25*93604a5aSThomas Zimmermann  * FB_MAX is the maximum number of framebuffer devices and also
26*93604a5aSThomas Zimmermann  * the maximum number of video= parameters. Although not directly
27*93604a5aSThomas Zimmermann  * related to each other, it makes sense to keep it that way.
28*93604a5aSThomas Zimmermann  */
29*93604a5aSThomas Zimmermann static const char *video_options[FB_MAX] __read_mostly;
30*93604a5aSThomas Zimmermann static const char *video_option __read_mostly;
31*93604a5aSThomas Zimmermann static int video_of_only __read_mostly;
32*93604a5aSThomas Zimmermann 
__video_get_option_string(const char * name)33*93604a5aSThomas Zimmermann static const char *__video_get_option_string(const char *name)
34*93604a5aSThomas Zimmermann {
35*93604a5aSThomas Zimmermann 	const char *options = NULL;
36*93604a5aSThomas Zimmermann 	size_t name_len = 0;
37*93604a5aSThomas Zimmermann 
38*93604a5aSThomas Zimmermann 	if (name)
39*93604a5aSThomas Zimmermann 		name_len = strlen(name);
40*93604a5aSThomas Zimmermann 
41*93604a5aSThomas Zimmermann 	if (name_len) {
42*93604a5aSThomas Zimmermann 		unsigned int i;
43*93604a5aSThomas Zimmermann 		const char *opt;
44*93604a5aSThomas Zimmermann 
45*93604a5aSThomas Zimmermann 		for (i = 0; i < ARRAY_SIZE(video_options); ++i) {
46*93604a5aSThomas Zimmermann 			if (!video_options[i])
47*93604a5aSThomas Zimmermann 				continue;
48*93604a5aSThomas Zimmermann 			if (video_options[i][0] == '\0')
49*93604a5aSThomas Zimmermann 				continue;
50*93604a5aSThomas Zimmermann 			opt = video_options[i];
51*93604a5aSThomas Zimmermann 			if (!strncmp(opt, name, name_len) && opt[name_len] == ':')
52*93604a5aSThomas Zimmermann 				options = opt + name_len + 1;
53*93604a5aSThomas Zimmermann 		}
54*93604a5aSThomas Zimmermann 	}
55*93604a5aSThomas Zimmermann 
56*93604a5aSThomas Zimmermann 	/* No match, return global options */
57*93604a5aSThomas Zimmermann 	if (!options)
58*93604a5aSThomas Zimmermann 		options = video_option;
59*93604a5aSThomas Zimmermann 
60*93604a5aSThomas Zimmermann 	return options;
61*93604a5aSThomas Zimmermann }
62*93604a5aSThomas Zimmermann 
63*93604a5aSThomas Zimmermann /**
64*93604a5aSThomas Zimmermann  * video_get_options - get kernel boot parameters
65*93604a5aSThomas Zimmermann  * @name:	name of the output as it would appear in the boot parameter
66*93604a5aSThomas Zimmermann  *		line (video=<name>:<options>)
67*93604a5aSThomas Zimmermann  *
68*93604a5aSThomas Zimmermann  * Looks up the video= options for the given name. Names are connector
69*93604a5aSThomas Zimmermann  * names with DRM, or driver names with fbdev. If no video option for
70*93604a5aSThomas Zimmermann  * the name has been specified, the function returns the global video=
71*93604a5aSThomas Zimmermann  * setting. A @name of NULL always returns the global video setting.
72*93604a5aSThomas Zimmermann  *
73*93604a5aSThomas Zimmermann  * Returns:
74*93604a5aSThomas Zimmermann  * The string of video options for the given name, or NULL if no video
75*93604a5aSThomas Zimmermann  * option has been specified.
76*93604a5aSThomas Zimmermann  */
video_get_options(const char * name)77*93604a5aSThomas Zimmermann const char *video_get_options(const char *name)
78*93604a5aSThomas Zimmermann {
79*93604a5aSThomas Zimmermann 	return __video_get_option_string(name);
80*93604a5aSThomas Zimmermann }
81*93604a5aSThomas Zimmermann EXPORT_SYMBOL(video_get_options);
82*93604a5aSThomas Zimmermann 
__video_get_options(const char * name,const char ** options,bool is_of)83*93604a5aSThomas Zimmermann bool __video_get_options(const char *name, const char **options, bool is_of)
84*93604a5aSThomas Zimmermann {
85*93604a5aSThomas Zimmermann 	bool enabled = true;
86*93604a5aSThomas Zimmermann 	const char *opt = NULL;
87*93604a5aSThomas Zimmermann 
88*93604a5aSThomas Zimmermann 	if (video_of_only && !is_of)
89*93604a5aSThomas Zimmermann 		enabled = false;
90*93604a5aSThomas Zimmermann 
91*93604a5aSThomas Zimmermann 	opt = __video_get_option_string(name);
92*93604a5aSThomas Zimmermann 
93*93604a5aSThomas Zimmermann 	if (options)
94*93604a5aSThomas Zimmermann 		*options = opt;
95*93604a5aSThomas Zimmermann 
96*93604a5aSThomas Zimmermann 	return enabled;
97*93604a5aSThomas Zimmermann }
98*93604a5aSThomas Zimmermann EXPORT_SYMBOL(__video_get_options);
99*93604a5aSThomas Zimmermann 
100*93604a5aSThomas Zimmermann /*
101*93604a5aSThomas Zimmermann  * Process command line options for video adapters. This function is
102*93604a5aSThomas Zimmermann  * a __setup and __init function. It only stores the options. Drivers
103*93604a5aSThomas Zimmermann  * have to call video_get_options() as necessary.
104*93604a5aSThomas Zimmermann  */
video_setup(char * options)105*93604a5aSThomas Zimmermann static int __init video_setup(char *options)
106*93604a5aSThomas Zimmermann {
107*93604a5aSThomas Zimmermann 	if (!options || !*options)
108*93604a5aSThomas Zimmermann 		goto out;
109*93604a5aSThomas Zimmermann 
110*93604a5aSThomas Zimmermann 	if (!strncmp(options, "ofonly", 6)) {
111*93604a5aSThomas Zimmermann 		video_of_only = true;
112*93604a5aSThomas Zimmermann 		goto out;
113*93604a5aSThomas Zimmermann 	}
114*93604a5aSThomas Zimmermann 
115*93604a5aSThomas Zimmermann 	if (strchr(options, ':')) {
116*93604a5aSThomas Zimmermann 		/* named */
117*93604a5aSThomas Zimmermann 		size_t i;
118*93604a5aSThomas Zimmermann 
119*93604a5aSThomas Zimmermann 		for (i = 0; i < ARRAY_SIZE(video_options); i++) {
120*93604a5aSThomas Zimmermann 			if (!video_options[i]) {
121*93604a5aSThomas Zimmermann 				video_options[i] = options;
122*93604a5aSThomas Zimmermann 				break;
123*93604a5aSThomas Zimmermann 			}
124*93604a5aSThomas Zimmermann 		}
125*93604a5aSThomas Zimmermann 	} else {
126*93604a5aSThomas Zimmermann 		/* global */
127*93604a5aSThomas Zimmermann 		video_option = options;
128*93604a5aSThomas Zimmermann 	}
129*93604a5aSThomas Zimmermann 
130*93604a5aSThomas Zimmermann out:
131*93604a5aSThomas Zimmermann 	return 1;
132*93604a5aSThomas Zimmermann }
133*93604a5aSThomas Zimmermann __setup("video=", video_setup);
134