⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 early.c

📁 linux 内核源代码
💻 C
字号:
/* *  arch/s390/kernel/early.c * *    Copyright IBM Corp. 2007 *    Author(s): Hongjie Yang <hongjie@us.ibm.com>, *		 Heiko Carstens <heiko.carstens@de.ibm.com> */#include <linux/init.h>#include <linux/errno.h>#include <linux/string.h>#include <linux/ctype.h>#include <linux/lockdep.h>#include <linux/module.h>#include <linux/pfn.h>#include <linux/uaccess.h>#include <asm/ipl.h>#include <asm/lowcore.h>#include <asm/processor.h>#include <asm/sections.h>#include <asm/setup.h>#include <asm/cpcmd.h>#include <asm/sclp.h>/* * Create a Kernel NSS if the SAVESYS= parameter is defined */#define DEFSYS_CMD_SIZE		96#define SAVESYS_CMD_SIZE	32char kernel_nss_name[NSS_NAME_SIZE + 1];#ifdef CONFIG_SHARED_KERNELstatic noinline __init void create_kernel_nss(void){	unsigned int i, stext_pfn, eshared_pfn, end_pfn, min_size;#ifdef CONFIG_BLK_DEV_INITRD	unsigned int sinitrd_pfn, einitrd_pfn;#endif	int response;	char *savesys_ptr;	char upper_command_line[COMMAND_LINE_SIZE];	char defsys_cmd[DEFSYS_CMD_SIZE];	char savesys_cmd[SAVESYS_CMD_SIZE];	/* Do nothing if we are not running under VM */	if (!MACHINE_IS_VM)		return;	/* Convert COMMAND_LINE to upper case */	for (i = 0; i < strlen(COMMAND_LINE); i++)		upper_command_line[i] = toupper(COMMAND_LINE[i]);	savesys_ptr = strstr(upper_command_line, "SAVESYS=");	if (!savesys_ptr)		return;	savesys_ptr += 8;    /* Point to the beginning of the NSS name */	for (i = 0; i < NSS_NAME_SIZE; i++) {		if (savesys_ptr[i] == ' ' || savesys_ptr[i] == '\0')			break;		kernel_nss_name[i] = savesys_ptr[i];	}	stext_pfn = PFN_DOWN(__pa(&_stext));	eshared_pfn = PFN_DOWN(__pa(&_eshared));	end_pfn = PFN_UP(__pa(&_end));	min_size = end_pfn << 2;	sprintf(defsys_cmd, "DEFSYS %s 00000-%.5X EW %.5X-%.5X SR %.5X-%.5X",		kernel_nss_name, stext_pfn - 1, stext_pfn, eshared_pfn - 1,		eshared_pfn, end_pfn);#ifdef CONFIG_BLK_DEV_INITRD	if (INITRD_START && INITRD_SIZE) {		sinitrd_pfn = PFN_DOWN(__pa(INITRD_START));		einitrd_pfn = PFN_UP(__pa(INITRD_START + INITRD_SIZE));		min_size = einitrd_pfn << 2;		sprintf(defsys_cmd, "%s EW %.5X-%.5X", defsys_cmd,		sinitrd_pfn, einitrd_pfn);	}#endif	sprintf(defsys_cmd, "%s EW MINSIZE=%.7iK", defsys_cmd, min_size);	sprintf(savesys_cmd, "SAVESYS %s \n IPL %s",		kernel_nss_name, kernel_nss_name);	__cpcmd(defsys_cmd, NULL, 0, &response);	if (response != 0)		return;	__cpcmd(savesys_cmd, NULL, 0, &response);	if (response != strlen(savesys_cmd))		return;	ipl_flags = IPL_NSS_VALID;}#else /* CONFIG_SHARED_KERNEL */static inline void create_kernel_nss(void) { }#endif /* CONFIG_SHARED_KERNEL *//* * Clear bss memory */static noinline __init void clear_bss_section(void){	memset(__bss_start, 0, __bss_stop - __bss_start);}/* * Initialize storage key for kernel pages */static noinline __init void init_kernel_storage_key(void){	unsigned long end_pfn, init_pfn;	end_pfn = PFN_UP(__pa(&_end));	for (init_pfn = 0 ; init_pfn < end_pfn; init_pfn++)		page_set_storage_key(init_pfn << PAGE_SHIFT, PAGE_DEFAULT_KEY);}static noinline __init void detect_machine_type(void){	struct cpuinfo_S390 *cpuinfo = &S390_lowcore.cpu_data;	get_cpu_id(&S390_lowcore.cpu_data.cpu_id);	/* Running under z/VM ? */	if (cpuinfo->cpu_id.version == 0xff)		machine_flags |= 1;	/* Running on a P/390 ? */	if (cpuinfo->cpu_id.machine == 0x7490)		machine_flags |= 4;}#ifdef CONFIG_64BITstatic noinline __init int memory_fast_detect(void){	unsigned long val0 = 0;	unsigned long val1 = 0xc;	int ret = -ENOSYS;	if (ipl_flags & IPL_NSS_VALID)		return -ENOSYS;	asm volatile(		"	diag	%1,%2,0x260\n"		"0:	lhi	%0,0\n"		"1:\n"		EX_TABLE(0b,1b)		: "+d" (ret), "+d" (val0), "+d" (val1) : : "cc");	if (ret || val0 != val1)		return -ENOSYS;	memory_chunk[0].size = val0 + 1;	return 0;}#elsestatic inline int memory_fast_detect(void){	return -ENOSYS;}#endifstatic inline __init unsigned long __tprot(unsigned long addr){	int cc = -1;	asm volatile(		"	tprot	0(%1),0\n"		"0:	ipm	%0\n"		"	srl	%0,28\n"		"1:\n"		EX_TABLE(0b,1b)		: "+d" (cc) : "a" (addr) : "cc");	return (unsigned long)cc;}/* Checking memory in 128KB increments. */#define CHUNK_INCR	(1UL << 17)#define ADDR2G		(1UL << 31)static noinline __init void find_memory_chunks(unsigned long memsize){	unsigned long addr = 0, old_addr = 0;	unsigned long old_cc = CHUNK_READ_WRITE;	unsigned long cc;	int chunk = 0;	while (chunk < MEMORY_CHUNKS) {		cc = __tprot(addr);		while (cc == old_cc) {			addr += CHUNK_INCR;			if (memsize && addr >= memsize)				break;#ifndef CONFIG_64BIT			if (addr == ADDR2G)				break;#endif			cc = __tprot(addr);		}		if (old_addr != addr &&		    (old_cc == CHUNK_READ_WRITE || old_cc == CHUNK_READ_ONLY)) {			memory_chunk[chunk].addr = old_addr;			memory_chunk[chunk].size = addr - old_addr;			memory_chunk[chunk].type = old_cc;			chunk++;		}		old_addr = addr;		old_cc = cc;#ifndef CONFIG_64BIT		if (addr == ADDR2G)			break;#endif		/*		 * Finish memory detection at the first hole		 * if storage size is unknown.		 */		if (cc == -1UL && !memsize)			break;		if (memsize && addr >= memsize)			break;	}}static __init void early_pgm_check_handler(void){	unsigned long addr;	const struct exception_table_entry *fixup;	addr = S390_lowcore.program_old_psw.addr;	fixup = search_exception_tables(addr & PSW_ADDR_INSN);	if (!fixup)		disabled_wait(0);	S390_lowcore.program_old_psw.addr = fixup->fixup | PSW_ADDR_AMODE;}static noinline __init void setup_lowcore_early(void){	psw_t psw;	psw.mask = PSW_BASE_BITS | PSW_DEFAULT_KEY;	psw.addr = PSW_ADDR_AMODE | (unsigned long) s390_base_ext_handler;	S390_lowcore.external_new_psw = psw;	psw.addr = PSW_ADDR_AMODE | (unsigned long) s390_base_pgm_handler;	S390_lowcore.program_new_psw = psw;	s390_base_pgm_handler_fn = early_pgm_check_handler;}/* * Save ipl parameters, clear bss memory, initialize storage keys * and create a kernel NSS at startup if the SAVESYS= parm is defined */void __init startup_init(void){	unsigned long long memsize;	ipl_save_parameters();	clear_bss_section();	init_kernel_storage_key();	lockdep_init();	lockdep_off();	detect_machine_type();	create_kernel_nss();	sort_main_extable();	setup_lowcore_early();	sclp_readinfo_early();	sclp_facilities_detect();	memsize = sclp_memory_detect();#ifndef CONFIG_64BIT	/*	 * Can't deal with more than 2G in 31 bit addressing mode, so	 * limit the value in order to avoid strange side effects.	 */	if (memsize > ADDR2G)		memsize = ADDR2G;#endif	if (memory_fast_detect() < 0)		find_memory_chunks((unsigned long) memsize);	lockdep_on();}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -