RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 766036
Accepted
Ruslan R. Laishev
Ruslan R. Laishev
Asked:2020-01-03 17:46:24 +0000 UTC2020-01-03 17:46:24 +0000 UTC 2020-01-03 17:46:24 +0000 UTC

为什么 bi_end_io 用带有/不带 biovecs 的 bio 调用?

  • 772

我正在制作一个块设备驱动程序,在处理 READ 请求时,我在 bio.bi_end_io 字段中设置了我的处理程序(I/O 完成例程)。正在调用我的处理程序,但 bio 中没有 biovec,我需要在块 I/O 子系统将数据返回给应用程序之前处理数据。这里有一些代码片段来说明。*

static  void    __iob_enc_dec   (
        struct bio *    iob
            )
{
struct bio_vec  bvec = {0};
struct  bvec_iter iter = {0};
sector_t    lbn, nlbn;

    $TRACE("Start %scrypting ...", bio_data_dir(iob) == WRITE ? "En" : "De");

    $SHOW_PTR(bio_data(iob));

    /* Do each segment independently. */
    **bio_for_each_segment**(bvec, iob, iter)
        {
        char *  iobuf;

        iobuf = __bio_kmap_atomic(iob, iter);

        $TRACE("#%02d: page=%p, off=%u, len=%u, iobuf=%p, lbn=%lu",
            iter.bi_idx, bvec.bv_page, bvec.bv_offset, bvec.bv_len, iobuf, iter.bi_sector);

        iobuf += bvec.bv_offset;

        lbn = iter.bi_sector;
        nlbn    = iter.bi_size/DUDRV$K_BLKSZ;

        for ( ;nlbn; nlbn--, lbn++ , iobuf += DUDRV$K_BLKSZ)
            {
            **...**
            }

        __bio_kunmap_atomic(iob);
        }
}


static  void    dua_bio_end_io  (
            struct bio *    iob
                )
{
IOB_ARGS * iob_args;

    $TRACE("Entering to %s completion I/O, bio=%p ...", bio_data_dir(iob) == WRITE ? "WRITE" : "READ", iob);

    $SHOW_UNSIGNED(iob->bi_flags);
    $SHOW_INT(iob->bi_vcnt);

    $SHOW_BOOL(bio_has_data(iob));

    iob_args = iob->bi_private;

    iob->bi_end_io = iob_args->bi_end_io;
    iob->bi_private = iob_args->bi_private;

    bio_put(iob);

    __ret_iob_args (iob_args);

    /* In case of READ request - we should  decrypt data buffer right now */
    if ( bio_data_dir(iob) == READ )
        **__iob_enc_dec(iob)**;

    bio_endio (iob);

}

static blk_qc_t dua_make_request_fn (
        struct request_queue *  ioq,
            struct bio *    iob
                )
{
int status = 0;

    $TRACE("Starting (%s), bio=%p, op=%d ...", bio_data_dir(iob) == WRITE ? "WRITE" : "READ", iob, bio_op(iob));


    $SHOW_UNSIGNED(iob->bi_flags);
    $SHOW_INT(iob->bi_vcnt);

    $SHOW_BOOL(bio_has_data(iob));

    /* In case of WRITE request - we can encrypt data buffer right now */
    if ( bio_data_dir(iob) == WRITE )
        __iob_enc_dec(iob);

    /*
     * A handling of the READ request is require 'enqueue read request' & 'wait for read completion' paradigm,
     * so we need to allocate IOB_ARGS block to carry data to the "Read Completion I/O" routine.
     */
    else if ( bio_data_dir(iob) == READ )
        {
        IOB_ARGS *iob_args = NULL;

        if ( __get_iob_args (&iob_args) )
            {
            printk(KERN_ERR  __MODULE__ ": Buffered I/O quota limit has been exhausted\n");

            iob->bi_error = -EBUSY;
            bio_endio(iob);
            }

        iob_args->bi_end_io = iob->bi_end_io;
        iob_args->bi_private = iob->bi_private;

        /*
         * Replace an address of the Completion I/O routine for 'read' operation,
         * save original address.
         */
        iob->bi_private = iob_args;
        iob->bi_end_io = dua_bio_end_io;

        bio_get(iob);
        }

    /* Just for sanity check ... */
    else    {
        printk(KERN_WARNING  __MODULE__ ": Unhandled I/O request %d\n", bio_data_dir(iob) );
        }


    /* Call original make_reques_fn() to performs a main work ... */
    status = backend_make_request_fn(ioq, iob);

    return  status;
}




[6463.418222] [DUDRIVER\dua_make_request_fn:479] Starting (READ), bio=ffff95c5f9552100, op=0 ...
[ 6463.418222] [DUDRIVER\dua_make_request_fn:482] : iob->bi_flags = 0x00000000
[ 6463.418223] [DUDRIVER\dua_make_request_fn:483] : iob->bi_vcnt = 1
[ 6463.418223] [DUDRIVER\dua_make_request_fn:485] : bio_has_data(iob) = ENABLED(TRUE)

[ 6463.418266] [DUDRIVER\dua_bio_end_io:445] Entering to READ completion I/O, bio=ffff95c5f9552100 ...
[ 6463.418266] [DUDRIVER\dua_bio_end_io:447] : iob->bi_flags = 0x00000102
[ 6463.418267] [DUDRIVER\dua_bio_end_io:448] : iob->bi_vcnt = 1
[ 6463.418267] [DUDRIVER\dua_bio_end_io:450] : bio_has_data(iob) = DISABLED(FALSE)

我需要做什么才能访问 bio_end_io() 中的数据缓冲区?这发生在 Ubuntu 16.x、Linux 内核 4.10.y 中。

更新1:设置了 BIO_CLONED 标志的 bio_for_each_segment 宏可能不允许您运行缓冲区段,但那该怎么办?

linux
  • 1 1 个回答
  • 10 Views

1 个回答

  • Voted
  1. Best Answer
    Ruslan R. Laishev
    2020-01-05T17:01:46Z2020-01-05T17:01:46Z
    static  void    __iob_enc_dec   (
            struct bio *    iob,
            sector_t    lbn
                )
    {
    struct bio_vec  *bvl;
    sector_t    nlbn;
    int i;
    
        $TRACE("Start %scrypting ...", bio_data_dir(iob) == WRITE ? "En" : "De");
    
    #if 0
    
        {
        for (i = 0, bvl = iob->bi_io_vec; i < iob->bi_vcnt; i++, bvl++)
            {
            $SHOW_PTR(bvl->bv_page);
            $SHOW_INT(bvl->bv_len);
            $SHOW_INT(bvl->bv_offset);
            }
        }
    
        /* Do each segment independently. */
        bio_for_each_segment(bvec, iob, iter)
            {
            char *  iobuf;
    
            iobuf = __bio_kmap_atomic(iob, iter);
    
            $TRACE("#%02d: page=%p, off=%u, len=%u, iobuf=%p, lbn=%lu",
                iter.bi_idx, bvec.bv_page, bvec.bv_offset, bvec.bv_len, iobuf, iter.bi_sector);
    
            iobuf += bvec.bv_offset;
    
            lbn = iter.bi_sector;
            nlbn    = iter.bi_size/DUDRV$K_BLKSZ;
    
            for ( ;nlbn; nlbn--, lbn++ , iobuf += DUDRV$K_BLKSZ)
                {
                bio_data_dir(iob) == WRITE
                        ? __encrypt (&gost89_ctx, lbn, iobuf, iobuf)
                        : __decrypt (&gost89_ctx, lbn, iobuf, iobuf);
                }
    
            __bio_kunmap_atomic(iob);
        }
    #endif
    
        /* Do each segment independently. */
        **for (i = 0, bvl = iob->bi_io_vec; i < iob->bi_vcnt; i++, bvl++)**
            {
            void     *kaddr = kmap_atomic(bvl->bv_page), *iobuf;
    
            iobuf   = kaddr + bvl->bv_offset;
    
            nlbn    = bvl->bv_len/DUDRV$K_BLKSZ;
    
            $TRACE("#%02d: page=%p, off=%u, len=%u, iobuf=%p, lbn=%lu, nlbn=%lu",
                i, bvl->bv_page, bvl->bv_offset, bvl->bv_len, iobuf, lbn, nlbn);
    
            for ( ;nlbn; nlbn--, lbn++ , iobuf += DUDRV$K_BLKSZ)
                {
                if ( bio_data_dir(iob) == WRITE )
                    __encrypt (&gost89_ctx, lbn, iobuf, iobuf);
                else    __decrypt (&gost89_ctx, lbn, iobuf, iobuf);
                }
    
            kunmap_atomic(kaddr);
        }
    }
    

    解决方案是用老式的向量运行替换愚蠢的宏。对于我的任务,我需要磁盘块的起始扇区号,它在 IOB_ARGS 上下文块中传输,但这些都是细节。

    • 0

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    是否可以在 C++ 中继承类 <---> 结构?

    • 2 个回答
  • Marko Smith

    这种神经网络架构适合文本分类吗?

    • 1 个回答
  • Marko Smith

    为什么分配的工作方式不同?

    • 3 个回答
  • Marko Smith

    控制台中的光标坐标

    • 1 个回答
  • Marko Smith

    如何在 C++ 中删除类的实例?

    • 4 个回答
  • Marko Smith

    点是否属于线段的问题

    • 2 个回答
  • Marko Smith

    json结构错误

    • 1 个回答
  • Marko Smith

    ServiceWorker 中的“获取”事件

    • 1 个回答
  • Marko Smith

    c ++控制台应用程序exe文件[重复]

    • 1 个回答
  • Marko Smith

    按多列从sql表中选择

    • 1 个回答
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Suvitruf - Andrei Apanasik 什么是空? 2020-08-21 01:48:09 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5