136 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0
 | |
| /*
 | |
|  * Based on the fbdev code in drivers/video/fbdev/core/fb_cmdline:
 | |
|  *
 | |
|  *  Copyright (C) 2014 Intel Corp
 | |
|  *  Copyright (C) 1994 Martin Schaller
 | |
|  *
 | |
|  *	2001 - Documented with DocBook
 | |
|  *	- Brad Douglas <brad@neruo.com>
 | |
|  *
 | |
|  * This file is subject to the terms and conditions of the GNU General Public
 | |
|  * License.  See the file COPYING in the main directory of this archive
 | |
|  * for more details.
 | |
|  *
 | |
|  * Authors:
 | |
|  *    Daniel Vetter <daniel.vetter@ffwll.ch>
 | |
|  */
 | |
| 
 | |
| #include <linux/fb.h> /* for FB_MAX */
 | |
| #include <linux/init.h>
 | |
| 
 | |
| #include <video/cmdline.h>
 | |
| 
 | |
| /*
 | |
|  * FB_MAX is the maximum number of framebuffer devices and also
 | |
|  * the maximum number of video= parameters. Although not directly
 | |
|  * related to each other, it makes sense to keep it that way.
 | |
|  */
 | |
| static const char *video_options[FB_MAX] __read_mostly;
 | |
| static const char *video_option __read_mostly;
 | |
| static int video_of_only __read_mostly;
 | |
| 
 | |
| static const char *__video_get_option_string(const char *name)
 | |
| {
 | |
| 	const char *options = NULL;
 | |
| 	size_t name_len = 0;
 | |
| 
 | |
| 	if (name)
 | |
| 		name_len = strlen(name);
 | |
| 
 | |
| 	if (name_len) {
 | |
| 		unsigned int i;
 | |
| 		const char *opt;
 | |
| 
 | |
| 		for (i = 0; i < ARRAY_SIZE(video_options); ++i) {
 | |
| 			if (!video_options[i])
 | |
| 				continue;
 | |
| 			if (video_options[i][0] == '\0')
 | |
| 				continue;
 | |
| 			opt = video_options[i];
 | |
| 			if (!strncmp(opt, name, name_len) && opt[name_len] == ':')
 | |
| 				options = opt + name_len + 1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/* No match, return global options */
 | |
| 	if (!options)
 | |
| 		options = video_option;
 | |
| 
 | |
| 	return options;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * video_get_options - get kernel boot parameters
 | |
|  * @name:	name of the output as it would appear in the boot parameter
 | |
|  *		line (video=<name>:<options>)
 | |
|  *
 | |
|  * Looks up the video= options for the given name. Names are connector
 | |
|  * names with DRM, or driver names with fbdev. If no video option for
 | |
|  * the name has been specified, the function returns the global video=
 | |
|  * setting. A @name of NULL always returns the global video setting.
 | |
|  *
 | |
|  * Returns:
 | |
|  * The string of video options for the given name, or NULL if no video
 | |
|  * option has been specified.
 | |
|  */
 | |
| const char *video_get_options(const char *name)
 | |
| {
 | |
| 	return __video_get_option_string(name);
 | |
| }
 | |
| EXPORT_SYMBOL(video_get_options);
 | |
| 
 | |
| #if IS_ENABLED(CONFIG_FB_CORE)
 | |
| bool __video_get_options(const char *name, const char **options, bool is_of)
 | |
| {
 | |
| 	bool enabled = true;
 | |
| 	const char *opt = NULL;
 | |
| 
 | |
| 	if (video_of_only && !is_of)
 | |
| 		enabled = false;
 | |
| 
 | |
| 	opt = __video_get_option_string(name);
 | |
| 
 | |
| 	if (options)
 | |
| 		*options = opt;
 | |
| 
 | |
| 	return enabled;
 | |
| }
 | |
| EXPORT_SYMBOL(__video_get_options);
 | |
| #endif
 | |
| 
 | |
| /*
 | |
|  * Process command line options for video adapters. This function is
 | |
|  * a __setup and __init function. It only stores the options. Drivers
 | |
|  * have to call video_get_options() as necessary.
 | |
|  */
 | |
| static int __init video_setup(char *options)
 | |
| {
 | |
| 	if (!options || !*options)
 | |
| 		goto out;
 | |
| 
 | |
| 	if (!strncmp(options, "ofonly", 6)) {
 | |
| 		video_of_only = true;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	if (strchr(options, ':')) {
 | |
| 		/* named */
 | |
| 		size_t i;
 | |
| 
 | |
| 		for (i = 0; i < ARRAY_SIZE(video_options); i++) {
 | |
| 			if (!video_options[i]) {
 | |
| 				video_options[i] = options;
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		/* global */
 | |
| 		video_option = options;
 | |
| 	}
 | |
| 
 | |
| out:
 | |
| 	return 1;
 | |
| }
 | |
| __setup("video=", video_setup);
 |