启动阶段LCD点亮,并且通过配置bootargs,可以将启动信息在LCD上,打印出来。之后LCD熄灭。
原因分析:
① LCD可用,只是背光没打开
② LCD驱动配置错误
echo 1 > /dev/backlight,可以点亮背光。此时发现/dev目录下什么都没有。
查资料发现dev的需要使用到mdev,再查看自己的rcS,发现其中也调用了mdev。
为了确认,调出之前的启动信息,发现:“/etc/init.d/rcS: line 6: mdev: not found”
进入busybox目录,make menuconfig,发现其中确实没有配置mdev。
make && make install
mkfs.cramfs xxx cfamfs.img-mdev.4.0
烧录,OK,dev下出现了很多注册的设备。
悲剧的是其中没有backlight设备文件。这从一方面证明了背光驱动出了问题。
以下为过程之后的总结,由于过程拖的比较长,所以没有一点一点在评论中贴出:
启动的过程中,LCD变亮,并且如果console配置了tty0的话,可以看到LCD上,会有启动信息打印。然而随即LCD熄灭。此时按住复位键不放,发现LCD变量,其上的LOGO也显示出来:这说明了是背光的问题。
代码中,关于LCD的位置,最明显的在于,mini2440_init函数了。其中:
static void __init mini2440_init(void)
{
struct mini2440_features_t features = { 0 };
int i;
printk(KERN_INFO "MINI2440: Option string mini2440=%s\n",
mini2440_features_str);
/* Parse the feature string */
mini2440_parse_features(&features, mini2440_features_str); 默认的特性为”0tb”,其中’b’代表backlight,打开背光
/* turn LCD on */
s3c_gpio_cfgpin(S3C2410_GPC(0), S3C2410_GPC0_LEND); 配置GPIO,开启LCD
/* Turn the backlight early on */
WARN_ON(gpio_request(S3C2410_GPG(4), "backlight"));
gpio_direction_output(S3C2410_GPG(4), 1); 配置GPB4,也就是屏幕背光
/* remove pullup on optional PWM backlight -- unused on 3.5 and 7"s */ 这段注释看不懂,不过下面的代码,把GPB1,配置成了输入口,
s3c_gpio_setpull(S3C2410_GPB(1), S3C_GPIO_PULL_UP); 而非PWM timer1输出口。
s3c2410_gpio_setpin(S3C2410_GPB(1), 0);
s3c_gpio_cfgpin(S3C2410_GPB(1), S3C2410_GPIO_INPUT);
/* Make sure the D+ pullup pin is output */
WARN_ON(gpio_request(S3C2410_GPC(5), "udc pup"));
gpio_direction_output(S3C2410_GPC(5), 0);
/* mark the key as input, without pullups (there is one on the board) */
for (i = 0; i < ARRAY_SIZE(mini2440_buttons); i++) {
s3c_gpio_setpull(mini2440_buttons[i].gpio, S3C_GPIO_PULL_UP);
s3c_gpio_cfgpin(mini2440_buttons[i].gpio, S3C2410_GPIO_INPUT);
}
if (features.lcd_index != -1) { 初始化LCD
int li;
mini2440_fb_info.displays =
&mini2440_lcd_cfg[features.lcd_index];
printk(KERN_INFO "MINI2440: LCD");
for (li = 0; li < ARRAY_SIZE(mini2440_lcd_cfg); li++)
if (li == features.lcd_index)
printk(" [%d:%dx%d]", li,
mini2440_lcd_cfg[li].width,
mini2440_lcd_cfg[li].height);
else
printk(" %d:%dx%d", li,
mini2440_lcd_cfg[li].width,
mini2440_lcd_cfg[li].height);
printk("\n");
s3c24xx_fb_set_platdata(&mini2440_fb_info);
}
s3c24xx_udc_set_platdata(&mini2440_udc_cfg);
s3c24xx_mci_set_platdata(&mini2440_mmc_cfg);
s3c_nand_set_platdata(&mini2440_nand_info);
s3c_i2c0_set_platdata(NULL);
i2c_register_board_info(0, mini2440_i2c_devs,
ARRAY_SIZE(mini2440_i2c_devs));
platform_add_devices(mini2440_devices, ARRAY_SIZE(mini2440_devices));
if (features.count) /* the optional features */
platform_add_devices(features.optional, features.count);
}
事实证明,LCD的配置是没有问题的,所以将焦点集中在背光之上。从上述代码的注释,以及在网上查询的资料中,都提到了PWM Backlight,于是乎搜寻PWM Backlight的资料;另外,检查配置,发现PWM Backlight配置正常,于是乎从代码入手,查看背光、PWM相关模块的代码。
第一个,对PWM背光的理解:
PWM控制LCD背光,LCD背光源有CCFL和LED。
PWM与LCD的关系并不大,PWM控制LED,只不过这个LED充当背光源的角色罢了。
PWM输出控制LED的亮暗程度原理?
利用改51单片机的定时器PWM输出控制LED灯的亮暗程度,通过一个参数调节其占空比,比如第一次0/10不亮;1/9 很暗;第二次2/8.....,第九次9/1很亮;第十次,10/0完全亮。 小弟不明白的是:LED的亮暗由其电流和电压决定,所以1/9时的电压比9/1的电压电流高,但是这个电压或者电流怎么计算的了? 1/9不是只是代表其高低电平的时间之比吗? 怎么与LED的电压联系起来呢?
把PWM的一个周期分成10块。
你说的1/9的意思是,这10块时间里,只有1块的时间让LED导通,剩下9块的时间里LED截止。假设PWM信号的幅值是5V,那么在导通的1块时间里LED电压的确是5V的,但是在剩下9块时间里,它的电压却是0V。也就是说从一个周期整体看来,LED的平均电压只有5*0.1+0*0.9=0.5V。
PWM信号频率很高的,我们无法通过肉眼来观察到每一个周期LED灯亮灭的变化过程,所以只好通过平均电压这样一种方式来决定这个LED的亮的程度了。
第二个,LCD相关的驱动代码在drivers/video,背光相关的在drivers/video/backlight。观察产生的.o文件,发现被编译了的有backlight.c/lcd.c/platform_lcd.c/pwm_bl.c。
File | Init | description |
Lcd.c | postcore_initcall | 在sysfs下创建lcd目录,另外,还有lcd_device_regist等函数 |
Platform_lcd.c | module_init | 注册LCD driver |
Backlight.c | postcore_initcall | 在sysfs下创建backlight目录,其中还有背光设备注册函数、亮度的调整函数 |
Pwm_bl.c | module_init | 注册pwm backlight驱动 |
查看postcore_initcall(include/linux/init.h)的定义:
#define pure_initcall(fn) __define_initcall("0",fn,0)
#define core_initcall(fn) __define_initcall("1",fn,1)
#define core_initcall_sync(fn) __define_initcall("1s",fn,1s)
#define postcore_initcall(fn) __define_initcall("2",fn,2)
#define postcore_initcall_sync(fn) __define_initcall("2s",fn,2s)
#define arch_initcall(fn) __define_initcall("3",fn,3)
#define arch_initcall_sync(fn) __define_initcall("3s",fn,3s)
#define subsys_initcall(fn) __define_initcall("4",fn,4)
#define subsys_initcall_sync(fn) __define_initcall("4s",fn,4s)
#define fs_initcall(fn) __define_initcall("5",fn,5)
#define fs_initcall_sync(fn) __define_initcall("5s",fn,5s)
#define rootfs_initcall(fn) __define_initcall("rootfs",fn,rootfs)
#define device_initcall(fn) __define_initcall("6",fn,6)
#define device_initcall_sync(fn) __define_initcall("6s",fn,6s)
#define late_initcall(fn) __define_initcall("7",fn,7)
#define late_initcall_sync(fn) __define_initcall("7s",fn,7s)
#define __initcall(fn) device_initcall(fn)
******
#define module_init(x) __initcall(x);
按顺序,postcore_initcall定义的函数,被存放在使用module_init定义的函数之前。
查看内核链接时生成的System.map文件,
__initcall_start
********
__early_initcall_end
********
__initcall_consistent_init1
__initcall_s3c2442_core_init1
__initcall_s3c2440_core_init1
__initcall_s3c24xx_gpiolib_init1
__initcall_s3c24xx_dma_sysclass_init1
__initcall_sysctl_init1
__initcall_ksysfs_init1
__initcall_async_init1
********
__initcall_lcd_class_init2
********
__initcall_pwm_backlight_init6
这些都会在start_kernel->rest_init->kernel_init->do_basic_setup-> do_initcalls,其中按顺序调用__initcall_start, __initcall_end之间的init函数。
综上,并没有发现对backlight的初始化配置部分。
再搜寻PWM配置部分,在arch/arm/plat-samsung/pwm.c中,有arch_initcall(pwm_init);定义的pwm_init函数。其中有对pwm driver的注册,驱动的probe函数中,有对pwm相关寄存器的配置。
返回mini2440_init函数,该函数被定义在mach_desc数据结构中。在start_kernel->setup_arch中,有init_machine = mdesc->init_machine; 在全代码树中搜索”init_machine”发现,就在setu_arch函数上,有一个customize_machine函数,其中调用init_machine函数,而customize_machine是使用arch_initcall(customize_machine);定义的。
现在解释另外一个问题,关于probe函数的调用时机问题,在各模块的init函数中,只是注册sys文件系统或注册驱动数据结构,绝非调用probe函数。跟踪platform_drive_register函数发现,在其中有对autoprobe的判断,如果是autoprobe的话,就会便利platform_bus上已挂载的设备链表,亦即调用probe函数。注册设备时,亦是如此。
由以上分析可知:
① pwm已被配置,backlight没被配置
② Mini2440_init在pwm_bl的init函数之前被调用
③ Backlight被注册在sys目录下,也证实了网上搜集的资料
尝试着,修改mini2440_init中GPB1的配置,将其修改为pwm timer1的输出。
/* remove pullup on optional PWM backlight -- unused on 3.5 and 7"s */
/*s3c_gpio_setpull(S3C2410_GPB(1), S3C_GPIO_PULL_UP);
trace();
s3c2410_gpio_setpin(S3C2410_GPB(1), 0);
trace();
s3c_gpio_cfgpin(S3C2410_GPB(1), S3C2410_GPIO_INPUT);*/
s3c_gpio_cfgpin(S3C2410_GPB(1), S3C2410_GPB1_TOUT1);
编译,烧写,运行
echo 1 > /sys/class/leds/backlight/brightness
LCD被点亮
之前LCD点亮之后又熄灭的原因,是驱动程序的初始化(pwm_bl.c中的pwm_backlight_probe函数)使然。