]> git.ipfire.org Git - thirdparty/cups.git/blob - backend/socket.c
-Ae is only for the HP C compiler, not the C++ compiler.
[thirdparty/cups.git] / backend / socket.c
1 /*
2 * "$Id: socket.c,v 1.19 2001/04/18 16:28:34 mike Exp $"
3 *
4 * AppSocket backend for the Common UNIX Printing System (CUPS).
5 *
6 * Copyright 1997-2001 by Easy Software Products, all rights reserved.
7 *
8 * These coded instructions, statements, and computer programs are the
9 * property of Easy Software Products and are protected by Federal
10 * copyright law. Distribution and use rights are outlined in the file
11 * "LICENSE" which should have been included with this file. If this
12 * file is missing or damaged please contact Easy Software Products
13 * at:
14 *
15 * Attn: CUPS Licensing Information
16 * Easy Software Products
17 * 44141 Airport View Drive, Suite 204
18 * Hollywood, Maryland 20636-3111 USA
19 *
20 * Voice: (301) 373-9603
21 * EMail: cups-info@cups.org
22 * WWW: http://www.cups.org
23 *
24 * Contents:
25 *
26 * main() - Send a file to the printer or server.
27 */
28
29 /*
30 * Include necessary headers.
31 */
32
33 #include <cups/cups.h>
34 #include <stdio.h>
35 #include <stdlib.h>
36 #include <stdarg.h>
37 #include <cups/string.h>
38 #include <errno.h>
39 #include <sys/types.h>
40 #include <sys/stat.h>
41 #include <signal.h>
42
43 #if defined(WIN32) || defined(__EMX__)
44 # include <winsock.h>
45 #else
46 # include <sys/socket.h>
47 # include <netinet/in.h>
48 # include <arpa/inet.h>
49 # include <netdb.h>
50 #endif /* WIN32 || __EMX__ */
51
52
53 /*
54 * 'main()' - Send a file to the printer or server.
55 *
56 * Usage:
57 *
58 * printer-uri job-id user title copies options [file]
59 */
60
61 int /* O - Exit status */
62 main(int argc, /* I - Number of command-line arguments (6 or 7) */
63 char *argv[]) /* I - Command-line arguments */
64 {
65 char method[255], /* Method in URI */
66 hostname[1024], /* Hostname */
67 username[255], /* Username info (not used) */
68 resource[1024]; /* Resource info (not used) */
69 FILE *fp; /* Print file */
70 int copies; /* Number of copies to print */
71 int port; /* Port number */
72 int delay; /* Delay for retries... */
73 int fd; /* AppSocket */
74 int error; /* Error code (if any) */
75 struct sockaddr_in addr; /* Socket address */
76 struct hostent *hostaddr; /* Host address */
77 int wbytes; /* Number of bytes written */
78 size_t nbytes, /* Number of bytes read */
79 tbytes; /* Total number of bytes written */
80 char buffer[8192], /* Output buffer */
81 *bufptr; /* Pointer into buffer */
82 struct timeval timeout; /* Timeout for select() */
83 fd_set input; /* Input set for select() */
84 #if defined(HAVE_SIGACTION) && !defined(HAVE_SIGSET)
85 struct sigaction action; /* Actions for POSIX signals */
86 #endif /* HAVE_SIGACTION && !HAVE_SIGSET */
87
88
89 /*
90 * Make sure status messages are not buffered...
91 */
92
93 setbuf(stderr, NULL);
94
95 /*
96 * Check command-line...
97 */
98
99 if (argc == 1)
100 {
101 puts("network socket \"Unknown\" \"AppSocket/HP JetDirect\"");
102 return (0);
103 }
104 else if (argc < 6 || argc > 7)
105 {
106 fprintf(stderr, "Usage: %s job-id user title copies options [file]\n",
107 argv[0]);
108 return (1);
109 }
110
111 /*
112 * If we have 7 arguments, print the file named on the command-line.
113 * Otherwise, send stdin instead...
114 */
115
116 if (argc == 6)
117 {
118 fp = stdin;
119 copies = 1;
120 }
121 else
122 {
123 /*
124 * Try to open the print file...
125 */
126
127 if ((fp = fopen(argv[6], "rb")) == NULL)
128 {
129 perror("ERROR: unable to open print file");
130 return (1);
131 }
132
133 copies = atoi(argv[4]);
134 }
135
136 /*
137 * Extract the hostname and port number from the URI...
138 */
139
140 httpSeparate(argv[0], method, username, hostname, &port, resource);
141
142 if (port == 0)
143 port = 9100; /* Default to HP JetDirect/Tektronix PhaserShare */
144
145 /*
146 * Then try to connect to the remote host...
147 */
148
149 if ((hostaddr = gethostbyname(hostname)) == NULL)
150 {
151 fprintf(stderr, "ERROR: Unable to locate printer \'%s\' - %s\n",
152 hostname, strerror(errno));
153 return (1);
154 }
155
156 fprintf(stderr, "INFO: Attempting to connect to host %s on port %d\n",
157 hostname, port);
158
159 memset(&addr, 0, sizeof(addr));
160 memcpy(&(addr.sin_addr), hostaddr->h_addr, hostaddr->h_length);
161 addr.sin_family = hostaddr->h_addrtype;
162 addr.sin_port = htons(port);
163
164 while (copies > 0)
165 {
166 for (delay = 5;;)
167 {
168 if ((fd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
169 {
170 perror("ERROR: Unable to create socket");
171 return (1);
172 }
173
174 if (connect(fd, (struct sockaddr *)&addr, sizeof(addr)) < 0)
175 {
176 error = errno;
177 close(fd);
178 fd = -1;
179
180 if (error == ECONNREFUSED)
181 {
182 fprintf(stderr, "INFO: Network host \'%s\' is busy; will retry in %d seconds...\n",
183 hostname, delay);
184 sleep(delay);
185
186 if (delay < 30)
187 delay += 5;
188 }
189 else
190 {
191 perror("ERROR: Unable to connect to printer (retrying in 30 seconds)");
192 sleep(30);
193 }
194 }
195 else
196 break;
197 }
198
199 /*
200 * Now that we are "connected" to the port, ignore SIGTERM so that we
201 * can finish out any page data the driver sends (e.g. to eject the
202 * current page...
203 */
204
205 #ifdef HAVE_SIGSET /* Use System V signals over POSIX to avoid bugs */
206 sigset(SIGTERM, SIG_IGN);
207 #elif defined(HAVE_SIGACTION)
208 memset(&action, 0, sizeof(action));
209
210 sigemptyset(&action.sa_mask);
211 action.sa_handler = SIG_IGN;
212 sigaction(SIGTERM, &action, NULL);
213 #else
214 signal(SIGTERM, SIG_IGN);
215 #endif /* HAVE_SIGSET */
216
217 /*
218 * Finally, send the print file...
219 */
220
221 copies --;
222
223 if (fp != stdin)
224 {
225 fputs("PAGE: 1 1\n", stderr);
226 rewind(fp);
227 }
228
229 fputs("INFO: Connected to host, sending print job...\n", stderr);
230
231 tbytes = 0;
232 while ((nbytes = fread(buffer, 1, sizeof(buffer), fp)) > 0)
233 {
234 /*
235 * Write the print data to the printer...
236 */
237
238 tbytes += nbytes;
239 bufptr = buffer;
240
241 while (nbytes > 0)
242 {
243 if ((wbytes = send(fd, bufptr, nbytes, 0)) < 0)
244 {
245 perror("ERROR: Unable to send print file to printer");
246 break;
247 }
248
249 nbytes -= wbytes;
250 bufptr += wbytes;
251 }
252
253 /*
254 * Check for possible data coming back from the printer...
255 */
256
257 timeout.tv_sec = 0;
258 timeout.tv_usec = 0;
259
260 FD_ZERO(&input);
261 FD_SET(fd, &input);
262 #ifdef __hpux
263 if (select(fd + 1, (int *)&input, NULL, NULL, &timeout) > 0)
264 #else
265 if (select(fd + 1, &input, NULL, NULL, &timeout) > 0)
266 #endif /* __hpux */
267 {
268 /*
269 * Grab the data coming back and spit it out to stderr...
270 */
271
272 if ((nbytes = recv(fd, buffer, sizeof(buffer), 0)) > 0)
273 fprintf(stderr, "INFO: Received %u bytes of back-channel data!\n",
274 nbytes);
275 }
276 else if (argc > 6)
277 fprintf(stderr, "INFO: Sending print file, %u bytes...\n", tbytes);
278 }
279
280 /*
281 * Shutdown the socket and wait for the other end to finish...
282 */
283
284 fputs("INFO: Print file sent, waiting for printer to finish...\n", stderr);
285
286 shutdown(fd, 1);
287
288 for (;;)
289 {
290 /*
291 * Wait a maximum of 90 seconds for backchannel data or a closed
292 * connection...
293 */
294
295 timeout.tv_sec = 90;
296 timeout.tv_usec = 0;
297
298 FD_ZERO(&input);
299 FD_SET(fd, &input);
300
301 #ifdef __hpux
302 if (select(fd + 1, (int *)&input, NULL, NULL, &timeout) > 0)
303 #else
304 if (select(fd + 1, &input, NULL, NULL, &timeout) > 0)
305 #endif /* __hpux */
306 {
307 /*
308 * Grab the data coming back and spit it out to stderr...
309 */
310
311 if ((nbytes = recv(fd, buffer, sizeof(buffer), 0)) > 0)
312 fprintf(stderr, "INFO: Received %u bytes of back-channel data!\n",
313 nbytes);
314 else
315 break;
316 }
317 else
318 break;
319 }
320
321 /*
322 * Close the socket connection...
323 */
324
325 close(fd);
326 }
327
328 /*
329 * Close the input file and return...
330 */
331
332 if (fp != stdin)
333 fclose(fp);
334
335 return (0);
336 }
337
338
339 /*
340 * End of "$Id: socket.c,v 1.19 2001/04/18 16:28:34 mike Exp $".
341 */