1 /*
2  * Backchannel functions for CUPS.
3  *
4  * Copyright 2007-2014 by Apple Inc.
5  * Copyright 1997-2007 by Easy Software Products.
6  *
7  * These coded instructions, statements, and computer programs are the
8  * property of Apple Inc. and are protected by Federal copyright
9  * law.  Distribution and use rights are outlined in the file "LICENSE.txt"
10  * which should have been included with this file.  If this file is
11  * missing or damaged, see the license at "http://www.cups.org/".
12  *
13  * This file is subject to the Apple OS-Developed Software exception.
14  */
15 
16 /*
17  * Include necessary headers...
18  */
19 
20 #include "cups.h"
21 #include <errno.h>
22 #ifdef WIN32
23 #  include <io.h>
24 #  include <fcntl.h>
25 #else
26 #  include <sys/time.h>
27 #endif /* WIN32 */
28 
29 
30 /*
31  * Local functions...
32  */
33 
34 static void	cups_setup(fd_set *set, struct timeval *tval,
35 		           double timeout);
36 
37 
38 /*
39  * 'cupsBackChannelRead()' - Read data from the backchannel.
40  *
41  * Reads up to "bytes" bytes from the backchannel/backend. The "timeout"
42  * parameter controls how many seconds to wait for the data - use 0.0 to
43  * return immediately if there is no data, -1.0 to wait for data indefinitely.
44  *
45  * @since CUPS 1.2/macOS 10.5@
46  */
47 
48 ssize_t					/* O - Bytes read or -1 on error */
cupsBackChannelRead(char * buffer,size_t bytes,double timeout)49 cupsBackChannelRead(char   *buffer,	/* I - Buffer to read into */
50                     size_t bytes,	/* I - Bytes to read */
51 		    double timeout)	/* I - Timeout in seconds, typically 0.0 to poll */
52 {
53   fd_set	input;			/* Input set */
54   struct timeval tval;			/* Timeout value */
55   int		status;			/* Select status */
56 
57 
58  /*
59   * Wait for input ready.
60   */
61 
62   do
63   {
64     cups_setup(&input, &tval, timeout);
65 
66     if (timeout < 0.0)
67       status = select(4, &input, NULL, NULL, NULL);
68     else
69       status = select(4, &input, NULL, NULL, &tval);
70   }
71   while (status < 0 && errno != EINTR && errno != EAGAIN);
72 
73   if (status < 0)
74     return (-1);			/* Timeout! */
75 
76  /*
77   * Read bytes from the pipe...
78   */
79 
80 #ifdef WIN32
81   return ((ssize_t)_read(3, buffer, (unsigned)bytes));
82 #else
83   return (read(3, buffer, bytes));
84 #endif /* WIN32 */
85 }
86 
87 
88 /*
89  * 'cupsBackChannelWrite()' - Write data to the backchannel.
90  *
91  * Writes "bytes" bytes to the backchannel/filter. The "timeout" parameter
92  * controls how many seconds to wait for the data to be written - use
93  * 0.0 to return immediately if the data cannot be written, -1.0 to wait
94  * indefinitely.
95  *
96  * @since CUPS 1.2/macOS 10.5@
97  */
98 
99 ssize_t					/* O - Bytes written or -1 on error */
cupsBackChannelWrite(const char * buffer,size_t bytes,double timeout)100 cupsBackChannelWrite(
101     const char *buffer,			/* I - Buffer to write */
102     size_t     bytes,			/* I - Bytes to write */
103     double     timeout)			/* I - Timeout in seconds, typically 1.0 */
104 {
105   fd_set	output;			/* Output set */
106   struct timeval tval;			/* Timeout value */
107   int		status;			/* Select status */
108   ssize_t	count;			/* Current bytes */
109   size_t	total;			/* Total bytes */
110 
111 
112  /*
113   * Write all bytes...
114   */
115 
116   total = 0;
117 
118   while (total < bytes)
119   {
120    /*
121     * Wait for write-ready...
122     */
123 
124     do
125     {
126       cups_setup(&output, &tval, timeout);
127 
128       if (timeout < 0.0)
129 	status = select(4, NULL, &output, NULL, NULL);
130       else
131 	status = select(4, NULL, &output, NULL, &tval);
132     }
133     while (status < 0 && errno != EINTR && errno != EAGAIN);
134 
135     if (status <= 0)
136       return (-1);			/* Timeout! */
137 
138    /*
139     * Write bytes to the pipe...
140     */
141 
142 #ifdef WIN32
143     count = (ssize_t)_write(3, buffer, (unsigned)(bytes - total));
144 #else
145     count = write(3, buffer, bytes - total);
146 #endif /* WIN32 */
147 
148     if (count < 0)
149     {
150      /*
151       * Write error - abort on fatal errors...
152       */
153 
154       if (errno != EINTR && errno != EAGAIN)
155         return (-1);
156     }
157     else
158     {
159      /*
160       * Write succeeded, update buffer pointer and total count...
161       */
162 
163       buffer += count;
164       total  += (size_t)count;
165     }
166   }
167 
168   return ((ssize_t)bytes);
169 }
170 
171 
172 /*
173  * 'cups_setup()' - Setup select()
174  */
175 
176 static void
cups_setup(fd_set * set,struct timeval * tval,double timeout)177 cups_setup(fd_set         *set,		/* I - Set for select() */
178            struct timeval *tval,	/* I - Timer value */
179 	   double         timeout)	/* I - Timeout in seconds */
180 {
181   tval->tv_sec = (int)timeout;
182   tval->tv_usec = (int)(1000000.0 * (timeout - tval->tv_sec));
183 
184   FD_ZERO(set);
185   FD_SET(3, set);
186 }
187