当前位置: 首页 > news >正文

Buildroot 增加系统启动项并解决后台无法获取输入(串口)

在这里插入图片描述

Buildroot 增加自启动项

  • 概述
  • 增加模块
  • 源码结构
  • 编写测试程序
  • 编译测试
  • 增加系统自启动
  • 一个问题
    • 解决方案:显式指定输入设备
  • 其他
    • /etc/init.d 目录下的 SXXxxx 文件作用解析‌

概述

Buildroot 是一款轻量级、高度可定制的开源工具集,专为嵌入式系统打造。它通过自动化构建流程(交叉编译工具链、Linux内核、根文件系统、引导程序),将复杂的嵌入式开发简化为“菜单配置”(make menuconfig)。开发者无需手动处理依赖与兼容性问题,只需勾选所需软件包,即可生成完整的可烧录镜像。在嵌入式场景中,服务进程的自动加载是刚需, 通常将启动脚本置于/etc/init.d/(SysV init)或编写systemd单元文件,实现服务的全自动托管。

增加模块

参考: Buildroot 添加自定义模块-内置文件到文件系统

–本文不再复述–

源码结构

package/auto_run/
├── CMakeLists.txt
├── Config.in
├── dumpsys.cpp
├── auto_run.mk
└── S99dumpsys

编写测试程序

dumpsys.cpp

#include <stdio.h>
#include <string.h>
#include <sys/statvfs.h>
#include <unistd.h>

void print_memory() {
    FILE *meminfo = fopen("/proc/meminfo", "r");
    if (!meminfo) {
        printf("无法读取内存信息\n");
        return;
    }

    char line[256];
    unsigned long total_mem = 0, free_mem = 0;

    while (fgets(line, sizeof(line), meminfo)) {
        if (strncmp(line, "MemTotal:", 9) == 0) {
            sscanf(line + 9, "%lu", &total_mem);
        } else if (strncmp(line, "MemFree:", 8) == 0) {
            sscanf(line + 8, "%lu", &free_mem);
        }
    }
    fclose(meminfo);

    printf("可用内存: %lu MB / 最大内存: %lu MB\n", free_mem / 1024, total_mem / 1024);
}

void print_disk() {
    struct statvfs stat;
    if (statvfs("/", &stat) != 0) {
        printf("无法读取磁盘信息\n");
        return;
    }

    unsigned long free_space = (stat.f_bfree * stat.f_frsize) / (1024 * 1024);
    unsigned long total_space = (stat.f_blocks * stat.f_frsize) / (1024 * 1024);

    printf("可用空间: %lu MB / 最大空间: %lu MB\n", free_space, total_space);
}

int main() {
    char input[10];

    while (1) {
        printf("输入命令 (mem/disk/exit): ");
        if (!fgets(input, sizeof(input), stdin)) {
            continue;
        }

        // 移除换行符
        input[strcspn(input, "\n")] = 0;

        if (strcmp(input, "mem") == 0) {
            print_memory();
        } else if (strcmp(input, "disk") == 0) {
            print_disk();
        } else if (strcmp(input, "exit") == 0) {
            break;
        } else {
            printf("无效命令\n");
        }
    }
    return 0;
}

编译测试

编译完成后, 可以直接将dumpsys 程序拷贝到/usr/data/dumpsys运行测试:

# /usr/data/dumpsys 
输入命令 (mem/disk/exit): mem
可用内存: 397 MB / 最大内存: 447 MB
输入命令 (mem/disk/exit): exit

增加系统自启动

auto_run.mk

AUTO_RUN_SITE = $(TOPDIR)/package/auto_run
AUTO_RUN_SITE_METHOD = local
AUTO_RUN_INSTALL_STAGING = YES

# dumpsys
define AUTO_RUN_INSTALL_INIT_SYSV
    $(INSTALL) -D -m 0755 $(AUTO_RUN_SITE)/S99dumpsys $(TARGET_DIR)/etc/init.d/S99dumpsys
endef

AUTO_RUN_POST_INSTALL_TARGET_HOOKS += AUTO_RUN_INSTALL_INIT_SYSV

$(eval $(cmake-package))

S99dumpsys

#!/bin/sh

check_run(){
    local file="/usr/data/dumpsys"
    local max_attempts=10
    local attempt=1

    while [ $attempt -le $max_attempts ]; do
        if [ -f "$file" ]; then
            echo "File $file found, starting dumpsys..."
            /usr/data/dumpsys &
            return 0
        else
            echo "Attempt $attempt: File $file not found, waiting..."
            sleep 1
            attempt=$((attempt + 1))
        fi
    done

    echo "file not found and exit"
    return 1
}

case "$1" in
    start)
        echo "Starting dumpsys..."
        check_run || exit 1
        ;;
    stop)
        echo "Stopping dumpsys..."
        killall dumpsys
        ;;
    restart)
        $0 stop
        $0 start
        ;;
    *)
        echo "Usage: $0 {start|stop|restart}"
        exit 1
        ;;
esac

exit 0

一切准备就绪! 打包烧录!

一个问题

为了方便测试, 默认/usr/data/目录下没有dumpsys, 通过命令拷贝并重启.
意外出现:

输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): 输入命令 (mem/disk/exit): ...

循环输出LOG: 输入命令 (mem/disk/exit):
调试发现问题出在

if (!fgets(input, sizeof(input), stdin))

原因是从stdin读取输入出现了问题.

在Buildroot环境中,fgets()在通过init脚本启动时失败但在手动启动时正常,这通常与标准输入(stdin)的环境差异有关。

由于 后台运行(&)的影响

  • &使程序在后台运行,可能导致stdin被关闭或重定向
  • init系统可能进一步处理了标准流

解决方案:显式指定输入设备

修改程序代码:

FILE *input_stream = fopen("/dev/console", "r");
if (!input_stream) {
    input_stream = fopen("/dev/tty", "r");
}
if (!input_stream) {
    input_stream = stdin;
}

if (!fgets(input, sizeof(input), input_stream)) {
    // 错误处理
}

根本原因是初始化环境和交互式shell环境的差异。在系统启动时,标准输入可能没有绑定到有效的终端设备,而手动启动时则会有完整的终端环境。

其他

/etc/init.d 目录下的 SXXxxx 文件作用解析‌

文件命名规则与核心作用‌

  • S 前缀‌:表示 ‌启动(Start)‌ 服务,与之对应的 K 前缀表示 ‌停止(Kill)‌ 服务‌。
    XX 数字‌:决定脚本执行顺序,数值小的优先执行(如 S10network 先于 S20apache 执行),用于解决服务间的依赖关系(例如网络接口需先于 Web 服务启动)‌。
    xxx 服务名‌:标识具体服务(如 S50ssh 表示 SSH 服务)。

  • SXXxxx 文件是 SysVinit/BusyBox 初始化系统中,通过数字排序和符号链接机制实现服务按需启动的核心配置单元,直接影响嵌入式设备或服务器在特定运行级别的服务状态‌。

相关文章:

  • PV操作指南
  • 【漏洞复现】Next.js中间件权限绕过漏洞 CVE-2025-29927
  • 穿越之程序员周树人的狂人日记Part7__教育战争2.0
  • C语言 —— 此去经年梦浪荡魂音 - 深入理解指针(卷四)
  • C++类与对象的第一个简单的实战练习-3.24笔记
  • 洛谷题单入门4-P5729 【深基5.例7】工艺品制作-python
  • 一文了解Gradle 依赖管理(五)- 依赖管理缓存依赖
  • K8S学习之基础四十五:k8s中部署elasticsearch
  • Spec2MP:项目管理之项目质量管理
  • oracle数据库(数据库启动关闭/sqlplus登录及基本操作/设置字符集/distinct去重)
  • 光流 | 基于光流的人体异常行为检测算法原理,公式,算法改进,matlab代码
  • 大模型应用(Java)2025/3/24
  • 运维面试题(八)
  • 《Python实战进阶》第31集:特征工程:特征选择与降维技术
  • PROE 与 STL 格式转换:开启 3D 打印及多元应用的大门
  • 【机器学习】什么是决策树?
  • FastStoneCapture免费版下载安装使用教程(附安装包)
  • 从零开始理解基于深度学习的语义分割模型:RCA与RCM模块的实现
  • 【项目合集】设备防丢失监控系统
  • C++List模拟实现|细节|难点|易错点|全面解析|类型转换|
  • 山西省委常委李金科添新职
  • 科创板年内第3家!健信超导IPO获受理,拟募资8.65亿
  • 季子文化与江南文化的根脉探寻与融合
  • 山东枣庄同一站点两名饿了么骑手先后猝死,当地热线:职能部门正调查
  • 全国层面首次!《防震减灾基本知识与技能大纲》发布
  • 湛江霞山通报渔船火灾:起火船舶共8艘,无人员伤亡或被困