原创

Linux管道函数使用

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://latelee.blog.csdn.net/article/details/7839282

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

#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, 1sizeof(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, 1sizeof(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文件。
还是那名话说得好,具体情况具体分析,因地制宜采用不同的策略方能制胜。


语法高亮由迟思堂工作室强力支持

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

文章最后发布于: 2012-08-07 15:22:09
展开阅读全文
0 个人打赏
私信求帮助

没有更多推荐了,返回首页

©️2019 CSDN 皮肤主题: 技术工厂 设计师: CSDN官方博客

分享到微信朋友圈

×

扫一扫,手机浏览