diff options
author | Hendrik Brueckner <brueckner@linux.vnet.ibm.com> | 2009-06-17 17:54:48 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2009-06-19 03:10:40 -0400 |
commit | 0ea920d211e0a870871965418923b08da2025b4a (patch) | |
tree | 5dc3451a2011f0d32839b5aef20d9aaec1da78cf | |
parent | bb664f49f8be17d7b8bf9821144e8a53d7fcfe8a (diff) |
af_iucv: Return -EAGAIN if iucv msg limit is exceeded
If the iucv message limit for a communication path is exceeded,
sendmsg() returns -EAGAIN instead of -EPIPE.
The calling application can then handle this error situtation,
e.g. to try again after waiting some time.
For blocking sockets, sendmsg() waits up to the socket timeout
before returning -EAGAIN. For the new wait condition, a macro
has been introduced and the iucv_sock_wait_state() has been
refactored to this macro.
Signed-off-by: Hendrik Brueckner <brueckner@linux.vnet.ibm.com>
Signed-off-by: Ursula Braun <ursula.braun@de.ibm.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r-- | include/net/iucv/af_iucv.h | 2 | ||||
-rw-r--r-- | net/iucv/af_iucv.c | 144 |
2 files changed, 103 insertions, 43 deletions
diff --git a/include/net/iucv/af_iucv.h b/include/net/iucv/af_iucv.h index 21ee49ffcbaf..f82a1e877372 100644 --- a/include/net/iucv/af_iucv.h +++ b/include/net/iucv/af_iucv.h | |||
@@ -94,8 +94,6 @@ unsigned int iucv_sock_poll(struct file *file, struct socket *sock, | |||
94 | poll_table *wait); | 94 | poll_table *wait); |
95 | void iucv_sock_link(struct iucv_sock_list *l, struct sock *s); | 95 | void iucv_sock_link(struct iucv_sock_list *l, struct sock *s); |
96 | void iucv_sock_unlink(struct iucv_sock_list *l, struct sock *s); | 96 | void iucv_sock_unlink(struct iucv_sock_list *l, struct sock *s); |
97 | int iucv_sock_wait_state(struct sock *sk, int state, int state2, | ||
98 | unsigned long timeo); | ||
99 | int iucv_sock_wait_cnt(struct sock *sk, unsigned long timeo); | 97 | int iucv_sock_wait_cnt(struct sock *sk, unsigned long timeo); |
100 | void iucv_accept_enqueue(struct sock *parent, struct sock *sk); | 98 | void iucv_accept_enqueue(struct sock *parent, struct sock *sk); |
101 | void iucv_accept_unlink(struct sock *sk); | 99 | void iucv_accept_unlink(struct sock *sk); |
diff --git a/net/iucv/af_iucv.c b/net/iucv/af_iucv.c index 42b7198a6883..abadb4a846cf 100644 --- a/net/iucv/af_iucv.c +++ b/net/iucv/af_iucv.c | |||
@@ -53,6 +53,38 @@ static const u8 iprm_shutdown[8] = | |||
53 | #define CB_TRGCLS(skb) ((skb)->cb + CB_TAG_LEN) /* iucv msg target class */ | 53 | #define CB_TRGCLS(skb) ((skb)->cb + CB_TAG_LEN) /* iucv msg target class */ |
54 | #define CB_TRGCLS_LEN (TRGCLS_SIZE) | 54 | #define CB_TRGCLS_LEN (TRGCLS_SIZE) |
55 | 55 | ||
56 | #define __iucv_sock_wait(sk, condition, timeo, ret) \ | ||
57 | do { \ | ||
58 | DEFINE_WAIT(__wait); \ | ||
59 | long __timeo = timeo; \ | ||
60 | ret = 0; \ | ||
61 | while (!(condition)) { \ | ||
62 | prepare_to_wait(sk->sk_sleep, &__wait, TASK_INTERRUPTIBLE); \ | ||
63 | if (!__timeo) { \ | ||
64 | ret = -EAGAIN; \ | ||
65 | break; \ | ||
66 | } \ | ||
67 | if (signal_pending(current)) { \ | ||
68 | ret = sock_intr_errno(__timeo); \ | ||
69 | break; \ | ||
70 | } \ | ||
71 | release_sock(sk); \ | ||
72 | __timeo = schedule_timeout(__timeo); \ | ||
73 | lock_sock(sk); \ | ||
74 | ret = sock_error(sk); \ | ||
75 | if (ret) \ | ||
76 | break; \ | ||
77 | } \ | ||
78 | finish_wait(sk->sk_sleep, &__wait); \ | ||
79 | } while (0) | ||
80 | |||
81 | #define iucv_sock_wait(sk, condition, timeo) \ | ||
82 | ({ \ | ||
83 | int __ret = 0; \ | ||
84 | if (!(condition)) \ | ||
85 | __iucv_sock_wait(sk, condition, timeo, __ret); \ | ||
86 | __ret; \ | ||
87 | }) | ||
56 | 88 | ||
57 | static void iucv_sock_kill(struct sock *sk); | 89 | static void iucv_sock_kill(struct sock *sk); |
58 | static void iucv_sock_close(struct sock *sk); | 90 | static void iucv_sock_close(struct sock *sk); |
@@ -121,6 +153,48 @@ static inline size_t iucv_msg_length(struct iucv_message *msg) | |||
121 | return msg->length; | 153 | return msg->length; |
122 | } | 154 | } |
123 | 155 | ||
156 | /** | ||
157 | * iucv_sock_in_state() - check for specific states | ||
158 | * @sk: sock structure | ||
159 | * @state: first iucv sk state | ||
160 | * @state: second iucv sk state | ||
161 | * | ||
162 | * Returns true if the socket in either in the first or second state. | ||
163 | */ | ||
164 | static int iucv_sock_in_state(struct sock *sk, int state, int state2) | ||
165 | { | ||
166 | return (sk->sk_state == state || sk->sk_state == state2); | ||
167 | } | ||
168 | |||
169 | /** | ||
170 | * iucv_below_msglim() - function to check if messages can be sent | ||
171 | * @sk: sock structure | ||
172 | * | ||
173 | * Returns true if the send queue length is lower than the message limit. | ||
174 | * Always returns true if the socket is not connected (no iucv path for | ||
175 | * checking the message limit). | ||
176 | */ | ||
177 | static inline int iucv_below_msglim(struct sock *sk) | ||
178 | { | ||
179 | struct iucv_sock *iucv = iucv_sk(sk); | ||
180 | |||
181 | if (sk->sk_state != IUCV_CONNECTED) | ||
182 | return 1; | ||
183 | return (skb_queue_len(&iucv->send_skb_q) < iucv->path->msglim); | ||
184 | } | ||
185 | |||
186 | /** | ||
187 | * iucv_sock_wake_msglim() - Wake up thread waiting on msg limit | ||
188 | */ | ||
189 | static void iucv_sock_wake_msglim(struct sock *sk) | ||
190 | { | ||
191 | read_lock(&sk->sk_callback_lock); | ||
192 | if (sk->sk_sleep && waitqueue_active(sk->sk_sleep)) | ||
193 | wake_up_interruptible_all(sk->sk_sleep); | ||
194 | sk_wake_async(sk, SOCK_WAKE_SPACE, POLL_OUT); | ||
195 | read_unlock(&sk->sk_callback_lock); | ||
196 | } | ||
197 | |||
124 | /* Timers */ | 198 | /* Timers */ |
125 | static void iucv_sock_timeout(unsigned long arg) | 199 | static void iucv_sock_timeout(unsigned long arg) |
126 | { | 200 | { |
@@ -212,7 +286,9 @@ static void iucv_sock_close(struct sock *sk) | |||
212 | timeo = sk->sk_lingertime; | 286 | timeo = sk->sk_lingertime; |
213 | else | 287 | else |
214 | timeo = IUCV_DISCONN_TIMEOUT; | 288 | timeo = IUCV_DISCONN_TIMEOUT; |
215 | err = iucv_sock_wait_state(sk, IUCV_CLOSED, 0, timeo); | 289 | err = iucv_sock_wait(sk, |
290 | iucv_sock_in_state(sk, IUCV_CLOSED, 0), | ||
291 | timeo); | ||
216 | } | 292 | } |
217 | 293 | ||
218 | case IUCV_CLOSING: /* fall through */ | 294 | case IUCV_CLOSING: /* fall through */ |
@@ -393,39 +469,6 @@ struct sock *iucv_accept_dequeue(struct sock *parent, struct socket *newsock) | |||
393 | return NULL; | 469 | return NULL; |
394 | } | 470 | } |
395 | 471 | ||
396 | int iucv_sock_wait_state(struct sock *sk, int state, int state2, | ||
397 | unsigned long timeo) | ||
398 | { | ||
399 | DECLARE_WAITQUEUE(wait, current); | ||
400 | int err = 0; | ||
401 | |||
402 | add_wait_queue(sk->sk_sleep, &wait); | ||
403 | while (sk->sk_state != state && sk->sk_state != state2) { | ||
404 | set_current_state(TASK_INTERRUPTIBLE); | ||
405 | |||
406 | if (!timeo) { | ||
407 | err = -EAGAIN; | ||
408 | break; | ||
409 | } | ||
410 | |||
411 | if (signal_pending(current)) { | ||
412 | err = sock_intr_errno(timeo); | ||
413 | break; | ||
414 | } | ||
415 | |||
416 | release_sock(sk); | ||
417 | timeo = schedule_timeout(timeo); | ||
418 | lock_sock(sk); | ||
419 | |||
420 | err = sock_error(sk); | ||
421 | if (err) | ||
422 | break; | ||
423 | } | ||
424 | set_current_state(TASK_RUNNING); | ||
425 | remove_wait_queue(sk->sk_sleep, &wait); | ||
426 | return err; | ||
427 | } | ||
428 | |||
429 | /* Bind an unbound socket */ | 472 | /* Bind an unbound socket */ |
430 | static int iucv_sock_bind(struct socket *sock, struct sockaddr *addr, | 473 | static int iucv_sock_bind(struct socket *sock, struct sockaddr *addr, |
431 | int addr_len) | 474 | int addr_len) |
@@ -570,8 +613,9 @@ static int iucv_sock_connect(struct socket *sock, struct sockaddr *addr, | |||
570 | } | 613 | } |
571 | 614 | ||
572 | if (sk->sk_state != IUCV_CONNECTED) { | 615 | if (sk->sk_state != IUCV_CONNECTED) { |
573 | err = iucv_sock_wait_state(sk, IUCV_CONNECTED, IUCV_DISCONN, | 616 | err = iucv_sock_wait(sk, iucv_sock_in_state(sk, IUCV_CONNECTED, |
574 | sock_sndtimeo(sk, flags & O_NONBLOCK)); | 617 | IUCV_DISCONN), |
618 | sock_sndtimeo(sk, flags & O_NONBLOCK)); | ||
575 | } | 619 | } |
576 | 620 | ||
577 | if (sk->sk_state == IUCV_DISCONN) { | 621 | if (sk->sk_state == IUCV_DISCONN) { |
@@ -725,9 +769,11 @@ static int iucv_sock_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
725 | struct iucv_message txmsg; | 769 | struct iucv_message txmsg; |
726 | struct cmsghdr *cmsg; | 770 | struct cmsghdr *cmsg; |
727 | int cmsg_done; | 771 | int cmsg_done; |
772 | long timeo; | ||
728 | char user_id[9]; | 773 | char user_id[9]; |
729 | char appl_id[9]; | 774 | char appl_id[9]; |
730 | int err; | 775 | int err; |
776 | int noblock = msg->msg_flags & MSG_DONTWAIT; | ||
731 | 777 | ||
732 | err = sock_error(sk); | 778 | err = sock_error(sk); |
733 | if (err) | 779 | if (err) |
@@ -799,8 +845,7 @@ static int iucv_sock_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
799 | * this is fine for SOCK_SEQPACKET (unless we want to support | 845 | * this is fine for SOCK_SEQPACKET (unless we want to support |
800 | * segmented records using the MSG_EOR flag), but | 846 | * segmented records using the MSG_EOR flag), but |
801 | * for SOCK_STREAM we might want to improve it in future */ | 847 | * for SOCK_STREAM we might want to improve it in future */ |
802 | skb = sock_alloc_send_skb(sk, len, msg->msg_flags & MSG_DONTWAIT, | 848 | skb = sock_alloc_send_skb(sk, len, noblock, &err); |
803 | &err); | ||
804 | if (!skb) | 849 | if (!skb) |
805 | goto out; | 850 | goto out; |
806 | if (memcpy_fromiovec(skb_put(skb, len), msg->msg_iov, len)) { | 851 | if (memcpy_fromiovec(skb_put(skb, len), msg->msg_iov, len)) { |
@@ -808,6 +853,18 @@ static int iucv_sock_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
808 | goto fail; | 853 | goto fail; |
809 | } | 854 | } |
810 | 855 | ||
856 | /* wait if outstanding messages for iucv path has reached */ | ||
857 | timeo = sock_sndtimeo(sk, noblock); | ||
858 | err = iucv_sock_wait(sk, iucv_below_msglim(sk), timeo); | ||
859 | if (err) | ||
860 | goto fail; | ||
861 | |||
862 | /* return -ECONNRESET if the socket is no longer connected */ | ||
863 | if (sk->sk_state != IUCV_CONNECTED) { | ||
864 | err = -ECONNRESET; | ||
865 | goto fail; | ||
866 | } | ||
867 | |||
811 | /* increment and save iucv message tag for msg_completion cbk */ | 868 | /* increment and save iucv message tag for msg_completion cbk */ |
812 | txmsg.tag = iucv->send_tag++; | 869 | txmsg.tag = iucv->send_tag++; |
813 | memcpy(CB_TAG(skb), &txmsg.tag, CB_TAG_LEN); | 870 | memcpy(CB_TAG(skb), &txmsg.tag, CB_TAG_LEN); |
@@ -844,9 +901,10 @@ static int iucv_sock_sendmsg(struct kiocb *iocb, struct socket *sock, | |||
844 | pr_err("Application %s on z/VM guest %s" | 901 | pr_err("Application %s on z/VM guest %s" |
845 | " exceeds message limit\n", | 902 | " exceeds message limit\n", |
846 | appl_id, user_id); | 903 | appl_id, user_id); |
847 | } | 904 | err = -EAGAIN; |
905 | } else | ||
906 | err = -EPIPE; | ||
848 | skb_unlink(skb, &iucv->send_skb_q); | 907 | skb_unlink(skb, &iucv->send_skb_q); |
849 | err = -EPIPE; | ||
850 | goto fail; | 908 | goto fail; |
851 | } | 909 | } |
852 | 910 | ||
@@ -1463,7 +1521,11 @@ static void iucv_callback_txdone(struct iucv_path *path, | |||
1463 | 1521 | ||
1464 | spin_unlock_irqrestore(&list->lock, flags); | 1522 | spin_unlock_irqrestore(&list->lock, flags); |
1465 | 1523 | ||
1466 | kfree_skb(this); | 1524 | if (this) { |
1525 | kfree_skb(this); | ||
1526 | /* wake up any process waiting for sending */ | ||
1527 | iucv_sock_wake_msglim(sk); | ||
1528 | } | ||
1467 | } | 1529 | } |
1468 | BUG_ON(!this); | 1530 | BUG_ON(!this); |
1469 | 1531 | ||