Linux管道函数使用

背景:项目有个功能是查看版本号(像什么内核版本号、驱动版本号,等等),是通过管道来实现的,网上也有这方面的介绍,就不多说了。 在一次测试过程中,发现不断查看版本号竟然会导致系统复位(比如查看20次、40次、100次),这个bug发现晚的原因是没有谁那么无聊连续查看100次版本号,当然,发现也是碰巧在几次查看之后系统就挂了。后来发现是因为某个地方卡住导致某个线程超时而复位。后来跟踪到了这个管道函数中来。解决的方法是使用读取文件的方法来获取系统版本号(如内核版本)。不过真正原因还没有找到,可能出在popen这类函数中。

下面给出示例代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>

static void OnSignal(int signalno)
{
    printf("receive signal num: %d\n", signalno);
    int bExit = true;
    for (int i = 0; i < 100; i++)
    {
        if (signalno == 17 || signalno == 13)
        {
            bExit = false;
            break;
        }
    }
    if (bExit)
    {
        printf("exit process.\n");
    }
}

// 通过管道读取执行命令后的信息
void GetSystemCmdInfo(const char* cmd, char* buf, int len)
{
    char info[256] = {0};

    FILE* fp = popen(cmd, "r");
    if (fp)
    {
        fread(info, 1, sizeof(info), fp);

        if ( buf != NULL && len > (int)strlen(info) )
        {
            strcpy(buf, info);
        }
        pclose(fp);
    }
}

// 通过文件读取到信息
// 这种方法是读文件,上面的方法是执行命令,两者不是一回事
void ReadKernelVersion(char* file, char* buf, int len)
{
    FILE* fp = NULL;
    char info[256] = {0};
    fp = fopen(file, "r");
    if (fp == NULL)
    {
        perror("open file error");
        return;
    }
    fread(info, 1, sizeof(info), fp);
    if ( buf != NULL && len > (int)strlen(info) )
    {
        strcpy(buf, info);
    }
    fclose(fp);
}

int main(void)
{
    char szValue[512] = {0};
    char buf[1024] = {0};
    int iLen = 0;
    for (int i = 0; i < 255; i++)
    {
        if (i != SIGILL && i != SIGBUS && i != SIGSEGV)
        {
            signal(i, OnSignal);
        }
    }
    // 执行此函数时,会有SIGCHLD信号
    GetSystemCmdInfo("cat /proc/version", szValue, sizeof(szValue));
    //ReadKernelVersion("/proc/version", szValue, sizeof(szValue));
    iLen = sprintf(buf, "[Kernel ver]: %s\r\n", szValue);
    printf("%s", buf);

    return 0;
}

popen算是重型武器了,一般的小场合可不必使用,像查看内核版本,可能使用cat /proc/version命令,也可以读取/proc/version文件。 还是那名话说得好,具体情况具体分析,因地制宜采用不同的策略方能制胜。
语法高亮由迟思堂工作室强力支持

声明:本文仅为学习之目的,代码仅是示例性代码,网络上也能找到类似用法,无意亦无胆涉及过多其它代码。

李迟 代码随笔 即日