// SPDX-License-Identifier: GPL-2.0-or-later /* * abituguru3.c * * Copyright (c) 2006-2008 Hans de Goede <hdegoede@redhat.com> * Copyright (c) 2008 Alistair John Strachan <alistair@devzero.co.uk>
*/ /* * This driver supports the sensor part of revision 3 of the custom Abit uGuru * chip found on newer Abit uGuru motherboards. Note: because of lack of specs * only reading the sensors and their settings is supported.
*/
/* uGuru3 bank addresses */ #define ABIT_UGURU3_SETTINGS_BANK 0x01 #define ABIT_UGURU3_SENSORS_BANK 0x08 #define ABIT_UGURU3_MISC_BANK 0x09 #define ABIT_UGURU3_ALARMS_START 0x1E #define ABIT_UGURU3_SETTINGS_START 0x24 #define ABIT_UGURU3_VALUES_START 0x80 #define ABIT_UGURU3_BOARD_ID 0x0A /* uGuru3 sensor bank flags */ /* Alarm if: */ #define ABIT_UGURU3_TEMP_HIGH_ALARM_ENABLE 0x01 /* temp over warn */ #define ABIT_UGURU3_VOLT_HIGH_ALARM_ENABLE 0x02 /* volt over max */ #define ABIT_UGURU3_VOLT_LOW_ALARM_ENABLE 0x04 /* volt under min */ #define ABIT_UGURU3_TEMP_HIGH_ALARM_FLAG 0x10 /* temp is over warn */ #define ABIT_UGURU3_VOLT_HIGH_ALARM_FLAG 0x20 /* volt is over max */ #define ABIT_UGURU3_VOLT_LOW_ALARM_FLAG 0x40 /* volt is under min */ #define ABIT_UGURU3_FAN_LOW_ALARM_ENABLE 0x01 /* fan under min */ #define ABIT_UGURU3_BEEP_ENABLE 0x08 /* beep if alarm */ #define ABIT_UGURU3_SHUTDOWN_ENABLE 0x80 /* shutdown if alarm */ /* sensor types */ #define ABIT_UGURU3_IN_SENSOR 0 #define ABIT_UGURU3_TEMP_SENSOR 1 #define ABIT_UGURU3_FAN_SENSOR 2
/* * Timeouts / Retries, if these turn out to need a lot of fiddling we could * convert them to params. Determined by trial and error. I assume this is * cpu-speed independent, since the ISA-bus and not the CPU should be the * bottleneck.
*/ #define ABIT_UGURU3_WAIT_TIMEOUT 250 /* * Normally the 0xAC at the end of synchronize() is reported after the * first read, but sometimes not and we need to poll
*/ #define ABIT_UGURU3_SYNCHRONIZE_TIMEOUT 5 /* utility macros */ #define ABIT_UGURU3_NAME "abituguru3" #define ABIT_UGURU3_DEBUG(format, arg...) \ do { \ if (verbose) \
pr_debug(format , ## arg); \
} while (0)
/* * All the macros below are named identical to the openguru2 program * reverse engineered by Louis Kruger, hence the names might not be 100% * logical. I could come up with better names, but I prefer keeping the names * identical so that this driver can be compared with his work more easily.
*/ /* Two i/o-ports are used by uGuru */ #define ABIT_UGURU3_BASE 0x00E0 #define ABIT_UGURU3_CMD 0x00 #define ABIT_UGURU3_DATA 0x04 #define ABIT_UGURU3_REGION_LENGTH 5 /* * The wait_xxx functions return this on success and the last contents * of the DATA register (0-255) on failure.
*/ #define ABIT_UGURU3_SUCCESS -1 /* uGuru status flags */ #define ABIT_UGURU3_STATUS_READY_FOR_READ 0x01 #define ABIT_UGURU3_STATUS_BUSY 0x02
/* Structures */ struct abituguru3_sensor_info { constchar *name; int port; int type; int multiplier; int divisor; int offset;
};
/* Avoid use of flexible array members */ #define ABIT_UGURU3_MAX_DMI_NAMES 2
struct abituguru3_motherboard_info {
u16 id; constchar *dmi_name[ABIT_UGURU3_MAX_DMI_NAMES + 1]; /* + 1 -> end of sensors indicated by a sensor with name == NULL */ struct abituguru3_sensor_info sensors[ABIT_UGURU3_MAX_NO_SENSORS + 1];
};
/* * For the Abit uGuru, we need to keep some data in memory. * The structure is dynamically allocated, at the same time when a new * abituguru3 device is allocated.
*/ struct abituguru3_data { struct device *hwmon_dev; /* hwmon registered device */ struct mutex update_lock; /* protect access to data and uGuru */ unsignedshort addr; /* uguru base address */ bool valid; /* true if following fields are valid */ unsignedlong last_updated; /* In jiffies */
/* * For convenience the sysfs attr and their names are generated * automatically. We have max 10 entries per sensor (for in sensors)
*/ struct sensor_device_attribute_2 sysfs_attr[ABIT_UGURU3_MAX_NO_SENSORS
* 10];
/* Buffer to store the dynamically generated sysfs names */ char sysfs_names[ABIT_UGURU3_SYSFS_NAMES_LENGTH];
/* Pointer to the sensors info for the detected motherboard */ conststruct abituguru3_sensor_info *sensors;
/* * The abituguru3 supports up to 48 sensors, and thus has registers * sets for 48 sensors, for convenience reasons / simplicity of the * code we always read and store all registers for all 48 sensors
*/
/* Alarms for all 48 sensors (1 bit per sensor) */
u8 alarms[48/8];
/* Value of all 48 sensors */
u8 value[48];
/* * Settings of all 48 sensors, note in and temp sensors (the first 32 * sensors) have 3 bytes of settings, while fans only have 2 bytes, * for convenience we use 3 bytes for all sensors
*/
u8 settings[48][3];
};
/* Insmod parameters */ staticbool force;
module_param(force, bool, 0);
MODULE_PARM_DESC(force, "Set to one to force detection."); /* Default verbose is 1, since this driver is still in the testing phase */ staticbool verbose = 1;
module_param(verbose, bool, 0644);
MODULE_PARM_DESC(verbose, "Enable/disable verbose error reporting");
staticconstchar *never_happen = "This should never happen."; staticconstchar *report_this = "Please report this to the abituguru3 maintainer (see MAINTAINERS)";
/* wait while the uguru is busy (usually after a write) */ staticint abituguru3_wait_while_busy(struct abituguru3_data *data)
{
u8 x; int timeout = ABIT_UGURU3_WAIT_TIMEOUT;
while ((x = inb_p(data->addr + ABIT_UGURU3_DATA)) &
ABIT_UGURU3_STATUS_BUSY) {
timeout--; if (timeout == 0) return x; /* * sleep a bit before our last try, to give the uGuru3 one * last chance to respond.
*/ if (timeout == 1)
msleep(1);
} return ABIT_UGURU3_SUCCESS;
}
/* wait till uguru is ready to be read */ staticint abituguru3_wait_for_read(struct abituguru3_data *data)
{
u8 x; int timeout = ABIT_UGURU3_WAIT_TIMEOUT;
while (!((x = inb_p(data->addr + ABIT_UGURU3_DATA)) &
ABIT_UGURU3_STATUS_READY_FOR_READ)) {
timeout--; if (timeout == 0) return x; /* * sleep a bit before our last try, to give the uGuru3 one * last chance to respond.
*/ if (timeout == 1)
msleep(1);
} return ABIT_UGURU3_SUCCESS;
}
/* * This synchronizes us with the uGuru3's protocol state machine, this * must be done before each command.
*/ staticint abituguru3_synchronize(struct abituguru3_data *data)
{ int x, timeout = ABIT_UGURU3_SYNCHRONIZE_TIMEOUT;
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("synchronize timeout during initial busy " "wait, status: 0x%02x\n", x); return -EIO;
}
outb(0x20, data->addr + ABIT_UGURU3_DATA);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("synchronize timeout after sending 0x20, " "status: 0x%02x\n", x); return -EIO;
}
outb(0x10, data->addr + ABIT_UGURU3_CMD);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("synchronize timeout after sending 0x10, " "status: 0x%02x\n", x); return -EIO;
}
outb(0x00, data->addr + ABIT_UGURU3_CMD);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("synchronize timeout after sending 0x00, " "status: 0x%02x\n", x); return -EIO;
}
x = abituguru3_wait_for_read(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("synchronize timeout waiting for read, " "status: 0x%02x\n", x); return -EIO;
}
while ((x = inb(data->addr + ABIT_UGURU3_CMD)) != 0xAC) {
timeout--; if (timeout == 0) {
ABIT_UGURU3_DEBUG("synchronize timeout cmd does not " "hold 0xAC after synchronize, cmd: 0x%02x\n",
x); return -EIO;
}
msleep(1);
} return 0;
}
/* * Read count bytes from sensor sensor_addr in bank bank_addr and store the * result in buf
*/ staticint abituguru3_read(struct abituguru3_data *data, u8 bank, u8 offset,
u8 count, u8 *buf)
{ int i, x;
x = abituguru3_synchronize(data); if (x) return x;
outb(0x1A, data->addr + ABIT_UGURU3_DATA);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("read from 0x%02x:0x%02x timed out after " "sending 0x1A, status: 0x%02x\n", (unsignedint)bank,
(unsignedint)offset, x); return -EIO;
}
outb(bank, data->addr + ABIT_UGURU3_CMD);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("read from 0x%02x:0x%02x timed out after " "sending the bank, status: 0x%02x\n",
(unsignedint)bank, (unsignedint)offset, x); return -EIO;
}
outb(offset, data->addr + ABIT_UGURU3_CMD);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("read from 0x%02x:0x%02x timed out after " "sending the offset, status: 0x%02x\n",
(unsignedint)bank, (unsignedint)offset, x); return -EIO;
}
outb(count, data->addr + ABIT_UGURU3_CMD);
x = abituguru3_wait_while_busy(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("read from 0x%02x:0x%02x timed out after " "sending the count, status: 0x%02x\n",
(unsignedint)bank, (unsignedint)offset, x); return -EIO;
}
for (i = 0; i < count; i++) {
x = abituguru3_wait_for_read(data); if (x != ABIT_UGURU3_SUCCESS) {
ABIT_UGURU3_DEBUG("timeout reading byte %d from " "0x%02x:0x%02x, status: 0x%02x\n", i,
(unsignedint)bank, (unsignedint)offset, x); break;
}
buf[i] = inb(data->addr + ABIT_UGURU3_CMD);
} return i;
}
/* * Sensor settings are stored 1 byte per offset with the bytes * placed add consecutive offsets.
*/ staticint abituguru3_read_increment_offset(struct abituguru3_data *data,
u8 bank, u8 offset, u8 count,
u8 *buf, int offset_count)
{ int i, x;
for (i = 0; i < offset_count; i++) {
x = abituguru3_read(data, bank, offset + i, count,
buf + i * count); if (x != count) { if (x < 0) return x; return i * count + x;
}
}
return i * count;
}
/* * Following are the sysfs callback functions. These functions expect: * sensor_device_attribute_2->index: index into the data->sensors array * sensor_device_attribute_2->nr: register offset, bitmask or NA.
*/ staticstruct abituguru3_data *abituguru3_update_device(struct device *dev);
/* are we reading a setting, or is this a normal read? */ if (attr->nr)
value = data->settings[sensor->port][attr->nr]; else
value = data->value[sensor->port];
/* convert the value */
value = (value * sensor->multiplier) / sensor->divisor +
sensor->offset;
/* * alternatively we could update the sensors settings struct for this, * but then its contents would differ from the windows sw ini files
*/ if (sensor->type == ABIT_UGURU3_TEMP_SENSOR)
value *= 1000;
/* * See if the alarm bit for this sensor is set and if a bitmask is * given in attr->nr also check if the alarm matches the type of alarm * we're looking for (for volt it can be either low or high). The type * is stored in a few readonly bits in the settings of the sensor.
*/ if ((data->alarms[port / 8] & (0x01 << (port % 8))) &&
(!attr->nr || (data->settings[port][0] & attr->nr))) return sprintf(buf, "1\n"); else return sprintf(buf, "0\n");
}
/* Fill the sysfs attr array */
sysfs_attr_i = 0;
sysfs_filename = data->sysfs_names;
sysfs_names_free = ABIT_UGURU3_SYSFS_NAMES_LENGTH; for (i = 0; data->sensors[i].name; i++) { /* Fail safe check, this should never happen! */ if (i >= ABIT_UGURU3_MAX_NO_SENSORS) {
pr_err("Fatal error motherboard has more sensors then ABIT_UGURU3_MAX_NO_SENSORS. %s %s\n",
never_happen, report_this);
res = -ENAMETOOLONG; goto abituguru3_probe_error;
}
type = data->sensors[i].type; for (j = 0; j < no_sysfs_attr[type]; j++) {
used = snprintf(sysfs_filename, sysfs_names_free,
abituguru3_sysfs_templ[type][j].dev_attr.attr.
name, sensor_index[type]) + 1;
data->sysfs_attr[sysfs_attr_i] =
abituguru3_sysfs_templ[type][j];
data->sysfs_attr[sysfs_attr_i].dev_attr.attr.name =
sysfs_filename;
data->sysfs_attr[sysfs_attr_i].index = i;
sysfs_filename += used;
sysfs_names_free -= used;
sysfs_attr_i++;
}
sensor_index[type]++;
} /* Fail safe check, this should never happen! */ if (sysfs_names_free < 0) {
pr_err("Fatal error ran out of space for sysfs attr names. %s %s\n",
never_happen, report_this);
res = -ENAMETOOLONG; goto abituguru3_probe_error;
}
/* Register sysfs hooks */ for (i = 0; i < sysfs_attr_i; i++) if (device_create_file(&pdev->dev,
&data->sysfs_attr[i].dev_attr)) goto abituguru3_probe_error; for (i = 0; i < ARRAY_SIZE(abituguru3_sysfs_attr); i++) if (device_create_file(&pdev->dev,
&abituguru3_sysfs_attr[i].dev_attr)) goto abituguru3_probe_error;
data->hwmon_dev = hwmon_device_register(&pdev->dev); if (IS_ERR(data->hwmon_dev)) {
res = PTR_ERR(data->hwmon_dev); goto abituguru3_probe_error;
}
return 0; /* success */
abituguru3_probe_error: for (i = 0; data->sysfs_attr[i].dev_attr.attr.name; i++)
device_remove_file(&pdev->dev, &data->sysfs_attr[i].dev_attr); for (i = 0; i < ARRAY_SIZE(abituguru3_sysfs_attr); i++)
device_remove_file(&pdev->dev,
&abituguru3_sysfs_attr[i].dev_attr); return res;
}
hwmon_device_unregister(data->hwmon_dev); for (i = 0; data->sysfs_attr[i].dev_attr.attr.name; i++)
device_remove_file(&pdev->dev, &data->sysfs_attr[i].dev_attr); for (i = 0; i < ARRAY_SIZE(abituguru3_sysfs_attr); i++)
device_remove_file(&pdev->dev,
&abituguru3_sysfs_attr[i].dev_attr);
}
mutex_lock(&data->update_lock); if (!data->valid || time_after(jiffies, data->last_updated + HZ)) { /* Clear data->valid while updating */
data->valid = false; /* Read alarms */ if (abituguru3_read_increment_offset(data,
ABIT_UGURU3_SETTINGS_BANK,
ABIT_UGURU3_ALARMS_START,
1, data->alarms, 48/8) != (48/8)) goto LEAVE_UPDATE; /* Read in and temp sensors (3 byte settings / sensor) */ for (i = 0; i < 32; i++) { if (abituguru3_read(data, ABIT_UGURU3_SENSORS_BANK,
ABIT_UGURU3_VALUES_START + i,
1, &data->value[i]) != 1) goto LEAVE_UPDATE; if (abituguru3_read_increment_offset(data,
ABIT_UGURU3_SETTINGS_BANK,
ABIT_UGURU3_SETTINGS_START + i * 3,
1,
data->settings[i], 3) != 3) goto LEAVE_UPDATE;
} /* Read temp sensors (2 byte settings / sensor) */ for (i = 0; i < 16; i++) { if (abituguru3_read(data, ABIT_UGURU3_SENSORS_BANK,
ABIT_UGURU3_VALUES_START + 32 + i,
1, &data->value[32 + i]) != 1) goto LEAVE_UPDATE; if (abituguru3_read_increment_offset(data,
ABIT_UGURU3_SETTINGS_BANK,
ABIT_UGURU3_SETTINGS_START + 32 * 3 +
i * 2, 1,
data->settings[32 + i], 2) != 2) goto LEAVE_UPDATE;
}
data->last_updated = jiffies;
data->valid = true;
}
LEAVE_UPDATE:
mutex_unlock(&data->update_lock); if (data->valid) return data; else return NULL;
}
staticint abituguru3_suspend(struct device *dev)
{ struct abituguru3_data *data = dev_get_drvdata(dev); /* * make sure all communications with the uguru3 are done and no new * ones are started
*/
mutex_lock(&data->update_lock); return 0;
}
board_vendor = dmi_get_system_info(DMI_BOARD_VENDOR); if (!board_vendor || strcmp(board_vendor, "http://www.abit.com.tw/")) return err;
board_name = dmi_get_system_info(DMI_BOARD_NAME); if (!board_name) return err;
/* * At the moment, we don't care about the part of the vendor * DMI string contained in brackets. Truncate the string at * the first occurrence of a bracket. Trim any trailing space * from the substring.
*/
sublen = strcspn(board_name, "("); while (sublen > 0 && board_name[sublen - 1] == ' ')
sublen--;
for (i = 0; abituguru3_motherboards[i].id; i++) {
dmi_name = abituguru3_motherboards[i].dmi_name; for ( ; *dmi_name; dmi_name++) { if (strlen(*dmi_name) != sublen) continue; if (!strncasecmp(board_name, *dmi_name, sublen)) return 0;
}
}
/* No match found */ return 1;
}
/* * FIXME: Manual detection should die eventually; we need to collect stable * DMI model names first before we can rely entirely on CONFIG_DMI.
*/
staticint __init abituguru3_detect(void)
{ /* * See if there is an uguru3 there. An idle uGuru3 will hold 0x00 or * 0x08 at DATA and 0xAC at CMD. Sometimes the uGuru3 will hold 0x05 * or 0x55 at CMD instead, why is unknown.
*/
u8 data_val = inb_p(ABIT_UGURU3_BASE + ABIT_UGURU3_DATA);
u8 cmd_val = inb_p(ABIT_UGURU3_BASE + ABIT_UGURU3_CMD); if (((data_val == 0x00) || (data_val == 0x08)) &&
((cmd_val == 0xAC) || (cmd_val == 0x05) ||
(cmd_val == 0x55))) return 0;
if (force) {
pr_info("Assuming Abit uGuru3 is present because of \"force\" parameter\n"); return 0;
}
/* No uGuru3 found */ return -ENODEV;
}
staticstruct platform_device *abituguru3_pdev;
staticint __init abituguru3_init(void)
{ struct resource res = { .flags = IORESOURCE_IO }; int err;
/* Attempt DMI detection first */
err = abituguru3_dmi_detect(); if (err < 0) return err;
/* * Fall back to manual detection if there was no exact * board name match, or force was specified.
*/ if (err > 0) {
err = abituguru3_detect(); if (err) return err;
pr_warn("this motherboard was not detected using DMI. " "Please send the output of \"dmidecode\" to the abituguru3 maintainer (see MAINTAINERS)\n");
}
err = platform_driver_register(&abituguru3_driver); if (err) gotoexit;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.