Now that we have Subversion's file renaming ability, it's time at
[u/mdw/putty] / windows / winprint.c
CommitLineData
09ad523a 1/*
2 * Printing interface for PuTTY.
3 */
4
09ad523a 5#include "putty.h"
4e95095a 6#include <winspool.h>
11773fc1 7
09ad523a 8struct printer_enum_tag {
9 int nprinters;
4c48c989 10 DWORD enum_level;
11 union {
12 LPPRINTER_INFO_4 i4;
13 LPPRINTER_INFO_5 i5;
14 } info;
09ad523a 15};
16
17struct printer_job_tag {
18 HANDLE hprinter;
19};
20
4c48c989 21static char *printer_add_enum(int param, DWORD level, char *buffer,
11773fc1 22 int offset, int *nprinters_ptr)
23{
24 DWORD needed, nprinters;
25
3d88e64d 26 buffer = sresize(buffer, offset+512, char);
11773fc1 27
01f7687f 28 /*
29 * Exploratory call to EnumPrinters to determine how much space
30 * we'll need for the output. Discard the return value since it
31 * will almost certainly be a failure due to lack of space.
32 */
4c48c989 33 EnumPrinters(param, NULL, level, buffer+offset, 512,
01f7687f 34 &needed, &nprinters);
11773fc1 35
36 if (needed < 512)
37 needed = 512;
38
3d88e64d 39 buffer = sresize(buffer, offset+needed, char);
11773fc1 40
4c48c989 41 if (EnumPrinters(param, NULL, level, buffer+offset,
11773fc1 42 needed, &needed, &nprinters) == 0)
43 return NULL;
44
45 *nprinters_ptr += nprinters;
46
47 return buffer;
48}
49
09ad523a 50printer_enum *printer_start_enum(int *nprinters_ptr)
51{
3d88e64d 52 printer_enum *ret = snew(printer_enum);
11773fc1 53 char *buffer = NULL, *retval;
09ad523a 54
55 *nprinters_ptr = 0; /* default return value */
3d88e64d 56 buffer = snewn(512, char);
11773fc1 57
4c48c989 58 /*
59 * Determine what enumeration level to use.
60 * When enumerating printers, we need to use PRINTER_INFO_4 on
61 * NT-class systems to avoid Windows looking too hard for them and
62 * slowing things down; and we need to avoid PRINTER_INFO_5 as
63 * we've seen network printers not show up.
64 * On 9x-class systems, PRINTER_INFO_4 isn't available and
65 * PRINTER_INFO_5 is recommended.
66 * Bletch.
67 */
68 if (osVersion.dwPlatformId != VER_PLATFORM_WIN32_NT) {
69 ret->enum_level = 5;
70 } else {
71 ret->enum_level = 4;
72 }
73
a5751e2d 74 retval = printer_add_enum(PRINTER_ENUM_LOCAL | PRINTER_ENUM_CONNECTIONS,
4c48c989 75 ret->enum_level, buffer, 0, nprinters_ptr);
11773fc1 76 if (!retval)
77 goto error;
78 else
79 buffer = retval;
80
4c48c989 81 switch (ret->enum_level) {
82 case 4:
83 ret->info.i4 = (LPPRINTER_INFO_4)buffer;
84 break;
85 case 5:
86 ret->info.i5 = (LPPRINTER_INFO_5)buffer;
87 break;
88 }
11773fc1 89 ret->nprinters = *nprinters_ptr;
09ad523a 90
91 return ret;
92
93 error:
94 sfree(buffer);
95 sfree(ret);
11773fc1 96 *nprinters_ptr = 0;
09ad523a 97 return NULL;
98}
99
100char *printer_get_name(printer_enum *pe, int i)
101{
102 if (!pe)
103 return NULL;
104 if (i < 0 || i >= pe->nprinters)
105 return NULL;
4c48c989 106 switch (pe->enum_level) {
107 case 4:
108 return pe->info.i4[i].pPrinterName;
109 case 5:
110 return pe->info.i5[i].pPrinterName;
111 default:
112 return NULL;
113 }
09ad523a 114}
115
116void printer_finish_enum(printer_enum *pe)
117{
118 if (!pe)
119 return;
4c48c989 120 switch (pe->enum_level) {
121 case 4:
122 sfree(pe->info.i4);
123 break;
124 case 5:
125 sfree(pe->info.i5);
126 break;
127 }
09ad523a 128 sfree(pe);
129}
130
131printer_job *printer_start_job(char *printer)
132{
3d88e64d 133 printer_job *ret = snew(printer_job);
09ad523a 134 DOC_INFO_1 docinfo;
135 int jobstarted = 0, pagestarted = 0;
136
137 ret->hprinter = NULL;
138 if (!OpenPrinter(printer, &ret->hprinter, NULL))
139 goto error;
140
141 docinfo.pDocName = "PuTTY remote printer output";
142 docinfo.pOutputFile = NULL;
143 docinfo.pDatatype = "RAW";
144
145 if (!StartDocPrinter(ret->hprinter, 1, (LPSTR)&docinfo))
146 goto error;
147 jobstarted = 1;
148
149 if (!StartPagePrinter(ret->hprinter))
150 goto error;
151 pagestarted = 1;
152
153 return ret;
154
155 error:
156 if (pagestarted)
157 EndPagePrinter(ret->hprinter);
158 if (jobstarted)
159 EndDocPrinter(ret->hprinter);
160 if (ret->hprinter)
161 ClosePrinter(ret->hprinter);
162 sfree(ret);
163 return NULL;
164}
165
166void printer_job_data(printer_job *pj, void *data, int len)
167{
168 DWORD written;
169
170 if (!pj)
171 return;
172
173 WritePrinter(pj->hprinter, data, len, &written);
174}
175
176void printer_finish_job(printer_job *pj)
177{
178 if (!pj)
179 return;
180
181 EndPagePrinter(pj->hprinter);
182 EndDocPrinter(pj->hprinter);
183 ClosePrinter(pj->hprinter);
184 sfree(pj);
185}