在写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);
    }
}

五、篇尾

谢谢关注~

相关文章
|
5月前
|
编解码 算法
掌握PWM:STM32F103实现PWM控制直流电机小风扇
PWM,即脉冲宽度调制(Pulse Width Modulation),是一种广泛应用于电子和电机控制领域的信号编码方法。PWM的核心思想是通过改变数字信号的脉冲宽度来模拟模拟信号的幅度变化,从而达到控制输出功率的目的。
717 0
|
8月前
|
IDE 开发工具
基于单片机的简易步进电机控制系统
基于单片机的简易步进电机控制系统
111 0
|
8月前
|
芯片 容器
CMOS电源稳压器LDO
一、基本概述 TX6213是一款300mA Low Power LDO,输入电压2.5V~6.5V,输出范围1.0V~3.3V,输出电流300mA,PSRR为75dB @1KHz,压差为220mV @ IOUT=200mA。 二、应用场景 MP3/MP4 Players Cellphones, radiophone, digital cameras Bluetooth, wireless handsets Others portable electronics device 三、电气特性 The following specifications apply for VOUT=2
|
8月前
|
芯片
PWM 调光的线性降压 LED 恒流驱动器
一、基本概述 TX6410 是一种带 PWM 调光功能的线性降压 LED 恒流驱动器,仅需外接一个电阻就可以构成一个完整的 LED 恒流驱动电路,调节该外接电阻可调节输出电流,输出电流范围为 10~2000mA。内置 30V 50 毫欧 MOS。内置过热保护功能,可有效保护芯片,避免因过热而造成损坏。具有很低的静态电流,典型值为49uA。带 PWM 调光功能, 可通过在DIM 脚加 PWM 信号调节 LED 电流。芯片采用 ESOP8 封装,内置散热片接LED脚。 二、产品的特点 VDD工作电压:2.5-6V 输出电流:10mA - 2A 内置 30V/50mΩ MOS PWM调光
139 1
|
8月前
|
芯片
带PWM 调光的线性降压 LED 恒流驱动器
一、基本概述 TX6410B是一种带 PWM 调光功能的线性降压 LED 恒流驱动器,仅需外接一个电阻就可以构成一个完整的 LED 恒流驱动电路,调节该外接电阻可调节输出电流,输出电流范围为 10~2000mA。TX6410B内置 30V 50 毫欧 MOS。TX6410B内置过热保护功能,可有效保护芯片,避免因过热而造成损坏。TX6410B具有很低的静态电流,典型值为 60uA。TX6410B带 PWM 调光功能,可通过在 DIM 脚加 PWM 信号调节 LED 电流。TX6410B采用ESOP8 封装。外露散热片接 LED 脚。 二、产品特点 内置 30V 50 毫欧 MOS 输出
220 0
|
传感器 芯片
可编程 USB 转串口适配器开发板与振弦传感器测量模块
当通过 IIC 接口修改 VM5xx 单个寄存器后,被修改的寄存器立即保存(断电不丢失),但连续寄存器的写入仅当时修改生效,模块重启后会自动恢复。为了能够使寄存器永久保存,可以单独向功能寄存器 03 写入指令码 0x000C 来强制保存所有寄存器。
可编程 USB 转串口适配器开发板与振弦传感器测量模块
单片机数码管显示热敏电阻实测温度,
单片机数码管显示热敏电阻实测温度,
115 0
|
测试技术 C语言 芯片
基于51单片机的自动打铃打鸣作息报时系统AT89C51数码管三极管时钟电路
基于51单片机的自动打铃打鸣作息报时系统AT89C51数码管三极管时钟电路
272 0
|
算法 机器人 芯片
利用单片机PWM信号占空比进行舵机控制
基于单片机的舵机控制方法具有简单、精度高、成本低、体积小的特点,并可根据不同的舵机数量加以灵活应用。
262 0
|
传感器 机器人 芯片
基于单片机的电机转速测量设计
基于单片机的电机转速测量设计
208 0
基于单片机的电机转速测量设计

热门文章

最新文章