/*------------------------------------------------------------------------- * drawElements Quality Program Test Executor * ------------------------------------------ * * 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 Tcp/Ip link that manages execserver process. *//*--------------------------------------------------------------------*/ #include "xeLocalTcpIpLink.hpp" #include "deClock.h" #include "deThread.h" #include <sstream> enum { SERVER_START_TIMEOUT = 1000, SERVER_START_IDLE_SLEEP = 50 }; namespace xe { LocalTcpIpLink::LocalTcpIpLink (void) : m_process(DE_NULL) { } LocalTcpIpLink::~LocalTcpIpLink (void) { stop(); } void LocalTcpIpLink::start (const char* execServerPath, const char* workDir, int port) { XE_CHECK(!m_process); std::ostringstream cmdLine; cmdLine << execServerPath << " --single --port=" << port; m_process = deProcess_create(); XE_CHECK(m_process); if (deProcess_start(m_process, cmdLine.str().c_str(), workDir) != DE_TRUE) { std::string err = deProcess_getLastError(m_process); deProcess_destroy(m_process); m_process = DE_NULL; XE_FAIL((std::string("Failed to start ExecServer '") + execServerPath + "' : " + err).c_str()); } try { de::SocketAddress address; address.setFamily (DE_SOCKETFAMILY_INET4); address.setProtocol (DE_SOCKETPROTOCOL_TCP); address.setHost ("127.0.0.1"); address.setPort (port); // Wait until server has started - \todo [2012-07-19 pyry] This could be improved by having server to signal when it is ready. deUint64 waitStart = deGetMicroseconds(); for (;;) { if (!deProcess_isRunning(m_process)) XE_FAIL("ExecServer died"); try { m_link.connect(address); break; } catch (const de::SocketError&) { if (deGetMicroseconds()-waitStart > SERVER_START_TIMEOUT*1000) XE_FAIL("Server start timeout"); deSleep(SERVER_START_IDLE_SLEEP); } } // Close stdout/stderr or otherwise process will hang once OS pipe buffers are full. // \todo [2012-07-19 pyry] Read and store stdout/stderr from execserver. XE_CHECK(deProcess_closeStdOut(m_process)); XE_CHECK(deProcess_closeStdErr(m_process)); } catch (const std::exception&) { stop(); throw; } } void LocalTcpIpLink::stop (void) { if (m_process) { try { m_link.disconnect(); } catch (...) { // Silently ignore since this is called in destructor. } // \note --single flag is used so execserver should kill itself once one connection is handled. // This is here to make sure it dies even in case of hang. deProcess_terminate (m_process); deProcess_waitForFinish (m_process); deProcess_destroy (m_process); m_process = DE_NULL; } } void LocalTcpIpLink::reset (void) { m_link.reset(); } CommLinkState LocalTcpIpLink::getState (void) const { if (!m_process) return COMMLINKSTATE_ERROR; else return m_link.getState(); } CommLinkState LocalTcpIpLink::getState (std::string& error) const { if (!m_process) { error = "Not started"; return COMMLINKSTATE_ERROR; } else return m_link.getState(); } void LocalTcpIpLink::setCallbacks (StateChangedFunc stateChangedCallback, LogDataFunc testLogDataCallback, LogDataFunc infoLogDataCallback, void* userPtr) { m_link.setCallbacks(stateChangedCallback, testLogDataCallback, infoLogDataCallback, userPtr); } void LocalTcpIpLink::startTestProcess (const char* name, const char* params, const char* workingDir, const char* caseList) { if (m_process) m_link.startTestProcess(name, params, workingDir, caseList); else XE_FAIL("Not started"); } void LocalTcpIpLink::stopTestProcess (void) { if (m_process) m_link.stopTestProcess(); else XE_FAIL("Not started"); } } // xe