All of lore.kernel.org
 help / color / mirror / Atom feed
* [PATCH] (1/3) allow earlier command line parsing
@ 2002-12-03  0:11 Dave Hansen
  2002-12-03  0:15 ` [PATCH] (2/3) do early command line parsing for 386 Dave Hansen
  2002-12-03  0:15 ` [PATCH] (3/3) early printk " Dave Hansen
  0 siblings, 2 replies; 3+ messages in thread
From: Dave Hansen @ 2002-12-03  0:11 UTC (permalink / raw)
  To: Linux Kernel Mailing List; +Cc: David Woodhouse

[-- Attachment #1: Type: text/plain, Size: 769 bytes --]

I stumbled upon this code when I was trying to get an early printk 
working for 386.  There is a real need for things to be able to parse 
the command line in setup_arch(), for many architectures.

parse_options() no longer runs the setup functions, it only parses.
It may be called more than once with no ill effects.  This way it can 
be called in setup_arch(), then again in the generic start_kernel().

run_setup() comes along later and actually runs the __setup functions.
   It can do this in many phases, as long as parse_options() is called
first.  The last run_setup() is special because it goes through and
picks up all of the environment variables and init arguments that are
left over, setting them with setup_setenv()

-- 
Dave Hansen
haveblue@us.ibm.com


[-- Attachment #2: A-ordered-setup-2.5.50-0.patch --]
[-- Type: text/plain, Size: 6632 bytes --]

# This is a BitKeeper generated patch for the following project:
# Project Name: Linux kernel tree
# This patch format is intended for GNU patch command version 2.5 or higher.
# This patch includes the following deltas:
#	           ChangeSet	1.968   -> 1.969  
#	include/linux/init.h	1.18    -> 1.19   
#	         init/main.c	1.83    -> 1.84   
#
# The following is the BitKeeper ChangeSet Log
# --------------------------------------------
# 02/12/02	haveblue@elm3b96.(none)	1.969
# c-1
# --------------------------------------------
#
diff -Nru a/include/linux/init.h b/include/linux/init.h
--- a/include/linux/init.h	Mon Dec  2 15:59:01 2002
+++ b/include/linux/init.h	Mon Dec  2 15:59:01 2002
@@ -86,21 +86,27 @@
 #define __exitcall(fn)							\
 	static exitcall_t __exitcall_##fn __exit_call = fn
 
+#define SETUP_ARCH_BEGIN	0
+#define SETUP_ARCH_LATE		1
+#define SETUP_DEFAULT		2
+#define SETUP_ENV		3
 /*
  * Used for kernel command line parameter setup
  */
 struct kernel_param {
 	const char *str;
 	int (*setup_func)(char *);
+	int level;
 };
 
 extern struct kernel_param __setup_start, __setup_end;
-
-#define __setup(str, fn)						\
+#define __setup(str, fn) __ordered_setup(SETUP_DEFAULT, str, fn)
+	
+#define __ordered_setup(level, str, fn)					\
 	static char __setup_str_##fn[] __initdata = str;		\
 	static struct kernel_param __setup_##fn				\
 		 __attribute__((unused,__section__ (".init.setup")))	\
-		= { __setup_str_##fn, fn }
+		= { __setup_str_##fn, fn, level }
 
 #endif /* __ASSEMBLY__ */
 
diff -Nru a/init/main.c b/init/main.c
--- a/init/main.c	Mon Dec  2 15:59:01 2002
+++ b/init/main.c	Mon Dec  2 15:59:01 2002
@@ -136,20 +136,73 @@
 
 __setup("profile=", profile_setup);
 
-static int __init checksetup(char *line)
+#define MAX_KERNEL_ARGS 64
+static struct kernel_arg {
+	char* str;
+	int handled;
+} kernel_argv[MAX_KERNEL_ARGS];
+static int kernel_argc = -1;
+
+/*
+ * Any cmd-line option is taken to be an environment variable if it contains
+ * the character '='.
+ *
+ * This routine also checks for options meant for the kernel.
+ * These options are not given to init - they are for internal kernel use only.
+ */
+static void __init setup_setenv(char *line)
+{
+	static int args = 0;
+	static int envs = 1; /* TERM is set to 'linux' by default */
+
+	if (!strncmp(line,"init=",5)) {
+		line += 5;
+		execute_command = line;
+		/* In case LILO is going to boot us with default command line,
+		 * it prepends "auto" before the whole cmdline which makes
+		 * the shell think it should execute a script with such name.
+		 * So we ignore all arguments entered _before_ init=... [MJ]
+		 */
+		args = 0;
+		argv_init[0] = NULL;
+		return;
+	}
+	
+	/*
+	 * check if it's an environment variable or
+	 * an option.
+	 */
+	
+	if (strchr(line,'=')) {
+		if (envs >= MAX_INIT_ENVS)
+			return;
+		envp_init[++envs] = line;
+		envp_init[envs+1] = NULL;
+	} else {
+		if (args >= MAX_INIT_ARGS)
+			return;
+		if (*line) {
+			argv_init[++args] = line;
+			argv_init[args+1] = NULL;
+		}
+	}
+}
+
+static void __init checksetup(int setup_level, struct kernel_arg *arg)
 {
 	struct kernel_param *p;
+	char *line = arg->str;
 
 	p = &__setup_start;
 	do {
 		int n = strlen(p->str);
-		if (!strncmp(line,p->str,n)) {
-			if (p->setup_func(line+n))
-				return 1;
+		if (p->level == setup_level && !strncmp(line, p->str, n)) {
+			arg->handled = p->setup_func(line+n);
+			if (arg->handled)
+				return;
 		}
 		p++;
 	} while (p < &__setup_end);
-	return 0;
 }
 
 /* this should be approx 2 Bo*oMips to start (note initial shift), and will
@@ -223,22 +276,25 @@
 
 /*
  * This is a simple kernel command line parsing function: it parses
- * the command line, and fills in the arguments/environment to init
- * as appropriate. Any cmd-line option is taken to be an environment
- * variable if it contains the character '='.
+ * the command line, and saves it for later processing by run_setup().
+ * This is designed to be called more than once.  This way, individual
+ * architectures can call it as early as they want, or they can ignore
+ * it and let the generic code call it.
  *
- * This routine also checks for options meant for the kernel.
- * These options are not given to init - they are for internal kernel use only.
+ * Make sure not to modify "line" after you call this, because the 
+ * argv array keeps references inside of it.  
  */
-static void __init parse_options(char *line)
+void __init parse_options(char *line)
 {
 	char *next,*quote;
-	int args, envs;
-
+	
 	if (!*line)
 		return;
-	args = 0;
-	envs = 1;	/* TERM is set to 'linux' by default */
+
+	if( kernel_argc != -1 )
+		return;
+	kernel_argc = 0;
+
 	next = line;
 	while ((line = next) != NULL) {
                 quote = strchr(line,'"');
@@ -255,39 +311,37 @@
                 }
                 if (next != NULL)
                         *next++ = 0;
-		if (!strncmp(line,"init=",5)) {
-			line += 5;
-			execute_command = line;
-			/* In case LILO is going to boot us with default command line,
-			 * it prepends "auto" before the whole cmdline which makes
-			 * the shell think it should execute a script with such name.
-			 * So we ignore all arguments entered _before_ init=... [MJ]
-			 */
-			args = 0;
-			continue;
-		}
-		if (checksetup(line))
-			continue;
-		
-		/*
-		 * Then check if it's an environment variable or
-		 * an option.
-		 */
-		if (strchr(line,'=')) {
-			if (envs >= MAX_INIT_ENVS)
-				break;
-			envp_init[++envs] = line;
-		} else {
-			if (args >= MAX_INIT_ARGS)
-				break;
-			if (*line)
-				argv_init[++args] = line;
+
+		kernel_argv[kernel_argc++].str = line;
+		if (kernel_argc == MAX_KERNEL_ARGS) {
+			kernel_argc--;
+			return;
 		}
 	}
-	argv_init[args+1] = NULL;
-	envp_init[envs+1] = NULL;
 }
 
+/*
+ * This routine depends on parse_options() being called first.  If you 
+ * want to use this in setup_arch(), make sure you can parse_options() 
+ * first.  
+ *
+ * during the SETUP_ENV level, all previously unused variables will be
+ * filled in as arguments/environment to init as appropriate. 
+ *
+ */
+void __init run_setup(int setup_level)
+{
+	int i;
+	
+	for( i=0; i<kernel_argc; i++ ) {
+		struct kernel_arg *arg = &kernel_argv[i];
+		if (setup_level == SETUP_ENV && !arg->handled)
+			setup_setenv(arg->str);
+		else
+			checksetup(setup_level,arg);
+	}
+
+}
 
 extern void setup_arch(char **);
 extern void cpu_idle(void);
@@ -394,6 +448,8 @@
 	page_alloc_init();
 	printk("Kernel command line: %s\n", saved_command_line);
 	parse_options(command_line);
+	run_setup(SETUP_DEFAULT);
+	run_setup(SETUP_ENV);
 	trap_init();
 	extable_init();
 	rcu_init();

^ permalink raw reply	[flat|nested] 3+ messages in thread

* Re: [PATCH] (2/3) do early command line parsing for 386
  2002-12-03  0:11 [PATCH] (1/3) allow earlier command line parsing Dave Hansen
@ 2002-12-03  0:15 ` Dave Hansen
  2002-12-03  0:15 ` [PATCH] (3/3) early printk " Dave Hansen
  1 sibling, 0 replies; 3+ messages in thread
From: Dave Hansen @ 2002-12-03  0:15 UTC (permalink / raw)
  To: Linux Kernel Mailing List; +Cc: David Woodhouse

[-- Attachment #1: Type: text/plain, Size: 326 bytes --]

This actually implements the early command-line parsing for i386.  It 
actually decreases the amount of code in setup.c, and makes it a lot 
more readable.

setup.c |  138 
+++++++++++++++++++++++++++++++---------------------------------
1 files changed, 68 insertions(+), 70 deletions(-)

-- 
Dave Hansen
haveblue@us.ibm.com

[-- Attachment #2: B-ordered-setup-i386-0.patch --]
[-- Type: text/plain, Size: 5378 bytes --]

# This is a BitKeeper generated patch for the following project:
# Project Name: Linux kernel tree
# This patch format is intended for GNU patch command version 2.5 or higher.
# This patch includes the following deltas:
#	           ChangeSet	1.969   -> 1.970  
#	arch/i386/kernel/setup.c	1.63    -> 1.64   
#
# The following is the BitKeeper ChangeSet Log
# --------------------------------------------
# 02/12/02	haveblue@elm3b96.(none)	1.970
# 1-2
# --------------------------------------------
#
diff -Nru a/arch/i386/kernel/setup.c b/arch/i386/kernel/setup.c
--- a/arch/i386/kernel/setup.c	Mon Dec  2 15:59:44 2002
+++ b/arch/i386/kernel/setup.c	Mon Dec  2 15:59:44 2002
@@ -498,82 +498,68 @@
 	print_memory_map(who);
 } /* setup_memory_region */
 
-
-static void __init parse_cmdline_early (char ** cmdline_p)
+/*
+ * "mem=nopentium" disables the 4MB page tables.
+ */
+static int arch_setup_mem_nopentium(char* arg)
 {
-	char c = ' ', *to = command_line, *from = COMMAND_LINE;
-	int len = 0;
-	int userdef = 0;
-
-	/* Save unparsed command line copy for /proc/cmdline */
-	memcpy(saved_command_line, COMMAND_LINE, COMMAND_LINE_SIZE);
-	saved_command_line[COMMAND_LINE_SIZE-1] = '\0';
-
-	for (;;) {
-		/*
-		 * "mem=nopentium" disables the 4MB page tables.
-		 * "mem=XXX[kKmM]" defines a memory region from HIGH_MEM
-		 * to <mem>, overriding the bios size.
-		 * "mem=XXX[KkmM]@XXX[KkmM]" defines a memory region from
-		 * <start> to <start>+<mem>, overriding the bios size.
-		 */
-		if (c == ' ' && !memcmp(from, "mem=", 4)) {
-			if (to != command_line)
-				to--;
-			if (!memcmp(from+4, "nopentium", 9)) {
-				from += 9+4;
-				clear_bit(X86_FEATURE_PSE, boot_cpu_data.x86_capability);
-			} else if (!memcmp(from+4, "exactmap", 8)) {
-				from += 8+4;
-				e820.nr_map = 0;
-				userdef = 1;
-			} else {
-				/* If the user specifies memory size, we
-				 * limit the BIOS-provided memory map to
-				 * that size. exactmap can be used to specify
-				 * the exact map. mem=number can be used to
-				 * trim the existing memory map.
-				 */
-				unsigned long long start_at, mem_size;
- 
-				mem_size = memparse(from+4, &from);
-				if (*from == '@') {
-					start_at = memparse(from+1, &from);
-					add_memory_region(start_at, mem_size, E820_RAM);
-				} else {
-					limit_regions(mem_size);
-					userdef=1;
-				}
-			}
-		}
+	clear_bit(X86_FEATURE_PSE, boot_cpu_data.x86_capability);
+	return 1;
+}
 
-		/* "acpi=off" disables both ACPI table parsing and interpreter init */
-		if (c == ' ' && !memcmp(from, "acpi=off", 8))
-			acpi_disabled = 1;
+static int arch_setup_mem_exactmap(char* arg)
+{
+	e820.nr_map = 0;
+	printk(KERN_INFO "user-defined physical RAM map:\n");
+	print_memory_map("user");
+	
+	return 1;
+}
 
-		/*
-		 * highmem=size forces highmem to be exactly 'size' bytes.
-		 * This works even on boxes that have no highmem otherwise.
-		 * This also works to reduce highmem size on bigger boxes.
-		 */
-		if (c == ' ' && !memcmp(from, "highmem=", 8))
-			highmem_pages = memparse(from+8, &from) >> PAGE_SHIFT;
+/*
+ * "mem=XXX[kKmM]" defines a memory region from HIGH_MEM
+ * to <mem>, overriding the bios size.
+ * "mem=XXX[KkmM]@XXX[KkmM]" defines a memory region from
+ * <start> to <start>+<mem>, overriding the bios size.
+ */
+static int arch_setup_mem(char* arg)
+{
+	unsigned long long start_at, mem_size;
+	char* next;
 	
-		c = *(from++);
-		if (!c)
-			break;
-		if (COMMAND_LINE_SIZE <= ++len)
-			break;
-		*(to++) = c;
-	}
-	*to = '\0';
-	*cmdline_p = command_line;
-	if (userdef) {
-		printk(KERN_INFO "user-defined physical RAM map:\n");
+	mem_size = memparse(arg, &next);
+	if( arg == next )
+		return 0;
+	if (*next == '@') {
+		start_at = memparse(next+1, &next);
+	add_memory_region(start_at, mem_size, E820_RAM);
+	} else {
+		limit_regions(mem_size);
+		printk(KERN_INFO "user-defined physical RAM size:\n");
 		print_memory_map("user");
 	}
+	return 1;
+}
+
+static int arch_setup_acpi_off(char* arg)
+{
+	acpi_disabled = 1;
+	return 1;
+}
+
+
+static int arch_setup_highmem(char* arg)
+{
+	highmem_pages = memparse(arg, &arg) >> PAGE_SHIFT;
+	return 1;
 }
 
+__ordered_setup(SETUP_ARCH_LATE, "mem=nopentium", arch_setup_mem_nopentium);
+__ordered_setup(SETUP_ARCH_LATE, "mem=exactmap", arch_setup_mem_exactmap);
+__ordered_setup(SETUP_ARCH_LATE, "mem=", arch_setup_mem);
+__ordered_setup(SETUP_ARCH_LATE, "acpi=off", arch_setup_acpi_off);
+__ordered_setup(SETUP_ARCH_LATE, "highmem=", arch_setup_highmem);
+
 /*
  * Find the highest page frame number we have available
  */
@@ -833,10 +819,22 @@
 		pci_mem_start = low_mem_size;
 }
 
+extern void __init parse_options(char *line);
+extern void __init run_setup(int setup_level);
+
 void __init setup_arch(char **cmdline_p)
 {
 	unsigned long max_low_pfn;
 
+	/* Save unparsed command line copy for /proc/cmdline */
+	memcpy(saved_command_line, COMMAND_LINE, COMMAND_LINE_SIZE);
+	saved_command_line[COMMAND_LINE_SIZE-1] = '\0';
+	strncpy(command_line, saved_command_line, COMMAND_LINE_SIZE);
+	*cmdline_p = command_line;
+
+	parse_options(command_line);
+	run_setup(SETUP_ARCH_BEGIN);
+	
 	pre_setup_arch_hook();
 	early_cpu_init();
 
@@ -875,8 +873,8 @@
 	data_resource.start = virt_to_phys(&_etext);
 	data_resource.end = virt_to_phys(&_edata)-1;
 
-	parse_cmdline_early(cmdline_p);
-
+	run_setup(SETUP_ARCH_LATE);
+	
 	max_low_pfn = setup_memory();
 
 	/*

^ permalink raw reply	[flat|nested] 3+ messages in thread

* Re: [PATCH] (3/3)  early printk for 386
  2002-12-03  0:11 [PATCH] (1/3) allow earlier command line parsing Dave Hansen
  2002-12-03  0:15 ` [PATCH] (2/3) do early command line parsing for 386 Dave Hansen
@ 2002-12-03  0:15 ` Dave Hansen
  1 sibling, 0 replies; 3+ messages in thread
From: Dave Hansen @ 2002-12-03  0:15 UTC (permalink / raw)
  To: Linux Kernel Mailing List; +Cc: David Woodhouse

[-- Attachment #1: Type: text/plain, Size: 351 bytes --]

Finally, do early printk for i386.  Only serial is supported for now, 
but there is nothing to stop other console types from being added.

  arch/i386/kernel/setup.c |   14 ++++++++++++++
  drivers/serial/8250.c    |    3 +++
  kernel/printk.c          |    2 +-
  3 files changed, 18 insertions(+), 1 deletion(-)

-- 
Dave Hansen
haveblue@us.ibm.com

[-- Attachment #2: C-early-printk-i386-0.patch --]
[-- Type: text/plain, Size: 2102 bytes --]

# This is a BitKeeper generated patch for the following project:
# Project Name: Linux kernel tree
# This patch format is intended for GNU patch command version 2.5 or higher.
# This patch includes the following deltas:
#	           ChangeSet	1.970   -> 1.971  
#	drivers/serial/8250.c	1.24    -> 1.25   
#	arch/i386/kernel/setup.c	1.64    -> 1.65   
#	     kernel/printk.c	1.15    -> 1.16   
#
# The following is the BitKeeper ChangeSet Log
# --------------------------------------------
# 02/12/02	haveblue@elm3b96.(none)	1.971
# 2-3
# --------------------------------------------
#
diff -Nru a/arch/i386/kernel/setup.c b/arch/i386/kernel/setup.c
--- a/arch/i386/kernel/setup.c	Mon Dec  2 16:00:27 2002
+++ b/arch/i386/kernel/setup.c	Mon Dec  2 16:00:27 2002
@@ -554,6 +554,20 @@
 	return 1;
 }
 
+static int __init early_printk_setup(char* arg) 
+{
+	/* early printk only works for serial ports now */
+	if (strncmp(arg,"ttyS",4) && strncmp(arg,"/dev/ttyS",9))
+		return 0;
+	
+	console_setup(arg);
+	serial8250_console_init();
+	
+	printk( "early printk enabled \n" );
+	return 1;
+}
+
+__ordered_setup(SETUP_ARCH_BEGIN,"console=",early_printk_setup);
 __ordered_setup(SETUP_ARCH_LATE, "mem=nopentium", arch_setup_mem_nopentium);
 __ordered_setup(SETUP_ARCH_LATE, "mem=exactmap", arch_setup_mem_exactmap);
 __ordered_setup(SETUP_ARCH_LATE, "mem=", arch_setup_mem);
diff -Nru a/drivers/serial/8250.c b/drivers/serial/8250.c
--- a/drivers/serial/8250.c	Mon Dec  2 16:00:27 2002
+++ b/drivers/serial/8250.c	Mon Dec  2 16:00:27 2002
@@ -1910,6 +1910,9 @@
 
 void __init serial8250_console_init(void)
 {
+	if (serial8250_console.flags & CON_ENABLED)
+		return;
+	
 	serial8250_isa_init_ports();
 	register_console(&serial8250_console);
 }
diff -Nru a/kernel/printk.c b/kernel/printk.c
--- a/kernel/printk.c	Mon Dec  2 16:00:27 2002
+++ b/kernel/printk.c	Mon Dec  2 16:00:27 2002
@@ -99,7 +99,7 @@
 /*
  *	Setup a list of consoles. Called from init/main.c
  */
-static int __init console_setup(char *str)
+int __init console_setup(char *str)
 {
 	struct console_cmdline *c;
 	char name[sizeof(c->name)];

^ permalink raw reply	[flat|nested] 3+ messages in thread

end of thread, other threads:[~2002-12-03  0:09 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2002-12-03  0:11 [PATCH] (1/3) allow earlier command line parsing Dave Hansen
2002-12-03  0:15 ` [PATCH] (2/3) do early command line parsing for 386 Dave Hansen
2002-12-03  0:15 ` [PATCH] (3/3) early printk " Dave Hansen

This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.