Linux编程之给你的程序开后门
- 在进程里开一个线程用于充当debug center
- 该线程通过fifo接收开发人员传给它的命令
- 解析这些命令
- 用脚本搭建简单的命令行界面
if(pthread_create(&debug_thread_id, NULL, (void*)debug_center, NULL))
{
MY_LOG(FATAL,"create debug center fail!\n");
return -;
}
system("rm /vob/ljsdpoenew3/exercise/debug_log"); //每次进入debug center我们都将原来的fifo文件删除,避免影响后面操作
rc = mkfifo("/vob/ljsdpoenew3/exercise/debug_log", ); //创建fifo
if(rc < )
{
MY_LOG(DEBUG, "make fifo fail!\n");
pthread_exit();
}
fp = fopen("/vob/ljsdpoenew3/exercise/debug_log", "r"); //打开fifo,读取指令
if(fp == NULL)
{
MY_LOG(DEBUG, "open debug_log fail!\n");
pthread_exit();
}
读fifo我们解决了,那怎么将我们的指令写进fifo呢?这里我打算使用shell的read指令,文章后面会解释如何实现。
- 将从fifo取得数据进行格式解析,比如我定义了d d的意思是display debug,即显示现在的debug级别,那么我们程序就得首先对原始数据进行格式处理。
- 将指令进行命令解析,执行相应操作。
static int get_args(FILE *inputFile)
{
char tmpBuffer[];
char *line = tmpBuffer;
char separator[] = " ,\n\t";
char *token;
int i;
char eof; int num = ; eof = !fgets(line, sizeof(tmpBuffer), inputFile);
if (eof)
return num; token = strtok(line, separator);
while (num < MAX_NUM_ARGS && token)
{
strcpy(args[num], token);
num++;
token = strtok(NULL, separator);
} for (i = num; i < MAX_NUM_ARGS; i++)
args[i][] = ; return num;
}
switch(args[][]) //解析指令,看每个指令对应哪些意思
{
case 'd': //display
switch(args[][])
{
case 'q': //display queue
show_MQ(fd);
break;
case 'd': //display debug
show_debug_level(fd);
break; default:
help_manual(fd);
break;
}
break; case 's': //set
switch(args[][])
{
case 'd': //set debug level
n = atoi(args[]); //将字符串转化为整数
fprintf(fd," debug level change from %d to %d",global.debug_level,n);
global.debug_level = n; //更改log级别
break; default:
help_manual(fd);
break;
}
break; default:
help_manual(fd);
break;
}

- 我们需将程序的打印输出重定向到一个文件里
- 使用shell脚本读出文件的内容
- 我们输入的命令需写入到fifo中
- 重定向输出
一开始我是打算将标准输出、标准错误都重定向到文件里的额,但是想了想,还是想直接把打印内容直接输出到文件就好了。比如这样:
fd = fopen("/vob/ljsdpoenew3/exercise/debug_log2", "w+");
if(fd == NULL)
{
MY_LOG(DEBUG, "open debug_log2 fail!\n");
pthread_exit();
}
fprintf(fd," debug level change from %d to %d",global.debug_level,n);
2.利用脚本读出内容且将内容写入fifo
#!/bin/bash my_inp_fifo=/vob/ljsdpoenew3/exercise/debug_log # 指定fifo文件
stty erase ^H while [ true ];
do read inp #循环读入用户输入
if [ "$inp" == "quit" ];then #quit代表结束该界面
exit
fi
echo $inp > $my_inp_fifo #写入fifo
cat debug_log2.txt #显示log的内容
done






#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <math.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/prctl.h>
#include "msg_def.h"
#include "global.h" extern Queue_t MsgQueue;
extern dashboard_t global; #define MAX_NUM_ARGS 20
#define MAX_ARGS_SIZE 56 static char args[MAX_NUM_ARGS][MAX_ARGS_SIZE]; static int get_args(FILE *inputFile,FILE *fd)
{
char tmpBuffer[];
char tmpBuffer2[];
char *line = tmpBuffer;
char separator[] = " ,\n\t";
char *token;
int i;
char eof; int num = ; eof = !fgets(line, sizeof(tmpBuffer), inputFile);
if (eof)
return num; memcpy(tmpBuffer2,tmpBuffer,); token = strtok(line, separator);
while (num < MAX_NUM_ARGS && token)
{
strcpy(args[num], token);
num++;
token = strtok(NULL, separator);
} for (i = num; i < MAX_NUM_ARGS; i++)
args[i][] = ; if(num > )
{
fprintf(fd, "%s", tmpBuffer2);
} return num;
} static void help_manual(FILE* fd)
{
fprintf(fd,"\nd d : display current debug level\n");
fprintf(fd,"d q : display msg queue length, head and tail\n");
fprintf(fd,"s d [level] : set debug [level] \n");
} static void show_MQ(FILE* fd)
{
fprintf(fd," msg queue length:%d head:%d tail:%d \n",abs(MsgQueue.head-MsgQueue.rear),MsgQueue.head,MsgQueue.rear);
} static void show_debug_level(FILE* fd)
{
fprintf(fd," current debug level: %d\n", global.debug_level);
} void debug_center()
{
int rc,num,n;
FILE* fp;
FILE* fd; MY_LOG(DEBUG,"Hi,debug!\n"); system("rm /vob/ljsdpoenew3/exercise/debug_log");
system("rm /vob/ljsdpoenew3/exercise/debug_log2");
rc = mkfifo("/vob/ljsdpoenew3/exercise/debug_log", );
if(rc < )
{
MY_LOG(DEBUG, "make fifo fail!\n");
pthread_exit();
} fp = fopen("/vob/ljsdpoenew3/exercise/debug_log", "r");
if(fp == NULL)
{
MY_LOG(DEBUG, "open debug_log fail!\n");
pthread_exit();
} fd = fopen("/vob/ljsdpoenew3/exercise/debug_log2", "w+");
if(fd == NULL)
{
MY_LOG(DEBUG, "open debug_log2 fail!\n");
pthread_exit();
}
//freopen("/vob/ljsdpoenew3/exercise/debug_log2.txt", "w+", fd); fprintf(fd," \n==================================================\n");
fprintf(fd," Welcme to Debug Center!");
fprintf(fd," \n==================================================\n\n");
help_manual(fd);
//fflush(fd); while()
{
fflush(fd);
fprintf(fd,"\n\nmy_diag>");
num = get_args(fp,fd);
if(num < )
{
freopen("/vob/ljsdpoenew3/exercise/debug_log", "r", fp);
fflush(fd);
continue;
}
fprintf(fd,"\n\nmy_diag>");
switch(args[][])
{
case 'd': //display
switch(args[][])
{
case 'q': //display queue
show_MQ(fd);
break;
case 'd': //display debug
show_debug_level(fd);
break; default:
help_manual(fd);
break;
}
break; case 's': //set
switch(args[][])
{
case 'd': //set debug level
n = atoi(args[]);
fprintf(fd," debug level change from %d to %d",global.debug_level,n);
global.debug_level = n;
break; default:
help_manual(fd);
break;
}
break; default:
help_manual(fd);
break;
} }
}
Linux编程之给你的程序开后门的更多相关文章
- 从裸机编程到嵌入式Linux编程思想的转变------分而治之:驱动和应用程序
笔者学习嵌入式Linux也有一段时间了,很奇怪的是很多书讲驱动编程方面的知识,也有很多书将ARM9方面的知识,但是从以前51形式的(对寄存器直接操作,初始化芯片的功能模块)编程方法,和思维模式,变换为 ...
- 牛人整理分享的面试知识:操作系统、计算机网络、设计模式、Linux编程,数据结构总结 转载
基础篇:操作系统.计算机网络.设计模式 一:操作系统 1. 进程的有哪几种状态,状态转换图,及导致转换的事件. 2. 进程与线程的区别. 3. 进程通信的几种方式. 4. 线程同步几种方式.(一定要会 ...
- 【转】牛人整理分享的面试知识:操作系统、计算机网络、设计模式、Linux编程,数据结构总结
基础篇:操作系统.计算机网络.设计模式 一:操作系统 1. 进程的有哪几种状态,状态转换图,及导致转换的事件. 2. 进程与线程的区别. 3. 进程通信的几种方式. 4. 线程同步几种方式.(一定要会 ...
- Linux 编程中的API函数和系统调用的关系【转】
转自:http://blog.chinaunix.net/uid-25968088-id-3426027.html 原文地址:Linux 编程中的API函数和系统调用的关系 作者:up哥小号 API: ...
- linux编程获取本机网络相关参数
getifaddrs()和struct ifaddrs的使用,获取本机IP 博客分类: Linux C编程 ifaddrs结构体定义如下: struct ifaddrs { struct ifad ...
- 面试知识:操作系统、计算机网络、设计模式、Linux编程,数据结构总结
基础篇:操作系统.计算机网络.设计模式 一:操作系统 1. 进程的有哪几种状态,状态转换图,及导致转换的事件. 2. 进程与线程的区别. 3. 进程通信的几种方式. 4. 线程同步几种方式.(一定要会 ...
- Linux编程简介
Linux编程可以分为Shell(如BASH.TCSH.GAWK.Perl.Tcl和Tk等)编程和高级语言(C语言,C++语言,java语言等)编程,Linux程序需要首先转化为低级机器语言即所谓的二 ...
- 在桌面Linux环境下开发图形界面程序的方案对比
在Linux下开发GUI程序的方法有很多,比如Gnome桌面使用GTK+作为默认的图形界面库,KDE桌面使用Qt作为默认的图形界面库,wxWidgets则是另一个使用广泛的图形库,此外使用Java中的 ...
- Linux编程return与exit区别
Linux编程return与exit区别 exit 是用来结束一个程序的执行的,而return只是用来从一个函数中返回. return return 表示从被调函数返回到主调函数继续执行,返回时可附 ...
随机推荐
- Lyx输入中文与代码高亮
如果您看了我的这个随笔:<OpenSUSE 13.2安装Texlive2014+Texmaker+Lyx> (一)LyX中文 打开Lyx直接新建开始使用,那么输入的中文会是编译失败的,疑? ...
- 使用Vim或Codeblocks格式化代码
在网上的代码,有很多的代码都是丢失缩进的,几行还好,手动改改,多了呢,不敢想象,没有缩进的代码.别说排错,就是阅读都是困难的,还好,有两个常用工具可以轻松的解决问题. (一)Vim(简单方便,可将代码 ...
- 单极型ADC如何测量负电压?
最常用的方法是使用一个运放做成加法器将负电压抬到0V以上,如果这样的输出超过了最大输出电压那么再使用比例衰减就可以办到了. 参考下面的讨论: http://www.amobbs.com/thread- ...
- spring bean范围
总结: 实例代码具体解释: 文件夹结构 Car.java package com.coslay.beans.autowire; public class Car { private String br ...
- 转: js快速分享代码
这是一款简单易用的文章分享工具,您只需将下面的html代码拷贝到模板中就可以实现文章快速分享功能.如果您想分享你的博客.个人网站或者企业网站等等,下面是两款不错的分享工具,值得拥有! 1. [html ...
- ADFS 2.0 配置简介 PartⅡ – 配置 ADFS 信任关系
ADFS 与应用程序间的各种验证是基于信任关系的,在 ADFS 服务器配置好要信赖的应用程序(以 URL 为标识)后,应用程序再通过指定认证服务器来将用户引导至 ADFS 登录页,登录完成后再将用户的 ...
- Visual Studio 自定义项目或工程模板
如何利用 Visual Studio 自定义项目或工程模板 在开发项目的时候,由其是商业性质的大型项目时,往往需要在每个代码文件上都加上一段关于版权.开发人员的信息,并且名称空间上都需要带有公司的标志 ...
- .NET 类库研究
.NET 类库研究必备参考 添加微软企业库源码 前不久,为大家提供了一个.NET 类库参考源码的网站,扣丁格鲁(谐音“coding guru”),使用了段时间,发现一些不方便的地方,特意做了一些更改, ...
- iOS基础 - UIWebView
一.UIWebView简介 是iOS内置的浏览器控件,可以浏览网页.打开文档等 能够加载html/htm.pdf.docx.txt等格式的文件 系统自带的Safari浏览器就是通过UIWebView实 ...
- 浅谈DevExpress<六>:为chart创建动态数据源
今天搞点稍微复杂些的东西,在列表中点击不同的行时,图表中显示和其数据关联的图,效果如下: