Re: libpgtcl: pq_recvbuf: unexpected EOF on client connection - Mailing list pgsql-interfaces
| From | Tom Lane |
|---|---|
| Subject | Re: libpgtcl: pq_recvbuf: unexpected EOF on client connection |
| Date | |
| Msg-id | 28530.1031010328@sss.pgh.pa.us Whole thread Raw |
| In response to | Re: libpgtcl: pq_recvbuf: unexpected EOF on client connection (Gerhard Hintermayer <g.hintermayer@inode.at>) |
| List | pgsql-interfaces |
Gerhard Hintermayer <g.hintermayer@inode.at> writes:
> I don't understand what you're thinking of, because the unregistering
> triggers the call to PgDelConnectionId, so who will close the
> connection if you remove the unregister call ?
I meant the unregister of the notifier_channel, which is not
Pg_disconnect's business anyway. Unfortunately, it seems this fixes
your problem but creates a bigger one: Tcl dumps core if you shut down
the interpreter without having done pg_disconnect.
As a compromise I've applied the attached patch, which seems to suppress
the "unexpected EOF" problem at the cost of leaking a small amount of
memory for each disconnect.
regards, tom lane
*** src/interfaces/libpgtcl/pgtclCmds.c.orig Mon Sep 2 17:51:47 2002
--- src/interfaces/libpgtcl/pgtclCmds.c Mon Sep 2 18:03:35 2002
***************
*** 399,405 **** int Pg_disconnect(ClientData cData, Tcl_Interp *interp, int argc, char *argv[]) {
- Pg_ConnectionId *connid; PGconn *conn; Tcl_Channel conn_chan;
--- 399,404 ----
***************
*** 413,427 **** if (conn_chan == NULL) { Tcl_ResetResult(interp);
! Tcl_AppendResult(interp, argv[1], " is not a valid connection\n", 0); return TCL_ERROR; }
! #if TCL_MAJOR_VERSION >= 8
! conn = PgGetConnectionId(interp, argv[1], &connid);
! if (connid->notifier_channel != NULL)
! Tcl_UnregisterChannel(interp, connid->notifier_channel);
! #endif return Tcl_UnregisterChannel(interp, conn_chan); }
--- 412,425 ---- if (conn_chan == NULL) { Tcl_ResetResult(interp);
! Tcl_AppendResult(interp, argv[1], " is not a valid connection", 0); return TCL_ERROR; }
! /* Check that it is a PG connection and not something else */
! conn = PgGetConnectionId(interp, argv[1], (Pg_ConnectionId **) NULL);
! if (conn == (PGconn *) NULL)
! return TCL_ERROR; return Tcl_UnregisterChannel(interp, conn_chan); }
*** src/interfaces/libpgtcl/pgtclId.c.orig Mon Sep 2 17:51:47 2002
--- src/interfaces/libpgtcl/pgtclId.c Mon Sep 2 19:38:22 2002
***************
*** 193,199 **** #if TCL_MAJOR_VERSION >= 8 connid->notifier_channel = Tcl_MakeTcpClientChannel((ClientData)
PQsocket(conn));
! Tcl_RegisterChannel(interp, connid->notifier_channel); #else connid->notifier_socket = -1; #endif
--- 194,200 ---- #if TCL_MAJOR_VERSION >= 8 connid->notifier_channel = Tcl_MakeTcpClientChannel((ClientData)
PQsocket(conn));
! Tcl_RegisterChannel(NULL, connid->notifier_channel); #else connid->notifier_socket = -1; #endif
***************
*** 286,291 ****
--- 289,313 ---- connid->conn = NULL; /*
+ * Kill the notifier channel, too. We must not do this until after
+ * we've closed the libpq connection, because Tcl will try to close
+ * the socket itself!
+ *
+ * XXX Unfortunately, while this works fine if we are closing due to
+ * explicit pg_disconnect, Tcl versions through 8.3.3 dump core if we
+ * try to do it during interpreter shutdown. Not clear why, or if
+ * there is a workaround. For now, accept leakage of the (fairly
+ * small) amount of memory taken for the channel state representation.
+ * Note we are not leaking a socket, since libpq closed that already.
+ */
+ #ifdef NOT_USED
+ #if TCL_MAJOR_VERSION >= 8
+ if (connid->notifier_channel != NULL)
+ Tcl_UnregisterChannel(NULL, connid->notifier_channel);
+ #endif
+ #endif
+
+ /* * We must use Tcl_EventuallyFree because we don't want the connid * struct to vanish instantly if
Pg_Notify_EventProcis active for it. * (Otherwise, closing the connection from inside a pg_listen callback
pgsql-interfaces by date: