Будет ли загрузка DLL динамически согласовывать ее stderr с основным приложением?Если так, то как ...? - PullRequest
2 голосов
/ 08 июля 2010

Я пишу приложение с графическим интерфейсом, использующее Qt, которое ссылается на стороннюю DLL, которая иногда отправляет сообщения об ошибках в stderr. Я хотел бы, чтобы эти сообщения об ошибках отображались в окне моего графического интерфейса.

Я не смог найти установленный способ перенаправления stderr (в отличие от std :: cerr) даже после долгих поисков, поэтому я сам написал следующий класс:

class StdErrRedirect : public QObject
{
    Q_OBJECT

public:
    // Constructor
    StdErrRedirect(QTextEdit *errorLog,
                   QObject   *parent = NULL);

    // Destructor
    ~StdErrRedirect();

private slots:
    void fileChanged(const QString &filename);

private:    
    QFile              tmp;
    QFileSystemWatcher watcher;
    QString            tmpFileNameQtFormat;
    QString            tmpFileNameNativeFormat;

    QTextEdit *m_errorLog;
    QString   oldContent;
};

StdErrRedirect::StdErrRedirect(QTextEdit *errorLog,
                               QObject   *parent)
    : QObject(parent)
{
    // Store the pointer to the error log window
    m_errorLog = errorLog;

    // Create a temporary filename: first find the path:
    tmpFileNameQtFormat = QDir::tempPath();

    // Make sure the closing slash is present:
    if (!tmpFileNameQtFormat.endsWith(QChar('/')))
        tmpFileNameQtFormat.append(QChar('/'));

    // Add the file name itself:
    tmpFileNameQtFormat.append("nb_stderrlog");

    // Obtain a version of the filename in the operating system's native format:
    tmpFileNameNativeFormat = QDir::toNativeSeparators(tmpFileNameQtFormat);

    // Set up redirection to this file:
    freopen(tmpFileNameNativeFormat.toAscii().constData(), "a+", stderr);

    // Initialise the QFileSystemWatcher:
    connect(&watcher, SIGNAL(fileChanged(const QString &)),
            this,     SLOT(fileChanged(const QString &)));
    watcher.addPath(tmpFileNameQtFormat);

    tmp.setFileName(tmpFileNameQtFormat);
}

StdErrRedirect::~StdErrRedirect()
{
    // Ensure the temporary file is properly deleted:
    fclose(stderr);
    tmp.close();
    tmp.open(QIODevice::ReadWrite);
    tmp.remove();
}

void StdErrRedirect::fileChanged(const QString &filename)
{
    tmp.open(QIODevice::ReadOnly);
    QTextStream stream(&tmp);
    QString content = stream.readAll();
    tmp.close();

    // Identify what's new, and just send this to the window:
    int newchars = content.size() - oldContent.size();
    if (newchars)
    {
        m_errorLog -> append(content.right(newchars));
        oldContent = content;
    }
}

Если я создаю это в главном окне, используя:

errorLog = new QTextEdit;
redirector = new StdErrRedirect(errorLog);

... тогда все, что я пишу в stderr, появляется в окне.

Пока все хорошо. Проблема заключается в том, что выход DLL все еще отсутствует. При вызове функции DLL, которая выдает ошибку, если я поставил код:

if (error != _OK)
{
    error.PrintErrorTrace();
    fprintf(stderr, "Should have printed an error \r\n");
    fflush(stderr);
    //fsync(_fileno(stderr));    Linux version
    _commit(_fileno(stderr));
    return;
}

... тогда появляется текст «Должен быть напечатан текст ошибки», а само сообщение об ошибке - нет.

Теперь я где-то читал, что это, вероятно, потому, что перенаправление устанавливается после , DLL была загружена в начале приложения, и поэтому ее собственный канал stderr не затронут. Следовательно, я должен быть в состоянии исправить это, динамически загружая DLL, после вместо установки перенаправления.

Вот мой вопрос: как мне это сделать? Я могу попробовать поместить следующий код в начале моего приложения:

QLibrary extlib;
extlib.setFileName("libname");
extlib.setLoadHints(QLibrary::ResolveAllSymbolsHint);
extlib.load();

... но само по себе это не имеет никакого эффекта. Я думаю это потому, что компоновщик все еще настраивает библиотеку на автоматическое открытие. Однако если я удалю DLL из компоновщика (я использую VS2008, поэтому я удаляю extlib.lib из списка зависимостей), то приложение не будет компилироваться, потому что компилятор не может найти символы из DLL.

Так что, очевидно, что-то глубоко не так с тем, что я пытаюсь сделать здесь. Кто-нибудь может помочь?

Спасибо, Стивен.

Ответы [ 2 ]

2 голосов
/ 08 июля 2010

Действительно ли DLL записывает в stderr?Или это пишет в GetStdHandle(STD_ERROR_HANDLE)?Первый отображается на второй, изначально.Но с freopen() вы просто меняете отображение.Все, что записано в STD_ERROR_HANDLE, все равно будет там.

Чтобы перенаправить вывод всех ошибок, вам потребуется SetStdHandle.

0 голосов
/ 09 июля 2010

Существует только один stderr, поэтому я предполагаю, что проблема не в том, что вам нужно динамически загружать dll, а где-то в вашем коде перенаправления .Ваш код перенаправления написан в стиле Linux, где в Windows все работает по-другому.

, если бы вы могли протестировать свое приложение в Linux, это помогло бы определить проблему.Если он работает в Linux, то это просто код перенаправления.

В любом случае, вам следует прочитать еще немного о перенаправлении и окнах , так как я не думаю, что вы пытаетесьсделать сейчас поможет вам.

...