diff options
Diffstat (limited to 'drivers/net/macvtap.c')
-rw-r--r-- | drivers/net/macvtap.c | 98 |
1 files changed, 77 insertions, 21 deletions
diff --git a/drivers/net/macvtap.c b/drivers/net/macvtap.c index 705099749766..e354501ab297 100644 --- a/drivers/net/macvtap.c +++ b/drivers/net/macvtap.c | |||
@@ -58,6 +58,8 @@ static unsigned int macvtap_major; | |||
58 | static struct class *macvtap_class; | 58 | static struct class *macvtap_class; |
59 | static struct cdev macvtap_cdev; | 59 | static struct cdev macvtap_cdev; |
60 | 60 | ||
61 | static const struct proto_ops macvtap_socket_ops; | ||
62 | |||
61 | /* | 63 | /* |
62 | * RCU usage: | 64 | * RCU usage: |
63 | * The macvtap_queue and the macvlan_dev are loosely coupled, the | 65 | * The macvtap_queue and the macvlan_dev are loosely coupled, the |
@@ -176,7 +178,7 @@ static int macvtap_forward(struct net_device *dev, struct sk_buff *skb) | |||
176 | return -ENOLINK; | 178 | return -ENOLINK; |
177 | 179 | ||
178 | skb_queue_tail(&q->sk.sk_receive_queue, skb); | 180 | skb_queue_tail(&q->sk.sk_receive_queue, skb); |
179 | wake_up(q->sk.sk_sleep); | 181 | wake_up_interruptible_poll(q->sk.sk_sleep, POLLIN | POLLRDNORM | POLLRDBAND); |
180 | return 0; | 182 | return 0; |
181 | } | 183 | } |
182 | 184 | ||
@@ -242,7 +244,7 @@ static void macvtap_sock_write_space(struct sock *sk) | |||
242 | return; | 244 | return; |
243 | 245 | ||
244 | if (sk->sk_sleep && waitqueue_active(sk->sk_sleep)) | 246 | if (sk->sk_sleep && waitqueue_active(sk->sk_sleep)) |
245 | wake_up_interruptible_sync(sk->sk_sleep); | 247 | wake_up_interruptible_poll(sk->sk_sleep, POLLOUT | POLLWRNORM | POLLWRBAND); |
246 | } | 248 | } |
247 | 249 | ||
248 | static int macvtap_open(struct inode *inode, struct file *file) | 250 | static int macvtap_open(struct inode *inode, struct file *file) |
@@ -270,6 +272,8 @@ static int macvtap_open(struct inode *inode, struct file *file) | |||
270 | init_waitqueue_head(&q->sock.wait); | 272 | init_waitqueue_head(&q->sock.wait); |
271 | q->sock.type = SOCK_RAW; | 273 | q->sock.type = SOCK_RAW; |
272 | q->sock.state = SS_CONNECTED; | 274 | q->sock.state = SS_CONNECTED; |
275 | q->sock.file = file; | ||
276 | q->sock.ops = &macvtap_socket_ops; | ||
273 | sock_init_data(&q->sock, &q->sk); | 277 | sock_init_data(&q->sock, &q->sk); |
274 | q->sk.sk_write_space = macvtap_sock_write_space; | 278 | q->sk.sk_write_space = macvtap_sock_write_space; |
275 | 279 | ||
@@ -387,32 +391,20 @@ static ssize_t macvtap_put_user(struct macvtap_queue *q, | |||
387 | 391 | ||
388 | rcu_read_lock_bh(); | 392 | rcu_read_lock_bh(); |
389 | vlan = rcu_dereference(q->vlan); | 393 | vlan = rcu_dereference(q->vlan); |
390 | macvlan_count_rx(vlan, len, ret == 0, 0); | 394 | if (vlan) |
395 | macvlan_count_rx(vlan, len, ret == 0, 0); | ||
391 | rcu_read_unlock_bh(); | 396 | rcu_read_unlock_bh(); |
392 | 397 | ||
393 | return ret ? ret : len; | 398 | return ret ? ret : len; |
394 | } | 399 | } |
395 | 400 | ||
396 | static ssize_t macvtap_aio_read(struct kiocb *iocb, const struct iovec *iv, | 401 | static ssize_t macvtap_do_read(struct macvtap_queue *q, struct kiocb *iocb, |
397 | unsigned long count, loff_t pos) | 402 | const struct iovec *iv, unsigned long len, |
403 | int noblock) | ||
398 | { | 404 | { |
399 | struct file *file = iocb->ki_filp; | ||
400 | struct macvtap_queue *q = file->private_data; | ||
401 | |||
402 | DECLARE_WAITQUEUE(wait, current); | 405 | DECLARE_WAITQUEUE(wait, current); |
403 | struct sk_buff *skb; | 406 | struct sk_buff *skb; |
404 | ssize_t len, ret = 0; | 407 | ssize_t ret = 0; |
405 | |||
406 | if (!q) { | ||
407 | ret = -ENOLINK; | ||
408 | goto out; | ||
409 | } | ||
410 | |||
411 | len = iov_length(iv, count); | ||
412 | if (len < 0) { | ||
413 | ret = -EINVAL; | ||
414 | goto out; | ||
415 | } | ||
416 | 408 | ||
417 | add_wait_queue(q->sk.sk_sleep, &wait); | 409 | add_wait_queue(q->sk.sk_sleep, &wait); |
418 | while (len) { | 410 | while (len) { |
@@ -421,7 +413,7 @@ static ssize_t macvtap_aio_read(struct kiocb *iocb, const struct iovec *iv, | |||
421 | /* Read frames from the queue */ | 413 | /* Read frames from the queue */ |
422 | skb = skb_dequeue(&q->sk.sk_receive_queue); | 414 | skb = skb_dequeue(&q->sk.sk_receive_queue); |
423 | if (!skb) { | 415 | if (!skb) { |
424 | if (file->f_flags & O_NONBLOCK) { | 416 | if (noblock) { |
425 | ret = -EAGAIN; | 417 | ret = -EAGAIN; |
426 | break; | 418 | break; |
427 | } | 419 | } |
@@ -440,7 +432,24 @@ static ssize_t macvtap_aio_read(struct kiocb *iocb, const struct iovec *iv, | |||
440 | 432 | ||
441 | current->state = TASK_RUNNING; | 433 | current->state = TASK_RUNNING; |
442 | remove_wait_queue(q->sk.sk_sleep, &wait); | 434 | remove_wait_queue(q->sk.sk_sleep, &wait); |
435 | return ret; | ||
436 | } | ||
437 | |||
438 | static ssize_t macvtap_aio_read(struct kiocb *iocb, const struct iovec *iv, | ||
439 | unsigned long count, loff_t pos) | ||
440 | { | ||
441 | struct file *file = iocb->ki_filp; | ||
442 | struct macvtap_queue *q = file->private_data; | ||
443 | ssize_t len, ret = 0; | ||
443 | 444 | ||
445 | len = iov_length(iv, count); | ||
446 | if (len < 0) { | ||
447 | ret = -EINVAL; | ||
448 | goto out; | ||
449 | } | ||
450 | |||
451 | ret = macvtap_do_read(q, iocb, iv, len, file->f_flags & O_NONBLOCK); | ||
452 | ret = min_t(ssize_t, ret, len); /* XXX copied from tun.c. Why? */ | ||
444 | out: | 453 | out: |
445 | return ret; | 454 | return ret; |
446 | } | 455 | } |
@@ -538,6 +547,53 @@ static const struct file_operations macvtap_fops = { | |||
538 | #endif | 547 | #endif |
539 | }; | 548 | }; |
540 | 549 | ||
550 | static int macvtap_sendmsg(struct kiocb *iocb, struct socket *sock, | ||
551 | struct msghdr *m, size_t total_len) | ||
552 | { | ||
553 | struct macvtap_queue *q = container_of(sock, struct macvtap_queue, sock); | ||
554 | return macvtap_get_user(q, m->msg_iov, total_len, | ||
555 | m->msg_flags & MSG_DONTWAIT); | ||
556 | } | ||
557 | |||
558 | static int macvtap_recvmsg(struct kiocb *iocb, struct socket *sock, | ||
559 | struct msghdr *m, size_t total_len, | ||
560 | int flags) | ||
561 | { | ||
562 | struct macvtap_queue *q = container_of(sock, struct macvtap_queue, sock); | ||
563 | int ret; | ||
564 | if (flags & ~(MSG_DONTWAIT|MSG_TRUNC)) | ||
565 | return -EINVAL; | ||
566 | ret = macvtap_do_read(q, iocb, m->msg_iov, total_len, | ||
567 | flags & MSG_DONTWAIT); | ||
568 | if (ret > total_len) { | ||
569 | m->msg_flags |= MSG_TRUNC; | ||
570 | ret = flags & MSG_TRUNC ? ret : total_len; | ||
571 | } | ||
572 | return ret; | ||
573 | } | ||
574 | |||
575 | /* Ops structure to mimic raw sockets with tun */ | ||
576 | static const struct proto_ops macvtap_socket_ops = { | ||
577 | .sendmsg = macvtap_sendmsg, | ||
578 | .recvmsg = macvtap_recvmsg, | ||
579 | }; | ||
580 | |||
581 | /* Get an underlying socket object from tun file. Returns error unless file is | ||
582 | * attached to a device. The returned object works like a packet socket, it | ||
583 | * can be used for sock_sendmsg/sock_recvmsg. The caller is responsible for | ||
584 | * holding a reference to the file for as long as the socket is in use. */ | ||
585 | struct socket *macvtap_get_socket(struct file *file) | ||
586 | { | ||
587 | struct macvtap_queue *q; | ||
588 | if (file->f_op != &macvtap_fops) | ||
589 | return ERR_PTR(-EINVAL); | ||
590 | q = file->private_data; | ||
591 | if (!q) | ||
592 | return ERR_PTR(-EBADFD); | ||
593 | return &q->sock; | ||
594 | } | ||
595 | EXPORT_SYMBOL_GPL(macvtap_get_socket); | ||
596 | |||
541 | static int macvtap_init(void) | 597 | static int macvtap_init(void) |
542 | { | 598 | { |
543 | int err; | 599 | int err; |