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

📄 misc.c

📁 Linux内核源代码 为压缩文件 是<<Linux内核>>一书中的源代码
💻 C
字号:
/* $Id: misc.c,v 1.19 2000/06/30 10:18:38 davem Exp $ * misc.c:  Miscellaneous prom functions that don't belong *          anywhere else. * * Copyright (C) 1995 David S. Miller (davem@caip.rutgers.edu) * Copyright (C) 1996,1997 Jakub Jelinek (jj@sunsite.mff.cuni.cz) */#include <linux/config.h>#include <linux/types.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/interrupt.h>#include <linux/delay.h>#include <asm/openprom.h>#include <asm/oplib.h>/* Reset and reboot the machine with the command 'bcommand'. */voidprom_reboot(char *bcommand){	p1275_cmd ("boot", P1275_ARG(0,P1275_ARG_IN_STRING)|		           P1275_INOUT(1,0), bcommand);}/* Forth evaluate the expression contained in 'fstring'. */voidprom_feval(char *fstring){	if(!fstring || fstring[0] == 0)		return;	p1275_cmd ("interpret", P1275_ARG(0,P1275_ARG_IN_STRING)|				P1275_INOUT(1,1), fstring);}/* We want to do this more nicely some day. */#ifdef CONFIG_SUN_CONSOLEextern void (*prom_palette)(int);extern int serial_console;#endif#ifdef CONFIG_SMPextern void smp_capture(void);extern void smp_release(void);#endif/* Drop into the prom, with the chance to continue with the 'go' * prom command. */voidprom_cmdline(void){	unsigned long flags;	__save_and_cli(flags);#ifdef CONFIG_SUN_CONSOLE	if(!serial_console && prom_palette)		prom_palette (1);#endif	/* We always arrive here via a serial interrupt.	 * So in order for everything to work reliably, even	 * on SMP, we need to drop the IRQ locks we hold.	 */#ifdef CONFIG_SMP	irq_exit(smp_processor_id(), 0);	smp_capture();#else	local_irq_count(smp_processor_id())--;#endif	p1275_cmd ("enter", P1275_INOUT(0,0));#ifdef CONFIG_SMP	smp_release();	irq_enter(smp_processor_id(), 0);	spin_unlock_wait(&__br_write_locks[BR_GLOBALIRQ_LOCK].lock);#else	local_irq_count(smp_processor_id())++;#endif#ifdef CONFIG_SUN_CONSOLE	if(!serial_console && prom_palette)		prom_palette (0);#endif	__restore_flags(flags);}#ifdef CONFIG_SMPextern void smp_promstop_others(void);#endif/* Drop into the prom, but completely terminate the program. * No chance of continuing. */voidprom_halt(void){#ifdef CONFIG_SMP	smp_promstop_others();	udelay(8000);#endifagain:	p1275_cmd ("exit", P1275_INOUT(0,0));	goto again; /* PROM is out to get me -DaveM */}/* Set prom sync handler to call function 'funcp'. */voidprom_setcallback(callback_func_t funcp){	if(!funcp) return;	p1275_cmd ("set-callback", P1275_ARG(0,P1275_ARG_IN_FUNCTION)|				   P1275_INOUT(1,1), funcp);}/* Get the idprom and stuff it into buffer 'idbuf'.  Returns the * format type.  'num_bytes' is the number of bytes that your idbuf * has space for.  Returns 0xff on error. */unsigned charprom_get_idprom(char *idbuf, int num_bytes){	int len;	len = prom_getproplen(prom_root_node, "idprom");	if((len>num_bytes) || (len==-1)) return 0xff;	if(!prom_getproperty(prom_root_node, "idprom", idbuf, num_bytes))		return idbuf[0];	return 0xff;}/* Get the major prom version number. */intprom_version(void){	return PROM_P1275;}/* Get the prom plugin-revision. */intprom_getrev(void){	return prom_rev;}/* Get the prom firmware print revision. */intprom_getprev(void){	return prom_prev;}/* Install Linux trap table so PROM uses that instead of it's own. */void prom_set_trap_table(unsigned long tba){	p1275_cmd("SUNW,set-trap-table", P1275_INOUT(1, 0), tba);}int mmu_ihandle_cache = 0;int prom_get_mmu_ihandle(void){	int node, ret;	if (mmu_ihandle_cache != 0)		return mmu_ihandle_cache;	node = prom_finddevice("/chosen");	ret = prom_getint(node, "mmu");	if(ret == -1 || ret == 0)		mmu_ihandle_cache = -1;	else		mmu_ihandle_cache = ret;	return ret;}static int prom_get_memory_ihandle(void){	static int memory_ihandle_cache = 0;	int node, ret;	if (memory_ihandle_cache != 0)		return memory_ihandle_cache;	node = prom_finddevice("/chosen");	ret = prom_getint(node, "memory");	if (ret == -1 || ret == 0)		memory_ihandle_cache = -1;	else		memory_ihandle_cache = ret;	return ret;}/* Load explicit I/D TLB entries. */long prom_itlb_load(unsigned long index,		    unsigned long tte_data,		    unsigned long vaddr){	return p1275_cmd("call-method",			 (P1275_ARG(0, P1275_ARG_IN_STRING) |			  P1275_ARG(2, P1275_ARG_IN_64B) |			  P1275_ARG(3, P1275_ARG_IN_64B) |			  P1275_INOUT(5, 1)),			 "SUNW,itlb-load",			 prom_get_mmu_ihandle(),			 /* And then our actual args are pushed backwards. */			 vaddr,			 tte_data,			 index);}long prom_dtlb_load(unsigned long index,		    unsigned long tte_data,		    unsigned long vaddr){	return p1275_cmd("call-method",			 (P1275_ARG(0, P1275_ARG_IN_STRING) |			  P1275_ARG(2, P1275_ARG_IN_64B) |			  P1275_ARG(3, P1275_ARG_IN_64B) |			  P1275_INOUT(5, 1)),			 "SUNW,dtlb-load",			 prom_get_mmu_ihandle(),			 /* And then our actual args are pushed backwards. */			 vaddr,			 tte_data,			 index);}int prom_map(int mode, unsigned long size,	     unsigned long vaddr, unsigned long paddr){	int ret = p1275_cmd("call-method",			    (P1275_ARG(0, P1275_ARG_IN_STRING) |			     P1275_ARG(3, P1275_ARG_IN_64B) |			     P1275_ARG(4, P1275_ARG_IN_64B) |			     P1275_ARG(6, P1275_ARG_IN_64B) |			     P1275_INOUT(7, 1)),			    "map",			    prom_get_mmu_ihandle(),			    mode,			    size,			    vaddr,			    0,			    paddr);	if (ret == 0)		ret = -1;	return ret;}void prom_unmap(unsigned long size, unsigned long vaddr){	p1275_cmd("call-method",		  (P1275_ARG(0, P1275_ARG_IN_STRING) |		   P1275_ARG(2, P1275_ARG_IN_64B) |		   P1275_ARG(3, P1275_ARG_IN_64B) |		   P1275_INOUT(4, 0)),		  "unmap",		  prom_get_mmu_ihandle(),		  size,		  vaddr);}/* Set aside physical memory which is not touched or modified * across soft resets. */unsigned long prom_retain(char *name,			  unsigned long pa_low, unsigned long pa_high,			  long size, long align){	/* XXX I don't think we return multiple values correctly.	 * XXX OBP supposedly returns pa_low/pa_high here, how does	 * XXX it work?	 */	/* If align is zero, the pa_low/pa_high args are passed,	 * else they are not.	 */	if(align == 0)		return p1275_cmd("SUNW,retain",				 (P1275_ARG(0, P1275_ARG_IN_BUF) | P1275_INOUT(5, 2)),				 name, pa_low, pa_high, size, align);	else		return p1275_cmd("SUNW,retain",				 (P1275_ARG(0, P1275_ARG_IN_BUF) | P1275_INOUT(3, 2)),				 name, size, align);}/* Get "Unumber" string for the SIMM at the given * memory address.  Usually this will be of the form * "Uxxxx" where xxxx is a decimal number which is * etched into the motherboard next to the SIMM slot * in question. */int prom_getunumber(int syndrome_code,		    unsigned long phys_addr,		    char *buf, int buflen){	return p1275_cmd("call-method",			 (P1275_ARG(0, P1275_ARG_IN_STRING)	|			  P1275_ARG(3, P1275_ARG_OUT_BUF)	|			  P1275_ARG(6, P1275_ARG_IN_64B)	|			  P1275_INOUT(8, 2)),			 "SUNW,get-unumber", prom_get_memory_ihandle(),			 buflen, buf, P1275_SIZE(buflen),			 0, phys_addr, syndrome_code);}/* Power management extensions. */void prom_sleepself(void){	p1275_cmd("SUNW,sleep-self", P1275_INOUT(0, 0));}int prom_sleepsystem(void){	return p1275_cmd("SUNW,sleep-system", P1275_INOUT(0, 1));}int prom_wakeupsystem(void){	return p1275_cmd("SUNW,wakeup-system", P1275_INOUT(0, 1));}#ifdef CONFIG_SMPvoid prom_startcpu(int cpunode, unsigned long pc, unsigned long o0){	p1275_cmd("SUNW,start-cpu", P1275_INOUT(3, 0), cpunode, pc, o0);}void prom_stopself(void){	p1275_cmd("SUNW,stop-self", P1275_INOUT(0, 0));}void prom_idleself(void){	p1275_cmd("SUNW,idle-self", P1275_INOUT(0, 0));}void prom_resumecpu(int cpunode){	p1275_cmd("SUNW,resume-cpu", P1275_INOUT(1, 0), cpunode);}#endif

⌨️ 快捷键说明

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