嵌入式Linux 优化开机启动时间

1、 U-Boot快速启动优化。

修改延时bootdelay为0秒,可缩减bootdelay延时

bash 复制代码
bootdelay=0

2、关闭内核校验

uboot在启动内核前可能会校验内核镜像是否正确,如果内核镜像出错,在uboot阶段做不做校验,系统都会挂死,因此不做校验理论上不会出问题;

bash 复制代码
setenv verify n

2、修改random

由于random设备初始化太慢,应用程序调用random的函数时可能被阻塞,导致应用启动慢,因此可通过关闭crng_ready()功能从而缩短启动时间,关闭crng_ready()功能后启动时间可缩短2~3秒。

cpp 复制代码
drivers/char/random.c
/*
 * crng_init =  0 --> Uninitialized
 *      1 --> Initialized
 *      2 --> Initialized from input_pool
 *
 * crng_init is protected by primary_crng->lock, and only increases
 * its value (from 0->1->2).
 */
static int crng_init = 0; 
#define crng_ready() (likely(crng_init > 0))                                                                                                     
//#define crng_ready() (likely(crng_init > 1))

3、修改LPG

LPJ可以用来缩短每次启动时调用calibrate_delay()来校准loops_per_jiffy消耗的时间。这个时间开销与CPU频率无关,在典型的嵌入式硬件环境下会消耗300ms左右。LPJ值对于固定硬件平台应该是一致的,可以只计算一次,在后续的启动中就可以在启动参数中强制指定LPJ值,而跳过实际的计算过程。具体方法是:在正常启动后记录下内核启动信息中的"Calibrating Delay"数值,在启动参数中以"lpj=xxxxxx"的形式强制指定。

bash 复制代码
[    0.008286] Console: colour dummy device 80x25
[    0.012560] Calibrating delay loop (skipped), value calculated using timer frequency.. 24.00 BogoMIPS (lpj=120000)
[    0.022866] pid_max: default: 32768 minimum: 301
bash 复制代码
setenv bootargs rootwait console=ttyS0,115200 root=/dev/mmcblk1p1 lpj=120000

4、修改输出日志

在嵌入式设备中,调试口基本使用UART口,系统启动过程中,串口为同步输出,十分影响启动时间。例如115200,8N1(1位起始位,8位数据,无校验,1位停止位)的常用串口输出格式,有效数据速率为115200 / 10 = 11520 字节/秒。启动时串口每输出11K的字符,即需要消耗1秒的启动时间。减少启动串口的输出,能大大减少系统的启动时间。

4.1、减少内核日志输出

4.1.1、修改日志输出级别

bash 复制代码
setenv bootargs earlycon=nvt_serial,0x2f0280000 rootwait console=ttyS0,115200 libata.force=1.5 rootfstype=ext4 rw lpj=120000 loglevel=0

4.1.2、添加quiet输出

启动过程默认打开控制台输出启动消息,但是控制台尤其是基于帧缓冲的控制台会减慢启动速度。因此在嵌入式Linux产品中,将启动过程中的控制台设为静默状态,方法是在内核启动参数中加入"quiet"

bash 复制代码
setenv bootargs earlycon=nvt_serial,0x2f0280000 rootwait console=ttyS0,115200 libata.force=1.5 rootfstype=ext4 rw  lpj=120000 quiet

4.2、减少uboot日志输出

现在有些uboot会给出取消uboot日志输出的uboot指令。如果没有的话,我们可以直接修改uboot底层代码,取消uboot的日志输出。uboot的串口启动一般位于 drivers/serial, 根据时间情况修改即可

cpp 复制代码
drivers/serial/serial_ns16550.c
static void _serial_putc(const char c, const int port)
{
    if (uart_disable_anchor == 1)
        return; 

    NS16550_putc(PORT, c); 
}

5、内核修改

删除不需要的Uboot驱动和内核驱动配置,减少Uboot和内核的体积

6、文件系统修改

这一部分需要根据实际情况来修改。基本上需要优化的是内核加载文件系统到应用启动这一段的时间间隔,修改的方向大概是减少文件系统挂载的体积,修改挂载顺序,异步加载内核驱动,删除不必要的系统配置。

相关推荐
小白同学_C7 小时前
Lab4-Lab: traps && MIT6.1810操作系统工程【持续更新】 _
linux·c/c++·操作系统os
今天只学一颗糖7 小时前
1、《深入理解计算机系统》--计算机系统介绍
linux·笔记·学习·系统架构
不做无法实现的梦~9 小时前
ros2实现路径规划---nav2部分
linux·stm32·嵌入式硬件·机器人·自动驾驶
默|笙10 小时前
【Linux】fd_重定向本质
linux·运维·服务器
陈苏同学11 小时前
[已解决] Solving environment: failed with repodata from current_repodata.json (python其实已经被AutoDL装好了!)
linux·python·conda
“αβ”11 小时前
网络层协议 -- ICMP协议
linux·服务器·网络·网络协议·icmp·traceroute·ping
不爱学习的老登12 小时前
Windows客户端与Linux服务器配置ssh无密码登录
linux·服务器·windows
小王C语言13 小时前
进程状态和进程优先级
linux·运维·服务器
xlp666hub13 小时前
【字符设备驱动】:从基础到实战(下)
linux·面试
弹幕教练宇宙起源14 小时前
cmake文件介绍及用法
android·linux·c++