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

📄 rtas-proc.c

📁 linux-2.6.15.6
💻 C
📖 第 1 页 / 共 2 页
字号:
/* *   arch/ppc64/kernel/rtas-proc.c *   Copyright (C) 2000 Tilmann Bitterberg *   (tilmann@bitterberg.de) * *   RTAS (Runtime Abstraction Services) stuff *   Intention is to provide a clean user interface *   to use the RTAS. * *   TODO: *   Split off a header file and maybe move it to a different *   location. Write Documentation on what the /proc/rtas/ entries *   actually do. */#include <linux/errno.h>#include <linux/sched.h>#include <linux/proc_fs.h>#include <linux/stat.h>#include <linux/ctype.h>#include <linux/time.h>#include <linux/string.h>#include <linux/init.h>#include <linux/seq_file.h>#include <linux/bitops.h>#include <linux/rtc.h>#include <asm/uaccess.h>#include <asm/processor.h>#include <asm/io.h>#include <asm/prom.h>#include <asm/rtas.h>#include <asm/machdep.h> /* for ppc_md */#include <asm/time.h>/* Token for Sensors */#define KEY_SWITCH		0x0001#define ENCLOSURE_SWITCH	0x0002#define THERMAL_SENSOR		0x0003#define LID_STATUS		0x0004#define POWER_SOURCE		0x0005#define BATTERY_VOLTAGE		0x0006#define BATTERY_REMAINING	0x0007#define BATTERY_PERCENTAGE	0x0008#define EPOW_SENSOR		0x0009#define BATTERY_CYCLESTATE	0x000a#define BATTERY_CHARGING	0x000b/* IBM specific sensors */#define IBM_SURVEILLANCE	0x2328 /* 9000 */#define IBM_FANRPM		0x2329 /* 9001 */#define IBM_VOLTAGE		0x232a /* 9002 */#define IBM_DRCONNECTOR		0x232b /* 9003 */#define IBM_POWERSUPPLY		0x232c /* 9004 *//* Status return values */#define SENSOR_CRITICAL_HIGH	13#define SENSOR_WARNING_HIGH	12#define SENSOR_NORMAL		11#define SENSOR_WARNING_LOW	10#define SENSOR_CRITICAL_LOW	 9#define SENSOR_SUCCESS		 0#define SENSOR_HW_ERROR		-1#define SENSOR_BUSY		-2#define SENSOR_NOT_EXIST	-3#define SENSOR_DR_ENTITY	-9000/* Location Codes */#define LOC_SCSI_DEV_ADDR	'A'#define LOC_SCSI_DEV_LOC	'B'#define LOC_CPU			'C'#define LOC_DISKETTE		'D'#define LOC_ETHERNET		'E'#define LOC_FAN			'F'#define LOC_GRAPHICS		'G'/* reserved / not used		'H' */#define LOC_IO_ADAPTER		'I'/* reserved / not used		'J' */#define LOC_KEYBOARD		'K'#define LOC_LCD			'L'#define LOC_MEMORY		'M'#define LOC_NV_MEMORY		'N'#define LOC_MOUSE		'O'#define LOC_PLANAR		'P'#define LOC_OTHER_IO		'Q'#define LOC_PARALLEL		'R'#define LOC_SERIAL		'S'#define LOC_DEAD_RING		'T'#define LOC_RACKMOUNTED		'U' /* for _u_nit is rack mounted */#define LOC_VOLTAGE		'V'#define LOC_SWITCH_ADAPTER	'W'#define LOC_OTHER		'X'#define LOC_FIRMWARE		'Y'#define LOC_SCSI		'Z'/* Tokens for indicators */#define TONE_FREQUENCY		0x0001 /* 0 - 1000 (HZ)*/#define TONE_VOLUME		0x0002 /* 0 - 100 (%) */#define SYSTEM_POWER_STATE	0x0003 #define WARNING_LIGHT		0x0004#define DISK_ACTIVITY_LIGHT	0x0005#define HEX_DISPLAY_UNIT	0x0006#define BATTERY_WARNING_TIME	0x0007#define CONDITION_CYCLE_REQUEST	0x0008#define SURVEILLANCE_INDICATOR	0x2328 /* 9000 */#define DR_ACTION		0x2329 /* 9001 */#define DR_INDICATOR		0x232a /* 9002 *//* 9003 - 9004: Vendor specific *//* 9006 - 9999: Vendor specific *//* other */#define MAX_SENSORS		 17  /* I only know of 17 sensors */    #define MAX_LINELENGTH          256#define SENSOR_PREFIX		"ibm,sensor-"#define cel_to_fahr(x)		((x*9/5)+32)/* Globals */static struct rtas_sensors sensors;static struct device_node *rtas_node = NULL;static unsigned long power_on_time = 0; /* Save the time the user set */static char progress_led[MAX_LINELENGTH];static unsigned long rtas_tone_frequency = 1000;static unsigned long rtas_tone_volume = 0;/* ****************STRUCTS******************************************* */struct individual_sensor {	unsigned int token;	unsigned int quant;};struct rtas_sensors {        struct individual_sensor sensor[MAX_SENSORS];	unsigned int quant;};/* ****************************************************************** *//* Declarations */static int ppc_rtas_sensors_show(struct seq_file *m, void *v);static int ppc_rtas_clock_show(struct seq_file *m, void *v);static ssize_t ppc_rtas_clock_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos);static int ppc_rtas_progress_show(struct seq_file *m, void *v);static ssize_t ppc_rtas_progress_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos);static int ppc_rtas_poweron_show(struct seq_file *m, void *v);static ssize_t ppc_rtas_poweron_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos);static ssize_t ppc_rtas_tone_freq_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos);static int ppc_rtas_tone_freq_show(struct seq_file *m, void *v);static ssize_t ppc_rtas_tone_volume_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos);static int ppc_rtas_tone_volume_show(struct seq_file *m, void *v);static int ppc_rtas_rmo_buf_show(struct seq_file *m, void *v);static int sensors_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_sensors_show, NULL);}struct file_operations ppc_rtas_sensors_operations = {	.open		= sensors_open,	.read		= seq_read,	.llseek		= seq_lseek,	.release	= single_release,};static int poweron_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_poweron_show, NULL);}struct file_operations ppc_rtas_poweron_operations = {	.open		= poweron_open,	.read		= seq_read,	.llseek		= seq_lseek,	.write		= ppc_rtas_poweron_write,	.release	= single_release,};static int progress_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_progress_show, NULL);}struct file_operations ppc_rtas_progress_operations = {	.open		= progress_open,	.read		= seq_read,	.llseek		= seq_lseek,	.write		= ppc_rtas_progress_write,	.release	= single_release,};static int clock_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_clock_show, NULL);}struct file_operations ppc_rtas_clock_operations = {	.open		= clock_open,	.read		= seq_read,	.llseek		= seq_lseek,	.write		= ppc_rtas_clock_write,	.release	= single_release,};static int tone_freq_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_tone_freq_show, NULL);}struct file_operations ppc_rtas_tone_freq_operations = {	.open		= tone_freq_open,	.read		= seq_read,	.llseek		= seq_lseek,	.write		= ppc_rtas_tone_freq_write,	.release	= single_release,};static int tone_volume_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_tone_volume_show, NULL);}struct file_operations ppc_rtas_tone_volume_operations = {	.open		= tone_volume_open,	.read		= seq_read,	.llseek		= seq_lseek,	.write		= ppc_rtas_tone_volume_write,	.release	= single_release,};static int rmo_buf_open(struct inode *inode, struct file *file){	return single_open(file, ppc_rtas_rmo_buf_show, NULL);}struct file_operations ppc_rtas_rmo_buf_ops = {	.open		= rmo_buf_open,	.read		= seq_read,	.llseek		= seq_lseek,	.release	= single_release,};static int ppc_rtas_find_all_sensors(void);static void ppc_rtas_process_sensor(struct seq_file *m,	struct individual_sensor *s, int state, int error, char *loc);static char *ppc_rtas_process_error(int error);static void get_location_code(struct seq_file *m,	struct individual_sensor *s, char *loc);static void check_location_string(struct seq_file *m, char *c);static void check_location(struct seq_file *m, char *c);static int __init proc_rtas_init(void){	struct proc_dir_entry *entry;	if (_machine != PLATFORM_PSERIES && _machine != PLATFORM_PSERIES_LPAR)		return 1;	rtas_node = of_find_node_by_name(NULL, "rtas");	if (rtas_node == NULL)		return 1;	entry = create_proc_entry("ppc64/rtas/progress", S_IRUGO|S_IWUSR, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_progress_operations;	entry = create_proc_entry("ppc64/rtas/clock", S_IRUGO|S_IWUSR, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_clock_operations;	entry = create_proc_entry("ppc64/rtas/poweron", S_IWUSR|S_IRUGO, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_poweron_operations;	entry = create_proc_entry("ppc64/rtas/sensors", S_IRUGO, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_sensors_operations;	entry = create_proc_entry("ppc64/rtas/frequency", S_IWUSR|S_IRUGO,				  NULL);	if (entry)		entry->proc_fops = &ppc_rtas_tone_freq_operations;	entry = create_proc_entry("ppc64/rtas/volume", S_IWUSR|S_IRUGO, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_tone_volume_operations;	entry = create_proc_entry("ppc64/rtas/rmo_buffer", S_IRUSR, NULL);	if (entry)		entry->proc_fops = &ppc_rtas_rmo_buf_ops;	return 0;}__initcall(proc_rtas_init);static int parse_number(const char __user *p, size_t count, unsigned long *val){	char buf[40];	char *end;	if (count > 39)		return -EINVAL;	if (copy_from_user(buf, p, count))		return -EFAULT;	buf[count] = 0;	*val = simple_strtoul(buf, &end, 10);	if (*end && *end != '\n')		return -EINVAL;	return 0;}/* ****************************************************************** *//* POWER-ON-TIME                                                      *//* ****************************************************************** */static ssize_t ppc_rtas_poweron_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos){	struct rtc_time tm;	unsigned long nowtime;	int error = parse_number(buf, count, &nowtime);	if (error)		return error;	power_on_time = nowtime; /* save the time */	to_tm(nowtime, &tm);	error = rtas_call(rtas_token("set-time-for-power-on"), 7, 1, NULL, 			tm.tm_year, tm.tm_mon, tm.tm_mday, 			tm.tm_hour, tm.tm_min, tm.tm_sec, 0 /* nano */);	if (error)		printk(KERN_WARNING "error: setting poweron time returned: %s\n", 				ppc_rtas_process_error(error));	return count;}/* ****************************************************************** */static int ppc_rtas_poweron_show(struct seq_file *m, void *v){	if (power_on_time == 0)		seq_printf(m, "Power on time not set\n");	else		seq_printf(m, "%lu\n",power_on_time);	return 0;}/* ****************************************************************** *//* PROGRESS                                                           *//* ****************************************************************** */static ssize_t ppc_rtas_progress_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos){	unsigned long hex;	if (count >= MAX_LINELENGTH)		count = MAX_LINELENGTH -1;	if (copy_from_user(progress_led, buf, count)) { /* save the string */		return -EFAULT;	}	progress_led[count] = 0;	/* Lets see if the user passed hexdigits */	hex = simple_strtoul(progress_led, NULL, 10);	rtas_progress ((char *)progress_led, hex);	return count;	/* clear the line */	/* rtas_progress("                   ", 0xffff);*/}/* ****************************************************************** */static int ppc_rtas_progress_show(struct seq_file *m, void *v){	if (progress_led)		seq_printf(m, "%s\n", progress_led);	return 0;}/* ****************************************************************** *//* CLOCK                                                              *//* ****************************************************************** */static ssize_t ppc_rtas_clock_write(struct file *file,		const char __user *buf, size_t count, loff_t *ppos){	struct rtc_time tm;	unsigned long nowtime;	int error = parse_number(buf, count, &nowtime);	if (error)		return error;	to_tm(nowtime, &tm);	error = rtas_call(rtas_token("set-time-of-day"), 7, 1, NULL, 			tm.tm_year, tm.tm_mon, tm.tm_mday, 			tm.tm_hour, tm.tm_min, tm.tm_sec, 0);	if (error)

⌨️ 快捷键说明

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