logo       

[GHC] #702: MingW ld.exe produces program which segfaults: msg#00068

lang.haskell.glasgow.bugs

Subject: [GHC] #702: MingW ld.exe produces program which segfaults

#702: MingW ld.exe produces program which segfaults
-------------------------------------+--------------------------------------
Reporter: alistair@xxxxxxxxxxx | Owner:
Type: bug | Status: new
Priority: normal | Milestone:
Component: Compiler | Version: 6.4.1
Severity: normal | Keywords:
Os: Windows | Difficulty: Unknown
Architecture: x86 |
-------------------------------------+--------------------------------------
(from Alistair Bayley - alistair@xxxxxxxxxxx)

The C program below works correctly when compiled with GHC and ld 2.13.90,
but segfaults on the first call to PQprepare when compiled with ld 2.15.91
(which ships with GHC 6.4.1) and also with ld 2.16.91. For now I have
replaced the ld in C:\ghc\ghc-6.4.1\gcc-lib with ld-2.13.90 (from my MingW
installation), but Sigbjorn Finn says that the more recent versions of ld
are necessary for large GHCi libraries, so we can't just go back.

One possibly interesting datum (or maybe just a red herring) is this
linker message emitted by 2.15.91 and 2.16.91, but not 2.13.90:

{{{
Info: resolving _PQprepare by linking to __imp__PQprepare (auto-import)
}}}

You'll need a full Postgres installation to reproduce this in its current
state, unfortunately. The commands I use to run it are (assuming the
default postgres database has been created, with user postgres, on
localhost):

{{{
ghc -o test.exe test.c "-LC:\Program Files\PostgreSQL\8.1\bin" -lpq
"-IC:\Program Files\PostgreSQL\8.1\include"
test.exe user=postgres
}}}


----

{{{
#!c
File: test.c

#include <stdio.h>
#include <stdlib.h>
#include "libpq-fe.h"

static void exit_nicely(PGconn *conn)
{
PQfinish(conn);
exit(1);
}

void check_error(PGconn *conn, PGresult *res, ExecStatusType rc, char
*msg)
{
if (PQresultStatus(res) != rc)
{
/* fprintf(stderr, msg, PQerrorMessage(conn)); */
fprintf(stderr, "%s: %s\n", msg, PQerrorMessage(conn));
PQclear(res);
exit_nicely(conn);
}
}

int main(int argc, char **argv)
{
const char *conninfo;
PGconn *conn;
PGresult *res;
int nFields;
int i,
j;
Oid paramTypes[10];

/*
* If the user supplies a parameter on the command line, use it as the
* conninfo string; otherwise default to setting dbname=postgres and
using
* environment variables or defaults for all other connection
parameters.
*/
if (argc > 1)
conninfo = argv[1];
else
conninfo = "dbname = postgres";

/* Make a connection to the database */
conn = PQconnectdb(conninfo);

/* Check to see that the backend connection was successfully made */
if (PQstatus(conn) != CONNECTION_OK)
{
fprintf(stderr, "Connection to database failed: %s",
PQerrorMessage(conn));
exit_nicely(conn);
}

res = PQprepare(conn, "x", "DECLARE myportal CURSOR FOR select * from
pg_database", 0, paramTypes);
check_error(conn, res, PGRES_COMMAND_OK, "Prepare failed");

/*
* Our test case here involves using a cursor, for which we must be
inside
* a transaction block. We could do the whole thing with a single
* PQexec() of "select * from pg_database", but that's too trivial to
make
* a good example.
*/

/* Start a transaction block */
res = PQexec(conn, "BEGIN");
check_error(conn, res, PGRES_COMMAND_OK, "BEGIN command failed");

/*
* Should PQclear PGresult whenever it is no longer needed to avoid
memory
* leaks
*/
PQclear(res);

/*
* Fetch rows from pg_database, the system catalog of databases
*/
res = PQexec(conn, "DECLARE myportal CURSOR FOR select * from
pg_database");
check_error(conn, res, PGRES_COMMAND_OK, "DECLARE CURSOR failed");
PQclear(res);

res = PQexec(conn, "FETCH ALL in myportal");
check_error(conn, res, PGRES_TUPLES_OK, "FETCH ALL failed");

/* first, print out the attribute names */
nFields = PQnfields(res);
for (i = 0; i < nFields; i++)
printf("%-15s", PQfname(res, i));
printf("\n\n");

/* next, print out the rows */
for (i = 0; i < PQntuples(res); i++)
{
for (j = 0; j < nFields; j++)
printf("%-15s", PQgetvalue(res, i, j));
printf("\n");
}

PQclear(res);

/* close the portal ... we don't bother to check for errors ... */
res = PQexec(conn, "CLOSE myportal");
PQclear(res);

/* end the transaction */
res = PQexec(conn, "END");
PQclear(res);

/* close the connection to the database and cleanup */
PQfinish(conn);

return 0;
}
}}}

--
Ticket URL: <http://hackage.haskell.org/trac/ghc/ticket/702>
GHC <http://www.haskell.org/ghc/>
The Glasgow Haskell Compiler_______________________________________________
Glasgow-haskell-bugs mailing list
Glasgow-haskell-bugs@xxxxxxxxxxx
http://www.haskell.org/mailman/listinfo/glasgow-haskell-bugs
<Prev in Thread] Current Thread [Next in Thread>
Google Custom Search

News | FAQ | advertise