// SPDX-License-Identifier: GPL-2.0 /* * Disk events - monitor disk events like media change and eject request.
*/ #include <linux/export.h> #include <linux/moduleparam.h> #include <linux/blkdev.h> #include"blk.h"
struct disk_events { struct list_head node; /* all disk_event's */ struct gendisk *disk; /* the associated disk */
spinlock_t lock;
struct mutex block_mutex; /* protects blocking */ int block; /* event blocking depth */ unsignedint pending; /* events already sent out */ unsignedint clearing; /* events being cleared */
long poll_msecs; /* interval, -1 for default */ struct delayed_work dwork;
};
/* * If device-specific poll interval is set, always use it. If * the default is being used, poll if the POLL flag is set.
*/ if (ev->poll_msecs >= 0)
intv_msecs = ev->poll_msecs; elseif (disk->event_flags & DISK_EVENT_FLAG_POLL)
intv_msecs = disk_events_dfl_poll_msecs;
return msecs_to_jiffies(intv_msecs);
}
/** * disk_block_events - block and flush disk event checking * @disk: disk to block events for * * On return from this function, it is guaranteed that event checking * isn't in progress and won't happen until unblocked by * disk_unblock_events(). Events blocking is counted and the actual * unblocking happens after the matching number of unblocks are done. * * Note that this intentionally does not block event checking from * disk_clear_events(). * * CONTEXT: * Might sleep.
*/ void disk_block_events(struct gendisk *disk)
{ struct disk_events *ev = disk->ev; unsignedlong flags; bool cancel;
if (!ev) return;
/* * Outer mutex ensures that the first blocker completes canceling * the event work before further blockers are allowed to finish.
*/
mutex_lock(&ev->block_mutex);
/** * disk_unblock_events - unblock disk event checking * @disk: disk to unblock events for * * Undo disk_block_events(). When the block count reaches zero, it * starts events polling if configured. * * CONTEXT: * Don't care. Safe to call from irq context.
*/ void disk_unblock_events(struct gendisk *disk)
{ if (disk->ev)
__disk_unblock_events(disk, false);
}
/** * disk_flush_events - schedule immediate event checking and flushing * @disk: disk to check and flush events for * @mask: events to flush * * Schedule immediate event checking on @disk if not blocked. Events in * @mask are scheduled to be cleared from the driver. Note that this * doesn't clear the events from @disk->ev. * * CONTEXT: * If @mask is non-zero must be called with disk->open_mutex held.
*/ void disk_flush_events(struct gendisk *disk, unsignedint mask)
{ struct disk_events *ev = disk->ev;
/* * Tell userland about new events. Only the events listed in @disk->events are * reported, and only if DISK_EVENT_FLAG_UEVENT is set. Otherwise, events are * processed internally but never get reported to userland.
*/ staticvoid disk_event_uevent(struct gendisk *disk, unsignedint events)
{ char *envp[ARRAY_SIZE(disk_uevents) + 1] = { }; int nr_events = 0, i;
for (i = 0; i < ARRAY_SIZE(disk_uevents); i++) if (events & disk->events & (1 << i))
envp[nr_events++] = disk_uevents[i];
if (nr_events)
kobject_uevent_env(&disk_to_dev(disk)->kobj, KOBJ_CHANGE, envp);
}
intv = disk_events_poll_jiffies(disk); if (!ev->block && intv)
queue_delayed_work(system_freezable_power_efficient_wq,
&ev->dwork, intv);
spin_unlock_irq(&ev->lock);
if (events & DISK_EVENT_MEDIA_CHANGE)
inc_diskseq(disk);
if (disk->event_flags & DISK_EVENT_FLAG_UEVENT)
disk_event_uevent(disk, events);
}
/** * disk_clear_events - synchronously check, clear and return pending events * @disk: disk to fetch and clear events from * @mask: mask of events to be fetched and cleared * * Disk events are synchronously checked and pending events in @mask * are cleared and returned. This ignores the block count. * * CONTEXT: * Might sleep.
*/ staticunsignedint disk_clear_events(struct gendisk *disk, unsignedint mask)
{ struct disk_events *ev = disk->ev; unsignedint pending; unsignedint clearing = mask;
if (!ev) return 0;
disk_block_events(disk);
/* * store the union of mask and ev->clearing on the stack so that the * race with disk_flush_events does not cause ambiguity (ev->clearing * can still be modified even if events are blocked).
*/
spin_lock_irq(&ev->lock);
clearing |= ev->clearing;
ev->clearing = 0;
spin_unlock_irq(&ev->lock);
disk_check_events(ev, &clearing); /* * if ev->clearing is not 0, the disk_flush_events got called in the * middle of this function, so we want to run the workfn without delay.
*/
__disk_unblock_events(disk, ev->clearing ? true : false);
/** * disk_check_media_change - check if a removable media has been changed * @disk: gendisk to check * * Returns %true and marks the disk for a partition rescan whether a removable * media has been changed, and %false if the media did not change.
*/ bool disk_check_media_change(struct gendisk *disk)
{ unsignedint events;
/** * disk_force_media_change - force a media change event * @disk: the disk which will raise the event * * Should be called when the media changes for @disk. Generates a uevent * and attempts to free all dentries and inodes and invalidates all block * device page cache entries in that case.
*/ void disk_force_media_change(struct gendisk *disk)
{
disk_event_uevent(disk, DISK_EVENT_MEDIA_CHANGE);
inc_diskseq(disk);
bdev_mark_dead(disk->part0, true);
set_bit(GD_NEED_PART_SCAN, &disk->state);
}
EXPORT_SYMBOL_GPL(disk_force_media_change);
/* * Separate this part out so that a different pointer for clearing_ptr can be * passed in for disk_clear_events.
*/ staticvoid disk_events_workfn(struct work_struct *work)
{ struct delayed_work *dwork = to_delayed_work(work); struct disk_events *ev = container_of(dwork, struct disk_events, dwork);
disk_check_events(ev, &ev->clearing);
}
/* * A disk events enabled device has the following sysfs nodes under * its /sys/block/X/ directory. * * events : list of all supported events * events_async : list of events which can be detected w/o polling * (always empty, only for backwards compatibility) * events_poll_msecs : polling interval, 0: disable, -1: system default
*/ static ssize_t __disk_events_show(unsignedint events, char *buf)
{ constchar *delim = "";
ssize_t pos = 0; int i;
for (i = 0; i < ARRAY_SIZE(disk_events_strs); i++) if (events & (1 << i)) {
pos += sprintf(buf + pos, "%s%s",
delim, disk_events_strs[i]);
delim = " ";
} if (pos)
pos += sprintf(buf + pos, "\n"); return pos;
}
/* * The default polling interval can be specified by the kernel * parameter block.events_dfl_poll_msecs which defaults to 0 * (disable). This can also be modified runtime by writing to * /sys/module/block/parameters/events_dfl_poll_msecs.
*/ staticint disk_events_set_dfl_poll_msecs(constchar *val, conststruct kernel_param *kp)
{ struct disk_events *ev; int ret;
ret = param_set_ulong(val, kp); if (ret < 0) return ret;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.