I.MX6 U-boot PWM hacking

简介: /******************************************************************************* * I.MX6 U-boot PWM hacking * 说明: * 本文主要记录I.MX6 U-boot是如何配置PWM,发现文中的pwm0对应的引脚和真正的 * 电路板的pwm0不是一个,也就意味着其实是引脚连接错了,另外之前一直有一个 * 疑问:为什么双屏显示的时候,有一个屏在U-boot阶段和Kernel阶段总是不亮。
/*******************************************************************************
 *                          I.MX6 U-boot PWM hacking
 * 说明:
 *     本文主要记录I.MX6 U-boot是如何配置PWM,发现文中的pwm0对应的引脚和真正的
 * 电路板的pwm0不是一个,也就意味着其实是引脚连接错了,另外之前一直有一个
 * 疑问:为什么双屏显示的时候,有一个屏在U-boot阶段和Kernel阶段总是不亮。原因
 * 应该是这里只初始化了一个显示控制部分,也就是不会初始两个PWM。
 *
 *                                          2016-3-7 深圳 南山平山村 曾剑锋
 ******************************************************************************/

cat bootable/bootloader/uboot-imx/board/freescale/mx6q_sabresd/mx6q_sabresd.c                       
    ......                                                                                          
    #ifdef CONFIG_LCD                                                                               
    void lcd_enable(void)                                                                           
    {                                                                                               
        ......                                                                                      
        imx_pwm_config(pwm0, 25000, 50000);                     ------------+                       
        imx_pwm_enable(pwm0);                                   ------------*-+                     
                                                                            | |                     
        #if defined CONFIG_MX6Q                                             | |                     
        /* PWM backlight */                                                 | |                     
        mxc_iomux_v3_setup_pad(MX6Q_PAD_SD1_DAT3__PWM1_PWMO);               | |                     
        /* LVDS panel CABC_EN0 */                                           | |                     
        mxc_iomux_v3_setup_pad(MX6Q_PAD_NANDF_CS2__GPIO_6_15);              | |                     
        /* LVDS panel CABC_EN1 */                                           | |                     
        mxc_iomux_v3_setup_pad(MX6Q_PAD_NANDF_CS3__GPIO_6_16);              | |                     
        #elif defined CONFIG_MX6DL                                          | |                     
        /* PWM backlight */                                                 | |                     
        mxc_iomux_v3_setup_pad(MX6DL_PAD_SD1_DAT3__PWM1_PWMO);              | |                     
        /* LVDS panel CABC_EN0 */                                           | |                     
        mxc_iomux_v3_setup_pad(MX6DL_PAD_NANDF_CS2__GPIO_6_15);             | |                     
        /* LVDS panel CABC_EN1 */                                           | |                     
        mxc_iomux_v3_setup_pad(MX6DL_PAD_NANDF_CS3__GPIO_6_16);             | |                     
        #endif                                                              | |                     
        ......                                                              | |                     
    }                                                                       | |                     
    ......                                                                  | |                     
                                                                            | |                     
static struct pwm_device pwm0 = {                               <-----------+ |                     
    .pwm_id = 0,                                                            | |                     
    .pwmo_invert = 0,                                                       | |                     
};                                                                          | |                     
                                                                            | |                     
                                                                            | |                     
int imx_pwm_config(struct pwm_device pwm, int duty_ns, int period_ns)  <----+ |                     
{                                                                             |                     
    unsigned long long c;                                                     |                     
    unsigned long period_cycles, duty_cycles, prescale;                       |                     
    u32 cr;                                                                   |                     
                                                                              |                     
    if (period_ns == 0 || duty_ns > period_ns)                                |                     
        return -1;                                                            |                     
                                                                              |                     
    pwm.mmio_base = pwm.pwm_id ? (unsigned long)IMX_PWM2_BASE:                |                     
                (unsigned long)IMX_PWM1_BASE;                                 |                     
                                                                              |                     
    if (pwm.pwmo_invert)                                                      |                     
        duty_ns = period_ns - duty_ns;                                        |                     
                                                                              |                     
    c = mxc_get_clock(MXC_IPG_PERCLK);                                        |                     
    c = c * period_ns;                                                        |                     
    do_div(c, 1000000000);                                                    |                     
    period_cycles = c;                                                        |                     
                                                                              |                     
    prescale = period_cycles / 0x10000 + 1;                                   |                     
                                                                              |                     
    period_cycles /= prescale;                                                |                     
    c = (unsigned long long)period_cycles * duty_ns;                          |                     
    do_div(c, period_ns);                                                     |                     
    duty_cycles = c;                                                          |                     
                                                                              |                     
    writel(duty_cycles, pwm.mmio_base + MX_PWMSAR);                           |                     
    writel(period_cycles, pwm.mmio_base + MX_PWMPR);                          |                     
                                                                              |                     
    cr = MX_PWMCR_PRESCALER(prescale) |                                       |                     
        MX_PWMCR_STOPEN | MX_PWMCR_DOZEEN |                                   |                     
        MX_PWMCR_WAITEN | MX_PWMCR_DBGEN;                                     |                     
                                                                              |                     
    cr |= MX_PWMCR_CLKSRC_IPG_HIGH;                                           |                     
                                                                              |                     
    writel(cr, pwm.mmio_base + MX_PWMCR);                                     |                     
                                                                              |                     
    return 0;                                                                 |                     
}                                                                             |                     
                                                                              |                     
int imx_pwm_enable(struct pwm_device pwm)                    <----------------+                     
{                                                                                                   
    unsigned long reg;                                                                              
    int rc = 0;                                                                                     
                                                                                                    
    if (pwm.enable_pwm_clk)                                                                         
        pwm.enable_pwm_clk();                                                                       
                                                                                                    
    pwm.mmio_base = pwm.pwm_id ? (unsigned long)IMX_PWM2_BASE:                                      
                (unsigned long)IMX_PWM1_BASE;                                                       
                                                                                                    
    reg = readl(pwm.mmio_base + MX_PWMCR);                                                          
    reg |= MX_PWMCR_EN;                                                                             
    writel(reg, pwm.mmio_base + MX_PWMCR);                                                          
                                                                                                    
    if (pwm.enable_pwm_pad)                                                                         
        pwm.enable_pwm_pad();                                                                       
                                                                                                    
    return rc;                                                                                      
}                                                                                                   

 

目录
打赏
0
0
0
0
12
分享
相关文章
I.MX6 U-boot GPIO hacking
/******************************************************************************* * I.MX6 U-boot GPIO hacking * 说明: * 本文主要记录I.MX6 U-boot是如何设置GPIO口输入输出的,主要是考虑到这个阶段 * 并没有像Linux内核中的gpio_request一系列函数使用。
1100 0
I.MX6 U-Boot mkconfig hacking
/**************************************************************************** * I.MX6 U-Boot mkconfig hacking * 声明: * 本文主要是为了知道U-Boot中的mkconfig是如何工作的,是如何将配置 * 中的内容进行汇总,同时又将这些内容放在了什么地方。
1105 0
I.MX6 u-boot 2009 lvds hdmi lcd 补丁
/************************************************************************* * I.MX6 u-boot 2009 lvds hdmi lcd 补丁 * 说明: * 最近一直在弄这个Parallel LCD的显示问题,找到了了这个补丁,不过打补丁 * 的时候会遇到一些问题。
1081 0
I.MX6 U-boot lvds display hacking
/*********************************************************************************** * I.MX6 U-boot lvds display hacking * 声明: * 本文主要是为了跟踪I.MX6中的U-boot中显示部分代码,查看是否支持24bit显示。
914 0
I.MX6 AT24Cxx eeprom Linux i2c-gpio
/************************************************************************** * I.
1384 0
【STM32 .Net MF开发板学习-06】蜂鸣器和LED数码管显示
无论是蜂鸣器还是LED数码管显示,其实这二者对代码编写来说没有太大区别,都是GPIO的一个典型应用。红牛开发板有一个蜂鸣器,而EM-STM3210E有一个四位LED数码管,代码都相对简单,不值的为二者单独写一篇博文,所以二者合一以一篇文章来说明,不过两个示例代码是独立的。
685 0
I.MX6 天嵌 E9 U-boot menu hacking
/************************************************************************************ * I.MX6 天嵌 E9 U-boot menu hacking * 说明: * 天嵌在U-boot中添加了自己的选择menu,想看一下怎么实现的。
966 0
【STM32 .Net MF开发板学习-25】LED数码管显示
在单片机系统中,LED数码管的应用是很常见的,不过对我们偏软件开发的人来说,要想研究LED还得储备一定的知识才能进行近一步的工作。不过采用.NET Micro Framework系统,我们马上就可以站在软件人员的角度来研究LED数码管显示0~F数字的原理了。
571 0
OK335xS U-boot GPIO control hacking
/**************************************************************************************** * OK335xS U-boot GPIO control hacking * 声明: * 本文主要是跟踪U-boot中如何设置GPIO口电平。
1050 0
IAR FOR STM8 学习笔记 固件库 GPIO
经过一番挣扎,还是决定使用官方的固件库了。。 从网上下一个STM8S的固件库,记得是FOR IAR的。 找到里面的IAR模板就可以开始用了。     这些都是直接写好的库函数,可以直接调用,但首先得先读懂,先了解些必备知识。
1759 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等