Add support for Phytium fan tacho driver support

Add a driver for fan tachometer and capture counter of Phytium SoCs.

Signed-off-by: wangzhimin <wangzhimin1179@phytium.com.cn>
(cherry picked from commit cf09b9c0f0)
Signed-off-by: Alex Shi <alexsshi@tencent.com>
Signed-off-by: Jianping Liu <frankjpliu@tencent.com>
This commit is contained in:
wangzhimin 2023-06-01 02:37:45 +00:00 committed by Jianping Liu
parent f6e379417f
commit 9ccf02f19f
3 changed files with 387 additions and 0 deletions

View File

@ -381,6 +381,15 @@ config SENSORS_ASPEED
This driver can also be built as a module. If so, the module
will be called aspeed_pwm_tacho.
config SENSORS_PHYTIUM
tristate "Phytium Fan tach and capture counter driver"
help
This driver provides support for Phytium Fan Tacho and capture
counter controllers.
This driver can also be built as a module. If so, the module
will be called tacho-phytium.
config SENSORS_ATXP1
tristate "Attansic ATXP1 VID controller"
depends on I2C

View File

@ -51,6 +51,7 @@ obj-$(CONFIG_SENSORS_ARM_SCPI) += scpi-hwmon.o
obj-$(CONFIG_SENSORS_AS370) += as370-hwmon.o
obj-$(CONFIG_SENSORS_ASC7621) += asc7621.o
obj-$(CONFIG_SENSORS_ASPEED) += aspeed-pwm-tacho.o
obj-$(CONFIG_SENSORS_PHYTIUM) += tacho-phytium.o
obj-$(CONFIG_SENSORS_ATXP1) += atxp1.o
obj-$(CONFIG_SENSORS_CORETEMP) += coretemp.o
obj-$(CONFIG_SENSORS_DA9052_ADC)+= da9052-hwmon.o

View File

@ -0,0 +1,377 @@
// SPDX-License-Identifier: GPL-2.0
/*
* Hwmon driver for Phytium tachometer.
*
* Copyright (C) 2021-2023, Phytium Technology Co., Ltd.
*/
#include <linux/module.h>
#include <linux/init.h>
#include <linux/slab.h>
#include <linux/clk.h>
#include <linux/delay.h>
#include <linux/of.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <linux/platform_device.h>
#include <linux/err.h>
#include <linux/mutex.h>
#include <linux/hwmon.h>
#include <linux/hwmon-sysfs.h>
#define TIMER_CTRL_REG 0x00
#define TIMER_CTRL_MODE_SHIFT 0//0:1
#define TIMER_CTRL_RESET_SHIFT BIT(2)
#define TIMER_CTRL_FORCE_SHIFT BIT(3)
#define TIMER_CTRL_CAPTURE_EN_SHIFT BIT(4)
#define TIMER_CTRL_CAPTURE_CNT_SHIFT 5//5:11
#define TIMER_CTRL_ANTI_JITTER_SHIFT 18//18:19
#define TIMER_CTRL_TACHO_MODE_SHIFT 20//20:21
#define TIMER_CTRL_TIMER_CNT_MODE_SHIFT BIT(22)
#define TIMER_CTRL_BIT_SET_SHIFT 24
#define TIMER_CTRL_CNT_EN_SHIFT BIT(25)
#define TIMER_CTRL_CNT_CLR_SHIFT BIT(26)
#define TIMER_CTRL_TIMER_MODE_SHIFT BIT(27)
#define TIMER_CTRL_PULSE_NUM_SHIFT 28//28:30
#define TIMER_CTRL_TACHO_EN_SHIFT BIT(31)
#define TIMER_TACHO_RES_REG 0x04
#define TIMER_TACHO_RES_VALID_SHIFT BIT(31)
#define TIMER_TACHO_RES_MASK GENMASK(30, 0)
#define TIMER_CMP_VALUE_UP_REG 0x08
#define TIMER_CMP_VALUE_LOW_REG 0x1C
#define TIMER_CNT_VALUE_UP_REG 0x20
#define TIMER_CNT_VALUE_LOW_REG 0x24
#define TIMER_INT_MASK_REG 0x28
#define TIMER_INT_STAT_REG 0x2C
#define TIMER_INT_CAPTURE_SHIFT BIT(5)
#define TIMER_INT_CYC_COMP_SHIFT BIT(4)
#define TIMER_INT_ONE_COMP_SHIFT BIT(3)
#define TIMER_INT_ROLLOVER_SHIFT BIT(2)
#define TIMER_INT_TACHO_UNDER_SHIFT BIT(1)
#define TIMER_INT_TACHO_OVER_SHIFT BIT(0)
#define TIMER_TACHO_OVER_REG 0x30
#define TIMER_TACHO_UNDER_REG 0x34
#define TIMER_START_VALUE_REG 0x38
#define TIMER_INT_CLR_MASK GENMASK(5, 0)
enum tacho_modes {
tacho_mode = 1,
capture_mode,
};
enum edge_modes {
rising_edge,
falling_edge,
double_edge,
};
struct phytium_tacho {
struct device *dev;
struct device *hwmon;
void __iomem *base;
struct clk *clk;
u32 freq;
int irq;
u8 work_mode;
u8 edge_mode;
u32 debounce;
};
static u16 capture_count;
static void phytium_tacho_init(struct phytium_tacho *tacho)
{
u32 val;
if (tacho->work_mode == tacho_mode) {
val = (TIMER_CTRL_TACHO_EN_SHIFT |
TIMER_CTRL_CNT_EN_SHIFT |
(tacho->edge_mode << TIMER_CTRL_TACHO_MODE_SHIFT) |
(tacho->debounce << TIMER_CTRL_ANTI_JITTER_SHIFT) |
(tacho->work_mode << TIMER_CTRL_MODE_SHIFT));
writel_relaxed(val, tacho->base + TIMER_CTRL_REG);
writel_relaxed(0x2faf07f, tacho->base + TIMER_CMP_VALUE_LOW_REG);
} else {
val = (TIMER_CTRL_TACHO_EN_SHIFT |
TIMER_CTRL_CNT_EN_SHIFT |
(tacho->edge_mode << TIMER_CTRL_TACHO_MODE_SHIFT) |
(tacho->debounce << TIMER_CTRL_ANTI_JITTER_SHIFT) |
TIMER_CTRL_CAPTURE_EN_SHIFT |
(0x7f << TIMER_CTRL_CAPTURE_CNT_SHIFT) |
(tacho->work_mode << TIMER_CTRL_MODE_SHIFT)),
writel_relaxed(val, tacho->base + TIMER_CTRL_REG);
writel_relaxed(0x20, tacho->base + TIMER_INT_MASK_REG);
}
}
static int phytium_get_fan_tach_rpm(struct phytium_tacho *priv)
{
u64 raw_data, tach_div, clk_source;
u8 mode, both;
unsigned long timeout;
unsigned long loopcounter;
timeout = jiffies + msecs_to_jiffies(500);
for (loopcounter = 0;; loopcounter++) {
raw_data = readl_relaxed(priv->base + TIMER_TACHO_RES_REG);
if (raw_data & TIMER_TACHO_RES_VALID_SHIFT)
break;
if (time_after(jiffies, timeout))
return -ETIMEDOUT;
if (loopcounter > 3000)
msleep(20);
else {
udelay(100);
cond_resched();
}
}
raw_data = raw_data & TIMER_TACHO_RES_MASK;
clk_source = priv->freq;
mode = priv->edge_mode;
both = (mode == double_edge) ? 1 : 0;
tach_div = 1 << both;
if (raw_data == 0)
return 0;
return (clk_source * 60 * raw_data) / 0x2faf080 / tach_div;
}
static ssize_t show_rpm(struct device *dev, struct device_attribute *attr,
char *buf)
{
int rpm;
struct phytium_tacho *priv = dev_get_drvdata(dev);
rpm = phytium_get_fan_tach_rpm(priv);
if (rpm < 0)
return rpm;
return sprintf(buf, "%d\n", rpm);
}
static SENSOR_DEVICE_ATTR(fan_input, 0444,
show_rpm, NULL, 0);
static struct attribute *tacho_dev_attrs[] = {
&sensor_dev_attr_fan_input.dev_attr.attr,
NULL
};
static umode_t tacho_dev_is_visible(struct kobject *kobj,
struct attribute *a, int index)
{
return a->mode;
}
static const struct attribute_group tacho_group = {
.attrs = tacho_dev_attrs,
.is_visible = tacho_dev_is_visible,
};
static const struct attribute_group *tacho_groups[] = {
&tacho_group,
NULL
};
static irqreturn_t capture_irq_handler(int irq, void *dev_id)
{
struct phytium_tacho *priv = dev_id;
u32 status = readl_relaxed(priv->base + TIMER_INT_STAT_REG);
if (status & TIMER_INT_CAPTURE_SHIFT) {
capture_count++;
if (capture_count == 0)
dev_err(priv->dev, "Capture counter is overflowed");
writel_relaxed(status, priv->base + TIMER_INT_STAT_REG);
return IRQ_HANDLED;
}
return IRQ_NONE;
}
static ssize_t show_capture(struct device *dev, struct device_attribute *attr,
char *buf)
{
int cnt;
struct phytium_tacho *priv = dev_get_drvdata(dev);
cnt = capture_count * 0x7f + readl_relaxed(priv->base + TIMER_CNT_VALUE_LOW_REG);
return sprintf(buf, "%d\n", cnt);
}
static SENSOR_DEVICE_ATTR(capture_input, 0444,
show_capture, NULL, 0);
static struct attribute *capture_dev_attrs[] = {
&sensor_dev_attr_capture_input.dev_attr.attr,
NULL
};
static umode_t capture_dev_is_visible(struct kobject *kobj,
struct attribute *a, int index)
{
return a->mode;
}
static const struct attribute_group capture_group = {
.attrs = capture_dev_attrs,
.is_visible = capture_dev_is_visible,
};
static const struct attribute_group *capture_groups[] = {
&capture_group,
NULL
};
static int phytium_tacho_get_work_mode(struct phytium_tacho *tacho)
{
struct device_node *nc = tacho->dev->of_node;
if (of_property_read_bool(nc, "tacho"))
return tacho_mode;
if (of_property_read_bool(nc, "capture"))
return capture_mode;
return tacho_mode;
}
static int phytium_tacho_get_edge_mode(struct phytium_tacho *tacho)
{
struct device_node *nc = tacho->dev->of_node;
if (of_property_read_bool(nc, "up"))
return rising_edge;
if (of_property_read_bool(nc, "down"))
return falling_edge;
if (of_property_read_bool(nc, "double"))
return double_edge;
return rising_edge;
}
static int phytium_tacho_get_debounce(struct phytium_tacho *tacho)
{
u32 value;
struct device_node *nc = tacho->dev->of_node;
if (!of_property_read_u32(nc, "debounce-level", &value))
return value;
else
return 0;
}
static void phytium_tacho_get_of_data(struct phytium_tacho *tacho)
{
tacho->work_mode = phytium_tacho_get_work_mode(tacho);
tacho->edge_mode = phytium_tacho_get_edge_mode(tacho);
tacho->debounce = phytium_tacho_get_debounce(tacho);
}
static int phytium_tacho_probe(struct platform_device *pdev)
{
struct device *dev = &pdev->dev;
struct resource *res;
struct phytium_tacho *tacho;
int ret;
tacho = devm_kzalloc(dev, sizeof(*tacho), GFP_KERNEL);
if (!tacho)
return -ENOMEM;
tacho->dev = &pdev->dev;
res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
if (!res)
return -ENOENT;
tacho->base = devm_ioremap_resource(&pdev->dev, res);
if (IS_ERR(tacho->base)) {
dev_err(&pdev->dev, "region map failed\n");
return PTR_ERR(tacho->base);
}
tacho->clk = devm_clk_get(&pdev->dev, NULL);
if (IS_ERR(tacho->clk))
return PTR_ERR(tacho->clk);
ret = clk_prepare_enable(tacho->clk);
if (ret)
return ret;
tacho->freq = clk_get_rate(tacho->clk);
tacho->irq = platform_get_irq(pdev, 0);
if (tacho->irq < 0) {
dev_err(&pdev->dev, "no irq resource?\n");
return tacho->irq;
}
ret = devm_request_irq(dev, tacho->irq, capture_irq_handler,
0, "phytium_tacho", tacho);
if (ret) {
dev_err(&pdev->dev, "Cannot request IRQ\n");
return ret;
}
phytium_tacho_get_of_data(tacho);
phytium_tacho_init(tacho);
if (tacho->work_mode == tacho_mode)
tacho->hwmon = devm_hwmon_device_register_with_groups(dev,
"phytium_tacho",
tacho, tacho_groups);
else
tacho->hwmon = devm_hwmon_device_register_with_groups(dev,
"phytium_capture",
tacho, capture_groups);
platform_set_drvdata(pdev, tacho);
return PTR_ERR_OR_ZERO(tacho->hwmon);
}
#ifdef CONFIG_PM_SLEEP
static int phytium_tacho_suspend(struct device *dev)
{
return 0;
}
static int phytium_tacho_resume(struct device *dev)
{
struct phytium_tacho *tacho = dev_get_drvdata(dev);
phytium_tacho_init(tacho);
return 0;
}
#endif
static SIMPLE_DEV_PM_OPS(phytium_tacho_pm, phytium_tacho_suspend, phytium_tacho_resume);
static const struct of_device_id tacho_of_match[] = {
{ .compatible = "phytium,tacho", },
{},
};
MODULE_DEVICE_TABLE(of, tacho_of_match);
static struct platform_driver phytium_tacho_driver = {
.probe = phytium_tacho_probe,
.driver = {
.name = "phytium_tacho",
.pm = &phytium_tacho_pm,
.of_match_table = of_match_ptr(tacho_of_match),
},
};
module_platform_driver(phytium_tacho_driver);
MODULE_AUTHOR("Zhang Yiqun <zhangyiqun@phytium.com.cn>");
MODULE_DESCRIPTION("Phytium tachometer driver");
MODULE_LICENSE("GPL");