// SPDX-License-Identifier: GPL-2.0-or-later /* * RDMA Network Block Driver * * Copyright (c) 2014 - 2018 ProfitBricks GmbH. All rights reserved. * Copyright (c) 2018 - 2019 1&1 IONOS Cloud GmbH. All rights reserved. * Copyright (c) 2019 - 2020 1&1 IONOS SE. All rights reserved. */ #undef pr_fmt #define pr_fmt(fmt) KBUILD_MODNAME " L" __stringify(__LINE__) ": " fmt #include "rnbd-srv-dev.h" #include "rnbd-log.h" struct rnbd_dev *rnbd_dev_open(const char *path, fmode_t flags, struct bio_set *bs) { struct rnbd_dev *dev; int ret; dev = kzalloc(sizeof(*dev), GFP_KERNEL); if (!dev) return ERR_PTR(-ENOMEM); dev->blk_open_flags = flags; dev->bdev = blkdev_get_by_path(path, flags, THIS_MODULE); ret = PTR_ERR_OR_ZERO(dev->bdev); if (ret) goto err; dev->blk_open_flags = flags; bdevname(dev->bdev, dev->name); dev->ibd_bio_set = bs; return dev; err: kfree(dev); return ERR_PTR(ret); } void rnbd_dev_close(struct rnbd_dev *dev) { blkdev_put(dev->bdev, dev->blk_open_flags); kfree(dev); } void rnbd_dev_bi_end_io(struct bio *bio) { struct rnbd_dev_blk_io *io = bio->bi_private; rnbd_endio(io->priv, blk_status_to_errno(bio->bi_status)); bio_put(bio); } /** * rnbd_bio_map_kern - map kernel address into bio * @data: pointer to buffer to map * @bs: bio_set to use. * @len: length in bytes * @gfp_mask: allocation flags for bio allocation * * Map the kernel address into a bio suitable for io to a block * device. Returns an error pointer in case of error. */ struct bio *rnbd_bio_map_kern(void *data, struct bio_set *bs, unsigned int len, gfp_t gfp_mask) { unsigned long kaddr = (unsigned long)data; unsigned long end = (kaddr + len + PAGE_SIZE - 1) >> PAGE_SHIFT; unsigned long start = kaddr >> PAGE_SHIFT; const int nr_pages = end - start; int offset, i; struct bio *bio; bio = bio_alloc_bioset(gfp_mask, nr_pages, bs); if (!bio) return ERR_PTR(-ENOMEM); offset = offset_in_page(kaddr); for (i = 0; i < nr_pages; i++) { unsigned int bytes = PAGE_SIZE - offset; if (len <= 0) break; if (bytes > len) bytes = len; if (bio_add_page(bio, virt_to_page(data), bytes, offset) < bytes) { /* we don't support partial mappings */ bio_put(bio); return ERR_PTR(-EINVAL); } data += bytes; len -= bytes; offset = 0; } return bio; }