]> Git Repo - linux.git/blob - drivers/misc/ntsync.c
Merge tag 'ti-k3-dt-for-v6.11-part2' into ti-k3-dts-next
[linux.git] / drivers / misc / ntsync.c
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * ntsync.c - Kernel driver for NT synchronization primitives
4  *
5  * Copyright (C) 2024 Elizabeth Figura <[email protected]>
6  */
7
8 #include <linux/anon_inodes.h>
9 #include <linux/file.h>
10 #include <linux/fs.h>
11 #include <linux/miscdevice.h>
12 #include <linux/module.h>
13 #include <linux/overflow.h>
14 #include <linux/slab.h>
15 #include <linux/spinlock.h>
16 #include <uapi/linux/ntsync.h>
17
18 #define NTSYNC_NAME     "ntsync"
19
20 enum ntsync_type {
21         NTSYNC_TYPE_SEM,
22 };
23
24 /*
25  * Individual synchronization primitives are represented by
26  * struct ntsync_obj, and each primitive is backed by a file.
27  *
28  * The whole namespace is represented by a struct ntsync_device also
29  * backed by a file.
30  *
31  * Both rely on struct file for reference counting. Individual
32  * ntsync_obj objects take a reference to the device when created.
33  */
34
35 struct ntsync_obj {
36         spinlock_t lock;
37
38         enum ntsync_type type;
39
40         struct file *file;
41         struct ntsync_device *dev;
42
43         /* The following fields are protected by the object lock. */
44         union {
45                 struct {
46                         __u32 count;
47                         __u32 max;
48                 } sem;
49         } u;
50 };
51
52 struct ntsync_device {
53         struct file *file;
54 };
55
56 /*
57  * Actually change the semaphore state, returning -EOVERFLOW if it is made
58  * invalid.
59  */
60 static int post_sem_state(struct ntsync_obj *sem, __u32 count)
61 {
62         __u32 sum;
63
64         lockdep_assert_held(&sem->lock);
65
66         if (check_add_overflow(sem->u.sem.count, count, &sum) ||
67             sum > sem->u.sem.max)
68                 return -EOVERFLOW;
69
70         sem->u.sem.count = sum;
71         return 0;
72 }
73
74 static int ntsync_sem_post(struct ntsync_obj *sem, void __user *argp)
75 {
76         __u32 __user *user_args = argp;
77         __u32 prev_count;
78         __u32 args;
79         int ret;
80
81         if (copy_from_user(&args, argp, sizeof(args)))
82                 return -EFAULT;
83
84         if (sem->type != NTSYNC_TYPE_SEM)
85                 return -EINVAL;
86
87         spin_lock(&sem->lock);
88
89         prev_count = sem->u.sem.count;
90         ret = post_sem_state(sem, args);
91
92         spin_unlock(&sem->lock);
93
94         if (!ret && put_user(prev_count, user_args))
95                 ret = -EFAULT;
96
97         return ret;
98 }
99
100 static int ntsync_obj_release(struct inode *inode, struct file *file)
101 {
102         struct ntsync_obj *obj = file->private_data;
103
104         fput(obj->dev->file);
105         kfree(obj);
106
107         return 0;
108 }
109
110 static long ntsync_obj_ioctl(struct file *file, unsigned int cmd,
111                              unsigned long parm)
112 {
113         struct ntsync_obj *obj = file->private_data;
114         void __user *argp = (void __user *)parm;
115
116         switch (cmd) {
117         case NTSYNC_IOC_SEM_POST:
118                 return ntsync_sem_post(obj, argp);
119         default:
120                 return -ENOIOCTLCMD;
121         }
122 }
123
124 static const struct file_operations ntsync_obj_fops = {
125         .owner          = THIS_MODULE,
126         .release        = ntsync_obj_release,
127         .unlocked_ioctl = ntsync_obj_ioctl,
128         .compat_ioctl   = compat_ptr_ioctl,
129         .llseek         = no_llseek,
130 };
131
132 static struct ntsync_obj *ntsync_alloc_obj(struct ntsync_device *dev,
133                                            enum ntsync_type type)
134 {
135         struct ntsync_obj *obj;
136
137         obj = kzalloc(sizeof(*obj), GFP_KERNEL);
138         if (!obj)
139                 return NULL;
140         obj->type = type;
141         obj->dev = dev;
142         get_file(dev->file);
143         spin_lock_init(&obj->lock);
144
145         return obj;
146 }
147
148 static int ntsync_obj_get_fd(struct ntsync_obj *obj)
149 {
150         struct file *file;
151         int fd;
152
153         fd = get_unused_fd_flags(O_CLOEXEC);
154         if (fd < 0)
155                 return fd;
156         file = anon_inode_getfile("ntsync", &ntsync_obj_fops, obj, O_RDWR);
157         if (IS_ERR(file)) {
158                 put_unused_fd(fd);
159                 return PTR_ERR(file);
160         }
161         obj->file = file;
162         fd_install(fd, file);
163
164         return fd;
165 }
166
167 static int ntsync_create_sem(struct ntsync_device *dev, void __user *argp)
168 {
169         struct ntsync_sem_args __user *user_args = argp;
170         struct ntsync_sem_args args;
171         struct ntsync_obj *sem;
172         int fd;
173
174         if (copy_from_user(&args, argp, sizeof(args)))
175                 return -EFAULT;
176
177         if (args.count > args.max)
178                 return -EINVAL;
179
180         sem = ntsync_alloc_obj(dev, NTSYNC_TYPE_SEM);
181         if (!sem)
182                 return -ENOMEM;
183         sem->u.sem.count = args.count;
184         sem->u.sem.max = args.max;
185         fd = ntsync_obj_get_fd(sem);
186         if (fd < 0) {
187                 kfree(sem);
188                 return fd;
189         }
190
191         return put_user(fd, &user_args->sem);
192 }
193
194 static int ntsync_char_open(struct inode *inode, struct file *file)
195 {
196         struct ntsync_device *dev;
197
198         dev = kzalloc(sizeof(*dev), GFP_KERNEL);
199         if (!dev)
200                 return -ENOMEM;
201
202         file->private_data = dev;
203         dev->file = file;
204         return nonseekable_open(inode, file);
205 }
206
207 static int ntsync_char_release(struct inode *inode, struct file *file)
208 {
209         struct ntsync_device *dev = file->private_data;
210
211         kfree(dev);
212
213         return 0;
214 }
215
216 static long ntsync_char_ioctl(struct file *file, unsigned int cmd,
217                               unsigned long parm)
218 {
219         struct ntsync_device *dev = file->private_data;
220         void __user *argp = (void __user *)parm;
221
222         switch (cmd) {
223         case NTSYNC_IOC_CREATE_SEM:
224                 return ntsync_create_sem(dev, argp);
225         default:
226                 return -ENOIOCTLCMD;
227         }
228 }
229
230 static const struct file_operations ntsync_fops = {
231         .owner          = THIS_MODULE,
232         .open           = ntsync_char_open,
233         .release        = ntsync_char_release,
234         .unlocked_ioctl = ntsync_char_ioctl,
235         .compat_ioctl   = compat_ptr_ioctl,
236         .llseek         = no_llseek,
237 };
238
239 static struct miscdevice ntsync_misc = {
240         .minor          = MISC_DYNAMIC_MINOR,
241         .name           = NTSYNC_NAME,
242         .fops           = &ntsync_fops,
243 };
244
245 module_misc_device(ntsync_misc);
246
247 MODULE_AUTHOR("Elizabeth Figura <[email protected]>");
248 MODULE_DESCRIPTION("Kernel driver for NT synchronization primitives");
249 MODULE_LICENSE("GPL");
This page took 0.046736 seconds and 4 git commands to generate.