793 lines
19 KiB
C++
793 lines
19 KiB
C++
/*-------------------------------------------------------------------------
|
|
* drawElements Quality Program Execution Server
|
|
* ---------------------------------------------
|
|
*
|
|
* Copyright 2014 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
*//*!
|
|
* \file
|
|
* \brief TestProcess implementation for Win32.
|
|
*//*--------------------------------------------------------------------*/
|
|
|
|
#include "xsWin32TestProcess.hpp"
|
|
#include "deFilePath.hpp"
|
|
#include "deString.h"
|
|
#include "deMemory.h"
|
|
#include "deClock.h"
|
|
#include "deFile.h"
|
|
|
|
#include <sstream>
|
|
#include <string.h>
|
|
|
|
using std::string;
|
|
using std::vector;
|
|
|
|
namespace xs
|
|
{
|
|
|
|
enum
|
|
{
|
|
MAX_OLD_LOGFILE_DELETE_ATTEMPTS = 20, //!< How many times execserver tries to delete old log file
|
|
LOGFILE_DELETE_SLEEP_MS = 50 //!< Sleep time (in ms) between log file delete attempts
|
|
};
|
|
|
|
namespace win32
|
|
{
|
|
|
|
// Error
|
|
|
|
static std::string formatErrMsg (DWORD error, const char* msg)
|
|
{
|
|
std::ostringstream str;
|
|
LPSTR msgBuf;
|
|
|
|
#if defined(UNICODE)
|
|
# error Unicode not supported.
|
|
#endif
|
|
|
|
if (FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER|FORMAT_MESSAGE_FROM_SYSTEM|FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
NULL, error, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPSTR)&msgBuf, 0, DE_NULL) > 0)
|
|
str << msg << ", error " << error << ": " << msgBuf;
|
|
else
|
|
str << msg << ", error " << error;
|
|
|
|
return str.str();
|
|
}
|
|
|
|
Error::Error (DWORD error, const char* msg)
|
|
: std::runtime_error(formatErrMsg(error, msg))
|
|
, m_error (error)
|
|
{
|
|
}
|
|
|
|
// Event
|
|
|
|
Event::Event (bool manualReset, bool initialState)
|
|
: m_handle(0)
|
|
{
|
|
m_handle = CreateEvent(NULL, manualReset ? TRUE : FALSE, initialState ? TRUE : FALSE, NULL);
|
|
if (!m_handle)
|
|
throw Error(GetLastError(), "CreateEvent() failed");
|
|
}
|
|
|
|
Event::~Event (void)
|
|
{
|
|
CloseHandle(m_handle);
|
|
}
|
|
|
|
void Event::setSignaled (void)
|
|
{
|
|
if (!SetEvent(m_handle))
|
|
throw Error(GetLastError(), "SetEvent() failed");
|
|
}
|
|
|
|
void Event::reset (void)
|
|
{
|
|
if (!ResetEvent(m_handle))
|
|
throw Error(GetLastError(), "ResetEvent() failed");
|
|
}
|
|
|
|
// CaseListWriter
|
|
|
|
CaseListWriter::CaseListWriter (void)
|
|
: m_dst (INVALID_HANDLE_VALUE)
|
|
, m_cancelEvent (true, false)
|
|
{
|
|
}
|
|
|
|
CaseListWriter::~CaseListWriter (void)
|
|
{
|
|
}
|
|
|
|
void CaseListWriter::start (const char* caseList, HANDLE dst)
|
|
{
|
|
DE_ASSERT(!isStarted());
|
|
|
|
m_dst = dst;
|
|
|
|
int caseListSize = (int)strlen(caseList)+1;
|
|
m_caseList.resize(caseListSize);
|
|
std::copy(caseList, caseList+caseListSize, m_caseList.begin());
|
|
|
|
de::Thread::start();
|
|
}
|
|
|
|
void CaseListWriter::run (void)
|
|
{
|
|
try
|
|
{
|
|
Event ioEvent (true, false); // Manual reset, non-signaled state.
|
|
HANDLE waitHandles[] = { ioEvent.getHandle(), m_cancelEvent.getHandle() };
|
|
OVERLAPPED overlapped;
|
|
int curPos = 0;
|
|
|
|
deMemset(&overlapped, 0, sizeof(overlapped));
|
|
overlapped.hEvent = ioEvent.getHandle();
|
|
|
|
while (curPos < (int)m_caseList.size())
|
|
{
|
|
const int maxWriteSize = 4096;
|
|
const int numToWrite = de::min(maxWriteSize, (int)m_caseList.size() - curPos);
|
|
DWORD waitRes = 0;
|
|
|
|
if (!WriteFile(m_dst, &m_caseList[curPos], (DWORD)numToWrite, NULL, &overlapped))
|
|
{
|
|
DWORD err = GetLastError();
|
|
if (err != ERROR_IO_PENDING)
|
|
throw Error(err, "WriteFile() failed");
|
|
}
|
|
|
|
waitRes = WaitForMultipleObjects(DE_LENGTH_OF_ARRAY(waitHandles), &waitHandles[0], FALSE, INFINITE);
|
|
|
|
if (waitRes == WAIT_OBJECT_0)
|
|
{
|
|
DWORD numBytesWritten = 0;
|
|
|
|
// \note GetOverlappedResult() will fail with ERROR_IO_INCOMPLETE if IO event is not complete (should be).
|
|
if (!GetOverlappedResult(m_dst, &overlapped, &numBytesWritten, FALSE))
|
|
throw Error(GetLastError(), "GetOverlappedResult() failed");
|
|
|
|
if (numBytesWritten == 0)
|
|
throw Error(GetLastError(), "Writing to pipe failed (pipe closed?)");
|
|
|
|
curPos += (int)numBytesWritten;
|
|
}
|
|
else if (waitRes == WAIT_OBJECT_0 + 1)
|
|
{
|
|
// Cancel.
|
|
if (!CancelIo(m_dst))
|
|
throw Error(GetLastError(), "CancelIo() failed");
|
|
break;
|
|
}
|
|
else
|
|
throw Error(GetLastError(), "WaitForMultipleObjects() failed");
|
|
}
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
// \todo [2013-08-13 pyry] What to do about this?
|
|
printf("win32::CaseListWriter::run(): %s\n", e.what());
|
|
}
|
|
}
|
|
|
|
void CaseListWriter::stop (void)
|
|
{
|
|
if (!isStarted())
|
|
return; // Nothing to do.
|
|
|
|
m_cancelEvent.setSignaled();
|
|
|
|
// Join thread.
|
|
join();
|
|
|
|
m_cancelEvent.reset();
|
|
|
|
m_dst = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
// FileReader
|
|
|
|
FileReader::FileReader (ThreadedByteBuffer* dst)
|
|
: m_dstBuf (dst)
|
|
, m_handle (INVALID_HANDLE_VALUE)
|
|
, m_cancelEvent (false, false)
|
|
{
|
|
}
|
|
|
|
FileReader::~FileReader (void)
|
|
{
|
|
}
|
|
|
|
void FileReader::start (HANDLE file)
|
|
{
|
|
DE_ASSERT(!isStarted());
|
|
|
|
m_handle = file;
|
|
|
|
de::Thread::start();
|
|
}
|
|
|
|
void FileReader::run (void)
|
|
{
|
|
try
|
|
{
|
|
Event ioEvent (true, false); // Manual reset, not signaled state.
|
|
HANDLE waitHandles[] = { ioEvent.getHandle(), m_cancelEvent.getHandle() };
|
|
OVERLAPPED overlapped;
|
|
std::vector<deUint8> tmpBuf (FILEREADER_TMP_BUFFER_SIZE);
|
|
deUint64 offset = 0; // Overlapped IO requires manual offset keeping.
|
|
|
|
deMemset(&overlapped, 0, sizeof(overlapped));
|
|
overlapped.hEvent = ioEvent.getHandle();
|
|
|
|
for (;;)
|
|
{
|
|
DWORD numBytesRead = 0;
|
|
DWORD waitRes;
|
|
|
|
overlapped.Offset = (DWORD)(offset & 0xffffffffu);
|
|
overlapped.OffsetHigh = (DWORD)(offset >> 32);
|
|
|
|
if (!ReadFile(m_handle, &tmpBuf[0], (DWORD)tmpBuf.size(), NULL, &overlapped))
|
|
{
|
|
DWORD err = GetLastError();
|
|
|
|
if (err == ERROR_BROKEN_PIPE)
|
|
break;
|
|
else if (err == ERROR_HANDLE_EOF)
|
|
{
|
|
if (m_dstBuf->isCanceled())
|
|
break;
|
|
|
|
deSleep(FILEREADER_IDLE_SLEEP);
|
|
|
|
if (m_dstBuf->isCanceled())
|
|
break;
|
|
else
|
|
continue;
|
|
}
|
|
else if (err != ERROR_IO_PENDING)
|
|
throw Error(err, "ReadFile() failed");
|
|
}
|
|
|
|
waitRes = WaitForMultipleObjects(DE_LENGTH_OF_ARRAY(waitHandles), &waitHandles[0], FALSE, INFINITE);
|
|
|
|
if (waitRes == WAIT_OBJECT_0)
|
|
{
|
|
// \note GetOverlappedResult() will fail with ERROR_IO_INCOMPLETE if IO event is not complete (should be).
|
|
if (!GetOverlappedResult(m_handle, &overlapped, &numBytesRead, FALSE))
|
|
{
|
|
DWORD err = GetLastError();
|
|
|
|
if (err == ERROR_HANDLE_EOF)
|
|
{
|
|
// End of file - for now.
|
|
// \note Should check for end of buffer here, or otherwise may end up in infinite loop.
|
|
if (m_dstBuf->isCanceled())
|
|
break;
|
|
|
|
deSleep(FILEREADER_IDLE_SLEEP);
|
|
|
|
if (m_dstBuf->isCanceled())
|
|
break;
|
|
else
|
|
continue;
|
|
}
|
|
else if (err == ERROR_BROKEN_PIPE)
|
|
break;
|
|
else
|
|
throw Error(err, "GetOverlappedResult() failed");
|
|
}
|
|
|
|
if (numBytesRead == 0)
|
|
throw Error(GetLastError(), "Reading from file failed");
|
|
else
|
|
offset += (deUint64)numBytesRead;
|
|
}
|
|
else if (waitRes == WAIT_OBJECT_0 + 1)
|
|
{
|
|
// Cancel.
|
|
if (!CancelIo(m_handle))
|
|
throw Error(GetLastError(), "CancelIo() failed");
|
|
break;
|
|
}
|
|
else
|
|
throw Error(GetLastError(), "WaitForMultipleObjects() failed");
|
|
|
|
try
|
|
{
|
|
m_dstBuf->write((int)numBytesRead, &tmpBuf[0]);
|
|
m_dstBuf->flush();
|
|
}
|
|
catch (const ThreadedByteBuffer::CanceledException&)
|
|
{
|
|
// Canceled.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
// \todo [2013-08-13 pyry] What to do?
|
|
printf("win32::FileReader::run(): %s\n", e.what());
|
|
}
|
|
}
|
|
|
|
void FileReader::stop (void)
|
|
{
|
|
if (!isStarted())
|
|
return; // Nothing to do.
|
|
|
|
m_cancelEvent.setSignaled();
|
|
|
|
// Join thread.
|
|
join();
|
|
|
|
m_cancelEvent.reset();
|
|
|
|
m_handle = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
// TestLogReader
|
|
|
|
TestLogReader::TestLogReader (void)
|
|
: m_logBuffer (LOG_BUFFER_BLOCK_SIZE, LOG_BUFFER_NUM_BLOCKS)
|
|
, m_logFile (INVALID_HANDLE_VALUE)
|
|
, m_reader (&m_logBuffer)
|
|
{
|
|
}
|
|
|
|
TestLogReader::~TestLogReader (void)
|
|
{
|
|
if (m_logFile != INVALID_HANDLE_VALUE)
|
|
CloseHandle(m_logFile);
|
|
}
|
|
|
|
void TestLogReader::start (const char* filename)
|
|
{
|
|
DE_ASSERT(m_logFile == INVALID_HANDLE_VALUE && !m_reader.isStarted());
|
|
|
|
m_logFile = CreateFile(filename,
|
|
GENERIC_READ,
|
|
FILE_SHARE_DELETE|FILE_SHARE_READ|FILE_SHARE_WRITE,
|
|
DE_NULL,
|
|
OPEN_EXISTING,
|
|
FILE_ATTRIBUTE_NORMAL|FILE_FLAG_OVERLAPPED,
|
|
DE_NULL);
|
|
|
|
if (m_logFile == INVALID_HANDLE_VALUE)
|
|
throw Error(GetLastError(), "Failed to open log file");
|
|
|
|
m_reader.start(m_logFile);
|
|
}
|
|
|
|
void TestLogReader::stop (void)
|
|
{
|
|
if (!m_reader.isStarted())
|
|
return; // Nothing to do.
|
|
|
|
m_logBuffer.cancel();
|
|
m_reader.stop();
|
|
|
|
CloseHandle(m_logFile);
|
|
m_logFile = INVALID_HANDLE_VALUE;
|
|
|
|
m_logBuffer.clear();
|
|
}
|
|
|
|
// Process
|
|
|
|
Process::Process (void)
|
|
: m_state (STATE_NOT_STARTED)
|
|
, m_exitCode (0)
|
|
, m_standardIn (INVALID_HANDLE_VALUE)
|
|
, m_standardOut (INVALID_HANDLE_VALUE)
|
|
, m_standardErr (INVALID_HANDLE_VALUE)
|
|
{
|
|
deMemset(&m_procInfo, 0, sizeof(m_procInfo));
|
|
}
|
|
|
|
Process::~Process (void)
|
|
{
|
|
try
|
|
{
|
|
if (isRunning())
|
|
{
|
|
kill();
|
|
waitForFinish();
|
|
}
|
|
}
|
|
catch (...)
|
|
{
|
|
}
|
|
|
|
cleanupHandles();
|
|
}
|
|
|
|
void Process::cleanupHandles (void)
|
|
{
|
|
DE_ASSERT(!isRunning());
|
|
|
|
if (m_standardErr != INVALID_HANDLE_VALUE)
|
|
CloseHandle(m_standardErr);
|
|
|
|
if (m_standardOut != INVALID_HANDLE_VALUE)
|
|
CloseHandle(m_standardOut);
|
|
|
|
if (m_standardIn != INVALID_HANDLE_VALUE)
|
|
CloseHandle(m_standardIn);
|
|
|
|
if (m_procInfo.hProcess)
|
|
CloseHandle(m_procInfo.hProcess);
|
|
|
|
if (m_procInfo.hThread)
|
|
CloseHandle(m_procInfo.hThread);
|
|
|
|
m_standardErr = INVALID_HANDLE_VALUE;
|
|
m_standardOut = INVALID_HANDLE_VALUE;
|
|
m_standardIn = INVALID_HANDLE_VALUE;
|
|
|
|
deMemset(&m_procInfo, 0, sizeof(m_procInfo));
|
|
}
|
|
|
|
__declspec(thread) static int t_pipeNdx = 0;
|
|
|
|
static void createPipeWithOverlappedIO (HANDLE* readHandleOut, HANDLE* writeHandleOut, deUint32 readMode, deUint32 writeMode, SECURITY_ATTRIBUTES* securityAttr)
|
|
{
|
|
const int defaultBufSize = 4096;
|
|
char pipeName[128];
|
|
HANDLE readHandle;
|
|
HANDLE writeHandle;
|
|
|
|
DE_ASSERT(((readMode | writeMode) & ~FILE_FLAG_OVERLAPPED) == 0);
|
|
|
|
deSprintf(pipeName, sizeof(pipeName), "\\\\.\\Pipe\\dEQP-ExecServer-%08x-%08x-%08x",
|
|
GetCurrentProcessId(),
|
|
GetCurrentThreadId(),
|
|
t_pipeNdx++);
|
|
|
|
readHandle = CreateNamedPipe(pipeName, /* Pipe name. */
|
|
PIPE_ACCESS_INBOUND|readMode, /* Open mode. */
|
|
PIPE_TYPE_BYTE|PIPE_WAIT, /* Pipe flags. */
|
|
1, /* Max number of instances. */
|
|
defaultBufSize, /* Output buffer size. */
|
|
defaultBufSize, /* Input buffer size. */
|
|
0, /* Use default timeout. */
|
|
securityAttr);
|
|
|
|
if (readHandle == INVALID_HANDLE_VALUE)
|
|
throw Error(GetLastError(), "CreateNamedPipe() failed");
|
|
|
|
writeHandle = CreateFile(pipeName,
|
|
GENERIC_WRITE, /* Access mode. */
|
|
0, /* No sharing. */
|
|
securityAttr,
|
|
OPEN_EXISTING, /* Assume existing object. */
|
|
FILE_ATTRIBUTE_NORMAL|writeMode, /* Open mode / flags. */
|
|
DE_NULL /* Template file. */);
|
|
|
|
if (writeHandle == INVALID_HANDLE_VALUE)
|
|
{
|
|
DWORD openErr = GetLastError();
|
|
CloseHandle(readHandle);
|
|
throw Error(openErr, "Failed to open created pipe, CreateFile() failed");
|
|
}
|
|
|
|
*readHandleOut = readHandle;
|
|
*writeHandleOut = writeHandle;
|
|
}
|
|
|
|
void Process::start (const char* commandLine, const char* workingDirectory)
|
|
{
|
|
// Pipes.
|
|
HANDLE stdInRead = INVALID_HANDLE_VALUE;
|
|
HANDLE stdInWrite = INVALID_HANDLE_VALUE;
|
|
HANDLE stdOutRead = INVALID_HANDLE_VALUE;
|
|
HANDLE stdOutWrite = INVALID_HANDLE_VALUE;
|
|
HANDLE stdErrRead = INVALID_HANDLE_VALUE;
|
|
HANDLE stdErrWrite = INVALID_HANDLE_VALUE;
|
|
|
|
if (m_state == STATE_RUNNING)
|
|
throw std::runtime_error("Process already running");
|
|
else if (m_state == STATE_FINISHED)
|
|
{
|
|
// Process finished, clean up old cruft.
|
|
cleanupHandles();
|
|
m_state = STATE_NOT_STARTED;
|
|
}
|
|
|
|
// Create pipes
|
|
try
|
|
{
|
|
SECURITY_ATTRIBUTES securityAttr;
|
|
STARTUPINFO startInfo;
|
|
|
|
deMemset(&startInfo, 0, sizeof(startInfo));
|
|
deMemset(&securityAttr, 0, sizeof(securityAttr));
|
|
|
|
// Security attributes for inheriting handle.
|
|
securityAttr.nLength = sizeof(SECURITY_ATTRIBUTES);
|
|
securityAttr.bInheritHandle = TRUE;
|
|
securityAttr.lpSecurityDescriptor = DE_NULL;
|
|
|
|
createPipeWithOverlappedIO(&stdInRead, &stdInWrite, 0, FILE_FLAG_OVERLAPPED, &securityAttr);
|
|
createPipeWithOverlappedIO(&stdOutRead, &stdOutWrite, FILE_FLAG_OVERLAPPED, 0, &securityAttr);
|
|
createPipeWithOverlappedIO(&stdErrRead, &stdErrWrite, FILE_FLAG_OVERLAPPED, 0, &securityAttr);
|
|
|
|
if (!SetHandleInformation(stdInWrite, HANDLE_FLAG_INHERIT, 0) ||
|
|
!SetHandleInformation(stdOutRead, HANDLE_FLAG_INHERIT, 0) ||
|
|
!SetHandleInformation(stdErrRead, HANDLE_FLAG_INHERIT, 0))
|
|
throw Error(GetLastError(), "SetHandleInformation() failed");
|
|
|
|
// Startup info for process.
|
|
startInfo.cb = sizeof(startInfo);
|
|
startInfo.hStdError = stdErrWrite;
|
|
startInfo.hStdOutput = stdOutWrite;
|
|
startInfo.hStdInput = stdInRead;
|
|
startInfo.dwFlags |= STARTF_USESTDHANDLES;
|
|
|
|
if (!CreateProcess(DE_NULL, (LPTSTR)commandLine, DE_NULL, DE_NULL, TRUE /* inherit handles */, 0, DE_NULL, workingDirectory, &startInfo, &m_procInfo))
|
|
throw Error(GetLastError(), "CreateProcess() failed");
|
|
}
|
|
catch (...)
|
|
{
|
|
if (stdInRead != INVALID_HANDLE_VALUE) CloseHandle(stdInRead);
|
|
if (stdInWrite != INVALID_HANDLE_VALUE) CloseHandle(stdInWrite);
|
|
if (stdOutRead != INVALID_HANDLE_VALUE) CloseHandle(stdOutRead);
|
|
if (stdOutWrite != INVALID_HANDLE_VALUE) CloseHandle(stdOutWrite);
|
|
if (stdErrRead != INVALID_HANDLE_VALUE) CloseHandle(stdErrRead);
|
|
if (stdErrWrite != INVALID_HANDLE_VALUE) CloseHandle(stdErrWrite);
|
|
throw;
|
|
}
|
|
|
|
// Store handles to be kept.
|
|
m_standardIn = stdInWrite;
|
|
m_standardOut = stdOutRead;
|
|
m_standardErr = stdErrRead;
|
|
|
|
// Close other ends of handles.
|
|
CloseHandle(stdErrWrite);
|
|
CloseHandle(stdOutWrite);
|
|
CloseHandle(stdInRead);
|
|
|
|
m_state = STATE_RUNNING;
|
|
}
|
|
|
|
bool Process::isRunning (void)
|
|
{
|
|
if (m_state == STATE_RUNNING)
|
|
{
|
|
int exitCode;
|
|
BOOL result = GetExitCodeProcess(m_procInfo.hProcess, (LPDWORD)&exitCode);
|
|
|
|
if (result != TRUE)
|
|
throw Error(GetLastError(), "GetExitCodeProcess() failed");
|
|
|
|
if (exitCode == STILL_ACTIVE)
|
|
return true;
|
|
else
|
|
{
|
|
// Done.
|
|
m_exitCode = exitCode;
|
|
m_state = STATE_FINISHED;
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
void Process::waitForFinish (void)
|
|
{
|
|
if (m_state == STATE_RUNNING)
|
|
{
|
|
if (WaitForSingleObject(m_procInfo.hProcess, INFINITE) != WAIT_OBJECT_0)
|
|
throw Error(GetLastError(), "Waiting for process failed, WaitForSingleObject() failed");
|
|
|
|
if (isRunning())
|
|
throw std::runtime_error("Process is still alive");
|
|
}
|
|
else
|
|
throw std::runtime_error("Process is not running");
|
|
}
|
|
|
|
void Process::stopProcess (bool kill)
|
|
{
|
|
if (m_state == STATE_RUNNING)
|
|
{
|
|
if (!TerminateProcess(m_procInfo.hProcess, kill ? -1 : 0))
|
|
throw Error(GetLastError(), "TerminateProcess() failed");
|
|
}
|
|
else
|
|
throw std::runtime_error("Process is not running");
|
|
}
|
|
|
|
void Process::terminate (void)
|
|
{
|
|
stopProcess(false);
|
|
}
|
|
|
|
void Process::kill (void)
|
|
{
|
|
stopProcess(true);
|
|
}
|
|
|
|
} // win32
|
|
|
|
Win32TestProcess::Win32TestProcess (void)
|
|
: m_process (DE_NULL)
|
|
, m_processStartTime (0)
|
|
, m_infoBuffer (INFO_BUFFER_BLOCK_SIZE, INFO_BUFFER_NUM_BLOCKS)
|
|
, m_stdOutReader (&m_infoBuffer)
|
|
, m_stdErrReader (&m_infoBuffer)
|
|
{
|
|
}
|
|
|
|
Win32TestProcess::~Win32TestProcess (void)
|
|
{
|
|
delete m_process;
|
|
}
|
|
|
|
void Win32TestProcess::start (const char* name, const char* params, const char* workingDir, const char* caseList)
|
|
{
|
|
bool hasCaseList = strlen(caseList) > 0;
|
|
|
|
XS_CHECK(!m_process);
|
|
|
|
de::FilePath logFilePath = de::FilePath::join(workingDir, "TestResults.qpa");
|
|
m_logFileName = logFilePath.getPath();
|
|
|
|
// Remove old file if such exists.
|
|
// \note Sometimes on Windows the test process dies slowly and may not release handle to log file
|
|
// until a bit later.
|
|
// \todo [2013-07-15 pyry] This should be solved by improving deProcess and killing all child processes as well.
|
|
{
|
|
int tryNdx = 0;
|
|
while (tryNdx < MAX_OLD_LOGFILE_DELETE_ATTEMPTS && deFileExists(m_logFileName.c_str()))
|
|
{
|
|
if (deDeleteFile(m_logFileName.c_str()))
|
|
break;
|
|
deSleep(LOGFILE_DELETE_SLEEP_MS);
|
|
tryNdx += 1;
|
|
}
|
|
|
|
if (deFileExists(m_logFileName.c_str()))
|
|
throw TestProcessException(string("Failed to remove '") + m_logFileName + "'");
|
|
}
|
|
|
|
// Construct command line.
|
|
string cmdLine = de::FilePath(name).isAbsolutePath() ? name : de::FilePath::join(workingDir, name).normalize().getPath();
|
|
cmdLine += string(" --deqp-log-filename=") + logFilePath.getBaseName();
|
|
|
|
if (hasCaseList)
|
|
cmdLine += " --deqp-stdin-caselist";
|
|
|
|
if (strlen(params) > 0)
|
|
cmdLine += string(" ") + params;
|
|
|
|
DE_ASSERT(!m_process);
|
|
m_process = new win32::Process();
|
|
|
|
try
|
|
{
|
|
m_process->start(cmdLine.c_str(), strlen(workingDir) > 0 ? workingDir : DE_NULL);
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
delete m_process;
|
|
m_process = DE_NULL;
|
|
throw TestProcessException(e.what());
|
|
}
|
|
|
|
m_processStartTime = deGetMicroseconds();
|
|
|
|
// Create stdout & stderr readers.
|
|
m_stdOutReader.start(m_process->getStdOut());
|
|
m_stdErrReader.start(m_process->getStdErr());
|
|
|
|
// Start case list writer.
|
|
if (hasCaseList)
|
|
m_caseListWriter.start(caseList, m_process->getStdIn());
|
|
}
|
|
|
|
void Win32TestProcess::terminate (void)
|
|
{
|
|
if (m_process)
|
|
{
|
|
try
|
|
{
|
|
m_process->kill();
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
printf("Win32TestProcess::terminate(): Failed to kill process: %s\n", e.what());
|
|
}
|
|
}
|
|
}
|
|
|
|
void Win32TestProcess::cleanup (void)
|
|
{
|
|
m_caseListWriter.stop();
|
|
|
|
// \note Buffers must be canceled before stopping readers.
|
|
m_infoBuffer.cancel();
|
|
|
|
m_stdErrReader.stop();
|
|
m_stdOutReader.stop();
|
|
m_testLogReader.stop();
|
|
|
|
// Reset buffers.
|
|
m_infoBuffer.clear();
|
|
|
|
if (m_process)
|
|
{
|
|
try
|
|
{
|
|
if (m_process->isRunning())
|
|
{
|
|
m_process->kill();
|
|
m_process->waitForFinish();
|
|
}
|
|
}
|
|
catch (const std::exception& e)
|
|
{
|
|
printf("Win32TestProcess::cleanup(): Failed to kill process: %s\n", e.what());
|
|
}
|
|
|
|
delete m_process;
|
|
m_process = DE_NULL;
|
|
}
|
|
}
|
|
|
|
int Win32TestProcess::readTestLog (deUint8* dst, int numBytes)
|
|
{
|
|
if (!m_testLogReader.isRunning())
|
|
{
|
|
if (deGetMicroseconds() - m_processStartTime > LOG_FILE_TIMEOUT*1000)
|
|
{
|
|
// Timeout, kill process.
|
|
terminate();
|
|
return 0; // \todo [2013-08-13 pyry] Throw exception?
|
|
}
|
|
|
|
if (!deFileExists(m_logFileName.c_str()))
|
|
return 0;
|
|
|
|
// Start reader.
|
|
m_testLogReader.start(m_logFileName.c_str());
|
|
}
|
|
|
|
DE_ASSERT(m_testLogReader.isRunning());
|
|
return m_testLogReader.read(dst, numBytes);
|
|
}
|
|
|
|
bool Win32TestProcess::isRunning (void)
|
|
{
|
|
if (m_process)
|
|
return m_process->isRunning();
|
|
else
|
|
return false;
|
|
}
|
|
|
|
int Win32TestProcess::getExitCode (void) const
|
|
{
|
|
if (m_process)
|
|
return m_process->getExitCode();
|
|
else
|
|
return -1;
|
|
}
|
|
|
|
} // xs
|