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 | 44 | <Option compilerVar="CC" /> |
| 45 | 45 | </Unit> |
| 46 | 46 | <Unit filename="src/plugin/linux/service.h" /> |
| 47 | + <Unit filename="src/plugin/windows/main.c"> | |
| 48 | + <Option compilerVar="CC" /> | |
| 49 | + </Unit> | |
| 47 | 50 | <Unit filename="src/testprogram/testprogram.c"> |
| 48 | 51 | <Option compilerVar="CC" /> |
| 49 | 52 | </Unit> | ... | ... |
src/hllapi/pluginmain.c
| ... | ... | @@ -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 @@ |
| 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 | + } | ... | ... |