summaryrefslogtreecommitdiffstats
path: root/package/lqtapi/src/tapi/tapi-port.c
blob: 0ef9555676acbdc600d1ac622dad3cd254ce1ca0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
#include <linux/cdev.h>
#include <linux/fs.h>
#include <linux/list.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/slab.h>

#include <linux/tapi/tapi.h>
#include <linux/tapi/tapi-ioctl.h>

static inline struct tapi_port *tapi_char_device_to_port(struct tapi_char_device *chrdev)
{
	return container_of(chrdev, struct tapi_port, chrdev);
}

static int tapi_port_open(struct inode *inode, struct file *file)
{
	struct tapi_device *tdev = cdev_to_tapi_char_device(inode->i_cdev)->tdev;

	get_device(&tdev->dev);
	file->private_data = cdev_to_tapi_char_device(inode->i_cdev);

	return 0;
}

static int tapi_port_release(struct inode *inode, struct file *file)
{
	struct tapi_device *tdev = cdev_to_tapi_char_device(inode->i_cdev)->tdev;

	put_device(&tdev->dev);

	return 0;
}

static long tapi_port_ioctl_get_endpoint(struct tapi_device *tdev,
	struct tapi_port *port, unsigned long arg)
{
	return port->ep.id;
}

static long tapi_port_ioctl_set_ring(struct tapi_device *tdev,
	struct tapi_port *port, unsigned long arg)
{
	tapi_port_set_ring(tdev, port, arg);
	return 0;
}

static long tapi_port_ioctl(struct file *file, unsigned int cmd,
	unsigned long arg)
{
	int ret;
	struct tapi_char_device *tchrdev = file->private_data;
	struct tapi_device *tdev = tchrdev->tdev;
	struct tapi_port *port = tapi_char_device_to_port(tchrdev);

	switch (cmd) {
	case TAPI_PORT_IOCTL_GET_ENDPOINT:
		ret = tapi_port_ioctl_get_endpoint(tdev, port, arg);
		break;
	case TAPI_PORT_IOCTL_SET_RING:
		ret = tapi_port_ioctl_set_ring(tdev, port, arg);
		break;
	default:
		ret = -EINVAL;
		break;
	}

	return ret;
}

static const struct file_operations tapi_port_file_ops = {
	.owner = THIS_MODULE,
	.open = tapi_port_open,
	.release = tapi_port_release,
	.unlocked_ioctl = tapi_port_ioctl,
};

int tapi_register_port_device(struct tapi_device* tdev, struct tapi_port *port)
{
	dev_set_name(&port->chrdev.dev, "tapi%uP%u", tdev->id, port->id);
	return tapi_char_device_register(tdev, &port->chrdev, &tapi_port_file_ops);
}