2.6.32内核sendfile的两个bug

在2.6.32内核中,sendfile存在两个已知的bug,其中一个有关安全性,是提权bug。比如http://hi.baidu.com/_kouu/item/b74558542f6b9ca9acc857d0,就详细的提到了。

另一个很少被人发现的bug是,当sendfile发送数据时的偏移(即offset参数)超过300G后,就会返回value too large的错误码。这是由于内核中的sendfile函数中对offset进行合法性检查时,判断出错.

在sendfile的系统调用中:

SYSCALL_DEFINE4(sendfile64, int, out_fd, int, in_fd, loff_t __user *, offset, size_t, count)
{
        loff_t pos;
        ssize_t ret;

        if (offset) {  /* 如果起始偏移不是0 */
                if (unlikely(copy_from_user(&pos, offset, sizeof(loff_t))))
                        return -EFAULT;
                ret = do_sendfile(out_fd, in_fd, &pos, count, 0);   /* 注意,最后一个参数为0 */
                if (unlikely(put_user(pos, offset)))
                        return -EFAULT;
                return ret;
        }

        return do_sendfile(out_fd, in_fd, NULL, count, 0);
}


static ssize_t do_sendfile(int out_fd, int in_fd, loff_t *ppos,
                           size_t count, loff_t max)   /* 当ppos的值不为空时,从系统调用传进来的max知为零 */
{
...

        if (!max)

/* bug发生在此处,sendfile的本意是想找出两个fd对应的磁盘空间总大小中最小的那个,但是由于out_inode是一个socket的文件描述符,所以这么比较是有误的,肯定会导致后面的判断出错 */

                max = min(in_inode->i_sb->s_maxbytes, out_inode->i_sb->s_maxbytes);

        pos = *ppos;
        if (unlikely(pos + count > max)) {
                retval = -EOVERFLOW;
                if (pos >= max)
                        goto fput_out;
                count = max - pos;
        }

...

}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值