genhd.h 11.4 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 block_device *bdev;
69
	struct device __dev;
70
	struct kobject *holder_dir;
L
Linus Torvalds 已提交
71
	int policy, partno;
72
	struct partition_meta_info *info;
73 74
#ifdef CONFIG_FAIL_MAKE_REQUEST
	int make_it_fail;
75
#endif
76
	struct rcu_work rcu_work;
L
Linus Torvalds 已提交
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 136
/**
 * 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 已提交
137

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

143 144 145 146 147 148 149
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 已提交
150 151 152
struct disk_part_tbl {
	struct rcu_head rcu_head;
	int len;
A
Arnd Bergmann 已提交
153 154
	struct hd_struct __rcu *last_lookup;
	struct hd_struct __rcu *part[];
T
Tejun Heo 已提交
155 156
};

157
struct disk_events;
158
struct badblocks;
159

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

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

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

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

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

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

	int flags;
195 196
	unsigned long state;
#define GD_NEED_PART_SCAN		0
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)
{
C
Christoph Hellwig 已提交
221 222 223
	if (unlikely(!part))
		return NULL;
	return part->bdev->bd_disk;
T
Tejun Heo 已提交
224 225
}

226 227
static inline int disk_max_parts(struct gendisk *disk)
{
228 229 230
	if (disk->flags & GENHD_FL_EXT_DEVT)
		return DISK_MAX_PARTS;
	return disk->minors;
T
Tejun Heo 已提交
231 232
}

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

static inline dev_t disk_devt(struct gendisk *disk)
{
241
	return MKDEV(disk->major, disk->first_minor);
242 243 244 245
}

static inline dev_t part_devt(struct hd_struct *part)
{
246
	return part_to_dev(part)->devt;
247 248
}

249 250 251 252 253
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))
254
		put_device(part_to_dev(part));
255 256
}

257 258 259 260 261 262 263
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
}

264 265 266 267 268
/*
 * 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 已提交
269
#define DISK_PITER_INCL_PART0	(1 << 2) /* include partition 0 */
270
#define DISK_PITER_INCL_EMPTY_PART0 (1 << 3) /* include empty partition 0 */
271 272 273 274 275 276 277 278 279 280 281 282

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);
283
extern bool disk_has_partitions(struct gendisk *disk);
L
Linus Torvalds 已提交
284

285
/* block/genhd.c */
286 287
extern void device_add_disk(struct device *parent, struct gendisk *disk,
			    const struct attribute_group **groups);
288 289
static inline void add_disk(struct gendisk *disk)
{
290
	device_add_disk(NULL, disk, NULL);
291
}
292 293 294 295 296
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);
}
297

L
Linus Torvalds 已提交
298
extern void del_gendisk(struct gendisk *gp);
299
extern struct block_device *bdget_disk(struct gendisk *disk, int partno);
L
Linus Torvalds 已提交
300 301 302

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

T
Tejun Heo 已提交
303 304 305 306 307
static inline int get_disk_ro(struct gendisk *disk)
{
	return disk->part0.policy;
}

308 309
extern void disk_block_events(struct gendisk *disk);
extern void disk_unblock_events(struct gendisk *disk);
310
extern void disk_flush_events(struct gendisk *disk, unsigned int mask);
311
bool set_capacity_and_notify(struct gendisk *disk, sector_t size);
312

L
Linus Torvalds 已提交
313
/* drivers/char/random.c */
314
extern void add_disk_randomness(struct gendisk *disk) __latent_entropy;
L
Linus Torvalds 已提交
315 316 317 318
extern void rand_initialize_disk(struct gendisk *disk);

static inline sector_t get_start_sect(struct block_device *bdev)
{
T
Tejun Heo 已提交
319
	return bdev->bd_part->start_sect;
L
Linus Torvalds 已提交
320 321 322
}
static inline sector_t get_capacity(struct gendisk *disk)
{
323
	return disk->part0.nr_sects;
L
Linus Torvalds 已提交
324 325 326
}
static inline void set_capacity(struct gendisk *disk, sector_t size)
{
327
	disk->part0.nr_sects = size;
L
Linus Torvalds 已提交
328 329
}

330
int bdev_disk_changed(struct block_device *bdev, bool invalidate);
331
int blk_add_partitions(struct gendisk *disk, struct block_device *bdev);
332
int blk_drop_partitions(struct block_device *bdev);
L
Linus Torvalds 已提交
333

334
extern struct gendisk *__alloc_disk_node(int minors, int node_id);
L
Linus Torvalds 已提交
335 336
extern void put_disk(struct gendisk *disk);

337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
#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)

355 356 357 358
int __register_blkdev(unsigned int major, const char *name,
		void (*probe)(dev_t devt));
#define register_blkdev(major, name) \
	__register_blkdev(major, name, NULL)
359 360
void unregister_blkdev(unsigned int major, const char *name);

361
void revalidate_disk_size(struct gendisk *disk, bool verbose);
362
bool bdev_check_media_change(struct block_device *bdev);
363
int __invalidate_device(struct block_device *bdev, bool kill_dirty);
364
void bd_set_nr_sectors(struct block_device *bdev, sector_t sectors);
365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384

/* 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 */

385 386
extern struct rw_semaphore bdev_lookup_sem;

C
Christoph Hellwig 已提交
387
dev_t blk_lookup_devt(const char *name, int partno);
388
void blk_request_module(dev_t devt);
389 390
#ifdef CONFIG_BLOCK
void printk_all_partitions(void);
391
#else /* CONFIG_BLOCK */
392 393 394
static inline void printk_all_partitions(void)
{
}
395
#endif /* CONFIG_BLOCK */
396

397
#endif /* _LINUX_GENHD_H */