Commit 99d3b0ed02e0500d81855cbf12dd24983d02de78
1 parent
9f0e9f3d
Exists in
master
and in
1 other branch
Refactoring IPC plugin for windows.
Showing
3 changed files
with
679 additions
and
673 deletions
Show diff stats
hllapi.cbp
@@ -44,6 +44,9 @@ | @@ -44,6 +44,9 @@ | ||
44 | <Option compilerVar="CC" /> | 44 | <Option compilerVar="CC" /> |
45 | </Unit> | 45 | </Unit> |
46 | <Unit filename="src/plugin/linux/service.h" /> | 46 | <Unit filename="src/plugin/linux/service.h" /> |
47 | + <Unit filename="src/plugin/windows/main.c"> | ||
48 | + <Option compilerVar="CC" /> | ||
49 | + </Unit> | ||
47 | <Unit filename="src/testprogram/testprogram.c"> | 50 | <Unit filename="src/testprogram/testprogram.c"> |
48 | <Option compilerVar="CC" /> | 51 | <Option compilerVar="CC" /> |
49 | </Unit> | 52 | </Unit> |
src/hllapi/pluginmain.c
@@ -1,673 +0,0 @@ | @@ -1,673 +0,0 @@ | ||
1 | -/* | ||
2 | - * "Software pw3270, desenvolvido com base nos códigos fontes do WC3270 e X3270 | ||
3 | - * (Paul Mattes Paul.Mattes@usa.net), de emulação de terminal 3270 para acesso a | ||
4 | - * aplicativos mainframe. Registro no INPI sob o nome G3270. Registro no INPI sob o nome G3270. | ||
5 | - * | ||
6 | - * Copyright (C) <2008> <Banco do Brasil S.A.> | ||
7 | - * | ||
8 | - * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob | ||
9 | - * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela | ||
10 | - * Free Software Foundation. | ||
11 | - * | ||
12 | - * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER | ||
13 | - * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO | ||
14 | - * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para | ||
15 | - * obter mais detalhes. | ||
16 | - * | ||
17 | - * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este | ||
18 | - * programa; se não, escreva para a Free Software Foundation, Inc., 51 Franklin | ||
19 | - * St, Fifth Floor, Boston, MA 02110-1301 USA | ||
20 | - * | ||
21 | - * Este programa está nomeado como pluginmain.c e possui - linhas de código. | ||
22 | - * | ||
23 | - * Contatos: | ||
24 | - * | ||
25 | - * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) | ||
26 | - * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) | ||
27 | - * | ||
28 | - * Agradecimento: | ||
29 | - * | ||
30 | - * Hélio Passos | ||
31 | - * | ||
32 | - */ | ||
33 | - | ||
34 | - #include "server.h" | ||
35 | - | ||
36 | -#ifdef _WIN32 | ||
37 | - #include <windows.h> | ||
38 | -#else | ||
39 | - #error HLLAPI is designed for windows. | ||
40 | -#endif // _WIN32 | ||
41 | - | ||
42 | - #include <pw3270/plugin.h> | ||
43 | - #include <v3270.h> | ||
44 | - #include <pw3270/ipcpackets.h> | ||
45 | - #include <lib3270/actions.h> | ||
46 | - #include <lib3270/charset.h> | ||
47 | - | ||
48 | -/*--[ Defines ]--------------------------------------------------------------------------------------*/ | ||
49 | - | ||
50 | - #pragma pack(1) | ||
51 | - | ||
52 | - enum PIPE_STATE | ||
53 | - { | ||
54 | - PIPE_STATE_WAITING, | ||
55 | - PIPE_STATE_READ, | ||
56 | - PIPE_STATE_PENDING_READ, | ||
57 | - PIPE_STATE_UNDEFINED | ||
58 | - }; | ||
59 | - | ||
60 | - typedef struct _pipe_source | ||
61 | - { | ||
62 | - GSource gsrc; | ||
63 | - HANDLE hPipe; | ||
64 | - | ||
65 | - enum PIPE_STATE state; | ||
66 | - | ||
67 | - OVERLAPPED overlap; | ||
68 | - unsigned char buffer[PIPE_BUFFER_LENGTH+1]; | ||
69 | - } pipe_source; | ||
70 | - | ||
71 | - #pragma pack() | ||
72 | - | ||
73 | - | ||
74 | -/*--[ Globals ]--------------------------------------------------------------------------------------*/ | ||
75 | - | ||
76 | -// static const gchar control_char = '@'; | ||
77 | - | ||
78 | -/*--[ Implement ]------------------------------------------------------------------------------------*/ | ||
79 | - | ||
80 | - static void IO_accept(pipe_source *source) | ||
81 | - { | ||
82 | - set_active(FALSE); | ||
83 | - | ||
84 | - if(ConnectNamedPipe(source->hPipe,&source->overlap)) | ||
85 | - { | ||
86 | - popup_lasterror("%s",_( "Error in ConnectNamedPipe" )); | ||
87 | - return; | ||
88 | - } | ||
89 | - | ||
90 | - switch(GetLastError()) | ||
91 | - { | ||
92 | - // The overlapped connection in progress. | ||
93 | - case ERROR_IO_PENDING: | ||
94 | - // trace("%s: ERROR_IO_PENDING",__FUNCTION__); | ||
95 | - source->state = PIPE_STATE_WAITING; | ||
96 | - break; | ||
97 | - | ||
98 | - // Client is already connected, so signal an event. | ||
99 | - case ERROR_PIPE_CONNECTED: | ||
100 | - trace("%s: ERROR_PIPE_CONNECTED",__FUNCTION__); | ||
101 | - set_active(TRUE); | ||
102 | - if(SetEvent(source->overlap.hEvent)) | ||
103 | - break; | ||
104 | - | ||
105 | - // If an error occurs during the connect operation... | ||
106 | - default: | ||
107 | - popup_lasterror("%s", _( "ConnectNamedPipe failed" )); | ||
108 | - } | ||
109 | - | ||
110 | - } | ||
111 | - | ||
112 | - static gboolean IO_prepare(GSource *source, gint *timeout) | ||
113 | - { | ||
114 | - /* | ||
115 | - * Called before all the file descriptors are polled. | ||
116 | - * If the source can determine that it is ready here | ||
117 | - * (without waiting for the results of the poll() call) | ||
118 | - * it should return TRUE. | ||
119 | - * | ||
120 | - * It can also return a timeout_ value which should be the maximum | ||
121 | - * timeout (in milliseconds) which should be passed to the poll() call. | ||
122 | - * The actual timeout used will be -1 if all sources returned -1, | ||
123 | - * or it will be the minimum of all the timeout_ values | ||
124 | - * returned which were >= 0. | ||
125 | - * | ||
126 | - */ | ||
127 | - if(WaitForSingleObject(((pipe_source *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | ||
128 | - return TRUE; | ||
129 | - | ||
130 | - *timeout = 10; | ||
131 | - return FALSE; | ||
132 | - } | ||
133 | - | ||
134 | - static gboolean IO_check(GSource *source) | ||
135 | - { | ||
136 | - /* | ||
137 | - * Called after all the file descriptors are polled. | ||
138 | - * The source should return TRUE if it is ready to be dispatched. | ||
139 | - * Note that some time may have passed since the previous prepare | ||
140 | - * function was called, so the source should be checked again here. | ||
141 | - * | ||
142 | - */ | ||
143 | - if(WaitForSingleObject(((pipe_source *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | ||
144 | - return TRUE; | ||
145 | - | ||
146 | - return FALSE; | ||
147 | - } | ||
148 | - | ||
149 | - static void send_text(pipe_source *source, char *text) | ||
150 | - { | ||
151 | - struct hllapi_packet_text *pkt; | ||
152 | - DWORD szBlock; | ||
153 | - int f; | ||
154 | - | ||
155 | - if(text) | ||
156 | - { | ||
157 | - szBlock = sizeof(struct hllapi_packet_text)+strlen(text); | ||
158 | - pkt = (struct hllapi_packet_text *) g_malloc0(szBlock); | ||
159 | - pkt->packet_id = 0; | ||
160 | - strcpy(pkt->text,text); | ||
161 | - lib3270_free(text); | ||
162 | - } | ||
163 | - else | ||
164 | - { | ||
165 | - szBlock = sizeof(struct hllapi_packet_text); | ||
166 | - pkt = (struct hllapi_packet_text *) g_malloc0(szBlock); | ||
167 | - pkt->packet_id = errno ? errno : -1; | ||
168 | - } | ||
169 | - | ||
170 | - trace("szBlock=%d text=\"%s\"",szBlock, ( (struct hllapi_packet_text *) pkt)->text); | ||
171 | - for(f=0;f< (int) szBlock;f++) | ||
172 | - { | ||
173 | - trace("rsp(%d)= %d \"%s\"",f,* (((char *) pkt)+f),((char *) pkt)+f); | ||
174 | - } | ||
175 | - | ||
176 | - WriteFile(source->hPipe,pkt,szBlock,&szBlock,NULL); | ||
177 | - | ||
178 | - g_free(pkt); | ||
179 | - } | ||
180 | - | ||
181 | - static void send_result(pipe_source *source, int rc) | ||
182 | - { | ||
183 | - struct hllapi_packet_result pkt = { rc }; | ||
184 | - DWORD wrote = sizeof(pkt); | ||
185 | - WriteFile(source->hPipe,&pkt,wrote,&wrote,NULL); | ||
186 | - } | ||
187 | - | ||
188 | - static int do_file_transfer(struct hllapi_packet_file_transfer * source) | ||
189 | - { | ||
190 | - /* | ||
191 | - const gchar * local = (const char *) source->text; | ||
192 | - const gchar * remote = (const char *) (local+strlen(local)+1); | ||
193 | - | ||
194 | - return v3270_transfer_file( v3270_get_default_widget(), | ||
195 | - (LIB3270_FT_OPTION) source->options, | ||
196 | - local, | ||
197 | - remote, | ||
198 | - source->lrecl, | ||
199 | - source->blksize, | ||
200 | - source->primspace, | ||
201 | - source->secspace, | ||
202 | - source->dft ); | ||
203 | - */ | ||
204 | - return EINVAL; | ||
205 | - } | ||
206 | - | ||
207 | - static void get_host(pipe_source *source) { | ||
208 | - send_text(source,strdup(lib3270_get_url(lib3270_get_default_session_handle()))); | ||
209 | - } | ||
210 | - | ||
211 | - static void process_input(pipe_source *source, DWORD cbRead) | ||
212 | - { | ||
213 | - const struct hllapi_packet_query * query = ((struct hllapi_packet_query *) source->buffer); | ||
214 | - | ||
215 | - trace("%s id=%d",__FUNCTION__,query->packet_id); | ||
216 | - | ||
217 | - switch(query->packet_id) | ||
218 | - { | ||
219 | - case HLLAPI_PACKET_CONNECT: | ||
220 | - send_result(source,lib3270_reconnect( lib3270_get_default_session_handle(), | ||
221 | - ((struct hllapi_packet_connect *) source->buffer)->wait)); | ||
222 | - break; | ||
223 | - | ||
224 | - case HLLAPI_PACKET_CONNECT_URL: | ||
225 | - send_result(source,lib3270_connect_url(lib3270_get_default_session_handle(),(const char *) (query+1),0)); | ||
226 | - break; | ||
227 | - | ||
228 | - case HLLAPI_PACKET_SET_HOST: | ||
229 | - send_result(source,lib3270_set_url(lib3270_get_default_session_handle(), | ||
230 | - ((struct hllapi_packet_text *) source->buffer)->text)); | ||
231 | - break; | ||
232 | - | ||
233 | - case HLLAPI_PACKET_GET_HOST: | ||
234 | - get_host(source); | ||
235 | - break; | ||
236 | - | ||
237 | - case HLLAPI_PACKET_DISCONNECT: | ||
238 | - send_result(source,lib3270_disconnect(lib3270_get_default_session_handle())); | ||
239 | - break; | ||
240 | - | ||
241 | - case HLLAPI_PACKET_IS_CONNECTED: | ||
242 | - send_result(source,lib3270_in_tn3270e(lib3270_get_default_session_handle())); | ||
243 | - break; | ||
244 | - | ||
245 | - case HLLAPI_PACKET_IS_READY: | ||
246 | - send_result(source,lib3270_is_ready(lib3270_get_default_session_handle())); | ||
247 | - break; | ||
248 | - | ||
249 | - case HLLAPI_PACKET_ENTER: | ||
250 | - send_result(source,lib3270_enter(lib3270_get_default_session_handle())); | ||
251 | - break; | ||
252 | - | ||
253 | - case HLLAPI_PACKET_PRINT: | ||
254 | - send_result(source,lib3270_print_all(lib3270_get_default_session_handle())); | ||
255 | - break; | ||
256 | - | ||
257 | - case HLLAPI_PACKET_ERASE: | ||
258 | - send_result(source,lib3270_erase(lib3270_get_default_session_handle())); | ||
259 | - break; | ||
260 | - | ||
261 | - case HLLAPI_PACKET_ERASE_EOF: | ||
262 | - send_result(source,lib3270_eraseeof(lib3270_get_default_session_handle())); | ||
263 | - break; | ||
264 | - | ||
265 | - case HLLAPI_PACKET_ERASE_EOL: | ||
266 | - send_result(source,lib3270_eraseeol(lib3270_get_default_session_handle())); | ||
267 | - break; | ||
268 | - | ||
269 | - case HLLAPI_PACKET_ERASE_INPUT: | ||
270 | - send_result(source,lib3270_eraseinput(lib3270_get_default_session_handle())); | ||
271 | - break; | ||
272 | - | ||
273 | - case HLLAPI_PACKET_PFKEY: | ||
274 | - send_result(source,lib3270_pfkey( lib3270_get_default_session_handle(), | ||
275 | - ((struct hllapi_packet_keycode *) source->buffer)->keycode)); | ||
276 | - break; | ||
277 | - | ||
278 | - case HLLAPI_PACKET_PAKEY: | ||
279 | - send_result(source,lib3270_pakey( lib3270_get_default_session_handle(), | ||
280 | - ((struct hllapi_packet_keycode *) source->buffer)->keycode)); | ||
281 | - break; | ||
282 | - | ||
283 | - case HLLAPI_PACKET_SET_CURSOR_POSITION: | ||
284 | - send_result(source,lib3270_set_cursor_position( lib3270_get_default_session_handle(), | ||
285 | - ((struct hllapi_packet_cursor *) source->buffer)->row, | ||
286 | - ((struct hllapi_packet_cursor *) source->buffer)->col)); | ||
287 | - break; | ||
288 | - | ||
289 | - case HLLAPI_PACKET_SET_TEXT_AT: | ||
290 | - send_result(source,lib3270_set_text_at( lib3270_get_default_session_handle(), | ||
291 | - ((struct hllapi_packet_text_at *) source->buffer)->row, | ||
292 | - ((struct hllapi_packet_text_at *) source->buffer)->col, | ||
293 | - (unsigned char *) ((struct hllapi_packet_text_at *) source->buffer)->text)); | ||
294 | - break; | ||
295 | - | ||
296 | - case HLLAPI_PACKET_GET_TEXT_AT: | ||
297 | - send_text(source,lib3270_get_string_at( lib3270_get_default_session_handle(), | ||
298 | - ((struct hllapi_packet_at *) source->buffer)->row, | ||
299 | - ((struct hllapi_packet_at *) source->buffer)->col, | ||
300 | - ((struct hllapi_packet_at *) source->buffer)->len, | ||
301 | - ((struct hllapi_packet_at *) source->buffer)->lf)); | ||
302 | - break; | ||
303 | - | ||
304 | - case HLLAPI_PACKET_GET_TEXT_AT_OFFSET: | ||
305 | - send_text(source,lib3270_get_string_at_address( lib3270_get_default_session_handle(), | ||
306 | - ((struct hllapi_packet_query_offset *) source->buffer)->addr, | ||
307 | - ((struct hllapi_packet_query_offset *) source->buffer)->len, | ||
308 | - ((struct hllapi_packet_query_offset *) source->buffer)->lf)); | ||
309 | - break; | ||
310 | - | ||
311 | - case HLLAPI_PACKET_CMP_TEXT_AT: | ||
312 | - send_result(source,lib3270_cmp_text_at( lib3270_get_default_session_handle(), | ||
313 | - ((struct hllapi_packet_text_at *) source->buffer)->row, | ||
314 | - ((struct hllapi_packet_text_at *) source->buffer)->col, | ||
315 | - ((struct hllapi_packet_text_at *) source->buffer)->text, | ||
316 | - ((struct hllapi_packet_text_at *) source->buffer)->lf)); | ||
317 | - break; | ||
318 | - | ||
319 | - case HLLAPI_PACKET_INPUT_STRING: | ||
320 | - send_result(source,lib3270_input_string(lib3270_get_default_session_handle(), | ||
321 | - (unsigned char *) ((struct hllapi_packet_text *) source->buffer)->text)); | ||
322 | - break; | ||
323 | - | ||
324 | - case HLLAPI_PACKET_EMULATE_INPUT: | ||
325 | - send_result(source,lib3270_emulate_input(lib3270_get_default_session_handle(), | ||
326 | - (const char *) ((struct hllapi_packet_emulate_input *) source->buffer)->text, | ||
327 | - (int) ((struct hllapi_packet_emulate_input *) source->buffer)->len, | ||
328 | - (int) ((struct hllapi_packet_emulate_input *) source->buffer)->pasting)); | ||
329 | - break; | ||
330 | - | ||
331 | - case HLLAPI_PACKET_SET_CURSOR: | ||
332 | - send_result(source,lib3270_set_cursor_address(lib3270_get_default_session_handle(), | ||
333 | - ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
334 | - break; | ||
335 | - | ||
336 | - case HLLAPI_PACKET_GET_CURSOR: | ||
337 | - send_result(source,lib3270_get_cursor_address(lib3270_get_default_session_handle())); | ||
338 | - break; | ||
339 | - | ||
340 | - case HLLAPI_PACKET_GET_WIDTH: | ||
341 | - send_result(source,lib3270_get_width(lib3270_get_default_session_handle())); | ||
342 | - break; | ||
343 | - | ||
344 | - case HLLAPI_PACKET_GET_HEIGHT: | ||
345 | - send_result(source,lib3270_get_height(lib3270_get_default_session_handle())); | ||
346 | - break; | ||
347 | - | ||
348 | - case HLLAPI_PACKET_GET_LENGTH: | ||
349 | - send_result(source,lib3270_get_length(lib3270_get_default_session_handle())); | ||
350 | - break; | ||
351 | - | ||
352 | - case HLLAPI_PACKET_GET_PROGRAM_MESSAGE: | ||
353 | - send_result(source,lib3270_get_program_message(lib3270_get_default_session_handle())); | ||
354 | - break; | ||
355 | - | ||
356 | - case HLLAPI_PACKET_GET_SSL_STATE: | ||
357 | - send_result(source,lib3270_get_secure(lib3270_get_default_session_handle())); | ||
358 | - break; | ||
359 | - | ||
360 | - case HLLAPI_PACKET_SET_UNLOCK_DELAY: | ||
361 | - lib3270_set_unlock_delay(lib3270_get_default_session_handle(),(unsigned short) ((struct hllapi_packet_set_int *) source->buffer)->value); | ||
362 | - send_result(source,0); | ||
363 | - break; | ||
364 | - | ||
365 | - case HLLAPI_PACKET_SET_TOGGLE: | ||
366 | - send_result(source,lib3270_set_toggle(lib3270_get_default_session_handle(), | ||
367 | - (LIB3270_TOGGLE) ((struct hllapi_packet_set *) source->buffer)->id, | ||
368 | - ((struct hllapi_packet_set *) source->buffer)->value)); | ||
369 | - break; | ||
370 | - | ||
371 | - case HLLAPI_PACKET_FIELD_START: | ||
372 | - send_result(source,lib3270_get_field_start(lib3270_get_default_session_handle(), | ||
373 | - ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
374 | - break; | ||
375 | - | ||
376 | - | ||
377 | - case HLLAPI_PACKET_FIELD_LEN: | ||
378 | - send_result(source,lib3270_get_field_len(lib3270_get_default_session_handle(), | ||
379 | - ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
380 | - break; | ||
381 | - | ||
382 | - case HLLAPI_PACKET_NEXT_UNPROTECTED: | ||
383 | - send_result(source,lib3270_get_next_unprotected(lib3270_get_default_session_handle(), | ||
384 | - ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
385 | - break; | ||
386 | - | ||
387 | - case HLLAPI_PACKET_IS_PROTECTED: | ||
388 | - send_result(source,lib3270_get_is_protected(lib3270_get_default_session_handle(), | ||
389 | - ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
390 | - break; | ||
391 | - | ||
392 | - case HLLAPI_PACKET_IS_PROTECTED_AT: | ||
393 | - send_result(source,lib3270_get_is_protected_at( lib3270_get_default_session_handle(), | ||
394 | - ((struct hllapi_packet_query_at *) source->buffer)->row, | ||
395 | - ((struct hllapi_packet_query_at *) source->buffer)->col)); | ||
396 | - break; | ||
397 | - | ||
398 | - case HLLAPI_PACKET_QUIT: | ||
399 | - gtk_main_quit(); | ||
400 | - send_result(source,0); | ||
401 | - break; | ||
402 | - | ||
403 | - case HLLAPI_PACKET_SET_HOST_CHARSET: | ||
404 | - send_result(source,lib3270_set_host_charset( lib3270_get_default_session_handle(), | ||
405 | - (const char *) ((struct hllapi_packet_set_text *) source->buffer)->text)); | ||
406 | - break; | ||
407 | - | ||
408 | - case HLLAPI_PACKET_ASC2EBC: | ||
409 | - send_text(source,(char *) lib3270_asc2ebc( | ||
410 | - lib3270_get_default_session_handle(), | ||
411 | - (unsigned char *) ((struct hllapi_packet_set_text *) source->buffer)->text,-1 | ||
412 | - )); | ||
413 | - break; | ||
414 | - | ||
415 | - case HLLAPI_PACKET_EBC2ASC: | ||
416 | - send_text(source,(char *) lib3270_ebc2asc( | ||
417 | - lib3270_get_default_session_handle(), | ||
418 | - (unsigned char *) ((struct hllapi_packet_set_text *) source->buffer)->text,-1 | ||
419 | - )); | ||
420 | - break; | ||
421 | - | ||
422 | - case HLLAPI_PACKET_FILE_TRANSFER: | ||
423 | - send_result(source,do_file_transfer((struct hllapi_packet_file_transfer *) source)); | ||
424 | - break; | ||
425 | - | ||
426 | - case HLLAPI_PACKET_GET_HOST_CHARSET: | ||
427 | - trace("%s","HLLAPI_PACKET_GET_HOST_CHARSET"); | ||
428 | - send_text(source,(char *) lib3270_get_host_charset(lib3270_get_default_session_handle())); | ||
429 | - break; | ||
430 | - | ||
431 | - case HLLAPI_PACKET_ACTION: | ||
432 | - send_result(source,lib3270_action(lib3270_get_default_session_handle(), | ||
433 | - (const char *) ((struct hllapi_packet_text *) source->buffer)->text)); | ||
434 | - break; | ||
435 | - | ||
436 | - | ||
437 | - default: | ||
438 | - send_result(source, EINVAL); | ||
439 | - g_message("Invalid remote request (id=%d)",source->buffer[0]); | ||
440 | - } | ||
441 | - | ||
442 | - } | ||
443 | - | ||
444 | - static void read_input_pipe(pipe_source *source) | ||
445 | - { | ||
446 | - DWORD cbRead = 0; | ||
447 | - | ||
448 | - if(ReadFile(source->hPipe,source->buffer,PIPE_BUFFER_LENGTH,&cbRead,&source->overlap) && cbRead > 0) | ||
449 | - process_input(source,cbRead); | ||
450 | - | ||
451 | - // The read operation is still pending. | ||
452 | - switch(GetLastError()) | ||
453 | - { | ||
454 | - case 0: | ||
455 | - break; | ||
456 | - | ||
457 | - case ERROR_IO_PENDING: | ||
458 | - // trace("%s: PIPE_STATE_PENDING_READ",__FUNCTION__); | ||
459 | - source->state = PIPE_STATE_PENDING_READ; | ||
460 | - break; | ||
461 | - | ||
462 | - case ERROR_PIPE_LISTENING: | ||
463 | - // trace("%s: ERROR_PIPE_LISTENING",__FUNCTION__); | ||
464 | - source->state = PIPE_STATE_READ; | ||
465 | - break; | ||
466 | - | ||
467 | - case ERROR_BROKEN_PIPE: | ||
468 | - trace("%s: ERROR_BROKEN_PIPE",__FUNCTION__); | ||
469 | - | ||
470 | - if(!DisconnectNamedPipe(source->hPipe)) | ||
471 | - { | ||
472 | - set_active(FALSE); | ||
473 | - popup_lasterror("%s",_( "Error in DisconnectNamedPipe" )); | ||
474 | - } | ||
475 | - else | ||
476 | - { | ||
477 | - IO_accept(source); | ||
478 | - } | ||
479 | - break; | ||
480 | - | ||
481 | - case ERROR_PIPE_NOT_CONNECTED: | ||
482 | - trace("%s: ERROR_PIPE_NOT_CONNECTED",__FUNCTION__); | ||
483 | - set_active(FALSE); | ||
484 | - break; | ||
485 | - | ||
486 | - default: | ||
487 | - if(source->hPipe != INVALID_HANDLE_VALUE) | ||
488 | - popup_lasterror("%s",_( "Error receiving message from pipe" ) ); | ||
489 | - } | ||
490 | - | ||
491 | - } | ||
492 | - | ||
493 | - static gboolean IO_dispatch(GSource *source, GSourceFunc callback, gpointer data) | ||
494 | - { | ||
495 | - /* | ||
496 | - * Called to dispatch the event source, | ||
497 | - * after it has returned TRUE in either its prepare or its check function. | ||
498 | - * The dispatch function is passed in a callback function and data. | ||
499 | - * The callback function may be NULL if the source was never connected | ||
500 | - * to a callback using g_source_set_callback(). The dispatch function | ||
501 | - * should call the callback function with user_data and whatever additional | ||
502 | - * parameters are needed for this type of event source. | ||
503 | - */ | ||
504 | - BOOL fSuccess; | ||
505 | - DWORD cbRead = 0; | ||
506 | -// DWORD dwErr = 0; | ||
507 | - | ||
508 | - fSuccess = GetOverlappedResult(((pipe_source *) source)->hPipe,&((pipe_source *) source)->overlap,&cbRead,FALSE ); | ||
509 | - | ||
510 | - // trace("%s: source=%p data=%p Result=%s cbRead=%d",__FUNCTION__,source,data,fSuccess ? "Success" : "Unsuccess",(int) cbRead); | ||
511 | - | ||
512 | - switch(((pipe_source *) source)->state) | ||
513 | - { | ||
514 | - case PIPE_STATE_WAITING: | ||
515 | - if(fSuccess) | ||
516 | - { | ||
517 | - trace("Pipe connected (cbRet=%d)",(int) cbRead); | ||
518 | - set_active(TRUE); | ||
519 | - ((pipe_source *) source)->state = PIPE_STATE_READ; | ||
520 | - } | ||
521 | - else | ||
522 | - { | ||
523 | - popup_lasterror("%s", _( "Pipe connection failed" )); | ||
524 | - } | ||
525 | - break; | ||
526 | - | ||
527 | - case PIPE_STATE_READ: | ||
528 | - // trace("Reading pipe (cbRead=%d)",(int) cbRead); | ||
529 | - read_input_pipe( (pipe_source *) source); | ||
530 | - break; | ||
531 | - | ||
532 | - case PIPE_STATE_PENDING_READ: | ||
533 | - if(fSuccess && cbRead > 0) | ||
534 | - process_input((pipe_source *) source,cbRead); | ||
535 | - ((pipe_source *) source)->state = PIPE_STATE_READ; | ||
536 | - break; | ||
537 | - | ||
538 | - case PIPE_STATE_UNDEFINED: | ||
539 | - break; | ||
540 | - | ||
541 | -//#ifdef DEBUG | ||
542 | -// default: | ||
543 | -// trace("%s: source=%p data=%p Unexpected mode %d",__FUNCTION__,source,data,((pipe_source *) source)->state); | ||
544 | -//#endif | ||
545 | - } | ||
546 | - | ||
547 | - return TRUE; | ||
548 | - } | ||
549 | - | ||
550 | - static void IO_finalize(GSource *source) | ||
551 | - { | ||
552 | -// trace("%s: source=%p",__FUNCTION__,source); | ||
553 | - | ||
554 | - if( ((pipe_source *) source)->hPipe != INVALID_HANDLE_VALUE) | ||
555 | - { | ||
556 | - CloseHandle(((pipe_source *) source)->hPipe); | ||
557 | - ((pipe_source *) source)->hPipe = INVALID_HANDLE_VALUE; | ||
558 | - } | ||
559 | - | ||
560 | - } | ||
561 | - | ||
562 | - static gboolean IO_closure(gpointer data) | ||
563 | - { | ||
564 | -// trace("%s: data=%p",__FUNCTION__,data); | ||
565 | - return 0; | ||
566 | - } | ||
567 | - | ||
568 | - void popup_lasterror(const gchar *fmt, ...) | ||
569 | - { | ||
570 | - char buffer[4096]; | ||
571 | - va_list arg_ptr; | ||
572 | - int sz; | ||
573 | - DWORD errcode = GetLastError(); | ||
574 | - char *ptr; | ||
575 | - LPVOID lpMsgBuf = 0; | ||
576 | - | ||
577 | - FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |FORMAT_MESSAGE_IGNORE_INSERTS, NULL, errcode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL); | ||
578 | - | ||
579 | - for(ptr= (char *) lpMsgBuf;*ptr && *ptr != '\n';ptr++); | ||
580 | - *ptr = 0; | ||
581 | - | ||
582 | - va_start(arg_ptr, fmt); | ||
583 | - vsnprintf(buffer,4095,fmt,arg_ptr); | ||
584 | - va_end(arg_ptr); | ||
585 | - | ||
586 | - sz = strlen(buffer); | ||
587 | - snprintf(buffer+sz,4096-sz,": %s\n(rc=%d)",lpMsgBuf,(int) errcode); | ||
588 | - | ||
589 | - printf("%s\n",buffer); | ||
590 | - | ||
591 | -#ifdef DEBUG | ||
592 | - fprintf(stderr,"%s\n",buffer); | ||
593 | - fflush(stderr); | ||
594 | -#endif | ||
595 | - | ||
596 | - LocalFree(lpMsgBuf); | ||
597 | - } | ||
598 | - | ||
599 | - LIB3270_EXPORT int pw3270_plugin_start(GtkWidget *window, GtkWidget *terminal) | ||
600 | - { | ||
601 | - char id; | ||
602 | - | ||
603 | - for(id='A';id < 'Z';id++) | ||
604 | - { | ||
605 | - gchar * pipename = g_strdup_printf("\\\\.\\pipe\\%s_%c",pw3270_get_session_name(window),id); | ||
606 | - gchar * ptr; | ||
607 | - HANDLE hPipe; | ||
608 | - | ||
609 | - for(ptr=pipename;*ptr;ptr++) | ||
610 | - *ptr = g_ascii_tolower(*ptr); | ||
611 | - | ||
612 | - hPipe = CreateNamedPipe( TEXT(pipename), // pipe name | ||
613 | - PIPE_ACCESS_DUPLEX | // read/write access | ||
614 | - FILE_FLAG_OVERLAPPED, // overlapped mode | ||
615 | - PIPE_TYPE_MESSAGE | // pipe type | ||
616 | - PIPE_READMODE_MESSAGE | // pipe mode | ||
617 | - PIPE_WAIT, // blocking mode | ||
618 | - 1, // number of instances | ||
619 | - PIPE_BUFFER_LENGTH, // output buffer size | ||
620 | - PIPE_BUFFER_LENGTH, // input buffer size | ||
621 | - NMPWAIT_USE_DEFAULT_WAIT, // client time-out | ||
622 | - NULL); // default security attributes | ||
623 | - | ||
624 | - trace("%s = %p",pipename,hPipe); | ||
625 | - g_free(pipename); | ||
626 | - | ||
627 | - if(hPipe != INVALID_HANDLE_VALUE) | ||
628 | - { | ||
629 | - static GSourceFuncs pipe_source_funcs = | ||
630 | - { | ||
631 | - IO_prepare, | ||
632 | - IO_check, | ||
633 | - IO_dispatch, | ||
634 | - IO_finalize, | ||
635 | - IO_closure, | ||
636 | - NULL | ||
637 | - }; | ||
638 | - pipe_source * source; | ||
639 | - gchar * session = g_strdup_printf("%s:%c",pw3270_get_session_name(window),id); | ||
640 | - | ||
641 | - pw3270_set_session_name(window,session); | ||
642 | - g_free(session); | ||
643 | - | ||
644 | - source = (pipe_source *) g_source_new(&pipe_source_funcs,sizeof(pipe_source)); | ||
645 | - | ||
646 | - source->hPipe = hPipe; | ||
647 | - source->state = PIPE_STATE_WAITING; | ||
648 | - source->overlap.hEvent = CreateEvent( NULL,TRUE,TRUE,NULL); | ||
649 | - | ||
650 | - g_source_attach((GSource *) source,NULL); | ||
651 | - IO_accept(source); | ||
652 | - | ||
653 | - return 0; | ||
654 | - } | ||
655 | - | ||
656 | - } | ||
657 | - | ||
658 | - popup_lasterror( "%s", _( "Can´t create remote control pipe" )); | ||
659 | - | ||
660 | - return -1; | ||
661 | - } | ||
662 | - | ||
663 | - LIB3270_EXPORT int pw3270_plugin_stop(GtkWidget *window, GtkWidget *terminal) | ||
664 | - { | ||
665 | - | ||
666 | - return 0; | ||
667 | - } | ||
668 | - | ||
669 | - G_GNUC_INTERNAL void set_active(gboolean on) | ||
670 | - { | ||
671 | - trace("%s(%s)",__FUNCTION__,on ? "Active" : "Inactive"); | ||
672 | - // v3270_set_script(v3270_get_default_widget(),'H'); | ||
673 | - } |
@@ -0,0 +1,676 @@ | @@ -0,0 +1,676 @@ | ||
1 | +/* | ||
2 | + * "Software pw3270, desenvolvido com base nos códigos fontes do WC3270 e X3270 | ||
3 | + * (Paul Mattes Paul.Mattes@usa.net), de emulação de terminal 3270 para acesso a | ||
4 | + * aplicativos mainframe. Registro no INPI sob o nome G3270. Registro no INPI sob o nome G3270. | ||
5 | + * | ||
6 | + * Copyright (C) <2008> <Banco do Brasil S.A.> | ||
7 | + * | ||
8 | + * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob | ||
9 | + * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela | ||
10 | + * Free Software Foundation. | ||
11 | + * | ||
12 | + * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER | ||
13 | + * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO | ||
14 | + * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para | ||
15 | + * obter mais detalhes. | ||
16 | + * | ||
17 | + * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este | ||
18 | + * programa; se não, escreva para a Free Software Foundation, Inc., 51 Franklin | ||
19 | + * St, Fifth Floor, Boston, MA 02110-1301 USA | ||
20 | + * | ||
21 | + * Este programa está nomeado como - e possui - linhas de código. | ||
22 | + * | ||
23 | + * Contatos: | ||
24 | + * | ||
25 | + * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) | ||
26 | + * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) | ||
27 | + * | ||
28 | + * Agradecimento: | ||
29 | + * | ||
30 | + * Hélio Passos | ||
31 | + * | ||
32 | + */ | ||
33 | + | ||
34 | + #include "server.h" | ||
35 | + | ||
36 | +#ifdef _WIN32 | ||
37 | + #include <windows.h> | ||
38 | +#else | ||
39 | + #error HLLAPI is designed for windows. | ||
40 | +#endif // _WIN32 | ||
41 | + | ||
42 | + #include <pw3270/plugin.h> | ||
43 | + #include <v3270.h> | ||
44 | + #include <pw3270/ipcpackets.h> | ||
45 | + #include <lib3270/actions.h> | ||
46 | + #include <lib3270/charset.h> | ||
47 | + | ||
48 | +/*--[ Defines ]--------------------------------------------------------------------------------------*/ | ||
49 | + | ||
50 | + #pragma pack(1) | ||
51 | + | ||
52 | + enum PIPE_STATE | ||
53 | + { | ||
54 | + PIPE_STATE_WAITING, | ||
55 | + PIPE_STATE_READ, | ||
56 | + PIPE_STATE_PENDING_READ, | ||
57 | + PIPE_STATE_UNDEFINED | ||
58 | + }; | ||
59 | + | ||
60 | + typedef struct _pipe_source | ||
61 | + { | ||
62 | + GSource gsrc; | ||
63 | + HANDLE hPipe; | ||
64 | + | ||
65 | + enum PIPE_STATE state; | ||
66 | + | ||
67 | + OVERLAPPED overlap; | ||
68 | + unsigned char buffer[PIPE_BUFFER_LENGTH+1]; | ||
69 | + } pipe_source; | ||
70 | + | ||
71 | + #pragma pack() | ||
72 | + | ||
73 | + | ||
74 | +/*--[ Globals ]--------------------------------------------------------------------------------------*/ | ||
75 | + | ||
76 | +// static const gchar control_char = '@'; | ||
77 | + | ||
78 | +/*--[ Implement ]------------------------------------------------------------------------------------*/ | ||
79 | + | ||
80 | + static void IO_accept(pipe_source *source) | ||
81 | + { | ||
82 | + set_active(FALSE); | ||
83 | + | ||
84 | + if(ConnectNamedPipe(source->hPipe,&source->overlap)) | ||
85 | + { | ||
86 | + popup_lasterror("%s",_( "Error in ConnectNamedPipe" )); | ||
87 | + return; | ||
88 | + } | ||
89 | + | ||
90 | + switch(GetLastError()) | ||
91 | + { | ||
92 | + // The overlapped connection in progress. | ||
93 | + case ERROR_IO_PENDING: | ||
94 | + // trace("%s: ERROR_IO_PENDING",__FUNCTION__); | ||
95 | + source->state = PIPE_STATE_WAITING; | ||
96 | + break; | ||
97 | + | ||
98 | + // Client is already connected, so signal an event. | ||
99 | + case ERROR_PIPE_CONNECTED: | ||
100 | + trace("%s: ERROR_PIPE_CONNECTED",__FUNCTION__); | ||
101 | + set_active(TRUE); | ||
102 | + if(SetEvent(source->overlap.hEvent)) | ||
103 | + break; | ||
104 | + | ||
105 | + // If an error occurs during the connect operation... | ||
106 | + default: | ||
107 | + popup_lasterror("%s", _( "ConnectNamedPipe failed" )); | ||
108 | + } | ||
109 | + | ||
110 | + } | ||
111 | + | ||
112 | + static gboolean IO_prepare(GSource *source, gint *timeout) | ||
113 | + { | ||
114 | + /* | ||
115 | + * Called before all the file descriptors are polled. | ||
116 | + * If the source can determine that it is ready here | ||
117 | + * (without waiting for the results of the poll() call) | ||
118 | + * it should return TRUE. | ||
119 | + * | ||
120 | + * It can also return a timeout_ value which should be the maximum | ||
121 | + * timeout (in milliseconds) which should be passed to the poll() call. | ||
122 | + * The actual timeout used will be -1 if all sources returned -1, | ||
123 | + * or it will be the minimum of all the timeout_ values | ||
124 | + * returned which were >= 0. | ||
125 | + * | ||
126 | + */ | ||
127 | + if(WaitForSingleObject(((pipe_source *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | ||
128 | + return TRUE; | ||
129 | + | ||
130 | + *timeout = 10; | ||
131 | + return FALSE; | ||
132 | + } | ||
133 | + | ||
134 | + static gboolean IO_check(GSource *source) | ||
135 | + { | ||
136 | + /* | ||
137 | + * Called after all the file descriptors are polled. | ||
138 | + * The source should return TRUE if it is ready to be dispatched. | ||
139 | + * Note that some time may have passed since the previous prepare | ||
140 | + * function was called, so the source should be checked again here. | ||
141 | + * | ||
142 | + */ | ||
143 | + if(WaitForSingleObject(((pipe_source *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | ||
144 | + return TRUE; | ||
145 | + | ||
146 | + return FALSE; | ||
147 | + } | ||
148 | + | ||
149 | + static void send_text(pipe_source *source, char *text) | ||
150 | + { | ||
151 | + struct hllapi_packet_text *pkt; | ||
152 | + DWORD szBlock; | ||
153 | + int f; | ||
154 | + | ||
155 | + if(text) | ||
156 | + { | ||
157 | + szBlock = sizeof(struct hllapi_packet_text)+strlen(text); | ||
158 | + pkt = (struct hllapi_packet_text *) g_malloc0(szBlock); | ||
159 | + pkt->packet_id = 0; | ||
160 | + strcpy(pkt->text,text); | ||
161 | + lib3270_free(text); | ||
162 | + } | ||
163 | + else | ||
164 | + { | ||
165 | + szBlock = sizeof(struct hllapi_packet_text); | ||
166 | + pkt = (struct hllapi_packet_text *) g_malloc0(szBlock); | ||
167 | + pkt->packet_id = errno ? errno : -1; | ||
168 | + } | ||
169 | + | ||
170 | + trace("szBlock=%d text=\"%s\"",szBlock, ( (struct hllapi_packet_text *) pkt)->text); | ||
171 | + for(f=0;f< (int) szBlock;f++) | ||
172 | + { | ||
173 | + trace("rsp(%d)= %d \"%s\"",f,* (((char *) pkt)+f),((char *) pkt)+f); | ||
174 | + } | ||
175 | + | ||
176 | + WriteFile(source->hPipe,pkt,szBlock,&szBlock,NULL); | ||
177 | + | ||
178 | + g_free(pkt); | ||
179 | + } | ||
180 | + | ||
181 | + static void send_result(pipe_source *source, int rc) | ||
182 | + { | ||
183 | + struct hllapi_packet_result pkt = { rc }; | ||
184 | + DWORD wrote = sizeof(pkt); | ||
185 | + WriteFile(source->hPipe,&pkt,wrote,&wrote,NULL); | ||
186 | + } | ||
187 | + | ||
188 | + static int do_file_transfer(struct hllapi_packet_file_transfer * source) | ||
189 | + { | ||
190 | + /* | ||
191 | + const gchar * local = (const char *) source->text; | ||
192 | + const gchar * remote = (const char *) (local+strlen(local)+1); | ||
193 | + | ||
194 | + return v3270_transfer_file( v3270_get_default_widget(), | ||
195 | + (LIB3270_FT_OPTION) source->options, | ||
196 | + local, | ||
197 | + remote, | ||
198 | + source->lrecl, | ||
199 | + source->blksize, | ||
200 | + source->primspace, | ||
201 | + source->secspace, | ||
202 | + source->dft ); | ||
203 | + */ | ||
204 | + return EINVAL; | ||
205 | + } | ||
206 | + | ||
207 | + static void get_host(pipe_source *source) { | ||
208 | + send_text(source,strdup(lib3270_get_url(lib3270_get_default_session_handle()))); | ||
209 | + } | ||
210 | + | ||
211 | + static void process_input(pipe_source *source, DWORD cbRead) | ||
212 | + { | ||
213 | + const struct hllapi_packet_query * query = ((struct hllapi_packet_query *) source->buffer); | ||
214 | + | ||
215 | + trace("%s id=%d",__FUNCTION__,query->packet_id); | ||
216 | + | ||
217 | + switch(query->packet_id) | ||
218 | + { | ||
219 | + case HLLAPI_PACKET_CONNECT: | ||
220 | + send_result(source,lib3270_reconnect( lib3270_get_default_session_handle(), | ||
221 | + ((struct hllapi_packet_connect *) source->buffer)->wait)); | ||
222 | + break; | ||
223 | + | ||
224 | + case HLLAPI_PACKET_CONNECT_URL: | ||
225 | + send_result(source,lib3270_connect_url(lib3270_get_default_session_handle(),(const char *) (query+1),0)); | ||
226 | + break; | ||
227 | + | ||
228 | + case HLLAPI_PACKET_SET_HOST: | ||
229 | + send_result(source,lib3270_set_url(lib3270_get_default_session_handle(), | ||
230 | + ((struct hllapi_packet_text *) source->buffer)->text)); | ||
231 | + break; | ||
232 | + | ||
233 | + case HLLAPI_PACKET_GET_HOST: | ||
234 | + get_host(source); | ||
235 | + break; | ||
236 | + | ||
237 | + case HLLAPI_PACKET_DISCONNECT: | ||
238 | + send_result(source,lib3270_disconnect(lib3270_get_default_session_handle())); | ||
239 | + break; | ||
240 | + | ||
241 | + case HLLAPI_PACKET_IS_CONNECTED: | ||
242 | + send_result(source,lib3270_in_tn3270e(lib3270_get_default_session_handle())); | ||
243 | + break; | ||
244 | + | ||
245 | + case HLLAPI_PACKET_IS_READY: | ||
246 | + send_result(source,lib3270_is_ready(lib3270_get_default_session_handle())); | ||
247 | + break; | ||
248 | + | ||
249 | + case HLLAPI_PACKET_ENTER: | ||
250 | + send_result(source,lib3270_enter(lib3270_get_default_session_handle())); | ||
251 | + break; | ||
252 | + | ||
253 | + case HLLAPI_PACKET_PRINT: | ||
254 | + send_result(source,lib3270_print_all(lib3270_get_default_session_handle())); | ||
255 | + break; | ||
256 | + | ||
257 | + case HLLAPI_PACKET_ERASE: | ||
258 | + send_result(source,lib3270_erase(lib3270_get_default_session_handle())); | ||
259 | + break; | ||
260 | + | ||
261 | + case HLLAPI_PACKET_ERASE_EOF: | ||
262 | + send_result(source,lib3270_eraseeof(lib3270_get_default_session_handle())); | ||
263 | + break; | ||
264 | + | ||
265 | + case HLLAPI_PACKET_ERASE_EOL: | ||
266 | + send_result(source,lib3270_eraseeol(lib3270_get_default_session_handle())); | ||
267 | + break; | ||
268 | + | ||
269 | + case HLLAPI_PACKET_ERASE_INPUT: | ||
270 | + send_result(source,lib3270_eraseinput(lib3270_get_default_session_handle())); | ||
271 | + break; | ||
272 | + | ||
273 | + case HLLAPI_PACKET_PFKEY: | ||
274 | + send_result(source,lib3270_pfkey( lib3270_get_default_session_handle(), | ||
275 | + ((struct hllapi_packet_keycode *) source->buffer)->keycode)); | ||
276 | + break; | ||
277 | + | ||
278 | + case HLLAPI_PACKET_PAKEY: | ||
279 | + send_result(source,lib3270_pakey( lib3270_get_default_session_handle(), | ||
280 | + ((struct hllapi_packet_keycode *) source->buffer)->keycode)); | ||
281 | + break; | ||
282 | + | ||
283 | + case HLLAPI_PACKET_SET_CURSOR_POSITION: | ||
284 | + send_result(source,lib3270_set_cursor_position( lib3270_get_default_session_handle(), | ||
285 | + ((struct hllapi_packet_cursor *) source->buffer)->row, | ||
286 | + ((struct hllapi_packet_cursor *) source->buffer)->col)); | ||
287 | + break; | ||
288 | + | ||
289 | + case HLLAPI_PACKET_SET_TEXT_AT: | ||
290 | + send_result(source,lib3270_set_text_at( lib3270_get_default_session_handle(), | ||
291 | + ((struct hllapi_packet_text_at *) source->buffer)->row, | ||
292 | + ((struct hllapi_packet_text_at *) source->buffer)->col, | ||
293 | + (unsigned char *) ((struct hllapi_packet_text_at *) source->buffer)->text)); | ||
294 | + break; | ||
295 | + | ||
296 | + case HLLAPI_PACKET_GET_TEXT_AT: | ||
297 | + send_text(source,lib3270_get_string_at( lib3270_get_default_session_handle(), | ||
298 | + ((struct hllapi_packet_at *) source->buffer)->row, | ||
299 | + ((struct hllapi_packet_at *) source->buffer)->col, | ||
300 | + ((struct hllapi_packet_at *) source->buffer)->len, | ||
301 | + ((struct hllapi_packet_at *) source->buffer)->lf)); | ||
302 | + break; | ||
303 | + | ||
304 | + case HLLAPI_PACKET_GET_TEXT_AT_OFFSET: | ||
305 | + send_text(source,lib3270_get_string_at_address( lib3270_get_default_session_handle(), | ||
306 | + ((struct hllapi_packet_query_offset *) source->buffer)->addr, | ||
307 | + ((struct hllapi_packet_query_offset *) source->buffer)->len, | ||
308 | + ((struct hllapi_packet_query_offset *) source->buffer)->lf)); | ||
309 | + break; | ||
310 | + | ||
311 | + case HLLAPI_PACKET_CMP_TEXT_AT: | ||
312 | + send_result(source,lib3270_cmp_text_at( lib3270_get_default_session_handle(), | ||
313 | + ((struct hllapi_packet_text_at *) source->buffer)->row, | ||
314 | + ((struct hllapi_packet_text_at *) source->buffer)->col, | ||
315 | + ((struct hllapi_packet_text_at *) source->buffer)->text, | ||
316 | + ((struct hllapi_packet_text_at *) source->buffer)->lf)); | ||
317 | + break; | ||
318 | + | ||
319 | + case HLLAPI_PACKET_INPUT_STRING: | ||
320 | + send_result(source,lib3270_input_string(lib3270_get_default_session_handle(), | ||
321 | + (unsigned char *) ((struct hllapi_packet_text *) source->buffer)->text)); | ||
322 | + break; | ||
323 | + | ||
324 | + case HLLAPI_PACKET_EMULATE_INPUT: | ||
325 | + send_result(source,lib3270_emulate_input(lib3270_get_default_session_handle(), | ||
326 | + (const char *) ((struct hllapi_packet_emulate_input *) source->buffer)->text, | ||
327 | + (int) ((struct hllapi_packet_emulate_input *) source->buffer)->len, | ||
328 | + (int) ((struct hllapi_packet_emulate_input *) source->buffer)->pasting)); | ||
329 | + break; | ||
330 | + | ||
331 | + case HLLAPI_PACKET_SET_CURSOR: | ||
332 | + send_result(source,lib3270_set_cursor_address(lib3270_get_default_session_handle(), | ||
333 | + ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
334 | + break; | ||
335 | + | ||
336 | + case HLLAPI_PACKET_GET_CURSOR: | ||
337 | + send_result(source,lib3270_get_cursor_address(lib3270_get_default_session_handle())); | ||
338 | + break; | ||
339 | + | ||
340 | + case HLLAPI_PACKET_GET_WIDTH: | ||
341 | + send_result(source,lib3270_get_width(lib3270_get_default_session_handle())); | ||
342 | + break; | ||
343 | + | ||
344 | + case HLLAPI_PACKET_GET_HEIGHT: | ||
345 | + send_result(source,lib3270_get_height(lib3270_get_default_session_handle())); | ||
346 | + break; | ||
347 | + | ||
348 | + case HLLAPI_PACKET_GET_LENGTH: | ||
349 | + send_result(source,lib3270_get_length(lib3270_get_default_session_handle())); | ||
350 | + break; | ||
351 | + | ||
352 | + case HLLAPI_PACKET_GET_PROGRAM_MESSAGE: | ||
353 | + send_result(source,lib3270_get_program_message(lib3270_get_default_session_handle())); | ||
354 | + break; | ||
355 | + | ||
356 | + case HLLAPI_PACKET_GET_SSL_STATE: | ||
357 | + send_result(source,lib3270_get_secure(lib3270_get_default_session_handle())); | ||
358 | + break; | ||
359 | + | ||
360 | + case HLLAPI_PACKET_SET_UNLOCK_DELAY: | ||
361 | + lib3270_set_unlock_delay(lib3270_get_default_session_handle(),(unsigned short) ((struct hllapi_packet_set_int *) source->buffer)->value); | ||
362 | + send_result(source,0); | ||
363 | + break; | ||
364 | + | ||
365 | + case HLLAPI_PACKET_SET_TOGGLE: | ||
366 | + send_result(source,lib3270_set_toggle(lib3270_get_default_session_handle(), | ||
367 | + (LIB3270_TOGGLE) ((struct hllapi_packet_set *) source->buffer)->id, | ||
368 | + ((struct hllapi_packet_set *) source->buffer)->value)); | ||
369 | + break; | ||
370 | + | ||
371 | + case HLLAPI_PACKET_FIELD_START: | ||
372 | + send_result(source,lib3270_get_field_start(lib3270_get_default_session_handle(), | ||
373 | + ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
374 | + break; | ||
375 | + | ||
376 | + | ||
377 | + case HLLAPI_PACKET_FIELD_LEN: | ||
378 | + send_result(source,lib3270_get_field_len(lib3270_get_default_session_handle(), | ||
379 | + ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
380 | + break; | ||
381 | + | ||
382 | + case HLLAPI_PACKET_NEXT_UNPROTECTED: | ||
383 | + send_result(source,lib3270_get_next_unprotected(lib3270_get_default_session_handle(), | ||
384 | + ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
385 | + break; | ||
386 | + | ||
387 | + case HLLAPI_PACKET_IS_PROTECTED: | ||
388 | + send_result(source,lib3270_get_is_protected(lib3270_get_default_session_handle(), | ||
389 | + ((struct hllapi_packet_addr *) source->buffer)->addr)); | ||
390 | + break; | ||
391 | + | ||
392 | + case HLLAPI_PACKET_IS_PROTECTED_AT: | ||
393 | + send_result(source,lib3270_get_is_protected_at( lib3270_get_default_session_handle(), | ||
394 | + ((struct hllapi_packet_query_at *) source->buffer)->row, | ||
395 | + ((struct hllapi_packet_query_at *) source->buffer)->col)); | ||
396 | + break; | ||
397 | + | ||
398 | + case HLLAPI_PACKET_QUIT: | ||
399 | + gtk_main_quit(); | ||
400 | + send_result(source,0); | ||
401 | + break; | ||
402 | + | ||
403 | + case HLLAPI_PACKET_SET_HOST_CHARSET: | ||
404 | + send_result(source,lib3270_set_host_charset( lib3270_get_default_session_handle(), | ||
405 | + (const char *) ((struct hllapi_packet_set_text *) source->buffer)->text)); | ||
406 | + break; | ||
407 | + | ||
408 | + case HLLAPI_PACKET_ASC2EBC: | ||
409 | + send_text(source,(char *) lib3270_asc2ebc( | ||
410 | + lib3270_get_default_session_handle(), | ||
411 | + (unsigned char *) ((struct hllapi_packet_set_text *) source->buffer)->text,-1 | ||
412 | + )); | ||
413 | + break; | ||
414 | + | ||
415 | + case HLLAPI_PACKET_EBC2ASC: | ||
416 | + send_text(source,(char *) lib3270_ebc2asc( | ||
417 | + lib3270_get_default_session_handle(), | ||
418 | + (unsigned char *) ((struct hllapi_packet_set_text *) source->buffer)->text,-1 | ||
419 | + )); | ||
420 | + break; | ||
421 | + | ||
422 | + case HLLAPI_PACKET_FILE_TRANSFER: | ||
423 | + send_result(source,do_file_transfer((struct hllapi_packet_file_transfer *) source)); | ||
424 | + break; | ||
425 | + | ||
426 | + case HLLAPI_PACKET_GET_HOST_CHARSET: | ||
427 | + trace("%s","HLLAPI_PACKET_GET_HOST_CHARSET"); | ||
428 | + send_text(source,(char *) lib3270_get_host_charset(lib3270_get_default_session_handle())); | ||
429 | + break; | ||
430 | + | ||
431 | + case HLLAPI_PACKET_ACTION: | ||
432 | + send_result(source,lib3270_action(lib3270_get_default_session_handle(), | ||
433 | + (const char *) ((struct hllapi_packet_text *) source->buffer)->text)); | ||
434 | + break; | ||
435 | + | ||
436 | + | ||
437 | + default: | ||
438 | + send_result(source, EINVAL); | ||
439 | + g_message("Invalid remote request (id=%d)",source->buffer[0]); | ||
440 | + } | ||
441 | + | ||
442 | + } | ||
443 | + | ||
444 | + static void read_input_pipe(pipe_source *source) | ||
445 | + { | ||
446 | + DWORD cbRead = 0; | ||
447 | + | ||
448 | + if(ReadFile(source->hPipe,source->buffer,PIPE_BUFFER_LENGTH,&cbRead,&source->overlap) && cbRead > 0) | ||
449 | + process_input(source,cbRead); | ||
450 | + | ||
451 | + // The read operation is still pending. | ||
452 | + switch(GetLastError()) | ||
453 | + { | ||
454 | + case 0: | ||
455 | + break; | ||
456 | + | ||
457 | + case ERROR_IO_PENDING: | ||
458 | + // trace("%s: PIPE_STATE_PENDING_READ",__FUNCTION__); | ||
459 | + source->state = PIPE_STATE_PENDING_READ; | ||
460 | + break; | ||
461 | + | ||
462 | + case ERROR_PIPE_LISTENING: | ||
463 | + // trace("%s: ERROR_PIPE_LISTENING",__FUNCTION__); | ||
464 | + source->state = PIPE_STATE_READ; | ||
465 | + break; | ||
466 | + | ||
467 | + case ERROR_BROKEN_PIPE: | ||
468 | + trace("%s: ERROR_BROKEN_PIPE",__FUNCTION__); | ||
469 | + | ||
470 | + if(!DisconnectNamedPipe(source->hPipe)) | ||
471 | + { | ||
472 | + set_active(FALSE); | ||
473 | + popup_lasterror("%s",_( "Error in DisconnectNamedPipe" )); | ||
474 | + } | ||
475 | + else | ||
476 | + { | ||
477 | + IO_accept(source); | ||
478 | + } | ||
479 | + break; | ||
480 | + | ||
481 | + case ERROR_PIPE_NOT_CONNECTED: | ||
482 | + trace("%s: ERROR_PIPE_NOT_CONNECTED",__FUNCTION__); | ||
483 | + set_active(FALSE); | ||
484 | + break; | ||
485 | + | ||
486 | + default: | ||
487 | + if(source->hPipe != INVALID_HANDLE_VALUE) | ||
488 | + popup_lasterror("%s",_( "Error receiving message from pipe" ) ); | ||
489 | + } | ||
490 | + | ||
491 | + } | ||
492 | + | ||
493 | + static gboolean IO_dispatch(GSource *source, GSourceFunc callback, gpointer data) | ||
494 | + { | ||
495 | + /* | ||
496 | + * Called to dispatch the event source, | ||
497 | + * after it has returned TRUE in either its prepare or its check function. | ||
498 | + * The dispatch function is passed in a callback function and data. | ||
499 | + * The callback function may be NULL if the source was never connected | ||
500 | + * to a callback using g_source_set_callback(). The dispatch function | ||
501 | + * should call the callback function with user_data and whatever additional | ||
502 | + * parameters are needed for this type of event source. | ||
503 | + */ | ||
504 | + BOOL fSuccess; | ||
505 | + DWORD cbRead = 0; | ||
506 | +// DWORD dwErr = 0; | ||
507 | + | ||
508 | + fSuccess = GetOverlappedResult(((pipe_source *) source)->hPipe,&((pipe_source *) source)->overlap,&cbRead,FALSE ); | ||
509 | + | ||
510 | + // trace("%s: source=%p data=%p Result=%s cbRead=%d",__FUNCTION__,source,data,fSuccess ? "Success" : "Unsuccess",(int) cbRead); | ||
511 | + | ||
512 | + switch(((pipe_source *) source)->state) | ||
513 | + { | ||
514 | + case PIPE_STATE_WAITING: | ||
515 | + if(fSuccess) | ||
516 | + { | ||
517 | + trace("Pipe connected (cbRet=%d)",(int) cbRead); | ||
518 | + set_active(TRUE); | ||
519 | + ((pipe_source *) source)->state = PIPE_STATE_READ; | ||
520 | + } | ||
521 | + else | ||
522 | + { | ||
523 | + popup_lasterror("%s", _( "Pipe connection failed" )); | ||
524 | + } | ||
525 | + break; | ||
526 | + | ||
527 | + case PIPE_STATE_READ: | ||
528 | + // trace("Reading pipe (cbRead=%d)",(int) cbRead); | ||
529 | + read_input_pipe( (pipe_source *) source); | ||
530 | + break; | ||
531 | + | ||
532 | + case PIPE_STATE_PENDING_READ: | ||
533 | + if(fSuccess && cbRead > 0) | ||
534 | + process_input((pipe_source *) source,cbRead); | ||
535 | + ((pipe_source *) source)->state = PIPE_STATE_READ; | ||
536 | + break; | ||
537 | + | ||
538 | + case PIPE_STATE_UNDEFINED: | ||
539 | + break; | ||
540 | + | ||
541 | +//#ifdef DEBUG | ||
542 | +// default: | ||
543 | +// trace("%s: source=%p data=%p Unexpected mode %d",__FUNCTION__,source,data,((pipe_source *) source)->state); | ||
544 | +//#endif | ||
545 | + } | ||
546 | + | ||
547 | + return TRUE; | ||
548 | + } | ||
549 | + | ||
550 | + static void IO_finalize(GSource *source) | ||
551 | + { | ||
552 | +// trace("%s: source=%p",__FUNCTION__,source); | ||
553 | + | ||
554 | + if( ((pipe_source *) source)->hPipe != INVALID_HANDLE_VALUE) | ||
555 | + { | ||
556 | + CloseHandle(((pipe_source *) source)->hPipe); | ||
557 | + ((pipe_source *) source)->hPipe = INVALID_HANDLE_VALUE; | ||
558 | + } | ||
559 | + | ||
560 | + } | ||
561 | + | ||
562 | + static gboolean IO_closure(gpointer data) | ||
563 | + { | ||
564 | +// trace("%s: data=%p",__FUNCTION__,data); | ||
565 | + return 0; | ||
566 | + } | ||
567 | + | ||
568 | + void popup_lasterror(const gchar *fmt, ...) | ||
569 | + { | ||
570 | + char buffer[4096]; | ||
571 | + va_list arg_ptr; | ||
572 | + int sz; | ||
573 | + DWORD errcode = GetLastError(); | ||
574 | + char *ptr; | ||
575 | + LPVOID lpMsgBuf = 0; | ||
576 | + | ||
577 | + FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |FORMAT_MESSAGE_IGNORE_INSERTS, NULL, errcode, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL); | ||
578 | + | ||
579 | + for(ptr= (char *) lpMsgBuf;*ptr && *ptr != '\n';ptr++); | ||
580 | + *ptr = 0; | ||
581 | + | ||
582 | + va_start(arg_ptr, fmt); | ||
583 | + vsnprintf(buffer,4095,fmt,arg_ptr); | ||
584 | + va_end(arg_ptr); | ||
585 | + | ||
586 | + sz = strlen(buffer); | ||
587 | + snprintf(buffer+sz,4096-sz,": %s\n(rc=%d)",lpMsgBuf,(int) errcode); | ||
588 | + | ||
589 | + printf("%s\n",buffer); | ||
590 | + | ||
591 | +#ifdef DEBUG | ||
592 | + fprintf(stderr,"%s\n",buffer); | ||
593 | + fflush(stderr); | ||
594 | +#endif | ||
595 | + | ||
596 | + LocalFree(lpMsgBuf); | ||
597 | + } | ||
598 | + | ||
599 | + LIB3270_EXPORT int pw3270_plugin_start(GtkWidget *window, GtkWidget *terminal) | ||
600 | + { | ||
601 | + char id; | ||
602 | + const gchar * name = "pw3270"; | ||
603 | + | ||
604 | + for(id='A';id < 'Z';id++) | ||
605 | + { | ||
606 | + gchar * pipename = g_strdup_printf("\\\\.\\pipe\\%s_%c",name,id); | ||
607 | + gchar * ptr; | ||
608 | + HANDLE hPipe; | ||
609 | + | ||
610 | + for(ptr=pipename;*ptr;ptr++) | ||
611 | + *ptr = g_ascii_tolower(*ptr); | ||
612 | + | ||
613 | + hPipe = CreateNamedPipe( TEXT(pipename), // pipe name | ||
614 | + PIPE_ACCESS_DUPLEX | // read/write access | ||
615 | + FILE_FLAG_OVERLAPPED, // overlapped mode | ||
616 | + PIPE_TYPE_MESSAGE | // pipe type | ||
617 | + PIPE_READMODE_MESSAGE | // pipe mode | ||
618 | + PIPE_WAIT, // blocking mode | ||
619 | + 1, // number of instances | ||
620 | + PIPE_BUFFER_LENGTH, // output buffer size | ||
621 | + PIPE_BUFFER_LENGTH, // input buffer size | ||
622 | + NMPWAIT_USE_DEFAULT_WAIT, // client time-out | ||
623 | + NULL); // default security attributes | ||
624 | + | ||
625 | + trace("%s = %p",pipename,hPipe); | ||
626 | + g_free(pipename); | ||
627 | + | ||
628 | + if(hPipe != INVALID_HANDLE_VALUE) | ||
629 | + { | ||
630 | + static GSourceFuncs pipe_source_funcs = | ||
631 | + { | ||
632 | + IO_prepare, | ||
633 | + IO_check, | ||
634 | + IO_dispatch, | ||
635 | + IO_finalize, | ||
636 | + IO_closure, | ||
637 | + NULL | ||
638 | + }; | ||
639 | + pipe_source * source; | ||
640 | + gchar * session = g_strdup_printf("%s:%c",name,id); | ||
641 | + | ||
642 | + // pw3270_set_session_name(window,session); | ||
643 | + v3270_set_session_name(terminal,session); | ||
644 | + | ||
645 | + g_free(session); | ||
646 | + | ||
647 | + source = (pipe_source *) g_source_new(&pipe_source_funcs,sizeof(pipe_source)); | ||
648 | + | ||
649 | + source->hPipe = hPipe; | ||
650 | + source->state = PIPE_STATE_WAITING; | ||
651 | + source->overlap.hEvent = CreateEvent( NULL,TRUE,TRUE,NULL); | ||
652 | + | ||
653 | + g_source_attach((GSource *) source,NULL); | ||
654 | + IO_accept(source); | ||
655 | + | ||
656 | + return 0; | ||
657 | + } | ||
658 | + | ||
659 | + } | ||
660 | + | ||
661 | + popup_lasterror( "%s", _( "Can´t create remote control pipe" )); | ||
662 | + | ||
663 | + return -1; | ||
664 | + } | ||
665 | + | ||
666 | + LIB3270_EXPORT int pw3270_plugin_stop(GtkWidget *window, GtkWidget *terminal) | ||
667 | + { | ||
668 | + | ||
669 | + return 0; | ||
670 | + } | ||
671 | + | ||
672 | + G_GNUC_INTERNAL void set_active(gboolean on) | ||
673 | + { | ||
674 | + trace("%s(%s)",__FUNCTION__,on ? "Active" : "Inactive"); | ||
675 | + // v3270_set_script(v3270_get_default_widget(),'H'); | ||
676 | + } |