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
83
84
85
86
87
88
89
90
91
92
|
/*++
/* NAME
/* timed_write 3
/* SUMMARY
/* write operation with pre-write timeout
/* SYNOPSIS
/* #include <iostuff.h>
/*
/* ssize_t timed_write(fd, buf, len, timeout, context)
/* int fd;
/* const void *buf;
/* size_t len;
/* int timeout;
/* void *context;
/* DESCRIPTION
/* timed_write() performs a write() operation when the specified
/* descriptor becomes writable within a user-specified deadline.
/*
/* Arguments:
/* .IP fd
/* File descriptor in the range 0..FD_SETSIZE.
/* .IP buf
/* Write buffer pointer.
/* .IP len
/* Write buffer size.
/* .IP timeout
/* The deadline in seconds. If this is <= 0, the deadline feature
/* is disabled.
/* .IP context
/* Application context. This parameter is unused. It exists only
/* for the sake of VSTREAM compatibility.
/* DIAGNOSTICS
/* When the operation does not complete within the deadline, the
/* result value is -1, and errno is set to ETIMEDOUT.
/* All other returns are identical to those of a write(2) operation.
/* LICENSE
/* .ad
/* .fi
/* The Secure Mailer license must be distributed with this software.
/* AUTHOR(S)
/* Wietse Venema
/* IBM T.J. Watson Research
/* P.O. Box 704
/* Yorktown Heights, NY 10598, USA
/*--*/
/* System library. */
#include <sys_defs.h>
#include <unistd.h>
#include <errno.h>
/* Utility library. */
#include <msg.h>
#include <iostuff.h>
/* timed_write - write with deadline */
ssize_t timed_write(int fd, const void *buf, size_t len,
int timeout, void *unused_context)
{
ssize_t ret;
/*
* Wait for a limited amount of time for something to happen. If nothing
* happens, report an ETIMEDOUT error.
*
* XXX Solaris 8 read() fails with EAGAIN after read-select() returns
* success. The code below exists just in case their write implementation
* is equally broken.
*
* This condition may also be found on systems where select() returns
* success on pipes with less than PIPE_BUF bytes of space, and with
* badly designed software where multiple writers are fighting for access
* to the same resource.
*/
for (;;) {
if (timeout > 0 && write_wait(fd, timeout) < 0)
return (-1);
if ((ret = write(fd, buf, len)) < 0 && timeout > 0 && errno == EAGAIN) {
msg_warn("write() returns EAGAIN on a writable file descriptor!");
msg_warn("pausing to avoid going into a tight select/write loop!");
sleep(1);
continue;
} else if (ret < 0 && errno == EINTR) {
continue;
} else {
return (ret);
}
}
}
|