Commit 5fd336a0580916aeea6609784986aae40cde9662
1 parent
445c656a
Exists in
master
and in
3 other branches
DNS Resolve and connect now runs on background thread.
Showing
4 changed files
with
973 additions
and
560 deletions
Show diff stats
lib3270.cbp
| @@ -157,6 +157,9 @@ | @@ -157,6 +157,9 @@ | ||
| 157 | <Unit filename="src/lib3270/kybd.c"> | 157 | <Unit filename="src/lib3270/kybd.c"> |
| 158 | <Option compilerVar="CC" /> | 158 | <Option compilerVar="CC" /> |
| 159 | </Unit> | 159 | </Unit> |
| 160 | + <Unit filename="src/lib3270/linux/connect.c"> | ||
| 161 | + <Option compilerVar="CC" /> | ||
| 162 | + </Unit> | ||
| 160 | <Unit filename="src/lib3270/linux/event_dispatcher.c"> | 163 | <Unit filename="src/lib3270/linux/event_dispatcher.c"> |
| 161 | <Option compilerVar="CC" /> | 164 | <Option compilerVar="CC" /> |
| 162 | </Unit> | 165 | </Unit> |
| @@ -245,6 +248,9 @@ | @@ -245,6 +248,9 @@ | ||
| 245 | <Unit filename="src/lib3270/version.c"> | 248 | <Unit filename="src/lib3270/version.c"> |
| 246 | <Option compilerVar="CC" /> | 249 | <Option compilerVar="CC" /> |
| 247 | </Unit> | 250 | </Unit> |
| 251 | + <Unit filename="src/lib3270/windows/connect.c"> | ||
| 252 | + <Option compilerVar="CC" /> | ||
| 253 | + </Unit> | ||
| 248 | <Unit filename="src/lib3270/windows/event_dispatcher.c"> | 254 | <Unit filename="src/lib3270/windows/event_dispatcher.c"> |
| 249 | <Option compilerVar="CC" /> | 255 | <Option compilerVar="CC" /> |
| 250 | </Unit> | 256 | </Unit> |
src/lib3270/connect.c
| @@ -1,560 +0,0 @@ | @@ -1,560 +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. | ||
| 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 connect.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 <config.h> | ||
| 31 | - | ||
| 32 | -#if defined(_WIN32) | ||
| 33 | - | ||
| 34 | - // Compiling for WinXP or later: Expose getaddrinfo()/freeaddrinfo(). | ||
| 35 | - #undef _WIN32_WINNT | ||
| 36 | - #define _WIN32_WINNT 0x0501 | ||
| 37 | - | ||
| 38 | - #include <winsock2.h> | ||
| 39 | - #include <windows.h> | ||
| 40 | - #include <ws2tcpip.h> | ||
| 41 | - | ||
| 42 | -#endif | ||
| 43 | - | ||
| 44 | -#include "private.h" | ||
| 45 | -#include <errno.h> | ||
| 46 | - | ||
| 47 | -#if defined(_WIN32) | ||
| 48 | - #include <ws2tcpip.h> | ||
| 49 | -#else | ||
| 50 | - #include <sys/types.h> | ||
| 51 | - #include <sys/socket.h> | ||
| 52 | - #include <sys/ioctl.h> | ||
| 53 | - #include <netinet/in.h> | ||
| 54 | - #include <netdb.h> | ||
| 55 | - #include <unistd.h> | ||
| 56 | - #include <fcntl.h> | ||
| 57 | -#endif | ||
| 58 | - | ||
| 59 | -#ifdef HAVE_ICONV | ||
| 60 | - #include <iconv.h> | ||
| 61 | -#endif // HAVE_ICONV | ||
| 62 | - | ||
| 63 | -#if defined(_WIN32) /*[*/ | ||
| 64 | - #define SOCK_CLOSE(s) closesocket(s->sock); s->sock = -1; | ||
| 65 | -#else /*][*/ | ||
| 66 | - #define SOCK_CLOSE(s) close(s->sock); s->sock = -1; | ||
| 67 | -#endif /*]*/ | ||
| 68 | - | ||
| 69 | -#include <stdlib.h> | ||
| 70 | -#include "statusc.h" | ||
| 71 | -#include "hostc.h" | ||
| 72 | -#include "trace_dsc.h" | ||
| 73 | -#include "utilc.h" | ||
| 74 | -#include "telnetc.h" | ||
| 75 | -#include "screen.h" | ||
| 76 | -#include <lib3270/internals.h> | ||
| 77 | - | ||
| 78 | -/*---[ Implement ]-------------------------------------------------------------------------------*/ | ||
| 79 | - | ||
| 80 | - | ||
| 81 | -//static void net_connected(H3270 *hSession) | ||
| 82 | -static void net_connected(H3270 *hSession, int fd unused, LIB3270_IO_FLAG flag unused, void *dunno unused) | ||
| 83 | -{ | ||
| 84 | - int err; | ||
| 85 | - socklen_t len = sizeof(err); | ||
| 86 | - | ||
| 87 | - if(hSession->xio.write) { | ||
| 88 | - trace("%s write=%p",__FUNCTION__,hSession->xio.write); | ||
| 89 | - lib3270_remove_poll(hSession, hSession->xio.write); | ||
| 90 | - hSession->xio.write = NULL; | ||
| 91 | - } | ||
| 92 | - | ||
| 93 | - if(getsockopt(hSession->sock, SOL_SOCKET, SO_ERROR, (char *) &err, &len) < 0) | ||
| 94 | - { | ||
| 95 | - lib3270_disconnect(hSession); | ||
| 96 | - lib3270_popup_dialog( hSession, | ||
| 97 | - LIB3270_NOTIFY_ERROR, | ||
| 98 | - _( "Network error" ), | ||
| 99 | - _( "Unable to get connection state." ), | ||
| 100 | -#ifdef _WIN32 | ||
| 101 | - "%s", lib3270_win32_strerror(WSAGetLastError()) | ||
| 102 | -#else | ||
| 103 | - _( "%s" ), strerror(errno) | ||
| 104 | -#endif // _WIN32 | ||
| 105 | - ); | ||
| 106 | - return; | ||
| 107 | - } | ||
| 108 | - else if(err) | ||
| 109 | - { | ||
| 110 | - char buffer[4096]; | ||
| 111 | - snprintf(buffer,4095,_( "Can't connect to %s" ), hSession->host.current ); | ||
| 112 | - | ||
| 113 | - lib3270_disconnect(hSession); | ||
| 114 | - lib3270_popup_dialog( hSession, | ||
| 115 | - LIB3270_NOTIFY_ERROR, | ||
| 116 | - _( "Connection failed" ), | ||
| 117 | - buffer, | ||
| 118 | -#ifdef _WIN32 | ||
| 119 | - _( "%s"), lib3270_win32_strerror(err) | ||
| 120 | -#else | ||
| 121 | - _( "%s" ), strerror(err) | ||
| 122 | -#endif // _WIN32 | ||
| 123 | - ); | ||
| 124 | - trace("%s",__FUNCTION__); | ||
| 125 | - return; | ||
| 126 | - } | ||
| 127 | - | ||
| 128 | - hSession->xio.except = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_EXCEPTION,net_exception,0); | ||
| 129 | - hSession->xio.read = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_READ,net_input,0); | ||
| 130 | - | ||
| 131 | -#if defined(HAVE_LIBSSL) | ||
| 132 | - if(hSession->ssl.con && hSession->ssl.state == LIB3270_SSL_UNDEFINED) | ||
| 133 | - { | ||
| 134 | - if(ssl_negotiate(hSession)) | ||
| 135 | - return; | ||
| 136 | - } | ||
| 137 | -#endif | ||
| 138 | - | ||
| 139 | - lib3270_setup_session(hSession); | ||
| 140 | - lib3270_set_connected_initial(hSession); | ||
| 141 | - | ||
| 142 | -} | ||
| 143 | - | ||
| 144 | - | ||
| 145 | -#if defined(_WIN32) | ||
| 146 | - | ||
| 147 | - static void sockstart(H3270 *session) | ||
| 148 | - { | ||
| 149 | - static int initted = 0; | ||
| 150 | - WORD wVersionRequested; | ||
| 151 | - WSADATA wsaData; | ||
| 152 | - | ||
| 153 | - if (initted) | ||
| 154 | - return; | ||
| 155 | - | ||
| 156 | - initted = 1; | ||
| 157 | - | ||
| 158 | - wVersionRequested = MAKEWORD(2, 2); | ||
| 159 | - | ||
| 160 | - if (WSAStartup(wVersionRequested, &wsaData) != 0) | ||
| 161 | - { | ||
| 162 | - lib3270_popup_dialog( session, | ||
| 163 | - LIB3270_NOTIFY_CRITICAL, | ||
| 164 | - N_( "Network startup error" ), | ||
| 165 | - N_( "WSAStartup failed" ), | ||
| 166 | - "%s", lib3270_win32_strerror(GetLastError()) ); | ||
| 167 | - | ||
| 168 | - _exit(1); | ||
| 169 | - } | ||
| 170 | - | ||
| 171 | - if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2) | ||
| 172 | - { | ||
| 173 | - lib3270_popup_dialog( session, | ||
| 174 | - LIB3270_NOTIFY_CRITICAL, | ||
| 175 | - N_( "Network startup error" ), | ||
| 176 | - N_( "Bad winsock version" ), | ||
| 177 | - N_( "Can't use winsock version %d.%d" ), LOBYTE(wsaData.wVersion), HIBYTE(wsaData.wVersion)); | ||
| 178 | - _exit(1); | ||
| 179 | - } | ||
| 180 | - } | ||
| 181 | -#endif // WIN32 | ||
| 182 | - | ||
| 183 | - LIB3270_EXPORT int lib3270_connect_url(H3270 *hSession, const char *url, int wait) | ||
| 184 | - { | ||
| 185 | - CHECK_SESSION_HANDLE(hSession); | ||
| 186 | - | ||
| 187 | - if(url && *url) | ||
| 188 | - { | ||
| 189 | - lib3270_set_url(hSession,url); | ||
| 190 | - } | ||
| 191 | - | ||
| 192 | - return lib3270_connect(hSession, wait); | ||
| 193 | - | ||
| 194 | - } | ||
| 195 | - | ||
| 196 | - LIB3270_EXPORT int lib3270_connect_host(H3270 *hSession, const char *hostname, const char *srvc, LIB3270_OPTION opt) | ||
| 197 | - { | ||
| 198 | - CHECK_SESSION_HANDLE(hSession); | ||
| 199 | - | ||
| 200 | - if(!hostname) | ||
| 201 | - return EINVAL; | ||
| 202 | - | ||
| 203 | - if(!srvc) | ||
| 204 | - srvc = "telnet"; | ||
| 205 | - | ||
| 206 | - if(*hostname == '$') | ||
| 207 | - { | ||
| 208 | - const char *name = getenv(hostname+1); | ||
| 209 | - if(!name) | ||
| 210 | - { | ||
| 211 | - lib3270_popup_dialog( hSession, | ||
| 212 | - LIB3270_NOTIFY_ERROR, | ||
| 213 | - _( "Connection error" ), | ||
| 214 | - _( "Unable to find selected hostname." ), | ||
| 215 | - _( "Can't determine value for environment variable \"%s\" " ), | ||
| 216 | - hostname); | ||
| 217 | - lib3270_set_disconnected(hSession); | ||
| 218 | - return ENOENT; | ||
| 219 | - } | ||
| 220 | - hostname = name; | ||
| 221 | - } | ||
| 222 | - | ||
| 223 | - hSession->options = opt & ~LIB3270_OPTION_WAIT; | ||
| 224 | - Replace(hSession->host.current,strdup(hostname)); | ||
| 225 | - Replace(hSession->host.srvc,strdup(srvc)); | ||
| 226 | - | ||
| 227 | - Replace(hSession->host.full, | ||
| 228 | - lib3270_strdup_printf( | ||
| 229 | - "%s%s:%s", | ||
| 230 | - opt&LIB3270_OPTION_SSL ? "tn3270s://" : "tn3270://", | ||
| 231 | - hostname, | ||
| 232 | - srvc )); | ||
| 233 | - | ||
| 234 | - trace("current_host=\"%s\"",hSession->host.current); | ||
| 235 | - | ||
| 236 | - return lib3270_connect(hSession,opt & LIB3270_OPTION_WAIT); | ||
| 237 | - | ||
| 238 | - } | ||
| 239 | - | ||
| 240 | - int lib3270_connect(H3270 *hSession, int seconds) | ||
| 241 | - { | ||
| 242 | - int s; | ||
| 243 | - int optval; | ||
| 244 | - struct addrinfo hints; | ||
| 245 | - struct addrinfo * result = NULL; | ||
| 246 | - struct addrinfo * rp = NULL; | ||
| 247 | - | ||
| 248 | - CHECK_SESSION_HANDLE(hSession); | ||
| 249 | - | ||
| 250 | - lib3270_main_iterate(hSession,0); | ||
| 251 | - | ||
| 252 | - if(hSession->auto_reconnect_inprogress) | ||
| 253 | - return errno = EAGAIN; | ||
| 254 | - | ||
| 255 | - if(hSession->sock > 0) | ||
| 256 | - return errno = EBUSY; | ||
| 257 | - | ||
| 258 | -#if defined(_WIN32) | ||
| 259 | - sockstart(hSession); | ||
| 260 | -#endif | ||
| 261 | - | ||
| 262 | -#if defined(HAVE_LIBSSL) | ||
| 263 | - set_ssl_state(hSession,LIB3270_SSL_UNSECURE); | ||
| 264 | -#endif // HAVE_LIBSSL | ||
| 265 | - | ||
| 266 | - snprintf(hSession->full_model_name,LIB3270_FULL_MODEL_NAME_LENGTH,"IBM-327%c-%d",hSession->m3279 ? '9' : '8', hSession->model_num); | ||
| 267 | - | ||
| 268 | - hSession->ever_3270 = False; | ||
| 269 | - | ||
| 270 | - memset(&hints, 0, sizeof(struct addrinfo)); | ||
| 271 | - hints.ai_family = AF_UNSPEC; /* Allow IPv4 or IPv6 */ | ||
| 272 | - hints.ai_socktype = SOCK_STREAM; /* Stream socket */ | ||
| 273 | - hints.ai_flags = AI_PASSIVE; /* For wildcard IP address */ | ||
| 274 | - hints.ai_protocol = 0; /* Any protocol */ | ||
| 275 | - hints.ai_canonname = NULL; | ||
| 276 | - hints.ai_addr = NULL; | ||
| 277 | - hints.ai_next = NULL; | ||
| 278 | - | ||
| 279 | - hSession->cstate = LIB3270_RESOLVING; | ||
| 280 | - lib3270_st_changed(hSession, LIB3270_STATE_RESOLVING, True); | ||
| 281 | - | ||
| 282 | - s = getaddrinfo(hSession->host.current, hSession->host.srvc, &hints, &result); | ||
| 283 | - | ||
| 284 | - if(s != 0) | ||
| 285 | - { | ||
| 286 | - char buffer[4096]; | ||
| 287 | - | ||
| 288 | - snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 289 | - | ||
| 290 | -#if defined(WIN32) && defined(HAVE_ICONV) | ||
| 291 | - { | ||
| 292 | - char tmpbuffer[4096]; | ||
| 293 | - const char * msg = gai_strerror(s); | ||
| 294 | - size_t in = strlen(msg); | ||
| 295 | - size_t out = 4096; | ||
| 296 | - char * ptr = tmpbuffer; | ||
| 297 | - | ||
| 298 | - iconv_t hConv = iconv_open(lib3270_win32_local_charset(),"UTF-8"); | ||
| 299 | - | ||
| 300 | - trace("Antes: [%s]",msg); | ||
| 301 | - if(iconv(hConv,&msg,&in,&ptr,&out) != ((size_t) -1)) | ||
| 302 | - msg = tmpbuffer; | ||
| 303 | - trace("Depois: [%s]",msg); | ||
| 304 | - | ||
| 305 | - iconv_close(hConv); | ||
| 306 | - | ||
| 307 | - lib3270_popup_dialog( hSession, | ||
| 308 | - LIB3270_NOTIFY_ERROR, | ||
| 309 | - _( "Connection error" ), | ||
| 310 | - buffer, | ||
| 311 | - "%s", | ||
| 312 | - msg); | ||
| 313 | - } | ||
| 314 | - | ||
| 315 | -#else | ||
| 316 | - lib3270_popup_dialog( hSession, | ||
| 317 | - LIB3270_NOTIFY_ERROR, | ||
| 318 | - _( "Connection error" ), | ||
| 319 | - buffer, | ||
| 320 | - "%s", | ||
| 321 | - gai_strerror(s)); | ||
| 322 | -#endif // WIN32 | ||
| 323 | - | ||
| 324 | - | ||
| 325 | - lib3270_set_disconnected(hSession); | ||
| 326 | - return errno = ENOENT; | ||
| 327 | - } | ||
| 328 | - | ||
| 329 | - | ||
| 330 | -#if !defined(_WIN32) | ||
| 331 | - /* don't share the socket with our children */ | ||
| 332 | - (void) fcntl(hSession->sock, F_SETFD, 1); | ||
| 333 | -#endif | ||
| 334 | - | ||
| 335 | - hSession->ever_3270 = False; | ||
| 336 | - hSession->ssl.host = 0; | ||
| 337 | - | ||
| 338 | - if(hSession->options&LIB3270_OPTION_SSL) | ||
| 339 | - { | ||
| 340 | -#if defined(HAVE_LIBSSL) | ||
| 341 | - hSession->ssl.host = 1; | ||
| 342 | - ssl_init(hSession); | ||
| 343 | -#else | ||
| 344 | - lib3270_popup_dialog( hSession, | ||
| 345 | - LIB3270_NOTIFY_ERROR, | ||
| 346 | - _( "SSL error" ), | ||
| 347 | - _( "Unable to connect to secure hosts" ), | ||
| 348 | - _( "This version of %s was built without support for secure sockets layer (SSL)." ), | ||
| 349 | - PACKAGE_NAME); | ||
| 350 | - | ||
| 351 | - return errno = EINVAL; | ||
| 352 | -#endif // HAVE_LIBSSL | ||
| 353 | - } | ||
| 354 | - | ||
| 355 | - /* connect */ | ||
| 356 | - status_connecting(hSession,1); | ||
| 357 | - | ||
| 358 | - for(rp = result; hSession->sock < 0 && rp != NULL; rp = rp->ai_next) | ||
| 359 | - { | ||
| 360 | - hSession->sock = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol); | ||
| 361 | - if(hSession->sock < 0) | ||
| 362 | - continue; | ||
| 363 | - | ||
| 364 | - trace("sock=%d",hSession->sock); | ||
| 365 | - | ||
| 366 | -#ifdef WIN32 | ||
| 367 | - | ||
| 368 | - WSASetLastError(0); | ||
| 369 | - u_long iMode=1; | ||
| 370 | - trace("sock=%d",hSession->sock); | ||
| 371 | - | ||
| 372 | - optval = lib3270_get_toggle(hSession,LIB3270_TOGGLE_KEEP_ALIVE) ? 1 : 0; | ||
| 373 | - if (setsockopt(hSession->sock, SOL_SOCKET, SO_KEEPALIVE, (char *)&optval, sizeof(optval)) < 0) | ||
| 374 | - { | ||
| 375 | - char buffer[4096]; | ||
| 376 | - snprintf(buffer,4095,N_( "Can't %s network keep-alive" ), optval ? _( "enable" ) : _( "disable" )); | ||
| 377 | - | ||
| 378 | - lib3270_popup_dialog( hSession, | ||
| 379 | - LIB3270_NOTIFY_ERROR, | ||
| 380 | - _( "Connection error" ), | ||
| 381 | - buffer, | ||
| 382 | - "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 383 | - SOCK_CLOSE(hSession); | ||
| 384 | - continue; | ||
| 385 | - } | ||
| 386 | - else | ||
| 387 | - { | ||
| 388 | - trace_dsn(hSession,"Network keep-alive is %s\n",optval ? "enabled" : "disabled" ); | ||
| 389 | - } | ||
| 390 | - | ||
| 391 | - if(ioctlsocket(hSession->sock,FIONBIO,&iMode)) | ||
| 392 | - { | ||
| 393 | - lib3270_popup_dialog( hSession, | ||
| 394 | - LIB3270_NOTIFY_ERROR, | ||
| 395 | - _( "Connection error" ), | ||
| 396 | - _( "ioctlsocket(FIONBIO) failed." ), | ||
| 397 | - "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 398 | - SOCK_CLOSE(hSession); | ||
| 399 | - continue; | ||
| 400 | - } | ||
| 401 | - else if(connect(hSession->sock, rp->ai_addr, rp->ai_addrlen)) | ||
| 402 | - { | ||
| 403 | - int err = WSAGetLastError(); | ||
| 404 | - if(err != WSAEWOULDBLOCK) | ||
| 405 | - { | ||
| 406 | - char buffer[4096]; | ||
| 407 | - snprintf(buffer,4095,_( "Can't connect to %s"), lib3270_get_host(hSession)); | ||
| 408 | - | ||
| 409 | - lib3270_popup_dialog( hSession, | ||
| 410 | - LIB3270_NOTIFY_ERROR, | ||
| 411 | - _( "Connection error" ), | ||
| 412 | - buffer, | ||
| 413 | - "%s", lib3270_win32_strerror(err)); | ||
| 414 | - SOCK_CLOSE(hSession); | ||
| 415 | - continue; | ||
| 416 | - | ||
| 417 | - } | ||
| 418 | - } | ||
| 419 | - | ||
| 420 | - optval = 1; | ||
| 421 | - if (setsockopt(hSession->sock, SOL_SOCKET, SO_OOBINLINE, (char *)&optval,sizeof(optval)) < 0) | ||
| 422 | - { | ||
| 423 | - lib3270_popup_dialog( hSession, | ||
| 424 | - LIB3270_NOTIFY_ERROR, | ||
| 425 | - _( "Connection error" ), | ||
| 426 | - _( "setsockopt(SO_OOBINLINE) has failed" ), | ||
| 427 | - "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 428 | - SOCK_CLOSE(hSession); | ||
| 429 | - continue; | ||
| 430 | - } | ||
| 431 | - | ||
| 432 | -#else | ||
| 433 | - fcntl(hSession->sock, F_SETFL,fcntl(hSession->sock,F_GETFL,0)|O_NONBLOCK); | ||
| 434 | - | ||
| 435 | - errno = 0; | ||
| 436 | - if(connect(hSession->sock, rp->ai_addr, rp->ai_addrlen)) | ||
| 437 | - { | ||
| 438 | - if( errno != EINPROGRESS ) | ||
| 439 | - { | ||
| 440 | - char buffer[4096]; | ||
| 441 | - snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 442 | - | ||
| 443 | - lib3270_popup_dialog( hSession, | ||
| 444 | - LIB3270_NOTIFY_ERROR, | ||
| 445 | - _( "Connection error" ), | ||
| 446 | - buffer, | ||
| 447 | - "%s", | ||
| 448 | - strerror(errno)); | ||
| 449 | - SOCK_CLOSE(hSession); | ||
| 450 | - continue; | ||
| 451 | - } | ||
| 452 | - } | ||
| 453 | - | ||
| 454 | - optval = 1; | ||
| 455 | - if (setsockopt(hSession->sock, SOL_SOCKET, SO_OOBINLINE, (char *)&optval,sizeof(optval)) < 0) | ||
| 456 | - { | ||
| 457 | - lib3270_popup_dialog( hSession, | ||
| 458 | - LIB3270_NOTIFY_ERROR, | ||
| 459 | - _( "Connection error" ), | ||
| 460 | - _( "setsockopt(SO_OOBINLINE) has failed" ), | ||
| 461 | - "%s", | ||
| 462 | - strerror(errno)); | ||
| 463 | - SOCK_CLOSE(hSession); | ||
| 464 | - continue; | ||
| 465 | - } | ||
| 466 | - | ||
| 467 | - optval = lib3270_get_toggle(hSession,LIB3270_TOGGLE_KEEP_ALIVE) ? 1 : 0; | ||
| 468 | - if (setsockopt(hSession->sock, SOL_SOCKET, SO_KEEPALIVE, (char *)&optval, sizeof(optval)) < 0) | ||
| 469 | - { | ||
| 470 | - char buffer[4096]; | ||
| 471 | - snprintf(buffer,4095,N_( "Can't %s network keep-alive" ), optval ? _( "enable" ) : _( "disable" )); | ||
| 472 | - | ||
| 473 | - lib3270_popup_dialog( hSession, | ||
| 474 | - LIB3270_NOTIFY_ERROR, | ||
| 475 | - _( "Connection error" ), | ||
| 476 | - buffer, | ||
| 477 | - "%s", | ||
| 478 | - strerror(errno)); | ||
| 479 | - SOCK_CLOSE(hSession); | ||
| 480 | - continue; | ||
| 481 | - } | ||
| 482 | - else | ||
| 483 | - { | ||
| 484 | - trace_dsn(hSession,"Network keep-alive is %s\n",optval ? "enabled" : "disabled" ); | ||
| 485 | - } | ||
| 486 | - | ||
| 487 | -#endif // WIN32 | ||
| 488 | - } | ||
| 489 | - | ||
| 490 | - freeaddrinfo(result); | ||
| 491 | - | ||
| 492 | - // set options for inline out-of-band data and keepalives | ||
| 493 | - | ||
| 494 | - /* | ||
| 495 | -#if defined(OMTU) | ||
| 496 | - else if (setsockopt(hSession->sock, SOL_SOCKET, SO_SNDBUF, (char *)&mtu,sizeof(mtu)) < 0) | ||
| 497 | - { | ||
| 498 | - popup_a_sockerr(hSession, N_( "setsockopt(%s)" ), "SO_SNDBUF"); | ||
| 499 | - SOCK_CLOSE(hSession); | ||
| 500 | - } | ||
| 501 | -#endif | ||
| 502 | - | ||
| 503 | - */ | ||
| 504 | - | ||
| 505 | - if(hSession->sock < 0) | ||
| 506 | - { | ||
| 507 | - lib3270_set_disconnected(hSession); | ||
| 508 | - return errno = ENOTCONN; | ||
| 509 | - } | ||
| 510 | - | ||
| 511 | - // Connecting, set callbacks, wait for connection | ||
| 512 | - hSession->cstate = LIB3270_PENDING; | ||
| 513 | - lib3270_st_changed(hSession, LIB3270_STATE_HALF_CONNECT, True); | ||
| 514 | - | ||
| 515 | - hSession->xio.write = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_WRITE,net_connected,0); | ||
| 516 | - // hSession->ns_write_id = AddOutput(hSession->sock, hSession, net_connected); | ||
| 517 | - | ||
| 518 | - trace("%s: Connection in progress",__FUNCTION__); | ||
| 519 | - | ||
| 520 | - if(seconds) | ||
| 521 | - { | ||
| 522 | - time_t end = time(0)+seconds; | ||
| 523 | - | ||
| 524 | - while(time(0) < end) | ||
| 525 | - { | ||
| 526 | - lib3270_main_iterate(hSession,1); | ||
| 527 | - | ||
| 528 | - switch(hSession->cstate) | ||
| 529 | - { | ||
| 530 | - case LIB3270_PENDING: | ||
| 531 | - case LIB3270_CONNECTED_INITIAL: | ||
| 532 | - case LIB3270_CONNECTED_ANSI: | ||
| 533 | - case LIB3270_CONNECTED_3270: | ||
| 534 | - case LIB3270_CONNECTED_INITIAL_E: | ||
| 535 | - case LIB3270_CONNECTED_NVT: | ||
| 536 | - case LIB3270_CONNECTED_SSCP: | ||
| 537 | - break; | ||
| 538 | - | ||
| 539 | - case LIB3270_NOT_CONNECTED: | ||
| 540 | - return errno = ENOTCONN; | ||
| 541 | - | ||
| 542 | - case LIB3270_CONNECTED_TN3270E: | ||
| 543 | - return 0; | ||
| 544 | - | ||
| 545 | - default: | ||
| 546 | - lib3270_write_log(hSession,"connect", "%s: State changed to unexpected state %d",__FUNCTION__,hSession->cstate); | ||
| 547 | - return -1; | ||
| 548 | - } | ||
| 549 | - | ||
| 550 | - } | ||
| 551 | - | ||
| 552 | - lib3270_disconnect(hSession); | ||
| 553 | - lib3270_write_log(hSession,"connect", "%s: %s",__FUNCTION__,strerror(ETIMEDOUT)); | ||
| 554 | - return errno = ETIMEDOUT; | ||
| 555 | - } | ||
| 556 | - | ||
| 557 | - return 0; | ||
| 558 | - | ||
| 559 | - } | ||
| 560 | - |
| @@ -0,0 +1,407 @@ | @@ -0,0 +1,407 @@ | ||
| 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 - 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 <config.h> | ||
| 31 | +#include "../private.h" | ||
| 32 | +#include <errno.h> | ||
| 33 | + | ||
| 34 | +#include <sys/types.h> | ||
| 35 | +#include <sys/socket.h> | ||
| 36 | +#include <sys/ioctl.h> | ||
| 37 | +#include <netinet/in.h> | ||
| 38 | +#include <netdb.h> | ||
| 39 | +#include <unistd.h> | ||
| 40 | +#include <fcntl.h> | ||
| 41 | + | ||
| 42 | +// #ifdef HAVE_ICONV | ||
| 43 | +// #include <iconv.h> | ||
| 44 | +// #endif // HAVE_ICONV | ||
| 45 | + | ||
| 46 | +#define SOCK_CLOSE(s) close(s->sock); s->sock = -1; | ||
| 47 | + | ||
| 48 | +#include <stdlib.h> | ||
| 49 | + | ||
| 50 | +//#include "statusc.h" | ||
| 51 | +#include "hostc.h" | ||
| 52 | +#include "trace_dsc.h" | ||
| 53 | +//#include "utilc.h" | ||
| 54 | +#include "telnetc.h" | ||
| 55 | +#include "screen.h" | ||
| 56 | + | ||
| 57 | +#include <lib3270/internals.h> | ||
| 58 | + | ||
| 59 | +/*---[ Implement ]-------------------------------------------------------------------------------*/ | ||
| 60 | + | ||
| 61 | + | ||
| 62 | +//static void net_connected(H3270 *hSession) | ||
| 63 | +static void net_connected(H3270 *hSession, int fd unused, LIB3270_IO_FLAG flag unused, void *dunno unused) | ||
| 64 | +{ | ||
| 65 | + int err; | ||
| 66 | + socklen_t len = sizeof(err); | ||
| 67 | + | ||
| 68 | + if(hSession->xio.write) { | ||
| 69 | + trace("%s write=%p",__FUNCTION__,hSession->xio.write); | ||
| 70 | + lib3270_remove_poll(hSession, hSession->xio.write); | ||
| 71 | + hSession->xio.write = NULL; | ||
| 72 | + } | ||
| 73 | + | ||
| 74 | + if(getsockopt(hSession->sock, SOL_SOCKET, SO_ERROR, (char *) &err, &len) < 0) | ||
| 75 | + { | ||
| 76 | + lib3270_disconnect(hSession); | ||
| 77 | + lib3270_popup_dialog( | ||
| 78 | + hSession, | ||
| 79 | + LIB3270_NOTIFY_ERROR, | ||
| 80 | + _( "Network error" ), | ||
| 81 | + _( "Unable to get connection state." ), | ||
| 82 | + _( "%s" ), strerror(errno) | ||
| 83 | + ); | ||
| 84 | + return; | ||
| 85 | + } | ||
| 86 | + else if(err) | ||
| 87 | + { | ||
| 88 | + char buffer[4096]; | ||
| 89 | + snprintf(buffer,4095,_( "Can't connect to %s" ), hSession->host.current ); | ||
| 90 | + | ||
| 91 | + lib3270_disconnect(hSession); | ||
| 92 | + lib3270_popup_dialog( | ||
| 93 | + hSession, | ||
| 94 | + LIB3270_NOTIFY_ERROR, | ||
| 95 | + _( "Connection failed" ), | ||
| 96 | + buffer, | ||
| 97 | + _( "%s" ), strerror(err) | ||
| 98 | + ); | ||
| 99 | + return; | ||
| 100 | + } | ||
| 101 | + | ||
| 102 | + hSession->xio.except = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_EXCEPTION,net_exception,0); | ||
| 103 | + hSession->xio.read = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_READ,net_input,0); | ||
| 104 | + | ||
| 105 | +#if defined(HAVE_LIBSSL) | ||
| 106 | + if(hSession->ssl.con && hSession->ssl.state == LIB3270_SSL_UNDEFINED) | ||
| 107 | + { | ||
| 108 | + if(ssl_negotiate(hSession)) | ||
| 109 | + return; | ||
| 110 | + } | ||
| 111 | +#endif | ||
| 112 | + | ||
| 113 | + lib3270_setup_session(hSession); | ||
| 114 | + lib3270_set_connected_initial(hSession); | ||
| 115 | + | ||
| 116 | +} | ||
| 117 | + | ||
| 118 | + LIB3270_EXPORT int lib3270_connect_url(H3270 *hSession, const char *url, int wait) | ||
| 119 | + { | ||
| 120 | + CHECK_SESSION_HANDLE(hSession); | ||
| 121 | + | ||
| 122 | + if(url && *url) | ||
| 123 | + { | ||
| 124 | + lib3270_set_url(hSession,url); | ||
| 125 | + } | ||
| 126 | + | ||
| 127 | + return lib3270_connect(hSession, wait); | ||
| 128 | + | ||
| 129 | + } | ||
| 130 | + | ||
| 131 | + LIB3270_EXPORT int lib3270_connect_host(H3270 *hSession, const char *hostname, const char *srvc, LIB3270_OPTION opt) | ||
| 132 | + { | ||
| 133 | + CHECK_SESSION_HANDLE(hSession); | ||
| 134 | + | ||
| 135 | + if(!hostname) | ||
| 136 | + return EINVAL; | ||
| 137 | + | ||
| 138 | + if(!srvc) | ||
| 139 | + srvc = "telnet"; | ||
| 140 | + | ||
| 141 | + if(*hostname == '$') | ||
| 142 | + { | ||
| 143 | + const char *name = getenv(hostname+1); | ||
| 144 | + if(!name) | ||
| 145 | + { | ||
| 146 | + lib3270_popup_dialog( hSession, | ||
| 147 | + LIB3270_NOTIFY_ERROR, | ||
| 148 | + _( "Connection error" ), | ||
| 149 | + _( "Unable to find selected hostname." ), | ||
| 150 | + _( "Can't determine value for environment variable \"%s\" " ), | ||
| 151 | + hostname); | ||
| 152 | + lib3270_set_disconnected(hSession); | ||
| 153 | + return ENOENT; | ||
| 154 | + } | ||
| 155 | + hostname = name; | ||
| 156 | + } | ||
| 157 | + | ||
| 158 | + hSession->options = opt & ~LIB3270_OPTION_WAIT; | ||
| 159 | + Replace(hSession->host.current,strdup(hostname)); | ||
| 160 | + Replace(hSession->host.srvc,strdup(srvc)); | ||
| 161 | + | ||
| 162 | + Replace(hSession->host.full, | ||
| 163 | + lib3270_strdup_printf( | ||
| 164 | + "%s%s:%s", | ||
| 165 | + opt&LIB3270_OPTION_SSL ? "tn3270s://" : "tn3270://", | ||
| 166 | + hostname, | ||
| 167 | + srvc )); | ||
| 168 | + | ||
| 169 | + trace("current_host=\"%s\"",hSession->host.current); | ||
| 170 | + | ||
| 171 | + return lib3270_connect(hSession,opt & LIB3270_OPTION_WAIT); | ||
| 172 | + | ||
| 173 | + } | ||
| 174 | + | ||
| 175 | + struct resolver | ||
| 176 | + { | ||
| 177 | + const char * message; | ||
| 178 | + }; | ||
| 179 | + | ||
| 180 | + static int background_connect(H3270 *hSession, void *host) | ||
| 181 | + { | ||
| 182 | + struct addrinfo hints; | ||
| 183 | + struct addrinfo * result = NULL; | ||
| 184 | + struct addrinfo * rp = NULL; | ||
| 185 | + | ||
| 186 | + memset(&hints,0,sizeof(hints)); | ||
| 187 | + hints.ai_family = AF_UNSPEC; // Allow IPv4 or IPv6 | ||
| 188 | + hints.ai_socktype = SOCK_STREAM; // Stream socket | ||
| 189 | + hints.ai_flags = AI_PASSIVE; // For wildcard IP address | ||
| 190 | + hints.ai_protocol = 0; // Any protocol | ||
| 191 | + | ||
| 192 | + int rc = getaddrinfo(hSession->host.current, hSession->host.srvc, &hints, &result); | ||
| 193 | + if(rc != 0) | ||
| 194 | + { | ||
| 195 | + ((struct resolver *) host)->message = gai_strerror(rc); | ||
| 196 | + return -1; | ||
| 197 | + } | ||
| 198 | + | ||
| 199 | + status_connecting(hSession,1); | ||
| 200 | + | ||
| 201 | + for(rp = result; hSession->sock < 0 && rp != NULL; rp = rp->ai_next) | ||
| 202 | + { | ||
| 203 | + hSession->sock = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol); | ||
| 204 | + if(hSession->sock < 0) | ||
| 205 | + { | ||
| 206 | + ((struct resolver *) host)->message = strerror(errno); | ||
| 207 | + continue; | ||
| 208 | + } | ||
| 209 | + | ||
| 210 | + // Connected! | ||
| 211 | + if(connect(hSession->sock, rp->ai_addr, rp->ai_addrlen)) | ||
| 212 | + { | ||
| 213 | + SOCK_CLOSE(hSession); | ||
| 214 | + ((struct resolver *) host)->message = strerror(errno); | ||
| 215 | + continue; | ||
| 216 | + } | ||
| 217 | + | ||
| 218 | + } | ||
| 219 | + | ||
| 220 | + freeaddrinfo(result); | ||
| 221 | + | ||
| 222 | + return 0; | ||
| 223 | + | ||
| 224 | + } | ||
| 225 | + | ||
| 226 | + int lib3270_connect(H3270 *hSession, int seconds) | ||
| 227 | + { | ||
| 228 | + int optval; | ||
| 229 | + struct resolver host; | ||
| 230 | + | ||
| 231 | + CHECK_SESSION_HANDLE(hSession); | ||
| 232 | + | ||
| 233 | + memset(&host,0,sizeof(host)); | ||
| 234 | + | ||
| 235 | + lib3270_main_iterate(hSession,0); | ||
| 236 | + | ||
| 237 | + if(hSession->auto_reconnect_inprogress) | ||
| 238 | + return errno = EAGAIN; | ||
| 239 | + | ||
| 240 | + if(hSession->sock > 0) | ||
| 241 | + return errno = EBUSY; | ||
| 242 | + | ||
| 243 | +#if defined(HAVE_LIBSSL) | ||
| 244 | + set_ssl_state(hSession,LIB3270_SSL_UNSECURE); | ||
| 245 | +#endif // HAVE_LIBSSL | ||
| 246 | + | ||
| 247 | + snprintf(hSession->full_model_name,LIB3270_FULL_MODEL_NAME_LENGTH,"IBM-327%c-%d",hSession->m3279 ? '9' : '8', hSession->model_num); | ||
| 248 | + | ||
| 249 | + hSession->ever_3270 = False; | ||
| 250 | + hSession->cstate = LIB3270_RESOLVING; | ||
| 251 | + | ||
| 252 | + lib3270_st_changed(hSession, LIB3270_STATE_RESOLVING, True); | ||
| 253 | + | ||
| 254 | + // s = getaddrinfo(hSession->host.current, hSession->host.srvc, &hints, &result); | ||
| 255 | + if(lib3270_run_task(hSession, background_connect, &host)) | ||
| 256 | + { | ||
| 257 | + char buffer[4096]; | ||
| 258 | + snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 259 | + | ||
| 260 | + lib3270_popup_dialog( hSession, | ||
| 261 | + LIB3270_NOTIFY_ERROR, | ||
| 262 | + _( "Connection error" ), | ||
| 263 | + buffer, | ||
| 264 | + "%s", | ||
| 265 | + host.message); | ||
| 266 | + | ||
| 267 | + lib3270_set_disconnected(hSession); | ||
| 268 | + return errno = ENOENT; | ||
| 269 | + } | ||
| 270 | + | ||
| 271 | + /* don't share the socket with our children */ | ||
| 272 | + (void) fcntl(hSession->sock, F_SETFD, 1); | ||
| 273 | + | ||
| 274 | + hSession->ever_3270 = False; | ||
| 275 | + hSession->ssl.host = 0; | ||
| 276 | + | ||
| 277 | + if(hSession->options&LIB3270_OPTION_SSL) | ||
| 278 | + { | ||
| 279 | +#if defined(HAVE_LIBSSL) | ||
| 280 | + hSession->ssl.host = 1; | ||
| 281 | + ssl_init(hSession); | ||
| 282 | +#else | ||
| 283 | + lib3270_popup_dialog( hSession, | ||
| 284 | + LIB3270_NOTIFY_ERROR, | ||
| 285 | + _( "SSL error" ), | ||
| 286 | + _( "Unable to connect to secure hosts" ), | ||
| 287 | + _( "This version of %s was built without support for secure sockets layer (SSL)." ), | ||
| 288 | + PACKAGE_NAME); | ||
| 289 | + | ||
| 290 | + return errno = EINVAL; | ||
| 291 | +#endif // HAVE_LIBSSL | ||
| 292 | + } | ||
| 293 | + | ||
| 294 | + /* connect */ | ||
| 295 | + if(hSession->sock < 0) | ||
| 296 | + { | ||
| 297 | + lib3270_set_disconnected(hSession); | ||
| 298 | + | ||
| 299 | + char buffer[4096]; | ||
| 300 | + snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 301 | + | ||
| 302 | + lib3270_popup_dialog( | ||
| 303 | + hSession, | ||
| 304 | + LIB3270_NOTIFY_ERROR, | ||
| 305 | + _( "Connection error" ), | ||
| 306 | + buffer, | ||
| 307 | + "%s", | ||
| 308 | + host.message | ||
| 309 | + ); | ||
| 310 | + | ||
| 311 | + return errno = ENOTCONN; | ||
| 312 | + } | ||
| 313 | + | ||
| 314 | + // set options for inline out-of-band data and keepalives | ||
| 315 | + optval = 1; | ||
| 316 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_OOBINLINE, (char *)&optval,sizeof(optval)) < 0) | ||
| 317 | + { | ||
| 318 | + lib3270_popup_dialog( hSession, | ||
| 319 | + LIB3270_NOTIFY_ERROR, | ||
| 320 | + _( "Connection error" ), | ||
| 321 | + _( "setsockopt(SO_OOBINLINE) has failed" ), | ||
| 322 | + "%s", | ||
| 323 | + strerror(errno)); | ||
| 324 | + SOCK_CLOSE(hSession); | ||
| 325 | + } | ||
| 326 | + | ||
| 327 | + optval = lib3270_get_toggle(hSession,LIB3270_TOGGLE_KEEP_ALIVE) ? 1 : 0; | ||
| 328 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_KEEPALIVE, (char *)&optval, sizeof(optval)) < 0) | ||
| 329 | + { | ||
| 330 | + char buffer[4096]; | ||
| 331 | + snprintf(buffer,4095,N_( "Can't %s network keep-alive" ), optval ? _( "enable" ) : _( "disable" )); | ||
| 332 | + | ||
| 333 | + lib3270_popup_dialog( hSession, | ||
| 334 | + LIB3270_NOTIFY_ERROR, | ||
| 335 | + _( "Connection error" ), | ||
| 336 | + buffer, | ||
| 337 | + "%s", | ||
| 338 | + strerror(errno)); | ||
| 339 | + SOCK_CLOSE(hSession); | ||
| 340 | + } | ||
| 341 | + else | ||
| 342 | + { | ||
| 343 | + trace_dsn(hSession,"Network keep-alive is %s\n",optval ? "enabled" : "disabled" ); | ||
| 344 | + } | ||
| 345 | + | ||
| 346 | + | ||
| 347 | + /* | ||
| 348 | +#if defined(OMTU) | ||
| 349 | + else if (setsockopt(hSession->sock, SOL_SOCKET, SO_SNDBUF, (char *)&mtu,sizeof(mtu)) < 0) | ||
| 350 | + { | ||
| 351 | + popup_a_sockerr(hSession, N_( "setsockopt(%s)" ), "SO_SNDBUF"); | ||
| 352 | + SOCK_CLOSE(hSession); | ||
| 353 | + } | ||
| 354 | +#endif | ||
| 355 | + | ||
| 356 | + */ | ||
| 357 | + | ||
| 358 | + // Connecting, set callbacks, wait for connection | ||
| 359 | + hSession->cstate = LIB3270_PENDING; | ||
| 360 | + lib3270_st_changed(hSession, LIB3270_STATE_HALF_CONNECT, True); | ||
| 361 | + | ||
| 362 | + hSession->xio.write = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_WRITE,net_connected,0); | ||
| 363 | + // hSession->ns_write_id = AddOutput(hSession->sock, hSession, net_connected); | ||
| 364 | + | ||
| 365 | + trace("%s: Connection in progress",__FUNCTION__); | ||
| 366 | + | ||
| 367 | + if(seconds) | ||
| 368 | + { | ||
| 369 | + time_t end = time(0)+seconds; | ||
| 370 | + | ||
| 371 | + while(time(0) < end) | ||
| 372 | + { | ||
| 373 | + lib3270_main_iterate(hSession,1); | ||
| 374 | + | ||
| 375 | + switch(hSession->cstate) | ||
| 376 | + { | ||
| 377 | + case LIB3270_PENDING: | ||
| 378 | + case LIB3270_CONNECTED_INITIAL: | ||
| 379 | + case LIB3270_CONNECTED_ANSI: | ||
| 380 | + case LIB3270_CONNECTED_3270: | ||
| 381 | + case LIB3270_CONNECTED_INITIAL_E: | ||
| 382 | + case LIB3270_CONNECTED_NVT: | ||
| 383 | + case LIB3270_CONNECTED_SSCP: | ||
| 384 | + break; | ||
| 385 | + | ||
| 386 | + case LIB3270_NOT_CONNECTED: | ||
| 387 | + return errno = ENOTCONN; | ||
| 388 | + | ||
| 389 | + case LIB3270_CONNECTED_TN3270E: | ||
| 390 | + return 0; | ||
| 391 | + | ||
| 392 | + default: | ||
| 393 | + lib3270_write_log(hSession,"connect", "%s: State changed to unexpected state %d",__FUNCTION__,hSession->cstate); | ||
| 394 | + return -1; | ||
| 395 | + } | ||
| 396 | + | ||
| 397 | + } | ||
| 398 | + | ||
| 399 | + lib3270_disconnect(hSession); | ||
| 400 | + lib3270_write_log(hSession,"connect", "%s: %s",__FUNCTION__,strerror(ETIMEDOUT)); | ||
| 401 | + return errno = ETIMEDOUT; | ||
| 402 | + } | ||
| 403 | + | ||
| 404 | + return 0; | ||
| 405 | + | ||
| 406 | + } | ||
| 407 | + |
| @@ -0,0 +1,560 @@ | @@ -0,0 +1,560 @@ | ||
| 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 connect.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 <config.h> | ||
| 31 | + | ||
| 32 | +#if defined(_WIN32) | ||
| 33 | + | ||
| 34 | + // Compiling for WinXP or later: Expose getaddrinfo()/freeaddrinfo(). | ||
| 35 | + #undef _WIN32_WINNT | ||
| 36 | + #define _WIN32_WINNT 0x0501 | ||
| 37 | + | ||
| 38 | + #include <winsock2.h> | ||
| 39 | + #include <windows.h> | ||
| 40 | + #include <ws2tcpip.h> | ||
| 41 | + | ||
| 42 | +#endif | ||
| 43 | + | ||
| 44 | +#include "private.h" | ||
| 45 | +#include <errno.h> | ||
| 46 | + | ||
| 47 | +#if defined(_WIN32) | ||
| 48 | + #include <ws2tcpip.h> | ||
| 49 | +#else | ||
| 50 | + #include <sys/types.h> | ||
| 51 | + #include <sys/socket.h> | ||
| 52 | + #include <sys/ioctl.h> | ||
| 53 | + #include <netinet/in.h> | ||
| 54 | + #include <netdb.h> | ||
| 55 | + #include <unistd.h> | ||
| 56 | + #include <fcntl.h> | ||
| 57 | +#endif | ||
| 58 | + | ||
| 59 | +#ifdef HAVE_ICONV | ||
| 60 | + #include <iconv.h> | ||
| 61 | +#endif // HAVE_ICONV | ||
| 62 | + | ||
| 63 | +#if defined(_WIN32) /*[*/ | ||
| 64 | + #define SOCK_CLOSE(s) closesocket(s->sock); s->sock = -1; | ||
| 65 | +#else /*][*/ | ||
| 66 | + #define SOCK_CLOSE(s) close(s->sock); s->sock = -1; | ||
| 67 | +#endif /*]*/ | ||
| 68 | + | ||
| 69 | +#include <stdlib.h> | ||
| 70 | +#include "statusc.h" | ||
| 71 | +#include "hostc.h" | ||
| 72 | +#include "trace_dsc.h" | ||
| 73 | +#include "utilc.h" | ||
| 74 | +#include "telnetc.h" | ||
| 75 | +#include "screen.h" | ||
| 76 | +#include <lib3270/internals.h> | ||
| 77 | + | ||
| 78 | +/*---[ Implement ]-------------------------------------------------------------------------------*/ | ||
| 79 | + | ||
| 80 | + | ||
| 81 | +//static void net_connected(H3270 *hSession) | ||
| 82 | +static void net_connected(H3270 *hSession, int fd unused, LIB3270_IO_FLAG flag unused, void *dunno unused) | ||
| 83 | +{ | ||
| 84 | + int err; | ||
| 85 | + socklen_t len = sizeof(err); | ||
| 86 | + | ||
| 87 | + if(hSession->xio.write) { | ||
| 88 | + trace("%s write=%p",__FUNCTION__,hSession->xio.write); | ||
| 89 | + lib3270_remove_poll(hSession, hSession->xio.write); | ||
| 90 | + hSession->xio.write = NULL; | ||
| 91 | + } | ||
| 92 | + | ||
| 93 | + if(getsockopt(hSession->sock, SOL_SOCKET, SO_ERROR, (char *) &err, &len) < 0) | ||
| 94 | + { | ||
| 95 | + lib3270_disconnect(hSession); | ||
| 96 | + lib3270_popup_dialog( hSession, | ||
| 97 | + LIB3270_NOTIFY_ERROR, | ||
| 98 | + _( "Network error" ), | ||
| 99 | + _( "Unable to get connection state." ), | ||
| 100 | +#ifdef _WIN32 | ||
| 101 | + "%s", lib3270_win32_strerror(WSAGetLastError()) | ||
| 102 | +#else | ||
| 103 | + _( "%s" ), strerror(errno) | ||
| 104 | +#endif // _WIN32 | ||
| 105 | + ); | ||
| 106 | + return; | ||
| 107 | + } | ||
| 108 | + else if(err) | ||
| 109 | + { | ||
| 110 | + char buffer[4096]; | ||
| 111 | + snprintf(buffer,4095,_( "Can't connect to %s" ), hSession->host.current ); | ||
| 112 | + | ||
| 113 | + lib3270_disconnect(hSession); | ||
| 114 | + lib3270_popup_dialog( hSession, | ||
| 115 | + LIB3270_NOTIFY_ERROR, | ||
| 116 | + _( "Connection failed" ), | ||
| 117 | + buffer, | ||
| 118 | +#ifdef _WIN32 | ||
| 119 | + _( "%s"), lib3270_win32_strerror(err) | ||
| 120 | +#else | ||
| 121 | + _( "%s" ), strerror(err) | ||
| 122 | +#endif // _WIN32 | ||
| 123 | + ); | ||
| 124 | + trace("%s",__FUNCTION__); | ||
| 125 | + return; | ||
| 126 | + } | ||
| 127 | + | ||
| 128 | + hSession->xio.except = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_EXCEPTION,net_exception,0); | ||
| 129 | + hSession->xio.read = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_READ,net_input,0); | ||
| 130 | + | ||
| 131 | +#if defined(HAVE_LIBSSL) | ||
| 132 | + if(hSession->ssl.con && hSession->ssl.state == LIB3270_SSL_UNDEFINED) | ||
| 133 | + { | ||
| 134 | + if(ssl_negotiate(hSession)) | ||
| 135 | + return; | ||
| 136 | + } | ||
| 137 | +#endif | ||
| 138 | + | ||
| 139 | + lib3270_setup_session(hSession); | ||
| 140 | + lib3270_set_connected_initial(hSession); | ||
| 141 | + | ||
| 142 | +} | ||
| 143 | + | ||
| 144 | + | ||
| 145 | +#if defined(_WIN32) | ||
| 146 | + | ||
| 147 | + static void sockstart(H3270 *session) | ||
| 148 | + { | ||
| 149 | + static int initted = 0; | ||
| 150 | + WORD wVersionRequested; | ||
| 151 | + WSADATA wsaData; | ||
| 152 | + | ||
| 153 | + if (initted) | ||
| 154 | + return; | ||
| 155 | + | ||
| 156 | + initted = 1; | ||
| 157 | + | ||
| 158 | + wVersionRequested = MAKEWORD(2, 2); | ||
| 159 | + | ||
| 160 | + if (WSAStartup(wVersionRequested, &wsaData) != 0) | ||
| 161 | + { | ||
| 162 | + lib3270_popup_dialog( session, | ||
| 163 | + LIB3270_NOTIFY_CRITICAL, | ||
| 164 | + N_( "Network startup error" ), | ||
| 165 | + N_( "WSAStartup failed" ), | ||
| 166 | + "%s", lib3270_win32_strerror(GetLastError()) ); | ||
| 167 | + | ||
| 168 | + _exit(1); | ||
| 169 | + } | ||
| 170 | + | ||
| 171 | + if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 2) | ||
| 172 | + { | ||
| 173 | + lib3270_popup_dialog( session, | ||
| 174 | + LIB3270_NOTIFY_CRITICAL, | ||
| 175 | + N_( "Network startup error" ), | ||
| 176 | + N_( "Bad winsock version" ), | ||
| 177 | + N_( "Can't use winsock version %d.%d" ), LOBYTE(wsaData.wVersion), HIBYTE(wsaData.wVersion)); | ||
| 178 | + _exit(1); | ||
| 179 | + } | ||
| 180 | + } | ||
| 181 | +#endif // WIN32 | ||
| 182 | + | ||
| 183 | + LIB3270_EXPORT int lib3270_connect_url(H3270 *hSession, const char *url, int wait) | ||
| 184 | + { | ||
| 185 | + CHECK_SESSION_HANDLE(hSession); | ||
| 186 | + | ||
| 187 | + if(url && *url) | ||
| 188 | + { | ||
| 189 | + lib3270_set_url(hSession,url); | ||
| 190 | + } | ||
| 191 | + | ||
| 192 | + return lib3270_connect(hSession, wait); | ||
| 193 | + | ||
| 194 | + } | ||
| 195 | + | ||
| 196 | + LIB3270_EXPORT int lib3270_connect_host(H3270 *hSession, const char *hostname, const char *srvc, LIB3270_OPTION opt) | ||
| 197 | + { | ||
| 198 | + CHECK_SESSION_HANDLE(hSession); | ||
| 199 | + | ||
| 200 | + if(!hostname) | ||
| 201 | + return EINVAL; | ||
| 202 | + | ||
| 203 | + if(!srvc) | ||
| 204 | + srvc = "telnet"; | ||
| 205 | + | ||
| 206 | + if(*hostname == '$') | ||
| 207 | + { | ||
| 208 | + const char *name = getenv(hostname+1); | ||
| 209 | + if(!name) | ||
| 210 | + { | ||
| 211 | + lib3270_popup_dialog( hSession, | ||
| 212 | + LIB3270_NOTIFY_ERROR, | ||
| 213 | + _( "Connection error" ), | ||
| 214 | + _( "Unable to find selected hostname." ), | ||
| 215 | + _( "Can't determine value for environment variable \"%s\" " ), | ||
| 216 | + hostname); | ||
| 217 | + lib3270_set_disconnected(hSession); | ||
| 218 | + return ENOENT; | ||
| 219 | + } | ||
| 220 | + hostname = name; | ||
| 221 | + } | ||
| 222 | + | ||
| 223 | + hSession->options = opt & ~LIB3270_OPTION_WAIT; | ||
| 224 | + Replace(hSession->host.current,strdup(hostname)); | ||
| 225 | + Replace(hSession->host.srvc,strdup(srvc)); | ||
| 226 | + | ||
| 227 | + Replace(hSession->host.full, | ||
| 228 | + lib3270_strdup_printf( | ||
| 229 | + "%s%s:%s", | ||
| 230 | + opt&LIB3270_OPTION_SSL ? "tn3270s://" : "tn3270://", | ||
| 231 | + hostname, | ||
| 232 | + srvc )); | ||
| 233 | + | ||
| 234 | + trace("current_host=\"%s\"",hSession->host.current); | ||
| 235 | + | ||
| 236 | + return lib3270_connect(hSession,opt & LIB3270_OPTION_WAIT); | ||
| 237 | + | ||
| 238 | + } | ||
| 239 | + | ||
| 240 | + int lib3270_connect(H3270 *hSession, int seconds) | ||
| 241 | + { | ||
| 242 | + int s; | ||
| 243 | + int optval; | ||
| 244 | + struct addrinfo hints; | ||
| 245 | + struct addrinfo * result = NULL; | ||
| 246 | + struct addrinfo * rp = NULL; | ||
| 247 | + | ||
| 248 | + CHECK_SESSION_HANDLE(hSession); | ||
| 249 | + | ||
| 250 | + lib3270_main_iterate(hSession,0); | ||
| 251 | + | ||
| 252 | + if(hSession->auto_reconnect_inprogress) | ||
| 253 | + return errno = EAGAIN; | ||
| 254 | + | ||
| 255 | + if(hSession->sock > 0) | ||
| 256 | + return errno = EBUSY; | ||
| 257 | + | ||
| 258 | +#if defined(_WIN32) | ||
| 259 | + sockstart(hSession); | ||
| 260 | +#endif | ||
| 261 | + | ||
| 262 | +#if defined(HAVE_LIBSSL) | ||
| 263 | + set_ssl_state(hSession,LIB3270_SSL_UNSECURE); | ||
| 264 | +#endif // HAVE_LIBSSL | ||
| 265 | + | ||
| 266 | + snprintf(hSession->full_model_name,LIB3270_FULL_MODEL_NAME_LENGTH,"IBM-327%c-%d",hSession->m3279 ? '9' : '8', hSession->model_num); | ||
| 267 | + | ||
| 268 | + hSession->ever_3270 = False; | ||
| 269 | + | ||
| 270 | + memset(&hints, 0, sizeof(struct addrinfo)); | ||
| 271 | + hints.ai_family = AF_UNSPEC; /* Allow IPv4 or IPv6 */ | ||
| 272 | + hints.ai_socktype = SOCK_STREAM; /* Stream socket */ | ||
| 273 | + hints.ai_flags = AI_PASSIVE; /* For wildcard IP address */ | ||
| 274 | + hints.ai_protocol = 0; /* Any protocol */ | ||
| 275 | + hints.ai_canonname = NULL; | ||
| 276 | + hints.ai_addr = NULL; | ||
| 277 | + hints.ai_next = NULL; | ||
| 278 | + | ||
| 279 | + hSession->cstate = LIB3270_RESOLVING; | ||
| 280 | + lib3270_st_changed(hSession, LIB3270_STATE_RESOLVING, True); | ||
| 281 | + | ||
| 282 | + s = getaddrinfo(hSession->host.current, hSession->host.srvc, &hints, &result); | ||
| 283 | + | ||
| 284 | + if(s != 0) | ||
| 285 | + { | ||
| 286 | + char buffer[4096]; | ||
| 287 | + | ||
| 288 | + snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 289 | + | ||
| 290 | +#if defined(WIN32) && defined(HAVE_ICONV) | ||
| 291 | + { | ||
| 292 | + char tmpbuffer[4096]; | ||
| 293 | + const char * msg = gai_strerror(s); | ||
| 294 | + size_t in = strlen(msg); | ||
| 295 | + size_t out = 4096; | ||
| 296 | + char * ptr = tmpbuffer; | ||
| 297 | + | ||
| 298 | + iconv_t hConv = iconv_open(lib3270_win32_local_charset(),"UTF-8"); | ||
| 299 | + | ||
| 300 | + trace("Antes: [%s]",msg); | ||
| 301 | + if(iconv(hConv,&msg,&in,&ptr,&out) != ((size_t) -1)) | ||
| 302 | + msg = tmpbuffer; | ||
| 303 | + trace("Depois: [%s]",msg); | ||
| 304 | + | ||
| 305 | + iconv_close(hConv); | ||
| 306 | + | ||
| 307 | + lib3270_popup_dialog( hSession, | ||
| 308 | + LIB3270_NOTIFY_ERROR, | ||
| 309 | + _( "Connection error" ), | ||
| 310 | + buffer, | ||
| 311 | + "%s", | ||
| 312 | + msg); | ||
| 313 | + } | ||
| 314 | + | ||
| 315 | +#else | ||
| 316 | + lib3270_popup_dialog( hSession, | ||
| 317 | + LIB3270_NOTIFY_ERROR, | ||
| 318 | + _( "Connection error" ), | ||
| 319 | + buffer, | ||
| 320 | + "%s", | ||
| 321 | + gai_strerror(s)); | ||
| 322 | +#endif // WIN32 | ||
| 323 | + | ||
| 324 | + | ||
| 325 | + lib3270_set_disconnected(hSession); | ||
| 326 | + return errno = ENOENT; | ||
| 327 | + } | ||
| 328 | + | ||
| 329 | + | ||
| 330 | +#if !defined(_WIN32) | ||
| 331 | + /* don't share the socket with our children */ | ||
| 332 | + (void) fcntl(hSession->sock, F_SETFD, 1); | ||
| 333 | +#endif | ||
| 334 | + | ||
| 335 | + hSession->ever_3270 = False; | ||
| 336 | + hSession->ssl.host = 0; | ||
| 337 | + | ||
| 338 | + if(hSession->options&LIB3270_OPTION_SSL) | ||
| 339 | + { | ||
| 340 | +#if defined(HAVE_LIBSSL) | ||
| 341 | + hSession->ssl.host = 1; | ||
| 342 | + ssl_init(hSession); | ||
| 343 | +#else | ||
| 344 | + lib3270_popup_dialog( hSession, | ||
| 345 | + LIB3270_NOTIFY_ERROR, | ||
| 346 | + _( "SSL error" ), | ||
| 347 | + _( "Unable to connect to secure hosts" ), | ||
| 348 | + _( "This version of %s was built without support for secure sockets layer (SSL)." ), | ||
| 349 | + PACKAGE_NAME); | ||
| 350 | + | ||
| 351 | + return errno = EINVAL; | ||
| 352 | +#endif // HAVE_LIBSSL | ||
| 353 | + } | ||
| 354 | + | ||
| 355 | + /* connect */ | ||
| 356 | + status_connecting(hSession,1); | ||
| 357 | + | ||
| 358 | + for(rp = result; hSession->sock < 0 && rp != NULL; rp = rp->ai_next) | ||
| 359 | + { | ||
| 360 | + hSession->sock = socket(rp->ai_family, rp->ai_socktype, rp->ai_protocol); | ||
| 361 | + if(hSession->sock < 0) | ||
| 362 | + continue; | ||
| 363 | + | ||
| 364 | + trace("sock=%d",hSession->sock); | ||
| 365 | + | ||
| 366 | +#ifdef WIN32 | ||
| 367 | + | ||
| 368 | + WSASetLastError(0); | ||
| 369 | + u_long iMode=1; | ||
| 370 | + trace("sock=%d",hSession->sock); | ||
| 371 | + | ||
| 372 | + optval = lib3270_get_toggle(hSession,LIB3270_TOGGLE_KEEP_ALIVE) ? 1 : 0; | ||
| 373 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_KEEPALIVE, (char *)&optval, sizeof(optval)) < 0) | ||
| 374 | + { | ||
| 375 | + char buffer[4096]; | ||
| 376 | + snprintf(buffer,4095,N_( "Can't %s network keep-alive" ), optval ? _( "enable" ) : _( "disable" )); | ||
| 377 | + | ||
| 378 | + lib3270_popup_dialog( hSession, | ||
| 379 | + LIB3270_NOTIFY_ERROR, | ||
| 380 | + _( "Connection error" ), | ||
| 381 | + buffer, | ||
| 382 | + "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 383 | + SOCK_CLOSE(hSession); | ||
| 384 | + continue; | ||
| 385 | + } | ||
| 386 | + else | ||
| 387 | + { | ||
| 388 | + trace_dsn(hSession,"Network keep-alive is %s\n",optval ? "enabled" : "disabled" ); | ||
| 389 | + } | ||
| 390 | + | ||
| 391 | + if(ioctlsocket(hSession->sock,FIONBIO,&iMode)) | ||
| 392 | + { | ||
| 393 | + lib3270_popup_dialog( hSession, | ||
| 394 | + LIB3270_NOTIFY_ERROR, | ||
| 395 | + _( "Connection error" ), | ||
| 396 | + _( "ioctlsocket(FIONBIO) failed." ), | ||
| 397 | + "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 398 | + SOCK_CLOSE(hSession); | ||
| 399 | + continue; | ||
| 400 | + } | ||
| 401 | + else if(connect(hSession->sock, rp->ai_addr, rp->ai_addrlen)) | ||
| 402 | + { | ||
| 403 | + int err = WSAGetLastError(); | ||
| 404 | + if(err != WSAEWOULDBLOCK) | ||
| 405 | + { | ||
| 406 | + char buffer[4096]; | ||
| 407 | + snprintf(buffer,4095,_( "Can't connect to %s"), lib3270_get_host(hSession)); | ||
| 408 | + | ||
| 409 | + lib3270_popup_dialog( hSession, | ||
| 410 | + LIB3270_NOTIFY_ERROR, | ||
| 411 | + _( "Connection error" ), | ||
| 412 | + buffer, | ||
| 413 | + "%s", lib3270_win32_strerror(err)); | ||
| 414 | + SOCK_CLOSE(hSession); | ||
| 415 | + continue; | ||
| 416 | + | ||
| 417 | + } | ||
| 418 | + } | ||
| 419 | + | ||
| 420 | + optval = 1; | ||
| 421 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_OOBINLINE, (char *)&optval,sizeof(optval)) < 0) | ||
| 422 | + { | ||
| 423 | + lib3270_popup_dialog( hSession, | ||
| 424 | + LIB3270_NOTIFY_ERROR, | ||
| 425 | + _( "Connection error" ), | ||
| 426 | + _( "setsockopt(SO_OOBINLINE) has failed" ), | ||
| 427 | + "%s", lib3270_win32_strerror(WSAGetLastError())); | ||
| 428 | + SOCK_CLOSE(hSession); | ||
| 429 | + continue; | ||
| 430 | + } | ||
| 431 | + | ||
| 432 | +#else | ||
| 433 | + fcntl(hSession->sock, F_SETFL,fcntl(hSession->sock,F_GETFL,0)|O_NONBLOCK); | ||
| 434 | + | ||
| 435 | + errno = 0; | ||
| 436 | + if(connect(hSession->sock, rp->ai_addr, rp->ai_addrlen)) | ||
| 437 | + { | ||
| 438 | + if( errno != EINPROGRESS ) | ||
| 439 | + { | ||
| 440 | + char buffer[4096]; | ||
| 441 | + snprintf(buffer,4095,_( "Can't connect to %s:%s"), hSession->host.current, hSession->host.srvc); | ||
| 442 | + | ||
| 443 | + lib3270_popup_dialog( hSession, | ||
| 444 | + LIB3270_NOTIFY_ERROR, | ||
| 445 | + _( "Connection error" ), | ||
| 446 | + buffer, | ||
| 447 | + "%s", | ||
| 448 | + strerror(errno)); | ||
| 449 | + SOCK_CLOSE(hSession); | ||
| 450 | + continue; | ||
| 451 | + } | ||
| 452 | + } | ||
| 453 | + | ||
| 454 | + optval = 1; | ||
| 455 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_OOBINLINE, (char *)&optval,sizeof(optval)) < 0) | ||
| 456 | + { | ||
| 457 | + lib3270_popup_dialog( hSession, | ||
| 458 | + LIB3270_NOTIFY_ERROR, | ||
| 459 | + _( "Connection error" ), | ||
| 460 | + _( "setsockopt(SO_OOBINLINE) has failed" ), | ||
| 461 | + "%s", | ||
| 462 | + strerror(errno)); | ||
| 463 | + SOCK_CLOSE(hSession); | ||
| 464 | + continue; | ||
| 465 | + } | ||
| 466 | + | ||
| 467 | + optval = lib3270_get_toggle(hSession,LIB3270_TOGGLE_KEEP_ALIVE) ? 1 : 0; | ||
| 468 | + if (setsockopt(hSession->sock, SOL_SOCKET, SO_KEEPALIVE, (char *)&optval, sizeof(optval)) < 0) | ||
| 469 | + { | ||
| 470 | + char buffer[4096]; | ||
| 471 | + snprintf(buffer,4095,N_( "Can't %s network keep-alive" ), optval ? _( "enable" ) : _( "disable" )); | ||
| 472 | + | ||
| 473 | + lib3270_popup_dialog( hSession, | ||
| 474 | + LIB3270_NOTIFY_ERROR, | ||
| 475 | + _( "Connection error" ), | ||
| 476 | + buffer, | ||
| 477 | + "%s", | ||
| 478 | + strerror(errno)); | ||
| 479 | + SOCK_CLOSE(hSession); | ||
| 480 | + continue; | ||
| 481 | + } | ||
| 482 | + else | ||
| 483 | + { | ||
| 484 | + trace_dsn(hSession,"Network keep-alive is %s\n",optval ? "enabled" : "disabled" ); | ||
| 485 | + } | ||
| 486 | + | ||
| 487 | +#endif // WIN32 | ||
| 488 | + } | ||
| 489 | + | ||
| 490 | + freeaddrinfo(result); | ||
| 491 | + | ||
| 492 | + // set options for inline out-of-band data and keepalives | ||
| 493 | + | ||
| 494 | + /* | ||
| 495 | +#if defined(OMTU) | ||
| 496 | + else if (setsockopt(hSession->sock, SOL_SOCKET, SO_SNDBUF, (char *)&mtu,sizeof(mtu)) < 0) | ||
| 497 | + { | ||
| 498 | + popup_a_sockerr(hSession, N_( "setsockopt(%s)" ), "SO_SNDBUF"); | ||
| 499 | + SOCK_CLOSE(hSession); | ||
| 500 | + } | ||
| 501 | +#endif | ||
| 502 | + | ||
| 503 | + */ | ||
| 504 | + | ||
| 505 | + if(hSession->sock < 0) | ||
| 506 | + { | ||
| 507 | + lib3270_set_disconnected(hSession); | ||
| 508 | + return errno = ENOTCONN; | ||
| 509 | + } | ||
| 510 | + | ||
| 511 | + // Connecting, set callbacks, wait for connection | ||
| 512 | + hSession->cstate = LIB3270_PENDING; | ||
| 513 | + lib3270_st_changed(hSession, LIB3270_STATE_HALF_CONNECT, True); | ||
| 514 | + | ||
| 515 | + hSession->xio.write = lib3270_add_poll_fd(hSession,hSession->sock,LIB3270_IO_FLAG_WRITE,net_connected,0); | ||
| 516 | + // hSession->ns_write_id = AddOutput(hSession->sock, hSession, net_connected); | ||
| 517 | + | ||
| 518 | + trace("%s: Connection in progress",__FUNCTION__); | ||
| 519 | + | ||
| 520 | + if(seconds) | ||
| 521 | + { | ||
| 522 | + time_t end = time(0)+seconds; | ||
| 523 | + | ||
| 524 | + while(time(0) < end) | ||
| 525 | + { | ||
| 526 | + lib3270_main_iterate(hSession,1); | ||
| 527 | + | ||
| 528 | + switch(hSession->cstate) | ||
| 529 | + { | ||
| 530 | + case LIB3270_PENDING: | ||
| 531 | + case LIB3270_CONNECTED_INITIAL: | ||
| 532 | + case LIB3270_CONNECTED_ANSI: | ||
| 533 | + case LIB3270_CONNECTED_3270: | ||
| 534 | + case LIB3270_CONNECTED_INITIAL_E: | ||
| 535 | + case LIB3270_CONNECTED_NVT: | ||
| 536 | + case LIB3270_CONNECTED_SSCP: | ||
| 537 | + break; | ||
| 538 | + | ||
| 539 | + case LIB3270_NOT_CONNECTED: | ||
| 540 | + return errno = ENOTCONN; | ||
| 541 | + | ||
| 542 | + case LIB3270_CONNECTED_TN3270E: | ||
| 543 | + return 0; | ||
| 544 | + | ||
| 545 | + default: | ||
| 546 | + lib3270_write_log(hSession,"connect", "%s: State changed to unexpected state %d",__FUNCTION__,hSession->cstate); | ||
| 547 | + return -1; | ||
| 548 | + } | ||
| 549 | + | ||
| 550 | + } | ||
| 551 | + | ||
| 552 | + lib3270_disconnect(hSession); | ||
| 553 | + lib3270_write_log(hSession,"connect", "%s: %s",__FUNCTION__,strerror(ETIMEDOUT)); | ||
| 554 | + return errno = ETIMEDOUT; | ||
| 555 | + } | ||
| 556 | + | ||
| 557 | + return 0; | ||
| 558 | + | ||
| 559 | + } | ||
| 560 | + |