嵌入式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、文件系统修改

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

相关推荐
用户23678298016842 分钟前
Linux pgrep 命令详解:按名称查找进程 PID 的高效方法
linux
zzipeng1 小时前
Linux LCD驱动
linux·运维·服务器
思麟呀1 小时前
C++11并发编程:call_once一次性执行+atomic原子类型+CAS无锁编程+自旋锁
linux·开发语言·jvm·c++·windows
浮生若城1 小时前
Linux库制作与原理(2):理解链接与加载
linux·运维·服务器
Cat_Rocky1 小时前
Gitlab安装与配置
linux·运维·gitlab
爱讲故事的1 小时前
操作系统第一讲复习:为什么学习操作系统,以及操作系统到底在做什么?
linux·开发语言·windows·学习·ubuntu·c#
荒--1 小时前
kali安装与下载、设置(2026)
linux·服务器
sulikey2 小时前
个人Linux操作系统学习笔记4 - makefile
linux·makefile·make·构建
_童年的回忆_2 小时前
【php】在linux下PHP安装amqp扩展
linux·开发语言·php
sxlishaobin2 小时前
linux 自动清除日志 脚本
linux·服务器·前端