Commit 9c1b63e1969f13aa7a66bdc429722902e10d56ed
1 parent
58bff019
Exists in
master
and in
1 other branch
Importing windows pipe service.
Showing
5 changed files
with
439 additions
and
4 deletions
Show diff stats
pw3270-plugin-ipc.cbp
| ... | ... | @@ -89,6 +89,13 @@ |
| 89 | 89 | <Unit filename="src/testprogram/testprogram.c"> |
| 90 | 90 | <Option compilerVar="CC" /> |
| 91 | 91 | </Unit> |
| 92 | + <Unit filename="src/windows/gobject.c"> | |
| 93 | + <Option compilerVar="CC" /> | |
| 94 | + </Unit> | |
| 95 | + <Unit filename="src/windows/gobject.h" /> | |
| 96 | + <Unit filename="src/windows/pipesource.c"> | |
| 97 | + <Option compilerVar="CC" /> | |
| 98 | + </Unit> | |
| 92 | 99 | <Extensions> |
| 93 | 100 | <code_completion /> |
| 94 | 101 | <envvars /> | ... | ... |
src/linux/gobject.h
| ... | ... | @@ -29,13 +29,13 @@ |
| 29 | 29 | */ |
| 30 | 30 | |
| 31 | 31 | /** |
| 32 | - * @brief Private definitions for pw3270 IPC linux plugin. | |
| 32 | + * @brief Private definitions for pw3270 IPC linux module. | |
| 33 | 33 | * |
| 34 | 34 | */ |
| 35 | 35 | |
| 36 | -#ifndef LINUX_PRIVATE_H_INCLUDED | |
| 36 | +#ifndef LINUX_GOBJECT_H_INCLUDED | |
| 37 | 37 | |
| 38 | - #define LINUX_PRIVATE_H_INCLUDED | |
| 38 | + #define LINUX_GOBJECT_H_INCLUDED | |
| 39 | 39 | |
| 40 | 40 | #include <config.h> |
| 41 | 41 | |
| ... | ... | @@ -72,4 +72,4 @@ |
| 72 | 72 | |
| 73 | 73 | G_END_DECLS |
| 74 | 74 | |
| 75 | -#endif // LINUX_PRIVATE_H_INCLUDED | |
| 75 | +#endif // LINUX_GOBJECT_H_INCLUDED | ... | ... |
| ... | ... | @@ -0,0 +1,125 @@ |
| 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. | |
| 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 gobject.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 | + */ | |
| 29 | + | |
| 30 | +#include "gobject.h" | |
| 31 | +#include <lib3270.h> | |
| 32 | +#include <lib3270/actions.h> | |
| 33 | +#include <lib3270/properties.h> | |
| 34 | + | |
| 35 | +G_DEFINE_TYPE(ipc3270, ipc3270, G_TYPE_OBJECT) | |
| 36 | + | |
| 37 | +static void ipc3270_finalize(GObject *object) { | |
| 38 | + | |
| 39 | + ipc3270 * ipc = IPC3270(object); | |
| 40 | + | |
| 41 | + | |
| 42 | + G_OBJECT_CLASS(ipc3270_parent_class)->finalize(object); | |
| 43 | +} | |
| 44 | + | |
| 45 | + | |
| 46 | +static void ipc3270_class_init(ipc3270Class *klass) { | |
| 47 | + | |
| 48 | + debug("%s",__FUNCTION__); | |
| 49 | + | |
| 50 | + GObjectClass *object_class; | |
| 51 | + object_class = G_OBJECT_CLASS (klass); | |
| 52 | + object_class->finalize = ipc3270_finalize; | |
| 53 | + | |
| 54 | +} | |
| 55 | + | |
| 56 | +static void ipc3270_init(ipc3270 *object) { | |
| 57 | + | |
| 58 | + debug("%s",__FUNCTION__); | |
| 59 | + object->error_domain = g_quark_from_static_string(PACKAGE_NAME); | |
| 60 | + | |
| 61 | +} | |
| 62 | + | |
| 63 | +GObject * ipc3270_new() { | |
| 64 | + return g_object_new(GLIB_TYPE_IPC3270, NULL); | |
| 65 | +} | |
| 66 | + | |
| 67 | +void ipc3270_set_session(GObject *object, H3270 *hSession, const char *name, GError **error) { | |
| 68 | + | |
| 69 | + char id; | |
| 70 | + | |
| 71 | + ipc3270 * ipc = IPC3270(object); | |
| 72 | + ipc->hSession = hSession; | |
| 73 | + | |
| 74 | + for(id='A';id < 'Z';id++) { | |
| 75 | + | |
| 76 | + gchar * pipename = g_strdup_printf("\\\\.\\pipe\\%s\\%c",name,id); | |
| 77 | + gchar * ptr; | |
| 78 | + HANDLE hPipe; | |
| 79 | + | |
| 80 | + for(ptr=pipename;*ptr;ptr++) | |
| 81 | + *ptr = g_ascii_tolower(*ptr); | |
| 82 | + | |
| 83 | + hPipe = CreateNamedPipe( TEXT(pipename), // pipe name | |
| 84 | + PIPE_ACCESS_DUPLEX | // read/write access | |
| 85 | + FILE_FLAG_OVERLAPPED, // overlapped mode | |
| 86 | + PIPE_TYPE_MESSAGE | // pipe type | |
| 87 | + PIPE_READMODE_MESSAGE | // pipe mode | |
| 88 | + PIPE_WAIT, // blocking mode | |
| 89 | + 1, // number of instances | |
| 90 | + PIPE_BUFFER_LENGTH, // output buffer size | |
| 91 | + PIPE_BUFFER_LENGTH, // input buffer size | |
| 92 | + NMPWAIT_USE_DEFAULT_WAIT, // client time-out | |
| 93 | + NULL); // default security attributes | |
| 94 | + | |
| 95 | + debug("%s = %p",pipename,hPipe); | |
| 96 | + g_free(pipename); | |
| 97 | + | |
| 98 | + if(hPipe != INVALID_HANDLE_VALUE) { | |
| 99 | + | |
| 100 | + ipc->source = (IPC3270_PIPE_SOURCE *) g_source_new(&pipe_source_funcs,sizeof(IPC3270_PIPE_SOURCE)); | |
| 101 | + | |
| 102 | + lib3270_set_session_id(ipc->hSession, id); | |
| 103 | + | |
| 104 | + ipc->source->hPipe = hPipe; | |
| 105 | + ipc->source->state = PIPE_STATE_WAITING; | |
| 106 | + ipc->source->overlap.hEvent = CreateEvent( NULL,TRUE,TRUE,NULL); | |
| 107 | + | |
| 108 | + g_source_attach((GSource *) ipc->source,NULL); | |
| 109 | + | |
| 110 | + // IO_accept(source); | |
| 111 | + ipc3270_wait_for_client(ipc->source); | |
| 112 | + | |
| 113 | + break; | |
| 114 | + } | |
| 115 | + | |
| 116 | + } | |
| 117 | + | |
| 118 | + | |
| 119 | +} | |
| 120 | + | |
| 121 | +const gchar * ipc3270_get_display_charset(GObject *object) { | |
| 122 | + return lib3270_get_display_charset(IPC3270(object)->hSession); | |
| 123 | +} | |
| 124 | + | |
| 125 | + | ... | ... |
| ... | ... | @@ -0,0 +1,93 @@ |
| 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 | |
| 5 | + * o nome G3270. | |
| 6 | + * | |
| 7 | + * Copyright (C) <2008> <Banco do Brasil S.A.> | |
| 8 | + * | |
| 9 | + * Este programa é software livre. Você pode redistribuí-lo e/ou modificá-lo sob | |
| 10 | + * os termos da GPL v.2 - Licença Pública Geral GNU, conforme publicado pela | |
| 11 | + * Free Software Foundation. | |
| 12 | + * | |
| 13 | + * Este programa é distribuído na expectativa de ser útil, mas SEM QUALQUER | |
| 14 | + * GARANTIA; sem mesmo a garantia implícita de COMERCIALIZAÇÃO ou de ADEQUAÇÃO | |
| 15 | + * A QUALQUER PROPÓSITO EM PARTICULAR. Consulte a Licença Pública Geral GNU para | |
| 16 | + * obter mais detalhes. | |
| 17 | + * | |
| 18 | + * Você deve ter recebido uma cópia da Licença Pública Geral GNU junto com este | |
| 19 | + * programa; se não, escreva para a Free Software Foundation, Inc., 51 Franklin | |
| 20 | + * St, Fifth Floor, Boston, MA 02110-1301 USA | |
| 21 | + * | |
| 22 | + * Este programa está nomeado como - e possui - linhas de código. | |
| 23 | + * | |
| 24 | + * Contatos: | |
| 25 | + * | |
| 26 | + * perry.werneck@gmail.com (Alexandre Perry de Souza Werneck) | |
| 27 | + * erico.mendonca@gmail.com (Erico Mascarenhas Mendonça) | |
| 28 | + * | |
| 29 | + */ | |
| 30 | + | |
| 31 | + /** | |
| 32 | + * @brief Private definitions for pw3270 IPC windows module. | |
| 33 | + * | |
| 34 | + */ | |
| 35 | + | |
| 36 | +#ifndef WINDOWS_GOBJECT_H_INCLUDED | |
| 37 | + | |
| 38 | + #define WINDOWS_GOBJECT_H_INCLUDED | |
| 39 | + | |
| 40 | + #include <windows.h> | |
| 41 | + #include <config.h> | |
| 42 | + | |
| 43 | + #define ENABLE_NLS | |
| 44 | + #define GETTEXT_PACKAGE PACKAGE_NAME | |
| 45 | + | |
| 46 | + #include <libintl.h> | |
| 47 | + #include <glib/gi18n.h> | |
| 48 | + #include <gio/gio.h> | |
| 49 | + | |
| 50 | + #include <lib3270.h> | |
| 51 | + #include <lib3270/ipc.h> | |
| 52 | + | |
| 53 | + #define PIPE_BUFFER_LENGTH 8192 | |
| 54 | + | |
| 55 | + G_BEGIN_DECLS | |
| 56 | + | |
| 57 | + typedef struct _ipc3270 ipc3270; | |
| 58 | + typedef struct _ipc3270Class ipc3270Class; | |
| 59 | + | |
| 60 | + typedef enum _ipc3270_pipe_state { | |
| 61 | + PIPE_STATE_WAITING, | |
| 62 | + PIPE_STATE_READ, | |
| 63 | + PIPE_STATE_PENDING_READ, | |
| 64 | + PIPE_STATE_UNDEFINED | |
| 65 | + } IPC3270_PIPE_STATE; | |
| 66 | + | |
| 67 | + typedef struct _ipc3270_pipe_source { | |
| 68 | + GSource gsrc; | |
| 69 | + HANDLE hPipe; | |
| 70 | + | |
| 71 | + enum PIPE_STATE state; | |
| 72 | + | |
| 73 | + OVERLAPPED overlap; | |
| 74 | + unsigned char buffer[PIPE_BUFFER_LENGTH+1]; | |
| 75 | + } IPC3270_PIPE_SOURCE; | |
| 76 | + | |
| 77 | + struct _ipc3270 { | |
| 78 | + H3270 * hSession; | |
| 79 | + IPC3270_PIPE_SOURCE * source; | |
| 80 | + GQuark error_domain; | |
| 81 | + }; | |
| 82 | + | |
| 83 | + struct _ipc3270Class { | |
| 84 | + GObjectClass parent; | |
| 85 | + }; | |
| 86 | + | |
| 87 | + G_GNUC_INTERNAL GSourceFuncs ipc3270_source_funcs; | |
| 88 | + | |
| 89 | + G_GNUC_INTERNAL void ipc3270_wait_for_client(IPC3270_PIPE_SOURCE *source); | |
| 90 | + | |
| 91 | + G_END_DECLS | |
| 92 | + | |
| 93 | +#endif // WINDOWS_GOBJECT_H_INCLUDED | ... | ... |
| ... | ... | @@ -0,0 +1,210 @@ |
| 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. | |
| 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 pipesource.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 | + */ | |
| 29 | + | |
| 30 | +#include "gobject.h" | |
| 31 | + | |
| 32 | +void ipc3270_wait_for_client(IPC3270_PIPE_SOURCE *source) { | |
| 33 | + | |
| 34 | + if(ConnectNamedPipe(source->hPipe,&source->overlap)) { | |
| 35 | + g_message("Error %u in ConnectNamedPipe",(unsigned int) getLastError()); | |
| 36 | + // popup_lasterror("%s",_( "Error in ConnectNamedPipe" )); | |
| 37 | + return; | |
| 38 | + } | |
| 39 | + | |
| 40 | + switch(GetLastError()) { | |
| 41 | + case ERROR_IO_PENDING: // The overlapped connection in progress. | |
| 42 | + source->state = PIPE_STATE_WAITING; | |
| 43 | + break; | |
| 44 | + | |
| 45 | + | |
| 46 | + case ERROR_PIPE_CONNECTED: // Client is already connected, so signal an event. | |
| 47 | + if(SetEvent(source->overlap.hEvent)) | |
| 48 | + break; | |
| 49 | + | |
| 50 | + default: | |
| 51 | + g_message("Error %u in ConnectNamedPipe",(unsigned int) getLastError()); | |
| 52 | + // popup_lasterror("%s", _( "ConnectNamedPipe failed" )); | |
| 53 | + } | |
| 54 | + | |
| 55 | +} | |
| 56 | + | |
| 57 | +static gboolean IO_prepare(GSource *source, gint *timeout) { | |
| 58 | + /* | |
| 59 | + * Called before all the file descriptors are polled. | |
| 60 | + * If the source can determine that it is ready here | |
| 61 | + * (without waiting for the results of the poll() call) | |
| 62 | + * it should return TRUE. | |
| 63 | + * | |
| 64 | + * It can also return a timeout_ value which should be the maximum | |
| 65 | + * timeout (in milliseconds) which should be passed to the poll() call. | |
| 66 | + * The actual timeout used will be -1 if all sources returned -1, | |
| 67 | + * or it will be the minimum of all the timeout_ values | |
| 68 | + * returned which were >= 0. | |
| 69 | + * | |
| 70 | + */ | |
| 71 | + if(WaitForSingleObject(((pipe_source *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | |
| 72 | + return TRUE; | |
| 73 | + | |
| 74 | + *timeout = 10; | |
| 75 | + | |
| 76 | + return FALSE; | |
| 77 | + | |
| 78 | +} | |
| 79 | + | |
| 80 | +static gboolean IO_check(GSource *source) { | |
| 81 | + /* | |
| 82 | + * Called after all the file descriptors are polled. | |
| 83 | + * The source should return TRUE if it is ready to be dispatched. | |
| 84 | + * Note that some time may have passed since the previous prepare | |
| 85 | + * function was called, so the source should be checked again here. | |
| 86 | + * | |
| 87 | + */ | |
| 88 | + if(WaitForSingleObject(((IPC3270_PIPE_SOURCE *) source)->overlap.hEvent,0) == WAIT_OBJECT_0) | |
| 89 | + return TRUE; | |
| 90 | + | |
| 91 | + return FALSE; | |
| 92 | +} | |
| 93 | + | |
| 94 | +static void process_input(IPC3270_PIPE_SOURCE *source, DWORD cbRead) { | |
| 95 | + | |
| 96 | +} | |
| 97 | + | |
| 98 | +static void read_input_pipe(IPC3270_PIPE_SOURCE *source) { | |
| 99 | + | |
| 100 | + DWORD cbRead = 0; | |
| 101 | + | |
| 102 | + if(ReadFile(source->hPipe,source->buffer,PIPE_BUFFER_LENGTH,&cbRead,&source->overlap) && cbRead > 0) | |
| 103 | + process_input(source,cbRead); | |
| 104 | + | |
| 105 | + // The read operation is still pending. | |
| 106 | + switch(GetLastError()) | |
| 107 | + { | |
| 108 | + case 0: | |
| 109 | + break; | |
| 110 | + | |
| 111 | + case ERROR_IO_PENDING: | |
| 112 | + source->state = PIPE_STATE_PENDING_READ; | |
| 113 | + break; | |
| 114 | + | |
| 115 | + case ERROR_PIPE_LISTENING: | |
| 116 | + source->state = PIPE_STATE_READ; | |
| 117 | + break; | |
| 118 | + | |
| 119 | + case ERROR_BROKEN_PIPE: | |
| 120 | + | |
| 121 | + if(!DisconnectNamedPipe(source->hPipe)) { | |
| 122 | + | |
| 123 | + g_message("Error %u on DisconnectNamedPipe", (unsigned int) GetLastError()); | |
| 124 | + | |
| 125 | + } else { | |
| 126 | + | |
| 127 | + ipc3270_wait_for_client(source); | |
| 128 | + | |
| 129 | + } | |
| 130 | + break; | |
| 131 | + | |
| 132 | + case ERROR_PIPE_NOT_CONNECTED: | |
| 133 | + g_message("Cliente was disconnected"); | |
| 134 | + break; | |
| 135 | + | |
| 136 | + default: | |
| 137 | + if(source->hPipe != INVALID_HANDLE_VALUE) { | |
| 138 | + g_message( "Error %u receiving message from pipe", (unsigned int) GetLastError()); | |
| 139 | + } | |
| 140 | + } | |
| 141 | + | |
| 142 | +} | |
| 143 | + | |
| 144 | + | |
| 145 | +static gboolean IO_dispatch(GSource *source, GSourceFunc callback, gpointer data) { | |
| 146 | + /* | |
| 147 | + * Called to dispatch the event source, | |
| 148 | + * after it has returned TRUE in either its prepare or its check function. | |
| 149 | + * The dispatch function is passed in a callback function and data. | |
| 150 | + * The callback function may be NULL if the source was never connected | |
| 151 | + * to a callback using g_source_set_callback(). The dispatch function | |
| 152 | + * should call the callback function with user_data and whatever additional | |
| 153 | + * parameters are needed for this type of event source. | |
| 154 | + */ | |
| 155 | + BOOL fSuccess; | |
| 156 | + DWORD cbRead = 0; | |
| 157 | + | |
| 158 | + fSuccess = GetOverlappedResult(((pipe_source *) source)->hPipe,&((pipe_source *) source)->overlap,&cbRead,FALSE ); | |
| 159 | + | |
| 160 | + switch(((pipe_source *) source)->state) { | |
| 161 | + case PIPE_STATE_WAITING: | |
| 162 | + | |
| 163 | + if(fSuccess) { | |
| 164 | + | |
| 165 | + g_message("Client is connected"); | |
| 166 | + ((pipe_source *) source)->state = PIPE_STATE_READ; | |
| 167 | + | |
| 168 | + } else { | |
| 169 | + | |
| 170 | + // popup_lasterror("%s", _( "Pipe connection failed" )); | |
| 171 | + | |
| 172 | + } | |
| 173 | + break; | |
| 174 | + | |
| 175 | + case PIPE_STATE_READ: | |
| 176 | + // trace("Reading pipe (cbRead=%d)",(int) cbRead); | |
| 177 | + read_input_pipe( (pipe_source *) source); | |
| 178 | + break; | |
| 179 | + | |
| 180 | + case PIPE_STATE_PENDING_READ: | |
| 181 | + if(fSuccess && cbRead > 0) | |
| 182 | + process_input((pipe_source *) source,cbRead); | |
| 183 | + ((pipe_source *) source)->state = PIPE_STATE_READ; | |
| 184 | + break; | |
| 185 | + | |
| 186 | + case PIPE_STATE_UNDEFINED: | |
| 187 | + break; | |
| 188 | + | |
| 189 | + } | |
| 190 | + | |
| 191 | + return TRUE; | |
| 192 | +} | |
| 193 | + | |
| 194 | +static void IO_finalize(GSource *source) { | |
| 195 | + | |
| 196 | + if( ((pipe_source *) source)->hPipe != INVALID_HANDLE_VALUE) { | |
| 197 | + CloseHandle(((pipe_source *) source)->hPipe); | |
| 198 | + ((pipe_source *) source)->hPipe = INVALID_HANDLE_VALUE; | |
| 199 | + } | |
| 200 | + | |
| 201 | +} | |
| 202 | + | |
| 203 | +GSourceFuncs ipc3270_source_funcs = { | |
| 204 | + IO_prepare, | |
| 205 | + IO_check, | |
| 206 | + IO_dispatch, | |
| 207 | + IO_finalize, | |
| 208 | + IO_closure, | |
| 209 | + NULL | |
| 210 | +}; | ... | ... |