2021-07-17 15:39:19 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
/*
|
|
|
|
* Driver for the DapperMime-JTAG USB multitool: base MFD driver
|
|
|
|
*
|
|
|
|
* Copyright (c) 2021 sys64738 and haskal
|
|
|
|
*
|
|
|
|
* Adapted from:
|
|
|
|
* jc42.c Copyright (c) 2010 Ericcson AB
|
|
|
|
* max197.c Copyright (c) 2012 Vivien Didelot, Savoir-faire Linux Inc.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/hwmon.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/kernel.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/platform_device.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
#include <linux/mfd/dmj.h>
|
|
|
|
#else
|
|
|
|
#include "dmj.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define HARDWARE_NAME "DapperMime-JTAG"
|
|
|
|
#define HWMON_NAME "dmj"
|
|
|
|
|
|
|
|
#define DMJ_TEMP_CMD_GET_ADDR 0x00
|
|
|
|
#define DMJ_TEMP_CMD_SET_ADDR 0x01
|
|
|
|
#define DMJ_TEMP_CMD_GET_TEMP 0x02
|
|
|
|
#define DMJ_TEMP_CMD_GET_MIN 0x03
|
|
|
|
#define DMJ_TEMP_CMD_GET_MAX 0x04
|
|
|
|
#define DMJ_TEMP_CMD_GET_CRIT 0x05
|
|
|
|
|
|
|
|
struct dmj_hwmon {
|
|
|
|
struct platform_device *pdev;
|
|
|
|
struct device *hwmon_dev;
|
|
|
|
};
|
|
|
|
|
|
|
|
static umode_t dmj_hwmon_is_visible(const void *data, enum hwmon_sensor_types type,
|
|
|
|
uint32_t attr, int ch)
|
|
|
|
{
|
|
|
|
switch (attr) {
|
2021-07-17 16:26:25 +00:00
|
|
|
case hwmon_temp_type:
|
2021-07-17 15:39:19 +00:00
|
|
|
case hwmon_temp_input:
|
|
|
|
case hwmon_temp_min:
|
|
|
|
case hwmon_temp_max:
|
|
|
|
case hwmon_temp_crit:
|
|
|
|
return 0444;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static int dmj_hwmon_read(struct device *dev, enum hwmon_sensor_types type,
|
|
|
|
uint32_t attr, int ch, long *val)
|
|
|
|
{
|
|
|
|
struct dmj_hwmon *dmjh = dev_get_drvdata(dev);
|
|
|
|
uint8_t subcmd, *rbuf;
|
|
|
|
uint16_t rval;
|
|
|
|
int ret, rlen;
|
|
|
|
|
|
|
|
switch (attr) {
|
2021-07-17 15:52:35 +00:00
|
|
|
case hwmon_temp_type:
|
|
|
|
*val = 1;
|
|
|
|
return 0;
|
2021-07-17 15:39:19 +00:00
|
|
|
case hwmon_temp_input:
|
2021-07-17 15:52:35 +00:00
|
|
|
subcmd = DMJ_TEMP_CMD_GET_TEMP;
|
2021-07-17 15:39:19 +00:00
|
|
|
break;
|
|
|
|
case hwmon_temp_min:
|
|
|
|
subcmd = DMJ_TEMP_CMD_GET_MIN;
|
|
|
|
break;
|
|
|
|
case hwmon_temp_max:
|
|
|
|
subcmd = DMJ_TEMP_CMD_GET_MAX;
|
|
|
|
break;
|
|
|
|
case hwmon_temp_crit:
|
|
|
|
subcmd = DMJ_TEMP_CMD_GET_CRIT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
return -ENOTSUPP;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = dmj_transfer(dmjh->pdev, DMJ_CMD_MODE1_TEMPSENSOR,
|
|
|
|
DMJ_XFER_FLAGS_PARSE_RESP, &subcmd, sizeof(subcmd),
|
|
|
|
(void**)&rbuf, &rlen);
|
|
|
|
ret = dmj_check_retval(ret, rlen, dev, "hwmon read", true, 2, 2);
|
|
|
|
if (!ret) {
|
|
|
|
/* rval is 8.4 fixed point, bit 0x1000 is the sign bit, 0xe000 are flags */
|
|
|
|
rval = (uint16_t)rbuf[0] | ((uint16_t)rbuf[1] << 8);
|
|
|
|
rval &= 0x1fff; /* only data and sign bits, no flag bits */
|
|
|
|
rval |= ((rval & 0x1000) << 1) | ((rval & 0x1000) << 2)
|
|
|
|
| ((rval & 0x1000) << 3); /* sign-extend */
|
|
|
|
*val = ((long)(int16_t)rval * 125) / 2; /* 8.4 fixed -> .001 */
|
|
|
|
}
|
|
|
|
if (rbuf) kfree(rbuf);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct hwmon_channel_info *dmj_hwmon_info[] = {
|
2021-07-17 15:52:35 +00:00
|
|
|
HWMON_CHANNEL_INFO(temp, HWMON_T_TYPE | HWMON_T_INPUT | HWMON_T_MIN |
|
|
|
|
HWMON_T_MAX | HWMON_T_CRIT),
|
2021-07-17 15:39:19 +00:00
|
|
|
NULL
|
|
|
|
};
|
|
|
|
static const struct hwmon_ops dmj_hwmon_ops = {
|
|
|
|
.is_visible = dmj_hwmon_is_visible,
|
|
|
|
.read = dmj_hwmon_read,
|
|
|
|
.write = NULL/*dmj_hwmon_write*/
|
|
|
|
};
|
|
|
|
static const struct hwmon_chip_info dmj_chip_info = {
|
|
|
|
.ops = &dmj_hwmon_ops,
|
|
|
|
.info = dmj_hwmon_info
|
|
|
|
};
|
|
|
|
|
2021-07-17 16:26:25 +00:00
|
|
|
static int dmj_hwmon_check_hw(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
uint16_t m1ver;
|
|
|
|
uint8_t curmode, m1feat;
|
|
|
|
const int ver_min = 0x0010, ver_max = 0x0010;
|
|
|
|
int ret = 0, len;
|
|
|
|
uint8_t *buf = NULL;
|
|
|
|
|
|
|
|
ret = dmj_transfer(pdev, DMJ_CMD_CFG_GET_CUR_MODE,
|
|
|
|
DMJ_XFER_FLAGS_PARSE_RESP, NULL, 0, (void**)&buf, &len);
|
|
|
|
ret = dmj_check_retval(ret, len, dev, "hwmon test 1", true, sizeof(curmode), sizeof(curmode));
|
|
|
|
if (ret < 0 || !buf) goto out;
|
|
|
|
|
|
|
|
curmode = buf[0];
|
|
|
|
kfree(buf); buf = NULL;
|
|
|
|
if (curmode != 0x1) {
|
|
|
|
dev_err(dev, "device must be in mode 1 for hwmon to work, but it is in mode %d\n", curmode);
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = dmj_transfer(pdev, (1<<4) | DMJ_CMD_MODE_GET_VERSION,
|
|
|
|
DMJ_XFER_FLAGS_PARSE_RESP, NULL, 0, (void**)&buf, &len);
|
|
|
|
ret = dmj_check_retval(ret, len, dev, "hwmon test 2", true, sizeof(m1ver), sizeof(m1ver));
|
|
|
|
if (ret < 0 || !buf) goto out;
|
|
|
|
|
|
|
|
m1ver = (uint16_t)buf[0] | ((uint16_t)buf[1] << 8);
|
|
|
|
kfree(buf); buf = NULL;
|
|
|
|
if (m1ver > ver_max || m1ver < ver_min) {
|
|
|
|
dev_err(dev, "bad mode 1 version %04x on device, must be between %04x and %04x\n",
|
|
|
|
m1ver, ver_min, ver_max);
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = dmj_transfer(pdev, (1<<4) | DMJ_CMD_MODE_GET_FEATURES,
|
|
|
|
DMJ_XFER_FLAGS_PARSE_RESP, NULL, 0, (void**)&buf, &len);
|
|
|
|
ret = dmj_check_retval(ret, len, dev, "hwmon test 3", true, sizeof(m1feat), sizeof(m1feat));
|
|
|
|
if (ret < 0 || !buf) goto out;
|
|
|
|
m1feat = buf[0];
|
|
|
|
kfree(buf); buf = NULL;
|
|
|
|
if (!(m1feat & DMJ_FEATURE_MODE1_I2C)) {
|
|
|
|
dev_err(dev, "device's mode 1 does not support hwmon tempsensor\n");
|
|
|
|
ret = -EIO;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-07-17 15:39:19 +00:00
|
|
|
static int dmj_hwmon_probe(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct dmj_hwmon *dmjh;
|
|
|
|
struct device *dev = &pdev->dev;
|
|
|
|
int ret;
|
|
|
|
|
2021-07-17 16:26:25 +00:00
|
|
|
ret = dmj_hwmon_check_hw(pdev);
|
|
|
|
if (ret) {
|
|
|
|
dev_err(dev, "hw check failed: %d\n", ret);
|
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2021-07-17 15:39:19 +00:00
|
|
|
dmjh = devm_kzalloc(dev, sizeof(*dmjh), GFP_KERNEL);
|
|
|
|
if (!dmjh) return -ENOMEM;
|
|
|
|
|
|
|
|
dmjh->pdev = pdev;
|
|
|
|
|
|
|
|
platform_set_drvdata(pdev, dmjh);
|
|
|
|
|
|
|
|
dmjh->hwmon_dev = hwmon_device_register_with_info(dev, HWMON_NAME, dmjh,
|
|
|
|
&dmj_chip_info, NULL);
|
|
|
|
if (IS_ERR(dmjh->hwmon_dev)) {
|
|
|
|
ret = PTR_ERR(dmjh->hwmon_dev);
|
|
|
|
dev_err(dev, "hwmon device registration failed\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
static int dmj_hwmon_remove(struct platform_device *pdev)
|
|
|
|
{
|
|
|
|
struct dmj_hwmon *dmjh = platform_get_drvdata(pdev);
|
|
|
|
|
|
|
|
hwmon_device_unregister(dmjh->hwmon_dev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct platform_driver dmj_hwmon_driver = {
|
|
|
|
.driver = {
|
|
|
|
"dmj-hwmon",
|
|
|
|
},
|
|
|
|
.probe = dmj_hwmon_probe,
|
|
|
|
.remove = dmj_hwmon_remove
|
|
|
|
};
|
|
|
|
module_platform_driver(dmj_hwmon_driver);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("sys64738 <sys64738@disroot.org>");
|
|
|
|
MODULE_AUTHOR("haskal <haskal@awoo.systems>");
|
|
|
|
MODULE_DESCRIPTION("Hwmon driver for the " HARDWARE_NAME " USB multitool");
|
|
|
|
MODULE_LICENSE("GPL v2");
|
|
|
|
MODULE_ALIAS("platform:dmj-hwmon");
|
|
|
|
|