summaryrefslogtreecommitdiff
path: root/include/linux/exportfs_block.h
blob: de519b7b599b3e182986c474090b239f2a9a05e9 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
/* SPDX-License-Identifier: GPL-2.0 */
/*
 * Copyright (c) 2014-2026 Christoph Hellwig.
 *
 * Support for exportfs-based layout grants for direct block device access.
 */
#ifndef LINUX_EXPORTFS_BLOCK_H
#define LINUX_EXPORTFS_BLOCK_H 1

#include <linux/blkdev.h>
#include <linux/exportfs.h>
#include <linux/fs.h>

struct inode;
struct iomap;
struct super_block;

/*
 * There are the two types of block-style layout support:
 *  - In-band implies a device identified by a unique cookie inside the actual
 *    device address space checked by the ->get_uuid method as used by the pNFS
 *    block layout.  This is a bit dangerous and deprecated.
 *  - Out of band implies identification by out of band unique identifiers
 *    specified by the storage protocol, which is much safer and used by the
 *    pNFS SCSI/NVMe layouts.
 */
typedef unsigned int __bitwise expfs_block_layouts_t;
#define EXPFS_BLOCK_FLAG(__bit) \
	((__force expfs_block_layouts_t)(1u << __bit))
#define EXPFS_BLOCK_IN_BAND_ID		EXPFS_BLOCK_FLAG(0)
#define EXPFS_BLOCK_OUT_OF_BAND_ID	EXPFS_BLOCK_FLAG(1)

struct exportfs_block_ops {
	/*
	 * Returns the EXPFS_BLOCK_* bitmap of supported layout types.
	 */
	expfs_block_layouts_t (*layouts_supported)(struct super_block *sb);

	/*
	 * Get the in-band device unique signature exposed to clients.
	 */
	int (*get_uuid)(struct super_block *sb, u8 *buf, u32 *len, u64 *offset);

	/*
	 * Map blocks for direct block access.
	 * If @write is %true, also allocate the blocks for the range if needed.
	 */
	int (*map_blocks)(struct inode *inode, loff_t offset, u64 len,
			struct iomap *iomap, bool write,
			u32 *device_generation);

	/*
	 * Commit blocks previously handed out by ->map_blocks and written to by
	 * the client.
	 */
	int (*commit_blocks)(struct inode *inode, struct iomap *iomaps,
			int nr_iomaps, loff_t new_size);
};

static inline bool
exportfs_bdev_supports_out_of_band_id(struct block_device *bdev)
{
	return bdev->bd_disk->fops->pr_ops &&
		bdev->bd_disk->fops->get_unique_id;
}

#ifdef CONFIG_EXPORTFS_BLOCK_OPS
static inline expfs_block_layouts_t
exportfs_layouts_supported(struct super_block *sb)
{
	const struct exportfs_block_ops *bops = sb->s_export_op->block_ops;

	if (!bops ||
	    !bops->layouts_supported ||
	    WARN_ON_ONCE(!bops->map_blocks) ||
	    WARN_ON_ONCE(!bops->commit_blocks))
		return 0;
	return bops->layouts_supported(sb);
}
#else
static inline expfs_block_layouts_t
exportfs_layouts_supported(struct super_block *sb)
{
	return 0;
}
#endif /* CONFIG_EXPORTFS_BLOCK_OPS */

#endif /* LINUX_EXPORTFS_BLOCK_H */