当一个可执行文件已经为write而open时,此时的可执行文件是不允许被执行的。反过来,一个文件正在执行时,它也是不允许同时被write模式而open的。这个约束很好理解,因为文件执行和文件被写应该需要同步保护,因此内核会保证这种同步。那么内核是如何实现该机制的呢?

Inode结点中包含一个数据项,叫做i_writecount,很明显是用于记录文件被写的个数的,用于同步的,其类型也是atomic_t. 内核中有两个我们需要了解的函数,与write操作有关,分别是:

int get_write_access(struct inode * inode)

{

spin_lock(&inode->i_lock);

if (atomic_read(&inode->i_writecount) < 0) {

spin_unlock(&inode->i_lock);

return -ETXTBSY;

}

atomic_inc(&inode->i_writecount);

spin_unlock(&inode->i_lock);

return 0;

}

int deny_write_access(struct file * file)

{

struct inode *inode = file->f_path.dentry->d_inode;

spin_lock(&inode->i_lock);

if (atomic_read(&inode->i_writecount) > 0) {//如果文件被打开了,返回失败

spin_unlock(&inode->i_lock);

return -ETXTBSY;

}

atomic_dec(&inode->i_writecount);

spin_unlock(&inode->i_lock);

}

这两个函数都很简单,get_write_acess作用就和名称一致,同样deny_write_access也是。如果一个文件被执行了,要保证它在执行的过程中不能被写,那么在开始执行前应该调用deny_write_access 来关闭写的权限。那就来检查execve系统调用有没有这么做。

Sys_execve中调用do_execve,然后又调用函数open_exec,看一下open_exec的代码:

struct file *open_exec(const char *name)

{

struct file *file;

int err;

file = do_filp_open(AT_FDCWD, name,

O_LARGEFILE | O_RDONLY | FMODE_EXEC, 0,

MAY_EXEC | MAY_OPEN);

if (IS_ERR(file))

goto out;

err = -EACCES;

if (!S_ISREG(file->f_path.dentry->d_inode->i_mode))

goto exit;

if (file->f_path.mnt->mnt_flags & MNT_NOEXEC)

goto exit;

fsnotify_open(file->f_path.dentry);

err = deny_write_access(file);//调用

if (err)

goto exit;

out:

return file;

exit:

fput(file);

return ERR_PTR(err);

}

明显看到了deny_write_access的调用,和预想的完全一致。在open的调用里,应该有get_write_access的调用。在open调用相关的__dentry_open函数中就包含了对该函数的调用。

if (f->f_mode & FMODE_WRITE) {

error = __get_file_write_access(inode, mnt);

if (error)

goto cleanup_file;

if (!special_file(inode->i_mode))

file_take_write(f);

}

其中__get_file_write_access(inode, mnt)封装了get_write_access.

那么内核又是如何保证一个正在被写的文件是不允许被执行的呢?这个同样也很简单,当一个文件已经为write而open时,它对应的inode的i_writecount会变成1,因此在执行execve时同样会调用deny_write_access 中读取到i_writecount>0之后就会返回失败,因此execve也就会失败返回。

这里是写文件与i_writecount相关的场景:

写打开一个文件时,在函数dentry_open中:

if (f->f_mode & FMODE_WRITE) {

error = get_write_access(inode);

if (error)

goto cleanup_file;

}

当然在文件关闭时,会将i_writecount--;关闭时会执行代码:

if (file->f_mode & FMODE_WRITE)

put_write_access(inode);

put_write_access 代码很简单:

static inline void put_write_access(struct inode * inode)

{

atomic_dec(&inode->i_writecount);

}

于是乎自己写了个简单的代码,一个空循环,文件在执行的时候,在bash中,echo 111 >>可执行文件,结果预期之中,返回失败,并提示信息 text file busy.

那么该机制是否同样适用于映射机制呢,在执行可执行文件时,会mmap一些关联的动态链接库,这些动态链接库是否被mmap之后就不允许被写以及正在写时不允许mmap呢?这个是需要考虑的,因为它关系到安全的问题。因为库文件也是可执行的代码,被篡改同样会引起安全问题。

Mmap在调用mmap_region的函数里,有一个相关的检查:

if (vm_flags & VM_DENYWRITE) {

error = deny_write_access(file);

if (error)

goto free_vma;

correct_wcount = 1;

}

其中,mmap调用中的flags参数会被正确的赋值给vm_flags,对应关系是MAP_DENYWRIRE被设置了,那么VM_DENYWRITE就对应的也被设置。下面写了个简单的代码,做一下测试:

#include

#include

#include

#include

#include

#include

int main()

{

int fd;

void *src = NULL;

fd = open("test.txt",O_RDONLY);

if (fd != 0)

{

if ((src = mmap(0,5,PROT_READ|PROT_EXEC  ,MAP_PRIVATE|        MAP_DENYWRITE,fd,0))== MAP_FAILED)

{

printf("MMAP error\n");

printf("%s\n",strerror(errno));

}else{

printf("%x\n",src);

}

}

FILE * fd_t = fopen("test.txt","w");

if( !fd_t)

{

printf("open for write error\n");

printf("%s\n",strerror(errno));

return 0;

}

if (fwrite("0000",sizeof(char),4,fd_t) != 4)

{

printf("fwrite error \n");

}

fclose(fd_t);

close(fd);

return 1;

}

最后的test.txt被写成了”0000”,很奇怪,貌似MAP_DENTWRITE不起作用了。于是man mmap查看,发现:

MAP_DENYWRITE

This  flag  is ignored.  (Long ago, it signaled that attempts to write to the underlying file should fail with ETXTBUSY. But this was a source of denial-of-service attacks.)

原来这个标识在用户层已经不起作用了啊,而且还说明了原因,容易引起拒绝式服务攻击。攻击者恶意的将某些系统程序要写的文件以MAP_DENYWRITE模式映射,会导致正常程序写文件失败。不过VM_DENYWRITE在内核里还是有使用的,在mmap中还是有对deny_write_access的调用, 但是对它的调用已经不是由mmap中的flag参数的MAP_DENYWRITE驱动的了。

那与可执行文件相关的动态链接库文件就悲剧了,大家都知道动态链接库使用的也是mmap,这也导致动态链接库在运行时可以被更改。其实我这就是为了确认这点。这也导致我需要自己写同步控制代码了。我们可以使用inode中的i_security以及file结构的f_secutiry变量来写自己的同步逻辑,就是麻烦了不少,还要写内核模块,哎,工作量又增加了啊。安全问题是个麻烦的问题...

linux写文件操作同步,linux 可执行文件与写操作的同步问题(文件读写操作产生的锁机制)...相关推荐

  1. [绍棠] iOS文件目录和文件操作 及NSFileManager的读写操作

    1.理解部分 1.1文件 <1>文件管理类NSFileManager 2.对文件进行管理操作 a.遍历查看目录下的文件 [深度遍历] [浅度遍历] b.创建文件/目录 c.拷贝文件/目录 ...

  2. SDRAM的数据存储实现并对其数据进行读写操作

    SDRAM的数据存储实现并对其数据进行读写操作 文章目录 SDRAM的数据存储实现并对其数据进行读写操作 1 SDRAM 1.1 简介 1.2 SDRAM结构和原理 1.3 SDRAM块 2 SDRA ...

  3. 【FPGA】SPI协议详解及对flash读写操作

    FPGA基于SPI实现对flash读写操作 概括 一.SPI协议.flash讲解 1.SPI协议 2.flash (1)WREN (2)RDID (3)WRSR (4)READ (5)PP (6)SE ...

  4. Linux写文件断电保存,linux 写文件操作,异常掉电后,文件损坏丢失(0kb)

    今天调试程序,在做异常断电测试时,再开机发现文件是0 kb, 通过log查询,文件已经写入完成.不明白为何掉电之后文件就没了. 查到下面的博文解决了该问题. 通过 fflush->fsync-& ...

  5. linux下 c语言 用write open二进制写文件,Linux下用C语言fopen、fread和fwrite函数对二进制文件的操作-Go语言中文社区...

    一.前言 在做一个镜头的初始化操作,需要加载一个648*522像素大小的文件,厂商提供的是一个excel表,如果要加载数据,可用加载txt文本的方式,我选用二进制方式加载文件:大家都知道电脑真正执行的 ...

  6. Linux赋予目录或文件任何人都可以读、写、执行的操作

    之前root用户导出的文件,现在要赋予任何人都可以写的权利: 示例 任何人可读.可写.可执行: chmod 777 文件 任何人可读.可写,不可执行: chmod 666 文件 详细解析 比如drwx ...

  7. linux 文件操作函数,Linux下的文件操作函数及creat用法

    编写Linux应用程序要用到如下工具: (1)编译器:GCC GCC是Linux平台下最重要的开发工具,它是GNU的C和C++编译器,其基本用法为:gcc [options] [filenames]. ...

  8. linux 文件操作 编程,Linux系统编程------------文件操作(基础)

    一.文件操作 1.1 Linux文件系统结构 1.1.1  Linux常见系统目录 /bin  :  存放普通系统可执行的命令(ls wc等) /sbin  :  存放系统管理程序(fsck等) /b ...

  9. linux gcc编译下的文件读写操作

    linux下的文件操作 所有目录             1.文件及文件系统的定义             2.linux文件的类型             3.linux文件的权限          ...

最新文章

  1. HTML 中的marquee标签详解
  2. Spring IOC和MVC基础知识
  3. mysql语句导出数据库文件_通过Mysql命令行语句来导入、导出数据库文件
  4. 【技术综述】基于弱监督深度学习的图像分割方法综述
  5. 如何将.py文件转换为.exe
  6. MyBatis——动态SQL语句——if标签和where标签复合使用
  7. python科学计算笔记(十三)pandas的merge、concat合并数据集
  8. 「最简单」的 Core Data 上手指南
  9. 手机上图片信息怎么拉一个矩形框_华为手机EMUI系统隐藏的10个功能,上手体验后,实用性无敌了...
  10. python中tolist_python 列表,数组,矩阵两两转换tolist()的实例
  11. 计算机网络class 3(速率的相关性能指标)
  12. 其它 博客园 自己写的文章 标题含有小写字母 查看文章时 标题就变成大写的了...
  13. 80X86 汇编指令符号大全
  14. MyBatis和Spring总结
  15. 用c语言做一个背单词的软件,安利一个超好用且自由度高的背单词app
  16. 配置修改Tomcat端口
  17. windows 7 远程桌面连接图文教程
  18. android 图标 自定义,巧用 Drawable 之实现一个最简单的自定义电池图标
  19. Flink 入门教程
  20. AudioTrack 分析

热门文章

  1. Docker下载与安装(win7,8,10,mac)
  2. linux sort 排序 性能,Linux中sort 排序
  3. c语言编程命令,C语言编程命令
  4. Exp3 免杀原理与实践 20164314
  5. 机器学习(课程笔记4)——逼近参数算法牛顿方法;
  6. 参数数组(params)的用法
  7. Float 运算的怪异性
  8. tika提取html,TIKA内容提取
  9. rust异步编程--理解并发/多线程/回调/异步/future/promise/async/await/tokio
  10. (40)Xilinx PLL IP核配置(一)(第8天)