| 1 | /* SPDX-License-Identifier: GPL-2.0 */ | 
|---|---|
| 2 | #ifndef IOPRIO_H | 
| 3 | #define IOPRIO_H | 
| 4 | |
| 5 | #include <linux/sched.h> | 
| 6 | #include <linux/sched/rt.h> | 
| 7 | #include <linux/iocontext.h> | 
| 8 | |
| 9 | #include <uapi/linux/ioprio.h> | 
| 10 | |
| 11 | /* | 
| 12 | * Default IO priority. | 
| 13 | */ | 
| 14 | #define IOPRIO_DEFAULT IOPRIO_PRIO_VALUE(IOPRIO_CLASS_NONE, 0) | 
| 15 | |
| 16 | /* | 
| 17 | * Check that a priority value has a valid class. | 
| 18 | */ | 
| 19 | static inline bool ioprio_valid(unsigned short ioprio) | 
| 20 | { | 
| 21 | unsigned short class = IOPRIO_PRIO_CLASS(ioprio); | 
| 22 | |
| 23 | return class > IOPRIO_CLASS_NONE && class <= IOPRIO_CLASS_IDLE; | 
| 24 | } | 
| 25 | |
| 26 | /* | 
| 27 | * if process has set io priority explicitly, use that. if not, convert | 
| 28 | * the cpu scheduler nice value to an io priority | 
| 29 | */ | 
| 30 | static inline int task_nice_ioprio(struct task_struct *task) | 
| 31 | { | 
| 32 | return (task_nice(p: task) + 20) / 5; | 
| 33 | } | 
| 34 | |
| 35 | /* | 
| 36 | * This is for the case where the task hasn't asked for a specific IO class. | 
| 37 | * Check for idle and rt task process, and return appropriate IO class. | 
| 38 | */ | 
| 39 | static inline int task_nice_ioclass(struct task_struct *task) | 
| 40 | { | 
| 41 | if (task->policy == SCHED_IDLE) | 
| 42 | return IOPRIO_CLASS_IDLE; | 
| 43 | else if (rt_or_dl_task_policy(tsk: task)) | 
| 44 | return IOPRIO_CLASS_RT; | 
| 45 | else | 
| 46 | return IOPRIO_CLASS_BE; | 
| 47 | } | 
| 48 | |
| 49 | #ifdef CONFIG_BLOCK | 
| 50 | /* | 
| 51 | * If the task has set an I/O priority, use that. Otherwise, return | 
| 52 | * the default I/O priority. | 
| 53 | * | 
| 54 | * Expected to be called for current task or with task_lock() held to keep | 
| 55 | * io_context stable. | 
| 56 | */ | 
| 57 | static inline int __get_task_ioprio(struct task_struct *p) | 
| 58 | { | 
| 59 | struct io_context *ioc = p->io_context; | 
| 60 | int prio; | 
| 61 | |
| 62 | if (!ioc) | 
| 63 | return IOPRIO_PRIO_VALUE(task_nice_ioclass(p), | 
| 64 | task_nice_ioprio(p)); | 
| 65 | |
| 66 | if (p != current) | 
| 67 | lockdep_assert_held(&p->alloc_lock); | 
| 68 | |
| 69 | prio = ioc->ioprio; | 
| 70 | if (IOPRIO_PRIO_CLASS(prio) == IOPRIO_CLASS_NONE) | 
| 71 | prio = IOPRIO_PRIO_VALUE(task_nice_ioclass(p), | 
| 72 | task_nice_ioprio(p)); | 
| 73 | return prio; | 
| 74 | } | 
| 75 | #else | 
| 76 | static inline int __get_task_ioprio(struct task_struct *p) | 
| 77 | { | 
| 78 | return IOPRIO_DEFAULT; | 
| 79 | } | 
| 80 | #endif /* CONFIG_BLOCK */ | 
| 81 | |
| 82 | static inline int get_current_ioprio(void) | 
| 83 | { | 
| 84 | return __get_task_ioprio(current); | 
| 85 | } | 
| 86 | |
| 87 | extern int set_task_ioprio(struct task_struct *task, int ioprio); | 
| 88 | |
| 89 | #ifdef CONFIG_BLOCK | 
| 90 | extern int ioprio_check_cap(int ioprio); | 
| 91 | #else | 
| 92 | static inline int ioprio_check_cap(int ioprio) | 
| 93 | { | 
| 94 | return -ENOTBLK; | 
| 95 | } | 
| 96 | #endif /* CONFIG_BLOCK */ | 
| 97 | |
| 98 | #endif | 
| 99 | 
