Qt5.2.0; Debian Wheezy: QSqlDatabase destructor causes segfault

1.3k Views Asked by At

I have an app that utilizes a wrapper class to access a MySQL database. Since the database is accessed at various places in the app, and the app is multithreaded, the wrapper class is used to minimize the amount of repetitive code needed to access the DB.

I read somewhere that a QSqlDatabase must be constructed, used and then destructed within the same thread. Undocumented/unsupported behavior may occur if you create the connection in one thread, then destruct it in another. To prevent this, the wrapper creates the connection just before setting up the QSqlQuery, and then closes and removes that connection when the query is done. Setting up a pool of connections and keeping them alive throughout app execution isn't really feasible because there are any number of threads running at any given time, each of which may need to do multiple DB tasks asynchrously, so there isn't exactly a 1-1 correlation between threads and database connections to begin with.

The problem is that after the app has run for a while, it crashes with a segfault which blames the QSqlDatabase destructor. Odd thing is that the error only shows up after the app has been running for a while and has done numerous queries. (Sometimes it crashes after 15 minutes, other times after a few hours). Most recent stack trace:

Program received signal SIGSEGV, Segmentation fault.
pthread_cond_timedwait@@GLIBC_2.3.2 ()
    at ../nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_timedwait.S:174
174     ../nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_timedwait.S: No such file or     directory.
(gdb) where
#0  pthread_cond_timedwait@@GLIBC_2.3.2 ()
    at ../nptl/sysdeps/unix/sysv/linux/x86_64/pthread_cond_timedwait.S:174
#1  0x00007ffff32f55b4 in my_thread_global_end ()
   from /opt/Qt/5.2.0/gcc_64/plugins/sqldrivers/libqsqlmysql.so
#2  0x00007ffff32f40a5 in my_end ()
   from /opt/Qt/5.2.0/gcc_64/plugins/sqldrivers/libqsqlmysql.so
#3  0x00007ffff32db5c7 in mysql_server_end ()
   from /opt/Qt/5.2.0/gcc_64/plugins/sqldrivers/libqsqlmysql.so
#4  0x00007ffff32cd806 in ?? ()
   from /opt/Qt/5.2.0/gcc_64/plugins/sqldrivers/libqsqlmysql.so
#5  0x00007ffff32cd829 in ?? ()
   from /opt/Qt/5.2.0/gcc_64/plugins/sqldrivers/libqsqlmysql.so
#6  0x00007ffff7657537 in ?? () from /opt/Qt/5.2.0/gcc_64/lib/libQt5Sql.so.5
#7  0x00007ffff7657e7d in QSqlDatabase::~QSqlDatabase() ()
   from /opt/Qt/5.2.0/gcc_64/lib/libQt5Sql.so.5
#8  0x00007ffff7659204 in ?? () from /opt/Qt/5.2.0/gcc_64/lib/libQt5Sql.so.5
#9  0x00000000004143c2 in DB::~DB (this=0x7ffff0829200,
    __in_chrg=<optimized out>) at db.cpp:32
#10 0x000000000042e758 in WeatherOutcome::MassInsert (
    this=0x7ffff0829550, inputVector=...) at weatheroutcome.cpp:96

Here's the wrapper class header file:

#ifndef DB_H
#define DB_H
#include <QString>
#include <QtSql> 
#include <QSqlQuery>
#include <QSqlError>
#include <QUuid>
class DB
{
public:
    DB();
    ~DB();
    bool SetQuery(QString query);
    bool Exec();
    void manualConnect();
protected:
    QSqlQuery query;
    QSqlDatabase* db;
};

#endif // DB_H

Wrapper class CPP File:

#include db.h
DB::DB()
{

}
DB::~DB()
{
    if (query.isActive())
    {
        query.finish();
        query.clear();
    }
    if (db != NULL)
    {
        QString connName = db->connectionName();
        db->close();
        delete db;
        db = NULL;
        try
        {
            QSqlDatabase::removeDatabase(connName);
        }


        catch(...)
        {

        }
    }
}

void DB::manualConnect()
{
    QUuid uniqueId = QUuid::createUuid();
    QString connectionID = uniqueId.toString();
    QSqlDatabase::addDatabase("QMYSQL", connectionID);

    db = new QSqlDatabase(QSqlDatabase::database(connectionID, false));
    db->setHostName("127.0.0.1");
    db->setDatabaseName("my_db_name");
    db->setUserName("username");
    db->setPassword("password");

    db->open();
}

bool DB::Exec()
{
    return query.exec()  ;
}

bool DB::SetQuery(QString queryString)
{
    manualConnect();
    if (db)
    {
        if (db->isOpen())
        {
            query = QSqlQuery(*db);
            return query.prepare(queryString);
        }
        return false;
    }
    else
        return false;
}

Why would this rather simple destructor cause a segfault and crash the app?

2

There are 2 best solutions below

2
On

it seems like you do not have correct way to create and destroy a QDatabase

create way should like

QUuid uniqueId = QUuid::createUuid();
db = new QSqlDatabase();
*db = QSqlDatabase::addDatabase("QMYSQL", uniqueId.toString());
db->setHostName(...);
db->setDatabaseName(...);
db->setUserName(...);
db->setPassword(...);

and destroy like

db->close();
delete db;
QSqlDatabase::removeDatabase(db->connectionName());
1
On

Have you tried using deleteLater() instead of calling delete db;.

http://qt-project.org/doc/qt-5.0/qtcore/qobject.html#deleteLater

Also you may want to try is to disconnect the db variable, so that some pending transaction doesn't try to do something to the db after its gone.

http://qt-project.org/doc/qt-5.0/qtcore/qobject.html#disconnect

http://qt-project.org/doc/qt-5.0/qtcore/objecttrees.html

Hope that helps.