genhd.h 12.2 KB
Newer Older
1
/* SPDX-License-Identifier: GPL-2.0 */
L
Linus Torvalds 已提交
2 3 4 5 6 7 8 9 10 11 12 13
#ifndef _LINUX_GENHD_H
#define _LINUX_GENHD_H

/*
 * 	genhd.h Copyright (C) 1992 Drew Eckhardt
 *	Generic hard disk header file by  
 * 		Drew Eckhardt
 *
 *		<drew@colorado.edu>
 */

#include <linux/types.h>
14
#include <linux/kdev_t.h>
15
#include <linux/rcupdate.h>
16
#include <linux/slab.h>
M
Ming Lei 已提交
17
#include <linux/percpu-refcount.h>
18
#include <linux/uuid.h>
19
#include <linux/blk_types.h>
20
#include <asm/local.h>
L
Linus Torvalds 已提交
21

T
Tejun Heo 已提交
22
#define dev_to_disk(device)	container_of((device), struct gendisk, part0.__dev)
23
#define dev_to_part(device)	container_of((device), struct hd_struct, __dev)
T
Tejun Heo 已提交
24
#define disk_to_dev(disk)	(&(disk)->part0.__dev)
25
#define part_to_dev(part)	(&((part)->__dev))
26

27
extern const struct device_type disk_type;
28 29 30
extern struct device_type part_type;
extern struct class block_class;

31
#define DISK_MAX_PARTS			256
32
#define DISK_NAME_LEN			32
33

34 35 36 37 38
#include <linux/major.h>
#include <linux/device.h>
#include <linux/smp.h>
#include <linux/string.h>
#include <linux/fs.h>
39
#include <linux/workqueue.h>
40

41
#define PARTITION_META_INFO_VOLNAMELTH	64
42 43 44 45
/*
 * Enough for the string representation of any kind of UUID plus NULL.
 * EFI UUID is 36 characters. MSDOS UUID is 11 characters.
 */
46
#define PARTITION_META_INFO_UUIDLTH	(UUID_STRING_LEN + 1)
47 48

struct partition_meta_info {
49
	char uuid[PARTITION_META_INFO_UUIDLTH];
50 51 52
	u8 volname[PARTITION_META_INFO_VOLNAMELTH];
};

L
Linus Torvalds 已提交
53 54
struct hd_struct {
	sector_t start_sect;
55 56 57 58 59
	/*
	 * nr_sects is protected by sequence counter. One might extend a
	 * partition while IO is happening to it and update of nr_sects
	 * can be non-atomic on 32bit machines with 64bit sector_t.
	 */
L
Linus Torvalds 已提交
60
	sector_t nr_sects;
61
#if BITS_PER_LONG==32 && defined(CONFIG_SMP)
62
	seqcount_t nr_sects_seq;
63
#endif
64 65 66 67
	unsigned long stamp;
	struct disk_stats __percpu *dkstats;
	struct percpu_ref ref;

68
	struct device __dev;
69
	struct kobject *holder_dir;
L
Linus Torvalds 已提交
70
	int policy, partno;
71
	struct partition_meta_info *info;
72 73
#ifdef CONFIG_FAIL_MAKE_REQUEST
	int make_it_fail;
74
#endif
75
	struct rcu_work rcu_work;
L
Linus Torvalds 已提交
76 77
};

78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
/**
 * DOC: genhd capability flags
 *
 * ``GENHD_FL_REMOVABLE`` (0x0001): indicates that the block device
 * gives access to removable media.
 * When set, the device remains present even when media is not
 * inserted.
 * Must not be set for devices which are removed entirely when the
 * media is removed.
 *
 * ``GENHD_FL_CD`` (0x0008): the block device is a CD-ROM-style
 * device.
 * Affects responses to the ``CDROM_GET_CAPABILITY`` ioctl.
 *
 * ``GENHD_FL_UP`` (0x0010): indicates that the block device is "up",
 * with a similar meaning to network interfaces.
 *
 * ``GENHD_FL_SUPPRESS_PARTITION_INFO`` (0x0020): don't include
 * partition information in ``/proc/partitions`` or in the output of
 * printk_all_partitions().
 * Used for the null block device and some MMC devices.
 *
 * ``GENHD_FL_EXT_DEVT`` (0x0040): the driver supports extended
 * dynamic ``dev_t``, i.e. it wants extended device numbers
 * (``BLOCK_EXT_MAJOR``).
 * This affects the maximum number of partitions.
 *
 * ``GENHD_FL_NATIVE_CAPACITY`` (0x0080): based on information in the
 * partition table, the device's capacity has been extended to its
 * native capacity; i.e. the device has hidden capacity used by one
 * of the partitions (this is a flag used so that native capacity is
 * only ever unlocked once).
 *
 * ``GENHD_FL_BLOCK_EVENTS_ON_EXCL_WRITE`` (0x0100): event polling is
 * blocked whenever a writer holds an exclusive lock.
 *
 * ``GENHD_FL_NO_PART_SCAN`` (0x0200): partition scanning is disabled.
 * Used for loop devices in their default settings and some MMC
 * devices.
 *
 * ``GENHD_FL_HIDDEN`` (0x0400): the block device is hidden; it
 * doesn't produce events, doesn't appear in sysfs, and doesn't have
 * an associated ``bdev``.
 * Implies ``GENHD_FL_SUPPRESS_PARTITION_INFO`` and
 * ``GENHD_FL_NO_PART_SCAN``.
 * Used for multipath devices.
 */
#define GENHD_FL_REMOVABLE			0x0001
/* 2 is unused (used to be GENHD_FL_DRIVERFS) */
/* 4 is unused (used to be GENHD_FL_MEDIA_CHANGE_NOTIFY) */
#define GENHD_FL_CD				0x0008
#define GENHD_FL_UP				0x0010
#define GENHD_FL_SUPPRESS_PARTITION_INFO	0x0020
#define GENHD_FL_EXT_DEVT			0x0040
#define GENHD_FL_NATIVE_CAPACITY		0x0080
#define GENHD_FL_BLOCK_EVENTS_ON_EXCL_WRITE	0x0100
#define GENHD_FL_NO_PART_SCAN			0x0200
#define GENHD_FL_HIDDEN				0x0400
L
Linus Torvalds 已提交
136

137 138 139 140 141
enum {
	DISK_EVENT_MEDIA_CHANGE			= 1 << 0, /* media changed */
	DISK_EVENT_EJECT_REQUEST		= 1 << 1, /* eject requested */
};

142 143 144 145 146 147 148
enum {
	/* Poll even if events_poll_msecs is unset */
	DISK_EVENT_FLAG_POLL			= 1 << 0,
	/* Forward events to udev */
	DISK_EVENT_FLAG_UEVENT			= 1 << 1,
};

T
Tejun Heo 已提交
149 150 151
struct disk_part_tbl {
	struct rcu_head rcu_head;
	int len;
A
Arnd Bergmann 已提交
152 153
	struct hd_struct __rcu *last_lookup;
	struct hd_struct __rcu *part[];
T
Tejun Heo 已提交
154 155
};

156
struct disk_events;
157
struct badblocks;
158

159
struct blk_integrity {
160 161 162 163 164
	const struct blk_integrity_profile	*profile;
	unsigned char				flags;
	unsigned char				tuple_size;
	unsigned char				interval_exp;
	unsigned char				tag_size;
165 166
};

L
Linus Torvalds 已提交
167
struct gendisk {
168 169
	/* major, first_minor and minors are input parameters only,
	 * don't use directly.  Use disk_devt() and disk_max_parts().
170
	 */
L
Linus Torvalds 已提交
171 172 173 174
	int major;			/* major number of driver */
	int first_minor;
	int minors;                     /* maximum number of minors, =1 for
                                         * disks that can't be partitioned. */
175

176
	char disk_name[DISK_NAME_LEN];	/* name of major driver */
177

178 179
	unsigned short events;		/* supported events */
	unsigned short event_flags;	/* flags related to event processing */
180

T
Tejun Heo 已提交
181
	/* Array of pointers to partitions indexed by partno.
182 183 184 185
	 * Protected with matching bdev lock but stat and other
	 * non-critical accesses use RCU.  Always access through
	 * helpers.
	 */
A
Arnd Bergmann 已提交
186
	struct disk_part_tbl __rcu *part_tbl;
T
Tejun Heo 已提交
187
	struct hd_struct part0;
188

189
	const struct block_device_operations *fops;
L
Linus Torvalds 已提交
190 191 192 193
	struct request_queue *queue;
	void *private_data;

	int flags;
194 195
	unsigned long state;
#define GD_NEED_PART_SCAN		0
J
Jan Kara 已提交
196
	struct rw_semaphore lookup_sem;
197
	struct kobject *slave_dir;
L
Linus Torvalds 已提交
198 199 200

	struct timer_rand_state *random;
	atomic_t sync_io;		/* RAID */
201
	struct disk_events *ev;
202
#ifdef  CONFIG_BLK_DEV_INTEGRITY
203
	struct kobject integrity_kobj;
204
#endif	/* CONFIG_BLK_DEV_INTEGRITY */
205 206 207
#if IS_ENABLED(CONFIG_CDROM)
	struct cdrom_device_info *cdi;
#endif
T
Tejun Heo 已提交
208
	int node_id;
209
	struct badblocks *bb;
210
	struct lockdep_map lockdep_map;
L
Linus Torvalds 已提交
211 212
};

213 214 215 216 217 218
#if IS_REACHABLE(CONFIG_CDROM)
#define disk_to_cdi(disk)	((disk)->cdi)
#else
#define disk_to_cdi(disk)	NULL
#endif

T
Tejun Heo 已提交
219 220
static inline struct gendisk *part_to_disk(struct hd_struct *part)
{
T
Tejun Heo 已提交
221 222 223 224 225 226
	if (likely(part)) {
		if (part->partno)
			return dev_to_disk(part_to_dev(part)->parent);
		else
			return dev_to_disk(part_to_dev(part));
	}
T
Tejun Heo 已提交
227 228 229
	return NULL;
}

230 231
static inline int disk_max_parts(struct gendisk *disk)
{
232 233 234
	if (disk->flags & GENHD_FL_EXT_DEVT)
		return DISK_MAX_PARTS;
	return disk->minors;
T
Tejun Heo 已提交
235 236
}

T
Tejun Heo 已提交
237
static inline bool disk_part_scan_enabled(struct gendisk *disk)
T
Tejun Heo 已提交
238
{
T
Tejun Heo 已提交
239 240
	return disk_max_parts(disk) > 1 &&
		!(disk->flags & GENHD_FL_NO_PART_SCAN);
241 242 243 244
}

static inline dev_t disk_devt(struct gendisk *disk)
{
245
	return MKDEV(disk->major, disk->first_minor);
246 247 248 249
}

static inline dev_t part_devt(struct hd_struct *part)
{
250
	return part_to_dev(part)->devt;
251 252
}

253
extern struct hd_struct *__disk_get_part(struct gendisk *disk, int partno);
254 255 256 257 258
extern struct hd_struct *disk_get_part(struct gendisk *disk, int partno);

static inline void disk_put_part(struct hd_struct *part)
{
	if (likely(part))
259
		put_device(part_to_dev(part));
260 261
}

262 263 264 265 266 267 268
static inline void hd_sects_seq_init(struct hd_struct *p)
{
#if BITS_PER_LONG==32 && defined(CONFIG_SMP)
	seqcount_init(&p->nr_sects_seq);
#endif
}

269 270 271 272 273
/*
 * Smarter partition iterator without context limits.
 */
#define DISK_PITER_REVERSE	(1 << 0) /* iterate in the reverse direction */
#define DISK_PITER_INCL_EMPTY	(1 << 1) /* include 0-sized parts */
T
Tejun Heo 已提交
274
#define DISK_PITER_INCL_PART0	(1 << 2) /* include partition 0 */
275
#define DISK_PITER_INCL_EMPTY_PART0 (1 << 3) /* include empty partition 0 */
276 277 278 279 280 281 282 283 284 285 286 287

struct disk_part_iter {
	struct gendisk		*disk;
	struct hd_struct	*part;
	int			idx;
	unsigned int		flags;
};

extern void disk_part_iter_init(struct disk_part_iter *piter,
				 struct gendisk *disk, unsigned int flags);
extern struct hd_struct *disk_part_iter_next(struct disk_part_iter *piter);
extern void disk_part_iter_exit(struct disk_part_iter *piter);
288
extern bool disk_has_partitions(struct gendisk *disk);
L
Linus Torvalds 已提交
289

290
/* block/genhd.c */
291 292
extern void device_add_disk(struct device *parent, struct gendisk *disk,
			    const struct attribute_group **groups);
293 294
static inline void add_disk(struct gendisk *disk)
{
295
	device_add_disk(NULL, disk, NULL);
296
}
297 298 299 300 301
extern void device_add_disk_no_queue_reg(struct device *parent, struct gendisk *disk);
static inline void add_disk_no_queue_reg(struct gendisk *disk)
{
	device_add_disk_no_queue_reg(NULL, disk);
}
302

L
Linus Torvalds 已提交
303
extern void del_gendisk(struct gendisk *gp);
304
extern struct gendisk *get_gendisk(dev_t dev, int *partno);
305
extern struct block_device *bdget_disk(struct gendisk *disk, int partno);
L
Linus Torvalds 已提交
306 307 308 309

extern void set_device_ro(struct block_device *bdev, int flag);
extern void set_disk_ro(struct gendisk *disk, int flag);

T
Tejun Heo 已提交
310 311 312 313 314
static inline int get_disk_ro(struct gendisk *disk)
{
	return disk->part0.policy;
}

315 316
extern void disk_block_events(struct gendisk *disk);
extern void disk_unblock_events(struct gendisk *disk);
317
extern void disk_flush_events(struct gendisk *disk, unsigned int mask);
318
bool set_capacity_revalidate_and_notify(struct gendisk *disk, sector_t size,
319
		bool update_bdev);
320

L
Linus Torvalds 已提交
321
/* drivers/char/random.c */
322
extern void add_disk_randomness(struct gendisk *disk) __latent_entropy;
L
Linus Torvalds 已提交
323 324 325 326
extern void rand_initialize_disk(struct gendisk *disk);

static inline sector_t get_start_sect(struct block_device *bdev)
{
T
Tejun Heo 已提交
327
	return bdev->bd_part->start_sect;
L
Linus Torvalds 已提交
328 329 330
}
static inline sector_t get_capacity(struct gendisk *disk)
{
331
	return disk->part0.nr_sects;
L
Linus Torvalds 已提交
332 333 334
}
static inline void set_capacity(struct gendisk *disk, sector_t size)
{
335
	disk->part0.nr_sects = size;
L
Linus Torvalds 已提交
336 337
}

338
int bdev_disk_changed(struct block_device *bdev, bool invalidate);
339
int blk_add_partitions(struct gendisk *disk, struct block_device *bdev);
340
int blk_drop_partitions(struct block_device *bdev);
L
Linus Torvalds 已提交
341

342
extern struct gendisk *__alloc_disk_node(int minors, int node_id);
343
extern struct kobject *get_disk_and_module(struct gendisk *disk);
L
Linus Torvalds 已提交
344
extern void put_disk(struct gendisk *disk);
345
extern void put_disk_and_module(struct gendisk *disk);
346
extern void blk_register_region(dev_t devt, unsigned long range,
L
Linus Torvalds 已提交
347 348 349 350
			struct module *module,
			struct kobject *(*probe)(dev_t, int *, void *),
			int (*lock)(dev_t, void *),
			void *data);
351
extern void blk_unregister_region(dev_t devt, unsigned long range);
352 353
extern void blk_delete_region(dev_t devt, unsigned long range,
			struct kobject *(*probe)(dev_t, int *, void *));
L
Linus Torvalds 已提交
354

355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372
#define alloc_disk_node(minors, node_id)				\
({									\
	static struct lock_class_key __key;				\
	const char *__name;						\
	struct gendisk *__disk;						\
									\
	__name = "(gendisk_completion)"#minors"("#node_id")";		\
									\
	__disk = __alloc_disk_node(minors, node_id);			\
									\
	if (__disk)							\
		lockdep_init_map(&__disk->lockdep_map, __name, &__key, 0); \
									\
	__disk;								\
})

#define alloc_disk(minors) alloc_disk_node(minors, NUMA_NO_NODE)

373 374 375
int register_blkdev(unsigned int major, const char *name);
void unregister_blkdev(unsigned int major, const char *name);

376
void revalidate_disk_size(struct gendisk *disk, bool verbose);
377
bool bdev_check_media_change(struct block_device *bdev);
378
int __invalidate_device(struct block_device *bdev, bool kill_dirty);
379
void bd_set_nr_sectors(struct block_device *bdev, sector_t sectors);
380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399

/* for drivers/char/raw.c: */
int blkdev_ioctl(struct block_device *, fmode_t, unsigned, unsigned long);
long compat_blkdev_ioctl(struct file *, unsigned, unsigned long);

#ifdef CONFIG_SYSFS
int bd_link_disk_holder(struct block_device *bdev, struct gendisk *disk);
void bd_unlink_disk_holder(struct block_device *bdev, struct gendisk *disk);
#else
static inline int bd_link_disk_holder(struct block_device *bdev,
				      struct gendisk *disk)
{
	return 0;
}
static inline void bd_unlink_disk_holder(struct block_device *bdev,
					 struct gendisk *disk)
{
}
#endif /* CONFIG_SYSFS */

400 401 402
#ifdef CONFIG_BLOCK
void printk_all_partitions(void);
dev_t blk_lookup_devt(const char *name, int partno);
403
#else /* CONFIG_BLOCK */
404 405 406
static inline void printk_all_partitions(void)
{
}
407
static inline dev_t blk_lookup_devt(const char *name, int partno)
408 409 410 411
{
	dev_t devt = MKDEV(0, 0);
	return devt;
}
412
#endif /* CONFIG_BLOCK */
413

414
#endif /* _LINUX_GENHD_H */