关于<LDD3> "setconsole.c" Alesssandro Rubini 的邮件回复

来源:互联网 发布:手机路由器软件下载 编辑:程序博客网 时间:2024/05/22 12:59

http://blog.csdn.net/cinmyheart/article/details/38960967


关于 "setconsole.c"  Alesssandro Rubini 的邮件回复


                   抱着试一试的心态给偶像写了一封求助e-mail,回复了!!!我的小心脏啊~


呵呵~Rubini是谁不解释了...


我的HELP原文主要的关于setconsole.c的问题,这个是LDD3的一个小程序


setconsole.c
[cpp] view plain copy
 print?
  1. /* 
  2.  * setconsole.c -- choose a console to receive kernel messages 
  3.  * 
  4.  * Copyright (C) 1998,2000,2001 Alessandro Rubini 
  5.  *  
  6.  *   This program is free software; you can redistribute it and/or modify 
  7.  *   it under the terms of the GNU General Public License as published by 
  8.  *   the Free Software Foundation; either version 2 of the License, or 
  9.  *   (at your option) any later version. 
  10.  * 
  11.  *   This program is distributed in the hope that it will be useful, 
  12.  *   but WITHOUT ANY WARRANTY; without even the implied warranty of 
  13.  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the 
  14.  *   GNU General Public License for more details. 
  15.  * 
  16.  *   You should have received a copy of the GNU General Public License 
  17.  *   along with this program; if not, write to the Free Software 
  18.  *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA. 
  19.  */  
  20.   
  21. #include <stdio.h>  
  22. #include <stdlib.h>  
  23. #include <string.h>  
  24. #include <errno.h>  
  25. #include <unistd.h>  
  26. #include <sys/ioctl.h>  
  27.   
  28. int main(int argc, char **argv)  
  29. {  
  30.     char bytes[2] = {11,0}; /* 11 is the TIOCLINUX cmd number */  
  31.   
  32.     if (argc==2) bytes[1] = atoi(argv[1]); /* the chosen console */                                                                                                 
  33.     else {  
  34.         fprintf(stderr, "%s: need a single arg\n",argv[0]); exit(1);  
  35.     }  
  36.     if (ioctl(STDIN_FILENO, TIOCLINUX, bytes)<0) {    /* use stdin */  
  37.         fprintf(stderr,"%s: ioctl(stdin, TIOCLINUX): %s\n",  
  38.                 argv[0], strerror(errno));  
  39.         exit(1);  
  40.     }  
  41.     exit(0);  
  42. }  

但是在虚拟控制台之间测试一直不成功,很纠结~
我用*某*搜索引擎检索了很多blog,都是水...都是直接copy原书的内容,仅发现一个有对程序进行测试的,但是测试没有成功,google不能用,不知道情况如果,用TTT检索出来的一样,没有实质性的对setconsole.c的讨论和测试!


下面是Rubini 对我的回复:




首先指出了我的误区

1. ioctl(TIOCLINUX) 影响的并不是printf,而是printk!

2.tty 或者 terminal 不是console!   通过ctrl + alt + F* (1~6 )达到的都是控制台,F7是图形环境,不是console!


Allan Cruse 在2007年对setconsole做了一点改进


[cpp] view plain copy
 print?
  1. //----------------------------------------------------------------  
  2. //  setconsole.cpp  
  3. //  
  4. //  This utility allows a user possessing root privilege   
  5. //  to redirect 'printk' output to a designated console.   
  6. //  
  7. //  compile using:  
  8. //      root# gcc -o setconsole setconsole.cpp  
  9. //      root# chmod a+s setconsole  
  10. //  
  11. //  execute using:  
  12. //      user$ setconsole 4    
  13. //        
  14. //  Code used here is from an example by Alesandro Rubini,    
  15. //  "Linux Device Drivers (2nd Edition)," pages 99-100.  
  16. //              
  17. //  programmer: ALLAN CRUSE  
  18. //  written on: 24 NOV 2002  
  19. //  revised on: 24 JUN 2007 -- to use "/dev/console" device  
  20. //----------------------------------------------------------------  
  21.   
  22. #include <fcntl.h>        // for open()       <--- added   
  23. #include <stdio.h>        // for fprintf()  
  24. #include <errno.h>        // for errno  
  25. #include <stdlib.h>       // for exit()  
  26. #include <unistd.h>       // for STDIN_FILENO  
  27. #include <string.h>       // for strerror()  
  28. #include <sys/ioctl.h>        // for ioctl()  
  29. #include <asm/ioctls.h>       // for TIOCLINUX  
  30.   
  31. int main( int argc, char **argv )  
  32. {     
  33.     char    bytes[ 2 ] = { 11, 0 }; // 11 is the TIOCLINUX command-number  
  34.   
  35.     if ( argc == 2 ) bytes[1] = atoi( argv[1] );    // console id-number  
  36.     else    {  
  37.         fprintf( stderr, "%s: need a single argument\n", argv[0] );  
  38.         exit(1);  
  39.         }  
  40.   
  41.     int fd = open( "/dev/console", O_RDWR );        // <--- added  
  42.     if ( fd < 0 ) { perror( "/dev/console" ); exit(1); } // <--- added  
  43.       
  44.     if ( ioctl( fd, TIOCLINUX, bytes ) < 0 )     // <--- changed  
  45.         {     
  46.         fprintf( stderr, "%s: ioctl( fd, TIOCLINUX ): %s\n"// <---   
  47.                         argv[0], strerror( errno ) );  
  48.         exit(1);  
  49.         }  
  50.   
  51.     exit(0);  
  52. }  


是可以在console之间重定向IO的!!



这幅图看到的是tty4,我把控制台的IO重定向到了console 3,也就是/dev/tty3

切换到tty2,我们插入hello.ko 模块,调用printk,打印hello world



我故意执行了tty这个shell程序,提示读者当前我们在tty2,进行insmod操作



明明是有hello world的为什么不打印呢?去哪儿了?console 3!

我们ctrl + alt + F3 切换到console 3看看



终于搞定了!利用ioctl实现了对于不同console之间的IO重定向!




最后谢谢偶像~ Alesssandro Rubini & Allan Cruse


0 0
原创粉丝点击