Linuxカーネル層-アプリケーション層

カーネル層はアプリケーション層プログラムを呼び出します

リファレンスブログ:
Linuxドライバーはアプリケーションを呼び出します(実行/実行します)
この関数はカーネルレイヤーでのみ実行する必要があります

extern int call_usermodehelper(char *path, char **argv, char **envp, int wait);

カーネル層プログラム:

#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/init.h>
#include <linux/kmod.h>

MODULE_AUTHOR("darui");
MODULE_LICENSE("GPL");

static int __init practiceCall(void)
{
    
    
		int ret = 0;
        char *path = "/home/darui/practice/a";
        char *arv[] = {
    
    path,"",NULL};//无参数,注意要按照这个格式path在 第一个位置,必须NULL结束
        char *env[] = {
    
    "/home/darui","/usr/local/sbin",NULL};//多路径可以用逗号隔开,也可以在一对引号中用分号隔开,必须NULL结束标志

        printk("%s:call_usermodehelper\n",__func__);
        ret = call_usermodehelper(path,arv,env,UMH_WAIT_PROC);
        if(ret < 0)
        {
    
    
        	printk("error");
        	return -1;
        }
        return 0;
}

static void __exit practiceCallExit(void)
{
    
    
        printk("%s:call_usermodehelper\n",__func__);
}

module_init(practiceCall);
module_exit(practiceCallExit);

落とし穴があり、実行プログラムをa.shや他のプログラムにすることはできず、コンパイルされたプログラムのみにする必要があり、シェルスクリプトを実行するとエラーが発生することに注意してください(長い間スタックしています)。

アプリケーション層プログラムのソースコード:

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int main()
{
    
    
        int fd = -1;

        fd = open("/home/darui/practice/file.txt",O_RDWR|O_CREAT);
        write(fd,":practice\n",10);
        close(fd);
}

gcc -oa acをコンパイルすると、file.txtファイルが作成されます

原則:
call_usermodehelper-
》 call_usermodehelper_setup + call_usermodehelper_exec
————》 queue_work
——————》 __ call_usermodehelper
————————》 do_execve
——————————
1。プロセススタック内プッシュアプリケーション層プログラムに、
2 IRET命令は、スタックの切り替えのためのオンサイトカーネル保存し
、アプリケーション層が実行された後3を、ret_from_sys_call戻ります

ピットにパッチを当てる

ピットにパッチを当てる

おすすめ

転載: blog.csdn.net/qq_42882717/article/details/114062512