Added a circular buffer to the Qt Lua output console to allow for messages to be passed in a thread safe way from the emulation thread to the GUI thread. This fixes the crash issue mentioned in Issue #190.
This commit is contained in:
parent
da0ab1359c
commit
ce2afe23e1
|
@ -1,5 +1,7 @@
|
|||
// LuaControl.cpp
|
||||
//
|
||||
#include <stdio.h>
|
||||
#include <string.h>
|
||||
#include <list>
|
||||
|
||||
#include <QTextEdit>
|
||||
|
@ -20,9 +22,76 @@
|
|||
#include "Qt/ConsoleUtilities.h"
|
||||
|
||||
static bool luaScriptRunning = false;
|
||||
static bool updateLuaDisplay = false;
|
||||
|
||||
struct luaConsoleOutputLine
|
||||
{
|
||||
char text[256];
|
||||
|
||||
luaConsoleOutputLine(void)
|
||||
{
|
||||
memset( text, 0, sizeof(text) );
|
||||
}
|
||||
|
||||
void clear(void)
|
||||
{
|
||||
memset( text, 0, sizeof(text) );
|
||||
}
|
||||
|
||||
void setText( const char *txt )
|
||||
{
|
||||
strncpy( text, txt, sizeof(text)-1 );
|
||||
text[sizeof(text)-1] = 0;
|
||||
}
|
||||
};
|
||||
|
||||
struct luaConsoleOutputBuffer
|
||||
{
|
||||
int head;
|
||||
int tail;
|
||||
int size;
|
||||
struct luaConsoleOutputLine *line;
|
||||
|
||||
luaConsoleOutputBuffer(void)
|
||||
{
|
||||
tail = head = 0;
|
||||
size = 64;
|
||||
|
||||
line = new luaConsoleOutputLine[size];
|
||||
}
|
||||
|
||||
~luaConsoleOutputBuffer(void)
|
||||
{
|
||||
if ( line )
|
||||
{
|
||||
delete [] line; line = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
void addLine( const char *l )
|
||||
{
|
||||
//printf("Adding Line %i: '%s'\n", head, l );
|
||||
line[head].setText( l );
|
||||
|
||||
head = (head + 1) % size;
|
||||
|
||||
if ( head == tail )
|
||||
{
|
||||
tail = (tail + 1) % size;
|
||||
}
|
||||
}
|
||||
|
||||
void clear(void)
|
||||
{
|
||||
tail = head = 0;
|
||||
}
|
||||
};
|
||||
|
||||
static luaConsoleOutputBuffer outBuf;
|
||||
|
||||
static std::string luaOutputText;
|
||||
static std::list <LuaControlDialog_t*> winList;
|
||||
|
||||
static void updateLuaWindows( void );
|
||||
//----------------------------------------------------
|
||||
LuaControlDialog_t::LuaControlDialog_t(QWidget *parent)
|
||||
: QDialog( parent )
|
||||
|
@ -95,6 +164,12 @@ LuaControlDialog_t::LuaControlDialog_t(QWidget *parent)
|
|||
setLayout( mainLayout );
|
||||
|
||||
winList.push_back( this );
|
||||
|
||||
periodicTimer = new QTimer( this );
|
||||
|
||||
connect( periodicTimer, &QTimer::timeout, this, &LuaControlDialog_t::updatePeriodic );
|
||||
|
||||
periodicTimer->start( 200 ); // 5hz
|
||||
}
|
||||
|
||||
//----------------------------------------------------
|
||||
|
@ -104,6 +179,8 @@ LuaControlDialog_t::~LuaControlDialog_t(void)
|
|||
|
||||
printf("Destroy Lua Control Window\n");
|
||||
|
||||
periodicTimer->stop();
|
||||
|
||||
for (it = winList.begin(); it != winList.end(); it++)
|
||||
{
|
||||
if ( (*it) == this )
|
||||
|
@ -130,6 +207,16 @@ void LuaControlDialog_t::closeWindow(void)
|
|||
deleteLater();
|
||||
}
|
||||
//----------------------------------------------------
|
||||
void LuaControlDialog_t::updatePeriodic(void)
|
||||
{
|
||||
//printf("Update Lua\n");
|
||||
if ( updateLuaDisplay )
|
||||
{
|
||||
updateLuaWindows();
|
||||
updateLuaDisplay = false;
|
||||
}
|
||||
}
|
||||
//----------------------------------------------------
|
||||
void LuaControlDialog_t::openLuaScriptFile(void)
|
||||
{
|
||||
#ifdef _S9XLUA_H
|
||||
|
@ -193,7 +280,7 @@ void LuaControlDialog_t::openLuaScriptFile(void)
|
|||
void LuaControlDialog_t::startLuaScript(void)
|
||||
{
|
||||
#ifdef _S9XLUA_H
|
||||
luaOutputText.clear();
|
||||
outBuf.clear();
|
||||
fceuWrapperLock();
|
||||
if ( 0 == FCEU_LoadLuaCode( scriptPath->text().toStdString().c_str(), scriptArgs->text().toStdString().c_str() ) )
|
||||
{
|
||||
|
@ -214,6 +301,9 @@ void LuaControlDialog_t::stopLuaScript(void)
|
|||
//----------------------------------------------------
|
||||
void LuaControlDialog_t::refreshState(void)
|
||||
{
|
||||
int i;
|
||||
std::string luaOutputText;
|
||||
|
||||
if ( luaScriptRunning )
|
||||
{
|
||||
stopButton->setEnabled( true );
|
||||
|
@ -224,10 +314,22 @@ void LuaControlDialog_t::refreshState(void)
|
|||
stopButton->setEnabled( false );
|
||||
startButton->setText( tr("Start") );
|
||||
}
|
||||
|
||||
i = outBuf.tail;
|
||||
|
||||
while ( i != outBuf.head )
|
||||
{
|
||||
luaOutputText.append( outBuf.line[i].text );
|
||||
|
||||
i = (i + 1) % outBuf.size;
|
||||
}
|
||||
|
||||
luaOutput->setText( luaOutputText.c_str() );
|
||||
|
||||
luaOutput->moveCursor( QTextCursor::End );
|
||||
}
|
||||
//----------------------------------------------------
|
||||
void updateLuaWindows( void )
|
||||
static void updateLuaWindows( void )
|
||||
{
|
||||
std::list <LuaControlDialog_t*>::iterator it;
|
||||
|
||||
|
@ -259,8 +361,8 @@ void PrintToWindowConsole(intptr_t hDlgAsInt, const char* str)
|
|||
{
|
||||
//printf("%s\n", str );
|
||||
|
||||
luaOutputText.append( str );
|
||||
outBuf.addLine( str );
|
||||
|
||||
updateLuaWindows();
|
||||
updateLuaDisplay = true;
|
||||
}
|
||||
//----------------------------------------------------
|
||||
|
|
|
@ -31,6 +31,7 @@ class LuaControlDialog_t : public QDialog
|
|||
protected:
|
||||
void closeEvent(QCloseEvent *bar);
|
||||
|
||||
QTimer *periodicTimer;
|
||||
QLineEdit *scriptPath;
|
||||
QLineEdit *scriptArgs;
|
||||
QPushButton *browseButton;
|
||||
|
@ -42,6 +43,7 @@ class LuaControlDialog_t : public QDialog
|
|||
public slots:
|
||||
void closeWindow(void);
|
||||
private slots:
|
||||
void updatePeriodic(void);
|
||||
void openLuaScriptFile(void);
|
||||
void startLuaScript(void);
|
||||
void stopLuaScript(void);
|
||||
|
|
Loading…
Reference in New Issue