gl520sm.c
来自「linux 内核源代码」· C语言 代码 · 共 815 行 · 第 1/2 页
C
815 行
/* gl520sm.c - Part of lm_sensors, Linux kernel modules for hardware monitoring Copyright (c) 1998, 1999 Frodo Looijaard <frodol@dds.nl>, Kyösti Mälkki <kmalkki@cc.hut.fi> Copyright (c) 2005 Maarten Deprez <maartendeprez@users.sourceforge.net> This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.*/#include <linux/module.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/jiffies.h>#include <linux/i2c.h>#include <linux/hwmon.h>#include <linux/hwmon-vid.h>#include <linux/err.h>#include <linux/mutex.h>#include <linux/sysfs.h>/* Type of the extra sensor */static unsigned short extra_sensor_type;module_param(extra_sensor_type, ushort, 0);MODULE_PARM_DESC(extra_sensor_type, "Type of extra sensor (0=autodetect, 1=temperature, 2=voltage)");/* Addresses to scan */static unsigned short normal_i2c[] = { 0x2c, 0x2d, I2C_CLIENT_END };/* Insmod parameters */I2C_CLIENT_INSMOD_1(gl520sm);/* Many GL520 constants specified below One of the inputs can be configured as either temp or voltage.That's why _TEMP2 and _IN4 access the same register *//* The GL520 registers */#define GL520_REG_CHIP_ID 0x00#define GL520_REG_REVISION 0x01#define GL520_REG_CONF 0x03#define GL520_REG_MASK 0x11#define GL520_REG_VID_INPUT 0x02#define GL520_REG_IN0_INPUT 0x15#define GL520_REG_IN0_LIMIT 0x0c#define GL520_REG_IN0_MIN GL520_REG_IN0_LIMIT#define GL520_REG_IN0_MAX GL520_REG_IN0_LIMIT#define GL520_REG_IN1_INPUT 0x14#define GL520_REG_IN1_LIMIT 0x09#define GL520_REG_IN1_MIN GL520_REG_IN1_LIMIT#define GL520_REG_IN1_MAX GL520_REG_IN1_LIMIT#define GL520_REG_IN2_INPUT 0x13#define GL520_REG_IN2_LIMIT 0x0a#define GL520_REG_IN2_MIN GL520_REG_IN2_LIMIT#define GL520_REG_IN2_MAX GL520_REG_IN2_LIMIT#define GL520_REG_IN3_INPUT 0x0d#define GL520_REG_IN3_LIMIT 0x0b#define GL520_REG_IN3_MIN GL520_REG_IN3_LIMIT#define GL520_REG_IN3_MAX GL520_REG_IN3_LIMIT#define GL520_REG_IN4_INPUT 0x0e#define GL520_REG_IN4_MAX 0x17#define GL520_REG_IN4_MIN 0x18#define GL520_REG_TEMP1_INPUT 0x04#define GL520_REG_TEMP1_MAX 0x05#define GL520_REG_TEMP1_MAX_HYST 0x06#define GL520_REG_TEMP2_INPUT 0x0e#define GL520_REG_TEMP2_MAX 0x17#define GL520_REG_TEMP2_MAX_HYST 0x18#define GL520_REG_FAN_INPUT 0x07#define GL520_REG_FAN_MIN 0x08#define GL520_REG_FAN_DIV 0x0f#define GL520_REG_FAN_OFF GL520_REG_FAN_DIV#define GL520_REG_ALARMS 0x12#define GL520_REG_BEEP_MASK 0x10#define GL520_REG_BEEP_ENABLE GL520_REG_CONF/* * Function declarations */static int gl520_attach_adapter(struct i2c_adapter *adapter);static int gl520_detect(struct i2c_adapter *adapter, int address, int kind);static void gl520_init_client(struct i2c_client *client);static int gl520_detach_client(struct i2c_client *client);static int gl520_read_value(struct i2c_client *client, u8 reg);static int gl520_write_value(struct i2c_client *client, u8 reg, u16 value);static struct gl520_data *gl520_update_device(struct device *dev);/* Driver data */static struct i2c_driver gl520_driver = { .driver = { .name = "gl520sm", }, .id = I2C_DRIVERID_GL520, .attach_adapter = gl520_attach_adapter, .detach_client = gl520_detach_client,};/* Client data */struct gl520_data { struct i2c_client client; struct device *hwmon_dev; struct mutex update_lock; char valid; /* zero until the following fields are valid */ unsigned long last_updated; /* in jiffies */ u8 vid; u8 vrm; u8 in_input[5]; /* [0] = VVD */ u8 in_min[5]; /* [0] = VDD */ u8 in_max[5]; /* [0] = VDD */ u8 fan_input[2]; u8 fan_min[2]; u8 fan_div[2]; u8 fan_off; u8 temp_input[2]; u8 temp_max[2]; u8 temp_max_hyst[2]; u8 alarms; u8 beep_enable; u8 beep_mask; u8 alarm_mask; u8 two_temps;};/* * Sysfs stuff */#define sysfs_r(type, n, item, reg) \static ssize_t get_##type##item (struct gl520_data *, char *, int); \static ssize_t get_##type##n##item (struct device *, struct device_attribute *attr, char *); \static ssize_t get_##type##n##item (struct device *dev, struct device_attribute *attr, char *buf) \{ \ struct gl520_data *data = gl520_update_device(dev); \ return get_##type##item(data, buf, (n)); \}#define sysfs_w(type, n, item, reg) \static ssize_t set_##type##item (struct i2c_client *, struct gl520_data *, const char *, size_t, int, int); \static ssize_t set_##type##n##item (struct device *, struct device_attribute *attr, const char *, size_t); \static ssize_t set_##type##n##item (struct device *dev, struct device_attribute *attr, const char *buf, size_t count) \{ \ struct i2c_client *client = to_i2c_client(dev); \ struct gl520_data *data = i2c_get_clientdata(client); \ return set_##type##item(client, data, buf, count, (n), reg); \}#define sysfs_rw_n(type, n, item, reg) \sysfs_r(type, n, item, reg) \sysfs_w(type, n, item, reg) \static DEVICE_ATTR(type##n##item, S_IRUGO | S_IWUSR, get_##type##n##item, set_##type##n##item);#define sysfs_ro_n(type, n, item, reg) \sysfs_r(type, n, item, reg) \static DEVICE_ATTR(type##n##item, S_IRUGO, get_##type##n##item, NULL);#define sysfs_rw(type, item, reg) \sysfs_r(type, 0, item, reg) \sysfs_w(type, 0, item, reg) \static DEVICE_ATTR(type##item, S_IRUGO | S_IWUSR, get_##type##0##item, set_##type##0##item);#define sysfs_ro(type, item, reg) \sysfs_r(type, 0, item, reg) \static DEVICE_ATTR(type##item, S_IRUGO, get_##type##0##item, NULL);#define sysfs_vid(n) \sysfs_ro_n(cpu, n, _vid, GL520_REG_VID_INPUT)#define sysfs_in(n) \sysfs_ro_n(in, n, _input, GL520_REG_IN##n##INPUT) \sysfs_rw_n(in, n, _min, GL520_REG_IN##n##_MIN) \sysfs_rw_n(in, n, _max, GL520_REG_IN##n##_MAX) \#define sysfs_fan(n) \sysfs_ro_n(fan, n, _input, GL520_REG_FAN_INPUT) \sysfs_rw_n(fan, n, _min, GL520_REG_FAN_MIN) \sysfs_rw_n(fan, n, _div, GL520_REG_FAN_DIV)#define sysfs_fan_off(n) \sysfs_rw_n(fan, n, _off, GL520_REG_FAN_OFF) \#define sysfs_temp(n) \sysfs_ro_n(temp, n, _input, GL520_REG_TEMP##n##_INPUT) \sysfs_rw_n(temp, n, _max, GL520_REG_TEMP##n##_MAX) \sysfs_rw_n(temp, n, _max_hyst, GL520_REG_TEMP##n##_MAX_HYST)#define sysfs_alarms() \sysfs_ro(alarms, , GL520_REG_ALARMS) \sysfs_rw(beep_enable, , GL520_REG_BEEP_ENABLE) \sysfs_rw(beep_mask, , GL520_REG_BEEP_MASK)sysfs_vid(0)sysfs_in(0)sysfs_in(1)sysfs_in(2)sysfs_in(3)sysfs_in(4)sysfs_fan(1)sysfs_fan(2)sysfs_fan_off(1)sysfs_temp(1)sysfs_temp(2)sysfs_alarms()static ssize_t get_cpu_vid(struct gl520_data *data, char *buf, int n){ return sprintf(buf, "%u\n", vid_from_reg(data->vid, data->vrm));}#define VDD_FROM_REG(val) (((val)*95+2)/4)#define VDD_TO_REG(val) (SENSORS_LIMIT((((val)*4+47)/95),0,255))#define IN_FROM_REG(val) ((val)*19)#define IN_TO_REG(val) (SENSORS_LIMIT((((val)+9)/19),0,255))static ssize_t get_in_input(struct gl520_data *data, char *buf, int n){ u8 r = data->in_input[n]; if (n == 0) return sprintf(buf, "%d\n", VDD_FROM_REG(r)); else return sprintf(buf, "%d\n", IN_FROM_REG(r));}static ssize_t get_in_min(struct gl520_data *data, char *buf, int n){ u8 r = data->in_min[n]; if (n == 0) return sprintf(buf, "%d\n", VDD_FROM_REG(r)); else return sprintf(buf, "%d\n", IN_FROM_REG(r));}static ssize_t get_in_max(struct gl520_data *data, char *buf, int n){ u8 r = data->in_max[n]; if (n == 0) return sprintf(buf, "%d\n", VDD_FROM_REG(r)); else return sprintf(buf, "%d\n", IN_FROM_REG(r));}static ssize_t set_in_min(struct i2c_client *client, struct gl520_data *data, const char *buf, size_t count, int n, int reg){ long v = simple_strtol(buf, NULL, 10); u8 r; mutex_lock(&data->update_lock); if (n == 0) r = VDD_TO_REG(v); else r = IN_TO_REG(v); data->in_min[n] = r; if (n < 4) gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0xff) | r); else gl520_write_value(client, reg, r); mutex_unlock(&data->update_lock); return count;}static ssize_t set_in_max(struct i2c_client *client, struct gl520_data *data, const char *buf, size_t count, int n, int reg){ long v = simple_strtol(buf, NULL, 10); u8 r; if (n == 0) r = VDD_TO_REG(v); else r = IN_TO_REG(v); mutex_lock(&data->update_lock); data->in_max[n] = r; if (n < 4) gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0xff00) | (r << 8)); else gl520_write_value(client, reg, r); mutex_unlock(&data->update_lock); return count;}#define DIV_FROM_REG(val) (1 << (val))#define FAN_FROM_REG(val,div) ((val)==0 ? 0 : (480000/((val) << (div))))#define FAN_TO_REG(val,div) ((val)<=0?0:SENSORS_LIMIT((480000 + ((val) << ((div)-1))) / ((val) << (div)), 1, 255));static ssize_t get_fan_input(struct gl520_data *data, char *buf, int n){ return sprintf(buf, "%d\n", FAN_FROM_REG(data->fan_input[n - 1], data->fan_div[n - 1]));}static ssize_t get_fan_min(struct gl520_data *data, char *buf, int n){ return sprintf(buf, "%d\n", FAN_FROM_REG(data->fan_min[n - 1], data->fan_div[n - 1]));}static ssize_t get_fan_div(struct gl520_data *data, char *buf, int n){ return sprintf(buf, "%d\n", DIV_FROM_REG(data->fan_div[n - 1]));}static ssize_t get_fan_off(struct gl520_data *data, char *buf, int n){ return sprintf(buf, "%d\n", data->fan_off);}static ssize_t set_fan_min(struct i2c_client *client, struct gl520_data *data, const char *buf, size_t count, int n, int reg){ unsigned long v = simple_strtoul(buf, NULL, 10); u8 r; mutex_lock(&data->update_lock); r = FAN_TO_REG(v, data->fan_div[n - 1]); data->fan_min[n - 1] = r; if (n == 1) gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0xff00) | (r << 8)); else gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0xff) | r); data->beep_mask = gl520_read_value(client, GL520_REG_BEEP_MASK); if (data->fan_min[n - 1] == 0) data->alarm_mask &= (n == 1) ? ~0x20 : ~0x40; else data->alarm_mask |= (n == 1) ? 0x20 : 0x40; data->beep_mask &= data->alarm_mask; gl520_write_value(client, GL520_REG_BEEP_MASK, data->beep_mask); mutex_unlock(&data->update_lock); return count;}static ssize_t set_fan_div(struct i2c_client *client, struct gl520_data *data, const char *buf, size_t count, int n, int reg){ unsigned long v = simple_strtoul(buf, NULL, 10); u8 r; switch (v) { case 1: r = 0; break; case 2: r = 1; break; case 4: r = 2; break; case 8: r = 3; break; default: dev_err(&client->dev, "fan_div value %ld not supported. Choose one of 1, 2, 4 or 8!\n", v); return -EINVAL; } mutex_lock(&data->update_lock); data->fan_div[n - 1] = r; if (n == 1) gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0xc0) | (r << 6)); else gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0x30) | (r << 4)); mutex_unlock(&data->update_lock); return count;}static ssize_t set_fan_off(struct i2c_client *client, struct gl520_data *data, const char *buf, size_t count, int n, int reg){ u8 r = simple_strtoul(buf, NULL, 10)?1:0; mutex_lock(&data->update_lock); data->fan_off = r; gl520_write_value(client, reg, (gl520_read_value(client, reg) & ~0x0c) | (r << 2)); mutex_unlock(&data->update_lock); return count;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?