xref: /netbsd-src/sys/arch/mips/pmon/pmon.c (revision 31d88a0f6f885d78ff0b79a3881d7ea201aa3ebf)
162c4d5a4Sbouyer /*	$OpenBSD: pmon.c,v 1.4 2010/02/16 21:29:54 miod Exp $	*/
262c4d5a4Sbouyer 
362c4d5a4Sbouyer /*
462c4d5a4Sbouyer  * Copyright (c) 2009 Miodrag Vallat.
562c4d5a4Sbouyer  *
662c4d5a4Sbouyer  * Permission to use, copy, modify, and distribute this software for any
762c4d5a4Sbouyer  * purpose with or without fee is hereby granted, provided that the above
862c4d5a4Sbouyer  * copyright notice and this permission notice appear in all copies.
962c4d5a4Sbouyer  *
1062c4d5a4Sbouyer  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
1162c4d5a4Sbouyer  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
1262c4d5a4Sbouyer  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
1362c4d5a4Sbouyer  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
1462c4d5a4Sbouyer  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
1562c4d5a4Sbouyer  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
1662c4d5a4Sbouyer  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
1762c4d5a4Sbouyer  */
1862c4d5a4Sbouyer 
1962c4d5a4Sbouyer #include <sys/param.h>
2062c4d5a4Sbouyer #include <sys/systm.h>
21*31d88a0fSmatt #include <sys/cpu.h>
2262c4d5a4Sbouyer #include <sys/proc.h>
2362c4d5a4Sbouyer 
24*31d88a0fSmatt #include <mips/cpuregs.h>
2562c4d5a4Sbouyer #include <mips/pmon/pmon.h>
2662c4d5a4Sbouyer 
2762c4d5a4Sbouyer int	pmon_argc;
2862c4d5a4Sbouyer int32_t	*pmon_argv;
2962c4d5a4Sbouyer int32_t	*pmon_envp;
3062c4d5a4Sbouyer 
3162c4d5a4Sbouyer void
pmon_init(int32_t argc,int32_t argv,int32_t envp,int32_t callvec)3262c4d5a4Sbouyer pmon_init(int32_t argc, int32_t argv, int32_t envp, int32_t callvec)
3362c4d5a4Sbouyer {
3462c4d5a4Sbouyer 	pmon_callvec = callvec;
3562c4d5a4Sbouyer 
3662c4d5a4Sbouyer 	pmon_argc = argc;
3762c4d5a4Sbouyer 	/* sign extend pointers */
3862c4d5a4Sbouyer 	pmon_argv = (int32_t *)(vaddr_t)argv;
3962c4d5a4Sbouyer 	pmon_envp = (int32_t *)(vaddr_t)envp;
4062c4d5a4Sbouyer }
4162c4d5a4Sbouyer 
4262c4d5a4Sbouyer const char *
pmon_getarg(const int argno)4362c4d5a4Sbouyer pmon_getarg(const int argno)
4462c4d5a4Sbouyer {
4562c4d5a4Sbouyer 	if (argno < 0 || argno >= pmon_argc)
4662c4d5a4Sbouyer 		return NULL;
4762c4d5a4Sbouyer 
4862c4d5a4Sbouyer 	return (const char *)(vaddr_t)pmon_argv[argno];
4962c4d5a4Sbouyer }
5062c4d5a4Sbouyer 
5162c4d5a4Sbouyer const char *
pmon_getenv(const char * var)5262c4d5a4Sbouyer pmon_getenv(const char *var)
5362c4d5a4Sbouyer {
5462c4d5a4Sbouyer 	int32_t *envptr = pmon_envp;
5562c4d5a4Sbouyer 	const char *envstr;
5662c4d5a4Sbouyer 	size_t varlen;
5762c4d5a4Sbouyer 
5862c4d5a4Sbouyer 	if (envptr == NULL)
5962c4d5a4Sbouyer 		return NULL;
6062c4d5a4Sbouyer 
6162c4d5a4Sbouyer 	varlen = strlen(var);
6262c4d5a4Sbouyer 	while (*envptr != 0) {
6362c4d5a4Sbouyer 		envstr = (const char *)(vaddr_t)*envptr;
6462c4d5a4Sbouyer 		/*
6562c4d5a4Sbouyer 		 * There is a PMON2000 bug, at least on Lemote Yeeloong,
6662c4d5a4Sbouyer 		 * which causes it to override part of the environment
6762c4d5a4Sbouyer 		 * pointers array with the environment data itself.
6862c4d5a4Sbouyer 		 *
6962c4d5a4Sbouyer 		 * This only happens on cold boot, and if the BSD kernel
7062c4d5a4Sbouyer 		 * is loaded without symbols (i.e. no option -k passed
7162c4d5a4Sbouyer 		 * to the boot command).
7262c4d5a4Sbouyer 		 *
7362c4d5a4Sbouyer 		 * Until a suitable workaround is found or the bug is
7462c4d5a4Sbouyer 		 * fixed, ignore broken environment information and
7562c4d5a4Sbouyer 		 * tell the user (in case this prevents us from finding
7662c4d5a4Sbouyer 		 * important information).
7762c4d5a4Sbouyer 		 */
7862c4d5a4Sbouyer 		if ((vaddr_t)envstr < (vaddr_t)MIPS_KSEG1_START ||
7962c4d5a4Sbouyer 		    (vaddr_t)envstr >= (vaddr_t)MIPS_KSEG2_START) {
8062c4d5a4Sbouyer 			printf("WARNING! CORRUPTED ENVIRONMENT!\n");
8162c4d5a4Sbouyer 			printf("Unable to search for %s.\n", var);
8262c4d5a4Sbouyer #ifdef _STANDALONE
8362c4d5a4Sbouyer 			printf("If boot fails, power-cycle the machine.\n");
8462c4d5a4Sbouyer #else
8562c4d5a4Sbouyer 			printf("If the kernel fails to identify the system"
8662c4d5a4Sbouyer 			    " type, please boot it again with `-k' option.\n");
8762c4d5a4Sbouyer #endif
8862c4d5a4Sbouyer 
8962c4d5a4Sbouyer 			/* terminate environment for further calls */
9062c4d5a4Sbouyer 			*envptr = 0;
9162c4d5a4Sbouyer 			break;
9262c4d5a4Sbouyer 		}
9362c4d5a4Sbouyer 		if (strncmp(envstr, var, varlen) == 0 &&
9462c4d5a4Sbouyer 		    envstr[varlen] == '=')
9562c4d5a4Sbouyer 			return envstr + varlen + 1;
9662c4d5a4Sbouyer 		envptr++;
9762c4d5a4Sbouyer 	}
9862c4d5a4Sbouyer 
9962c4d5a4Sbouyer 	return NULL;
10062c4d5a4Sbouyer }
101