Re: pgin.tcl pg_exec_prepared slow (was: Released...) - Mailing list pgsql-interfaces

From L J Bayuk
Subject Re: pgin.tcl pg_exec_prepared slow (was: Released...)
Date
Msg-id 20040706010731.GA15183@bxlbisnugqvi.mindspring.com
Whole thread Raw
In response to Re: Released updated Tcl interfaces: pgin.tcl-2.1.0,  ("Nigel J. Andrews" <nandrews@investsystems.co.uk>)
Responses Re: pgin.tcl pg_exec_prepared slow (was: Released...)
Re: pgin.tcl pg_exec_prepared slow (was: Released...)
Re: pgin.tcl pg_exec_prepared slow (was: Released...)
List pgsql-interfaces
On Wed, Jun 30, 2004 at 11:29:32PM +0100, Nigel J. Andrews wrote:
> ... (description of prepared query execution in pgin.tcl taking 10 times
>      longer than pg_exec doing the same query)
> Obviously the prepared query execution is an order of magnitude slower and
> it's the TCL interface that's making it so.

My own tests show equivalent performance of pg_exec and pg_exec_prepared,
but that is for medium-to-large result sets. With tiny result sets - one
row, 2 values - I also see an unexpected 40 ms delay in pg_exec_prepared
when using pgin.tcl.  (Very interesting that it's the same delay on different
hardware.)

After looking at it for a while, I know where the problem is, but not
what is causing it. I also know how to fix it, but not why the fix works.
I'm going to explain what I know, hoping someone can fill in some missing
parts.

pg_exec sends 1 message: Query.  pg_exec_prepared sends 4 messages: Bind,
DescribePortal, Execute, and Sync. After that, they are identical.

The delay we are seeing (about 40 ms) is from the time the client sends the
Bind message to PostgreSQL until it gets a TCP ACK back. (It should get
a BindComplete message back, but the backend does't flush after sending
this - no pq_flush() at the end of exec_bind_message(). Hello, Tom, is this
a bug? Changing it just moves the delay around, though.) Other messages
from the client are typically replied to and/or ACK'd in under 1 ms,
according to Ethereal.  I don't know why this message takes so much longer
to ACK, nor why the client waits for the ACK. I'm sure it isn't the
backend; I don't think it is Tcl; it could be something in the TCP stack.

The fix: pgin.tcl sets its Tcl connection socket channel to "unbuffered",
which means each PostgreSQL message it sends will be go into the TCP
buffers immediately, since each message is written in a single "puts".  And
it seems to work fine with all messages except that Bind. My fix is to
change it to use Tcl "buffering full" mode; then I have it flush after each
sent message, except in pg_exec_prepared (and pg_exec_params), where it
flushes only after sending the final Sync.  The result is to combine the 4
messages from pg_exec_prepared so they get dumped into the TCP buffers at
once. The problem goes away, but I don't know why it works, and that
bothers me. Before:   39987 microseconds per iteration - pg_exec_prepared   39992 microseconds per iteration -
pg_exec_params   4133 microseconds per iteration - pg_exec After:    4483 microseconds per iteration - pg_exec_prepared
  5214 microseconds per iteration - pg_exec_params    4150 microseconds per iteration - pg_exec
 


pgsql-interfaces by date:

Previous
From: "Greg Sabino Mullane"
Date:
Subject: Re: DBD::Pg and $sth->{ParamValues}
Next
From: Tom Lane
Date:
Subject: Re: pgin.tcl pg_exec_prepared slow (was: Released...)