在写PWM风扇驱动之前:如何透过FG信号,获取风扇转速?

简介: 本文介绍了如何通过PWM风扇的FG信号线获取风扇转速的方法,包括风扇规格的解读、FG信号的公式推导、软件算法的解析,以及在全志H713平台上实现风扇转速获取的驱动代码示例。

一、前言

  • PWM风扇也是日常SOC智能设备开发中常见的外围小设备,而对于驱动工程师而言,主要工作就是实现风扇的控制驱动,除了设定风扇的速度之外,还要获取风扇的转速信息,而PWM风扇就
    是一个可以通过FG信号线反馈转速信息的设备。
  • 本文即基于全志H713平台,介绍一下FG信号线的用法,以及PWM风扇转速获取的算法。

二、风扇规格

2.1 线序

  • 从风扇规格书中,可得知YELLOW黄色线为FB线
  • 根据规定线序,接好线
    在这里插入图片描述

2.2 FG 信号说明

  • 下图就是风扇规格书中的FG信号的说明,可以看到最下方有四个公式
  • 对于新手而言,最开始看到下面公式会有些疑惑,后续我们可以猜猜(推导)下公式的具体含义
    在这里插入图片描述

三、FG公式推导

  • FG 信号是由风扇,根据其转速反馈的一个方波频率信号,从图中可知,风扇转一圈的周期 T=T1+T2+T3+T4
  • 假设方波的频率为 f, 则 f = 1 / T
  • N = R.P.M ,此处的R.P.M代表的是每分钟的转速,例如风扇每分钟转60圈,则N = R.P.M = 60 rpm,所以N就是转速
  • 下面就是一些公式的推导和验证,通过自己推导,就可以 理解最终 N = FG * 30 的含义
• N=R.P.M  电机铭牌中_R.P.M_就是指电机的额定转速,单位是:转/分。
• T = T1+T2+T3+T4, 可知 T 代表周期
• N / 60秒 = 每秒钟 n 转, 即 HZ (用 f 表示) 
• 60秒 / N = 1 / n = 1 / f   = T 周期
• 如按上升沿次数计算,一个周期里面有 2个上升沿,如1秒钟内有 a 个上升沿,a其实就是FG,我们可以通过下面来验证:
  • 则 f = a / 2 =>  T = 2 / a  ==> T = 2 / FG  ==>  FG = 2 / T = 1 / (2T) 
  • 则 频率 f = 1 / T , 转速 N = fx60秒 =  60秒 x  (1/ T) = 60 / ( 2/ FG) = FGx30 , 即 N = FGx30
  • 综上,可得知FG = 1秒内出现的上升沿的次数,即上升沿中断的次数,
  • 之后可通过 N = FG x 30 得出风扇的R.P.M, 每分钟转速

四、软件算法解析

  • 此软件算法已集成在全志公版代码之中,此处仅贴出必要部分做讲解

4.1 HZ 的概念

  • Jiffy是在中声明的内核时间单位。
  • 为了理解Jiffy,需要引入一个新的常量HZ,它是jiffies在1s内递增的次数,每个增量被称为一个Tick。
  • 在全志内核中 #define HZ 1000 ,它代表的含义是1秒内有1000个Tick

4.2 算法

  • 从FG信号的原理,我们如何进行编码呢?
  • 答案是,我们只需要统计1秒内,或者说单位时间内 FG信号的上升沿次数就可以了,也就是利用中断触发的次数来统计,每次有FG的上升沿到来时,就将 nFgCount + 1
  • 比如,设定1个定时器,这个定时器每秒钟触发1次,每次触发,都读取nFgCount ,将其除以2就是频率f
  • 下面,演示的就是这个算法

//(1)FG信号的中断处理函数,每次给自己+1
static irqreturn_t pulse_handler(int irq, void *dev_id)
{
   
   
    struct pwm_fan_ctx *ctx = dev_id;
        //每次FG引脚的中断触发,让pluses + 1
    atomic_inc(&ctx->pulses);

    return IRQ_HANDLED;
}

//(2)定时器的中断处理函数,每次触发,先获取pulse,用完后又将ctx->pulses归零,方便统计单位时间内次数
static void sample_timer(struct timer_list *t)
{
   
   
    struct pwm_fan_ctx *ctx = from_timer(ctx, t, rpm_timer);
    unsigned int delta = ktime_ms_delta(ktime_get(), ctx->sample_start);
    int pulses;

    if (delta) {
   
                   
        pulses = atomic_read(&ctx->pulses);//1秒内发生的脉冲次数
        atomic_sub(pulses, &ctx->pulses);// 归零

                //此处delta的理论值应该是1000ms,但会有些小小误差
                //可见此处,最终计算每分钟转速的公式: RPM=FGx30
        ctx->rpm = (unsigned int)(pulses * 1000 * 60) /
            (ctx->pulses_per_revolution * delta);

        ctx->sample_start = ktime_get();
    }

    mod_timer(&ctx->rpm_timer, jiffies + HZ);//设置+HZ后超时,继续触发sample_timer函数
}


//(3)最后,来看一下这个风扇设备的Probe过程,其中绑定了中断和中断处理函数,初始化了计时器
static int pwm_fan_probe(struct platform_device *pdev)
{
   
       
        …… 略 ……
    //从DTS中解析中断 FG信号线的中断,interrupts 
    ctx->irq = platform_get_irq_optional(pdev, 0);
    if (ctx->irq == -EPROBE_DEFER)
        return ctx->irq;
        …… 略 ……
        timer_setup(&ctx->rpm_timer, sample_timer, 0);//初始化定时器和定时函数
        …… 略 ……

    of_property_read_u32(dev->of_node, "pulses-per-revolution", &ppr);
    ctx->pulses_per_revolution = ppr;//获取一个周期有几个pulses,PWM风扇一般为2个,具体见规格书
    if (!ctx->pulses_per_revolution) {
   
   
        dev_err(dev, "pulses-per-revolution can't be zero.\n");
        return -EINVAL;
    }

    if (ctx->irq > 0) {
   
   
        ret = devm_request_irq(dev, ctx->irq, pulse_handler, 0,
                       pdev->name, ctx);//绑定FG中断响应函数
        if (ret) {
   
   
            dev_err(dev, "Failed to request interrupt: %d\n", ret);
            return ret;
        }
        ctx->sample_start = ktime_get();//初始化开始时间
        mod_timer(&ctx->rpm_timer, jiffies + HZ);//定时器开始计时,+HZ 后响应定时中断
    }
        …… 略 ……

        //此处,将fan注册为 cooling device, 并提供了查询和设置当前风扇转速等级的函数,供Thermal使用
       ctx->pwm_fan_state = ctx->pwm_fan_max_state;
    if (IS_ENABLED(CONFIG_THERMAL)) {
   
   
        cdev = devm_thermal_of_cooling_device_register(dev,
            dev->of_node, "pwm-fan", ctx, &pwm_fan_cooling_ops);
        if (IS_ERR(cdev)) {
   
   
            ret = PTR_ERR(cdev);
            dev_err(dev,
                "Failed to register pwm-fan as cooling device: %d\n",
                ret);
            return ret;
        }
        ctx->cdev = cdev;
        thermal_cdev_update(cdev);
    }
}

五、篇尾

谢谢关注~

相关文章
|
4月前
|
传感器 C语言 智能硬件
基于单片机的温度控制系统
基于单片机的温度控制系统
90 0
|
4月前
|
传感器 芯片
毕业设计 基于51单片机霍尔电机转速测量温度PWM调速设计
毕业设计 基于51单片机霍尔电机转速测量温度PWM调速设计
|
传感器
Arduino光敏传感器控制LED灯亮度
Arduino光敏传感器控制LED灯亮度
|
1月前
|
编解码 算法
掌握PWM:STM32F103实现PWM控制直流电机小风扇
PWM,即脉冲宽度调制(Pulse Width Modulation),是一种广泛应用于电子和电机控制领域的信号编码方法。PWM的核心思想是通过改变数字信号的脉冲宽度来模拟模拟信号的幅度变化,从而达到控制输出功率的目的。
111 0
|
4月前
串口通信如何控制步进电机转动?
串口通信如何控制步进电机转动?
62 3
|
算法 机器人 芯片
利用单片机PWM信号占空比进行舵机控制
基于单片机的舵机控制方法具有简单、精度高、成本低、体积小的特点,并可根据不同的舵机数量加以灵活应用。
222 0
|
传感器 前端开发 机器人
基于单片机的智能循迹避障小车STC89C52红外对管L298N驱动PWM波控制速度
利用红外对管检测黑线与障碍物,并以STC89C52单片机为控制芯片控制电动小汽车的速度及转向,从而实现自动循迹避障的功能。其中小车驱动由L298N驱动电路完成,速度由单片机输出的PWM波控制。
296 0
|
传感器 机器人 芯片
基于单片机的电机转速测量设计
基于单片机的电机转速测量设计
181 0
基于单片机的电机转速测量设计