blob: f060974222019f548fc835f00410a550334e6814 [file] [edit]
// SPDX-License-Identifier: GPL-2.0-or-later
/*
* Hardware monitoring driver for LattePanda Sigma EC.
*
* The LattePanda Sigma is an x86 SBC made by DFRobot with an ITE IT8613E
* Embedded Controller that manages a CPU fan and thermal sensors.
*
* The BIOS declares the ACPI Embedded Controller (PNP0C09) with _STA
* returning 0 and provides only stub ECRD/ECWT methods that return Zero
* for all registers. Since the kernel's ACPI EC subsystem never initializes,
* ec_read() is not available and direct port I/O to the standard ACPI EC
* ports (0x62/0x66) is used instead.
*
* Because ACPI never initializes the EC, there is no concurrent firmware
* access to these ports, and no ACPI Global Lock or namespace mutex is
* required. The hwmon with_info API serializes all sysfs callbacks,
* so no additional driver-level locking is needed.
*
* The EC register map was discovered by dumping all 256 registers,
* identifying those that change in real-time, and validating by physically
* stopping the fan and observing the RPM register drop to zero. The map
* has been verified on BIOS version 5.27; other versions may differ.
*
* Copyright (c) 2026 Mariano Abad <weimaraner@gmail.com>
*/
#include <linux/delay.h>
#include <linux/dmi.h>
#include <linux/hwmon.h>
#include <linux/io.h>
#include <linux/ioport.h>
#include <linux/module.h>
#include <linux/platform_device.h>
#define DRIVER_NAME "lattepanda_sigma_ec"
/* EC I/O ports (standard ACPI EC interface) */
#define EC_DATA_PORT 0x62
#define EC_CMD_PORT 0x66 /* also status port */
/* EC commands */
#define EC_CMD_READ 0x80
/* EC status register bits */
#define EC_STATUS_OBF 0x01 /* Output Buffer Full */
#define EC_STATUS_IBF 0x02 /* Input Buffer Full */
/* EC register offsets for LattePanda Sigma (BIOS 5.27) */
#define EC_REG_FAN_RPM_HI 0x2E
#define EC_REG_FAN_RPM_LO 0x2F
#define EC_REG_TEMP_BOARD 0x60
#define EC_REG_TEMP_CPU 0x70
#define EC_REG_FAN_DUTY 0x93
/*
* EC polling uses udelay() because the EC typically responds within a
* few microseconds. The kernel's own ACPI EC driver (drivers/acpi/ec.c)
* likewise uses udelay() for busy-polling with a per-poll delay of 550us.
*
* usleep_range() was tested but caused EC protocol failures: the EC
* clears its status flags within microseconds, and sleeping for 50-100us
* between polls allowed the flags to transition past the expected state.
*
* The worst-case total busy-wait of 25ms covers EC recovery after errors.
* In practice the EC responds within 10us so the loop exits immediately.
*/
#define EC_TIMEOUT_US 25000
#define EC_POLL_US 1
static bool force;
module_param(force, bool, 0444);
MODULE_PARM_DESC(force,
"Force loading on untested BIOS versions (default: false)");
static struct platform_device *lps_ec_pdev;
static int ec_wait_ibf_clear(void)
{
int i;
for (i = 0; i < EC_TIMEOUT_US; i++) {
if (!(inb(EC_CMD_PORT) & EC_STATUS_IBF))
return 0;
udelay(EC_POLL_US);
}
return -ETIMEDOUT;
}
static int ec_wait_obf_set(void)
{
int i;
for (i = 0; i < EC_TIMEOUT_US; i++) {
if (inb(EC_CMD_PORT) & EC_STATUS_OBF)
return 0;
udelay(EC_POLL_US);
}
return -ETIMEDOUT;
}
static int ec_read_reg(u8 reg, u8 *val)
{
int ret;
ret = ec_wait_ibf_clear();
if (ret)
return ret;
outb(EC_CMD_READ, EC_CMD_PORT);
ret = ec_wait_ibf_clear();
if (ret)
return ret;
outb(reg, EC_DATA_PORT);
ret = ec_wait_obf_set();
if (ret)
return ret;
*val = inb(EC_DATA_PORT);
return 0;
}
/*
* Read a 16-bit big-endian value from two consecutive EC registers.
*
* The EC may update the register pair between reading the high and low
* bytes, which could produce a corrupted value if the high byte rolls
* over (e.g., 0x0100 -> 0x00FF read as 0x01FF). Guard against this by
* re-reading the high byte after reading the low byte. If the high byte
* changed, re-read the low byte to get a consistent pair.
* See also lm90_read16() which uses the same approach.
*/
static int ec_read_reg16(u8 reg_hi, u8 reg_lo, u16 *val)
{
int ret;
u8 oldh, newh, lo;
ret = ec_read_reg(reg_hi, &oldh);
if (ret)
return ret;
ret = ec_read_reg(reg_lo, &lo);
if (ret)
return ret;
ret = ec_read_reg(reg_hi, &newh);
if (ret)
return ret;
if (oldh != newh) {
ret = ec_read_reg(reg_lo, &lo);
if (ret)
return ret;
}
*val = ((u16)newh << 8) | lo;
return 0;
}
static int
lps_ec_read_string(struct device *dev,
enum hwmon_sensor_types type,
u32 attr, int channel,
const char **str)
{
switch (type) {
case hwmon_fan:
*str = "CPU Fan";
return 0;
case hwmon_temp:
*str = channel == 0 ? "Board Temp" : "CPU Temp";
return 0;
default:
return -EOPNOTSUPP;
}
}
static umode_t
lps_ec_is_visible(const void *drvdata,
enum hwmon_sensor_types type,
u32 attr, int channel)
{
switch (type) {
case hwmon_fan:
if (attr == hwmon_fan_input || attr == hwmon_fan_label)
return 0444;
break;
case hwmon_temp:
if (attr == hwmon_temp_input || attr == hwmon_temp_label)
return 0444;
break;
default:
break;
}
return 0;
}
static int
lps_ec_read(struct device *dev,
enum hwmon_sensor_types type,
u32 attr, int channel, long *val)
{
u16 rpm;
u8 v;
int ret;
switch (type) {
case hwmon_fan:
if (attr != hwmon_fan_input)
return -EOPNOTSUPP;
ret = ec_read_reg16(EC_REG_FAN_RPM_HI,
EC_REG_FAN_RPM_LO, &rpm);
if (ret)
return ret;
*val = rpm;
return 0;
case hwmon_temp:
if (attr != hwmon_temp_input)
return -EOPNOTSUPP;
ret = ec_read_reg(channel == 0 ? EC_REG_TEMP_BOARD
: EC_REG_TEMP_CPU,
&v);
if (ret)
return ret;
/* EC reports unsigned 8-bit temperature in degrees Celsius */
*val = (unsigned long)v * 1000;
return 0;
default:
return -EOPNOTSUPP;
}
}
static const struct hwmon_channel_info * const lps_ec_info[] = {
HWMON_CHANNEL_INFO(fan, HWMON_F_INPUT | HWMON_F_LABEL),
HWMON_CHANNEL_INFO(temp,
HWMON_T_INPUT | HWMON_T_LABEL,
HWMON_T_INPUT | HWMON_T_LABEL),
NULL
};
static const struct hwmon_ops lps_ec_ops = {
.is_visible = lps_ec_is_visible,
.read = lps_ec_read,
.read_string = lps_ec_read_string,
};
static const struct hwmon_chip_info lps_ec_chip_info = {
.ops = &lps_ec_ops,
.info = lps_ec_info,
};
static int lps_ec_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct device *hwmon;
u8 test;
int ret;
if (!devm_request_region(dev, EC_DATA_PORT, 1, DRIVER_NAME))
return dev_err_probe(dev, -EBUSY,
"Failed to request EC data port 0x%x\n",
EC_DATA_PORT);
if (!devm_request_region(dev, EC_CMD_PORT, 1, DRIVER_NAME))
return dev_err_probe(dev, -EBUSY,
"Failed to request EC cmd port 0x%x\n",
EC_CMD_PORT);
/* Sanity check: verify EC is responsive */
ret = ec_read_reg(EC_REG_FAN_DUTY, &test);
if (ret)
return dev_err_probe(dev, ret,
"EC not responding on ports 0x%x/0x%x\n",
EC_DATA_PORT, EC_CMD_PORT);
hwmon = devm_hwmon_device_register_with_info(dev, DRIVER_NAME, NULL,
&lps_ec_chip_info, NULL);
if (IS_ERR(hwmon))
return dev_err_probe(dev, PTR_ERR(hwmon),
"Failed to register hwmon device\n");
dev_info(dev, "EC hwmon registered (fan duty: %u%%)\n", test);
return 0;
}
/* DMI table with strict BIOS version match (override with force=1) */
static const struct dmi_system_id lps_ec_dmi_table[] = {
{
.ident = "LattePanda Sigma",
.matches = {
DMI_MATCH(DMI_SYS_VENDOR, "LattePanda"),
DMI_MATCH(DMI_PRODUCT_NAME, "LattePanda Sigma"),
DMI_MATCH(DMI_BIOS_VERSION, "5.27"),
},
},
{ } /* terminator */
};
MODULE_DEVICE_TABLE(dmi, lps_ec_dmi_table);
/* Loose table (vendor + product only) for use with force=1 */
static const struct dmi_system_id lps_ec_dmi_table_force[] = {
{
.ident = "LattePanda Sigma",
.matches = {
DMI_MATCH(DMI_SYS_VENDOR, "LattePanda"),
DMI_MATCH(DMI_PRODUCT_NAME, "LattePanda Sigma"),
},
},
{ } /* terminator */
};
static struct platform_driver lps_ec_driver = {
.probe = lps_ec_probe,
.driver = {
.name = DRIVER_NAME,
},
};
static int __init lps_ec_init(void)
{
int ret;
if (!dmi_check_system(lps_ec_dmi_table)) {
if (!force || !dmi_check_system(lps_ec_dmi_table_force))
return -ENODEV;
pr_warn("%s: BIOS version not verified, loading due to force=1\n",
DRIVER_NAME);
}
ret = platform_driver_register(&lps_ec_driver);
if (ret)
return ret;
lps_ec_pdev = platform_device_register_simple(DRIVER_NAME, -1,
NULL, 0);
if (IS_ERR(lps_ec_pdev)) {
platform_driver_unregister(&lps_ec_driver);
return PTR_ERR(lps_ec_pdev);
}
return 0;
}
static void __exit lps_ec_exit(void)
{
platform_device_unregister(lps_ec_pdev);
platform_driver_unregister(&lps_ec_driver);
}
module_init(lps_ec_init);
module_exit(lps_ec_exit);
MODULE_AUTHOR("Mariano Abad <weimaraner@gmail.com>");
MODULE_DESCRIPTION("Hardware monitoring driver for LattePanda Sigma EC");
MODULE_LICENSE("GPL");