158 lines
4.8 KiB
Diff
158 lines
4.8 KiB
Diff
From dc8c2d594063d74d11f4ba2d2e8b34452f766dfe Mon Sep 17 00:00:00 2001
|
|
From: yangwei1 <yangwei1@eswincomputing.com>
|
|
Date: Fri, 24 May 2024 19:55:33 +0800
|
|
Subject: [PATCH 033/222] fix:modify fan control and read fan speed
|
|
|
|
Changelogs:
|
|
eswin,pwm_inverted enable pwm invert
|
|
---
|
|
arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts | 1 +
|
|
.../dts/eswin/eswin-win2030-die0-soc.dtsi | 2 +-
|
|
drivers/hwmon/eswin-fan-control.c | 42 ++++++++++++++-----
|
|
3 files changed, 33 insertions(+), 12 deletions(-)
|
|
|
|
diff --git a/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts b/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
|
|
index 93fa224c75b7..922db5ee1d4c 100644
|
|
--- a/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
|
|
+++ b/arch/riscv/boot/dts/eswin/eic7700-evb-a2.dts
|
|
@@ -688,6 +688,7 @@ &d0_mbox7 {
|
|
|
|
&fan_control {
|
|
status = "okay";
|
|
+ eswin,pwm_inverted;
|
|
};
|
|
|
|
&d0_i2c0 {
|
|
diff --git a/arch/riscv/boot/dts/eswin/eswin-win2030-die0-soc.dtsi b/arch/riscv/boot/dts/eswin/eswin-win2030-die0-soc.dtsi
|
|
index 430c5410bdda..0371d532d2ec 100644
|
|
--- a/arch/riscv/boot/dts/eswin/eswin-win2030-die0-soc.dtsi
|
|
+++ b/arch/riscv/boot/dts/eswin/eswin-win2030-die0-soc.dtsi
|
|
@@ -1289,7 +1289,7 @@ fan_control: fan_control@50b50000 {
|
|
interrupt-parent = <&plic0>;
|
|
interrupt-names = "fanirq";
|
|
interrupts = <354>;
|
|
- pulses-per-revolution = <1>;
|
|
+ pulses-per-revolution = <2>;
|
|
pwm-minimun-period = <1000>;
|
|
pwms = <&pwm0 0 100000>;
|
|
pinctrl-names = "default";
|
|
diff --git a/drivers/hwmon/eswin-fan-control.c b/drivers/hwmon/eswin-fan-control.c
|
|
index 9c8ab39dee30..f7ca5c29520b 100644
|
|
--- a/drivers/hwmon/eswin-fan-control.c
|
|
+++ b/drivers/hwmon/eswin-fan-control.c
|
|
@@ -62,6 +62,7 @@ struct eswin_fan_control_data {
|
|
u32 ppr;
|
|
/* revolutions per minute */
|
|
u32 rpm;
|
|
+ u8 pwm_inverted;
|
|
};
|
|
|
|
static inline void fan_iowrite(const u32 val, const u32 reg,
|
|
@@ -81,9 +82,14 @@ static ssize_t eswin_fan_pwm_ctl_show(struct device *dev, struct device_attribut
|
|
struct eswin_fan_control_data *ctl = dev_get_drvdata(dev);
|
|
struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
|
|
long temp = 0;
|
|
-
|
|
+ long period = 0;
|
|
if (FAN_PWM_DUTY == attr->index) {
|
|
temp = pwm_get_duty_cycle(ctl->pwm);
|
|
+ if(1 == ctl->pwm_inverted)
|
|
+ {
|
|
+ period = pwm_get_period(ctl->pwm);
|
|
+ temp = period- temp;
|
|
+ }
|
|
}
|
|
else if (FAN_PWM_PERIOD == attr->index) {
|
|
temp = pwm_get_period(ctl->pwm);
|
|
@@ -110,8 +116,14 @@ static ssize_t eswin_fan_pwm_ctl_store(struct device *dev, struct device_attribu
|
|
ret = kstrtoul(buf, 10, &val);
|
|
if (ret)
|
|
return ret;
|
|
-
|
|
- state.duty_cycle = val;
|
|
+ if(1 == ctl->pwm_inverted)
|
|
+ {
|
|
+ state.duty_cycle = state.period - val;
|
|
+ }
|
|
+ else
|
|
+ {
|
|
+ state.duty_cycle = val;
|
|
+ }
|
|
}
|
|
else if (FAN_PWM_PERIOD == attr->index) {
|
|
long val = 0;
|
|
@@ -168,9 +180,7 @@ static long eswin_fan_control_get_fan_rpm(struct eswin_fan_control_data *ctl)
|
|
|
|
ctl->wait_flag = false;
|
|
period = pwm_get_period(ctl->pwm);
|
|
- timeout = TIMEOUT(period);
|
|
- if(!timeout)
|
|
- timeout = TIMEOUT(ctl->min_period);
|
|
+ timeout = msecs_to_jiffies(1500);
|
|
|
|
val = fan_ioread(REG_FAN_INT, ctl);
|
|
val = val | 0x1;
|
|
@@ -185,7 +195,6 @@ static long eswin_fan_control_get_fan_rpm(struct eswin_fan_control_data *ctl)
|
|
/* timeout, set rpm to 0 */
|
|
ctl->rpm = 0;
|
|
}
|
|
-
|
|
if(ctl->rpm)
|
|
ctl->rpm = DIV_ROUND_CLOSEST(60 * ctl->clk_rate, ctl->ppr * ctl->rpm);
|
|
|
|
@@ -215,6 +224,10 @@ static int eswin_fan_control_read_pwm(struct device *dev, u32 attr, long *val)
|
|
switch (attr) {
|
|
case hwmon_pwm_input:
|
|
*val = eswin_fan_control_get_pwm_duty(ctl);
|
|
+ if(1 == ctl->pwm_inverted)
|
|
+ {
|
|
+ *val = 100 - *val;
|
|
+ }
|
|
return 0;
|
|
default:
|
|
return -ENOTSUPP;
|
|
@@ -235,13 +248,21 @@ static int eswin_fan_control_set_pwm_duty(const long val, struct eswin_fan_contr
|
|
static int eswin_fan_control_write_pwm(struct device *dev, u32 attr, long val)
|
|
{
|
|
struct eswin_fan_control_data *ctl = dev_get_drvdata(dev);
|
|
-
|
|
switch (attr) {
|
|
case hwmon_pwm_input:
|
|
- if((val < 0)||(val > 100))
|
|
+ if((val < 10) || (val > 99))
|
|
+ {
|
|
+ dev_err(dev,"pwm range is form 10 to 99\n");
|
|
return -EINVAL;
|
|
+ }
|
|
else
|
|
+ {
|
|
+ if(1 == ctl->pwm_inverted)
|
|
+ {
|
|
+ val = 100 - val;
|
|
+ }
|
|
return eswin_fan_control_set_pwm_duty(val, ctl);
|
|
+ }
|
|
default:
|
|
return -ENOTSUPP;
|
|
}
|
|
@@ -413,7 +434,6 @@ MODULE_DEVICE_TABLE(of, eswin_fan_control_of_match);
|
|
static int eswin_fan_control_probe(struct platform_device *pdev)
|
|
{
|
|
struct eswin_fan_control_data *ctl;
|
|
- struct clk *clk;
|
|
const struct of_device_id *id;
|
|
const char *name = "eswin_fan_control";
|
|
struct pwm_state state;
|
|
@@ -455,7 +475,7 @@ static int eswin_fan_control_probe(struct platform_device *pdev)
|
|
}
|
|
ret = reset_control_reset(ctl->fan_rst);
|
|
WARN_ON(0 != ret);
|
|
-
|
|
+ ctl->pwm_inverted = of_property_read_bool(pdev->dev.of_node, "eswin,pwm_inverted");
|
|
init_waitqueue_head(&ctl->wq);
|
|
|
|
ctl->irq = platform_get_irq(pdev, 0);
|
|
--
|
|
2.47.0
|
|
|