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