113 lines
2.7 KiB
C++
113 lines
2.7 KiB
C++
|
#include "decoderthread.h"
|
||
|
|
||
|
DecoderThread::DecoderThread(QObject *parent) :
|
||
|
QThread(parent)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
DecoderThread::~DecoderThread()
|
||
|
{
|
||
|
wait();
|
||
|
}
|
||
|
|
||
|
void DecoderThread::decode(QByteArray program, QByteArray input)
|
||
|
{
|
||
|
this->program = program;
|
||
|
this->input = input;
|
||
|
start();
|
||
|
}
|
||
|
|
||
|
void DecoderThread::run()
|
||
|
{
|
||
|
char* program = new char[this->program.size() + 1];
|
||
|
strcpy(program, this->program.data());
|
||
|
int programLength = this->program.size();
|
||
|
|
||
|
char* input = new char[this->input.size() + 1];
|
||
|
strcpy(input, this->input.data());
|
||
|
int inputLength = this->program.size();
|
||
|
|
||
|
actualDecode(program, programLength, input, inputLength);
|
||
|
|
||
|
delete program;
|
||
|
delete input;
|
||
|
}
|
||
|
|
||
|
void DecoderThread::actualDecode(char* program, int programLength, char* input, int inputLength)
|
||
|
{
|
||
|
// Don't use QList or QByteArray, it's too slow
|
||
|
char memory[65536];
|
||
|
int ptr = 0;
|
||
|
|
||
|
QString text;
|
||
|
QMap<int,int> correspondingBraces;
|
||
|
QStack<int> loopStack;
|
||
|
|
||
|
int programIndex = 0;
|
||
|
int inputIndex = 0;
|
||
|
|
||
|
for (programIndex = 0; programIndex < programLength; ++programIndex) {
|
||
|
switch (program[programIndex]) {
|
||
|
case '[':
|
||
|
loopStack.push(programIndex);
|
||
|
break;
|
||
|
case ']':
|
||
|
if(loopStack.isEmpty()) {
|
||
|
emit errorMessage("Unmatched end of loop");
|
||
|
return;
|
||
|
}
|
||
|
int old_i = loopStack.pop();
|
||
|
correspondingBraces.insert(programIndex, old_i);
|
||
|
correspondingBraces.insert(old_i, programIndex);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if(!loopStack.isEmpty()) {
|
||
|
emit errorMessage("Unmatched begin of loop");
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
for (programIndex = 0; programIndex < programLength; ++programIndex) {
|
||
|
switch (program[programIndex]) {
|
||
|
case '>':
|
||
|
++ptr;
|
||
|
break;
|
||
|
case '<':
|
||
|
--ptr;
|
||
|
break;
|
||
|
case '+':
|
||
|
++memory[ptr];
|
||
|
break;
|
||
|
case '-':
|
||
|
--memory[ptr];
|
||
|
break;
|
||
|
case '.':
|
||
|
text += (QString(memory[ptr]));
|
||
|
if (memory[ptr] == '\n') {
|
||
|
emit programDecoded(text);
|
||
|
}
|
||
|
break;
|
||
|
case ',':
|
||
|
if (inputIndex < inputLength) {
|
||
|
memory[ptr] = input[inputIndex];
|
||
|
++inputIndex;
|
||
|
} else {
|
||
|
memory[ptr] = 0;
|
||
|
}
|
||
|
break;
|
||
|
case '[':
|
||
|
if (memory[ptr] == 0) {
|
||
|
programIndex = correspondingBraces.value(programIndex);
|
||
|
}
|
||
|
break;
|
||
|
case ']':
|
||
|
if (memory[ptr] != 0) {
|
||
|
programIndex = correspondingBraces.value(programIndex);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
emit programDecoded(text);
|
||
|
}
|