Linu移植随笔:由ts_config:Success想到的

背景:

其时正在搞qtopia-2.2.0,结果花费大量时间却收获甚少。本文只是以Tslib为背景介绍一下一些程序的出错处理。不涉及具体的移植、操作等等。

 

Tslib校准的测试如下:

# ./ts_calibrate
No raw modules loaded.
ts_config: Success

这个“错误”信息来自./tests/ts_calibrate.c文件第215行(注:./表示Tsllib所在的目录,下同):

if (ts_config(ts)) {
        perror( "ts_config");
        exit(1);
 }

当ts_config函数返回值为非零时,打印出错信息,并退出。

 

ts_config函数在./src/ts_config.c文件第91行:

if (ts->list_raw == NULL) {
        ts_error( "No raw modules loaded./n");
        ret = -1;
 }

它调用ts_error函数打印“No raw modules loaded.”,返回非零值。

tslib打印出错信息比较经典,有必要在这里重现一下(注意函数指针的用法):

static int stderrfn(const char *fmt, va_list ap)
{
        return vfprintf(stderr, fmt, ap);
}

/*
 * Change this hook to point to your custom error handling function.
 */

int (*ts_error_fn)(const char *fmt, va_list ap) = stderrfn;

int ts_error(const char *fmt, ...)
{
        va_list ap;
        int ret;

        va_start(ap, fmt);
        ret = ts_error_fn(fmt, ap);
        va_end(ap);

 return ret;
}

 

罪魁祸首的就是这句:
perror("ts_config");

perror的手册描述如下:
The  routine  perror() produces a message on the standard error output,describing the last error encountered during a  call  to  a  system  or library  function.
网上有的资料将“last error”翻译成“最后的错误(号)”,这不太正确,应该说是“上一次的错误(号)”。正如英文中的“last night”一样。
这里将错误分两种:“系统错误”与“自定义错误”。系统错误信息由perror或strerror来打印。其中strerror是将errno转换成可读的字符串。比如errno为2的信息为“No such file or directory”,当没有发生错误时,errno为0号,即“Success”。错误号及其对应的信息可以在errno.h头文件中找到,在我的系统中,实际在/usr/include/asm-generic目录下的errno-base.h和errno.h 文件中定义。


前面的perror("ts_config");打印了“ts_config: Success”,就是因为上一次的系统(库)调用没有发生错误,errno返回值为0,为0者,当然是“Success”。

 

下面以打开串口的函数作例子。

int open_port( int port)
{
         int fd = -1;  /* File descriptor for the port, we return it. */

         if (port < 1 || port > 4)
        {
                printf( "Sorry, the port number must be 1~4./n");
                 return -1;
        }

        fd = open(device, O_RDWR | O_NOCTTY | O_NDELAY);
         if (fd == -1)
        {
                perror( "Unable to open the port");
                 return -1;
                }
         return fd;
}

示例代码中有两种错误,自定义的错误,即串口号只能在1~4之间,超出范围即提示错误,这里只使用简单的printf语句;另一个是调用系统调用open时的错误处理,使用perror来输出错误信息。
如果调用open_port(6),则会提示如下信息:
Sorry, the port number must be 1~4.
如果串口号范围正常,但由于调用open时发生错误,比如下面的两个:
Unable to open the port: Permission denied
Unable to open the port: No such file or directory
一个是权限不够(因为我使用普通用户来执行),另一个是设备文件不存在(因为我将device改为一个不存在的串口设备)。

 

下面使用perror和strerro作一些小测试:

perror("This is not an error!");
printf("errno %d:%s /nerrno %d:%s/n", 0, strerror(0), 2, strerror(2));

打印如下信息:

This is not an error!: Success
errno 0:Success
errno 2:No such file or directory

出错处理对于程序来说非常重要,一个很不好的出错处理例子如下:

if (ret == -1)
        printf( "发生错误,请联系管理员");
else  if (ret == -2)
        printf( "发生错误,请联系管理员");
else  if (ret == -3)
        printf( "发生错误,请联系管理员");
else  if (ret == -4)
        printf( "发生错误,请联系管理员");
else  if (ret == -5)
        printf( "发生错误,请联系管理员");

就是说,不能将所有的错误都交给“管理员”来处理,“管理员”能力再强,也有忘记ret所对应的各种错误信息的时候。正如为函数、变量命名要使用一些有意义的词语一样,最好不要用“process_data()”来表示“处理数据”,因为这个范围太大了,还不如用一些更具体的名称,比如是过滤字符串中某个字符、解析字符串,等等。
一般地,在Linux中调用系统调用都应该判断一下返回值(通常返回-1表示发生错误),同时使用perror或strerro来打印错误信息。但是不要将“系统信息”和“自定义错误”搞混了。此外,错误信息也应该全面一些,在易出错处,不能只打印一行“出错了”就了事。这个范围同样也太大了。到底是什么错误呢?权限不足?文件不存在?……太多了。

 

我们写代码应该有责任心。不为他人,亦为自己。

本文首发自http://www.latelee.org/embedded-linux/119-porting-linux-from-tsconfig.html

 

你可能感兴趣的:(linux,测试,File,System,library,output)