1 // SPDX-License-Identifier: GPL-2.0
5 * Copyright (C) 2018 Johan Hovold <johan@kernel.org>
8 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
10 #include <linux/cdev.h>
11 #include <linux/errno.h>
13 #include <linux/gnss.h>
14 #include <linux/idr.h>
15 #include <linux/init.h>
16 #include <linux/kernel.h>
17 #include <linux/module.h>
18 #include <linux/poll.h>
19 #include <linux/slab.h>
20 #include <linux/uaccess.h>
21 #include <linux/wait.h>
23 #define GNSS_FLAG_HAS_WRITE_RAW BIT(0)
25 #define GNSS_MINORS 16
27 static DEFINE_IDA(gnss_minors
);
28 static dev_t gnss_first
;
30 /* FIFO size must be a power of two */
31 #define GNSS_READ_FIFO_SIZE 4096
32 #define GNSS_WRITE_BUF_SIZE 1024
34 #define to_gnss_device(d) container_of((d), struct gnss_device, dev)
36 static int gnss_open(struct inode
*inode
, struct file
*file
)
38 struct gnss_device
*gdev
;
41 gdev
= container_of(inode
->i_cdev
, struct gnss_device
, cdev
);
43 get_device(&gdev
->dev
);
45 stream_open(inode
, file
);
46 file
->private_data
= gdev
;
48 down_write(&gdev
->rwsem
);
49 if (gdev
->disconnected
) {
54 if (gdev
->count
++ == 0) {
55 ret
= gdev
->ops
->open(gdev
);
60 up_write(&gdev
->rwsem
);
63 put_device(&gdev
->dev
);
68 static int gnss_release(struct inode
*inode
, struct file
*file
)
70 struct gnss_device
*gdev
= file
->private_data
;
72 down_write(&gdev
->rwsem
);
73 if (gdev
->disconnected
)
76 if (--gdev
->count
== 0) {
77 gdev
->ops
->close(gdev
);
78 kfifo_reset(&gdev
->read_fifo
);
81 up_write(&gdev
->rwsem
);
83 put_device(&gdev
->dev
);
88 static ssize_t
gnss_read(struct file
*file
, char __user
*buf
,
89 size_t count
, loff_t
*pos
)
91 struct gnss_device
*gdev
= file
->private_data
;
95 mutex_lock(&gdev
->read_mutex
);
96 while (kfifo_is_empty(&gdev
->read_fifo
)) {
97 mutex_unlock(&gdev
->read_mutex
);
99 if (gdev
->disconnected
)
102 if (file
->f_flags
& O_NONBLOCK
)
105 ret
= wait_event_interruptible(gdev
->read_queue
,
106 gdev
->disconnected
||
107 !kfifo_is_empty(&gdev
->read_fifo
));
111 mutex_lock(&gdev
->read_mutex
);
114 ret
= kfifo_to_user(&gdev
->read_fifo
, buf
, count
, &copied
);
118 mutex_unlock(&gdev
->read_mutex
);
123 static ssize_t
gnss_write(struct file
*file
, const char __user
*buf
,
124 size_t count
, loff_t
*pos
)
126 struct gnss_device
*gdev
= file
->private_data
;
130 if (gdev
->disconnected
)
136 if (!(gdev
->flags
& GNSS_FLAG_HAS_WRITE_RAW
))
139 /* Ignoring O_NONBLOCK, write_raw() is synchronous. */
141 ret
= mutex_lock_interruptible(&gdev
->write_mutex
);
146 size_t n
= count
- written
;
148 if (n
> GNSS_WRITE_BUF_SIZE
)
149 n
= GNSS_WRITE_BUF_SIZE
;
151 if (copy_from_user(gdev
->write_buf
, buf
, n
)) {
157 * Assumes write_raw can always accept GNSS_WRITE_BUF_SIZE
162 down_read(&gdev
->rwsem
);
163 if (!gdev
->disconnected
)
164 ret
= gdev
->ops
->write_raw(gdev
, gdev
->write_buf
, n
);
167 up_read(&gdev
->rwsem
);
175 if (written
== count
)
182 mutex_unlock(&gdev
->write_mutex
);
187 static __poll_t
gnss_poll(struct file
*file
, poll_table
*wait
)
189 struct gnss_device
*gdev
= file
->private_data
;
192 poll_wait(file
, &gdev
->read_queue
, wait
);
194 if (!kfifo_is_empty(&gdev
->read_fifo
))
195 mask
|= EPOLLIN
| EPOLLRDNORM
;
196 if (gdev
->disconnected
)
202 static const struct file_operations gnss_fops
= {
203 .owner
= THIS_MODULE
,
205 .release
= gnss_release
,
211 static struct class *gnss_class
;
213 static void gnss_device_release(struct device
*dev
)
215 struct gnss_device
*gdev
= to_gnss_device(dev
);
217 kfree(gdev
->write_buf
);
218 kfifo_free(&gdev
->read_fifo
);
219 ida_free(&gnss_minors
, gdev
->id
);
223 struct gnss_device
*gnss_allocate_device(struct device
*parent
)
225 struct gnss_device
*gdev
;
230 gdev
= kzalloc(sizeof(*gdev
), GFP_KERNEL
);
234 id
= ida_alloc_max(&gnss_minors
, GNSS_MINORS
- 1, GFP_KERNEL
);
243 device_initialize(dev
);
244 dev
->devt
= gnss_first
+ id
;
245 dev
->class = gnss_class
;
246 dev
->parent
= parent
;
247 dev
->release
= gnss_device_release
;
248 dev_set_drvdata(dev
, gdev
);
249 dev_set_name(dev
, "gnss%d", id
);
251 init_rwsem(&gdev
->rwsem
);
252 mutex_init(&gdev
->read_mutex
);
253 mutex_init(&gdev
->write_mutex
);
254 init_waitqueue_head(&gdev
->read_queue
);
256 ret
= kfifo_alloc(&gdev
->read_fifo
, GNSS_READ_FIFO_SIZE
, GFP_KERNEL
);
260 gdev
->write_buf
= kzalloc(GNSS_WRITE_BUF_SIZE
, GFP_KERNEL
);
261 if (!gdev
->write_buf
)
264 cdev_init(&gdev
->cdev
, &gnss_fops
);
265 gdev
->cdev
.owner
= THIS_MODULE
;
274 EXPORT_SYMBOL_GPL(gnss_allocate_device
);
276 void gnss_put_device(struct gnss_device
*gdev
)
278 put_device(&gdev
->dev
);
280 EXPORT_SYMBOL_GPL(gnss_put_device
);
282 int gnss_register_device(struct gnss_device
*gdev
)
286 /* Set a flag which can be accessed without holding the rwsem. */
287 if (gdev
->ops
->write_raw
!= NULL
)
288 gdev
->flags
|= GNSS_FLAG_HAS_WRITE_RAW
;
290 ret
= cdev_device_add(&gdev
->cdev
, &gdev
->dev
);
292 dev_err(&gdev
->dev
, "failed to add device: %d\n", ret
);
298 EXPORT_SYMBOL_GPL(gnss_register_device
);
300 void gnss_deregister_device(struct gnss_device
*gdev
)
302 down_write(&gdev
->rwsem
);
303 gdev
->disconnected
= true;
305 wake_up_interruptible(&gdev
->read_queue
);
306 gdev
->ops
->close(gdev
);
308 up_write(&gdev
->rwsem
);
310 cdev_device_del(&gdev
->cdev
, &gdev
->dev
);
312 EXPORT_SYMBOL_GPL(gnss_deregister_device
);
315 * Caller guarantees serialisation.
317 * Must not be called for a closed device.
319 int gnss_insert_raw(struct gnss_device
*gdev
, const unsigned char *buf
,
324 ret
= kfifo_in(&gdev
->read_fifo
, buf
, count
);
326 wake_up_interruptible(&gdev
->read_queue
);
330 EXPORT_SYMBOL_GPL(gnss_insert_raw
);
332 static const char * const gnss_type_names
[GNSS_TYPE_COUNT
] = {
333 [GNSS_TYPE_NMEA
] = "NMEA",
334 [GNSS_TYPE_SIRF
] = "SiRF",
335 [GNSS_TYPE_UBX
] = "UBX",
336 [GNSS_TYPE_MTK
] = "MTK",
339 static const char *gnss_type_name(const struct gnss_device
*gdev
)
341 const char *name
= NULL
;
343 if (gdev
->type
< GNSS_TYPE_COUNT
)
344 name
= gnss_type_names
[gdev
->type
];
347 dev_WARN(&gdev
->dev
, "type name not defined\n");
352 static ssize_t
type_show(struct device
*dev
, struct device_attribute
*attr
,
355 struct gnss_device
*gdev
= to_gnss_device(dev
);
357 return sprintf(buf
, "%s\n", gnss_type_name(gdev
));
359 static DEVICE_ATTR_RO(type
);
361 static struct attribute
*gnss_attrs
[] = {
365 ATTRIBUTE_GROUPS(gnss
);
367 static int gnss_uevent(const struct device
*dev
, struct kobj_uevent_env
*env
)
369 const struct gnss_device
*gdev
= to_gnss_device(dev
);
372 ret
= add_uevent_var(env
, "GNSS_TYPE=%s", gnss_type_name(gdev
));
379 static int __init
gnss_module_init(void)
383 ret
= alloc_chrdev_region(&gnss_first
, 0, GNSS_MINORS
, "gnss");
385 pr_err("failed to allocate device numbers: %d\n", ret
);
389 gnss_class
= class_create("gnss");
390 if (IS_ERR(gnss_class
)) {
391 ret
= PTR_ERR(gnss_class
);
392 pr_err("failed to create class: %d\n", ret
);
393 goto err_unregister_chrdev
;
396 gnss_class
->dev_groups
= gnss_groups
;
397 gnss_class
->dev_uevent
= gnss_uevent
;
399 pr_info("GNSS driver registered with major %d\n", MAJOR(gnss_first
));
403 err_unregister_chrdev
:
404 unregister_chrdev_region(gnss_first
, GNSS_MINORS
);
408 module_init(gnss_module_init
);
410 static void __exit
gnss_module_exit(void)
412 class_destroy(gnss_class
);
413 unregister_chrdev_region(gnss_first
, GNSS_MINORS
);
414 ida_destroy(&gnss_minors
);
416 module_exit(gnss_module_exit
);
418 MODULE_AUTHOR("Johan Hovold <johan@kernel.org>");
419 MODULE_DESCRIPTION("GNSS receiver core");
420 MODULE_LICENSE("GPL v2");