1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program Tester Core
3 * ----------------------------------------
4 *
5 * Copyright 2014 The Android Open Source Project
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 *//*!
20 * \file
21 * \brief Android ExecServer.
22 *//*--------------------------------------------------------------------*/
23
24 #include "tcuAndroidExecService.hpp"
25 #include "deFile.h"
26 #include "deClock.h"
27
28 #if 0
29 # define DBG_PRINT(ARGS) print ARGS
30 #else
31 # define DBG_PRINT(ARGS)
32 #endif
33
34 namespace tcu
35 {
36 namespace Android
37 {
38
39 static const char* LOG_FILE_NAME = "/sdcard/dEQP-log.qpa";
40
41 enum
42 {
43 PROCESS_START_TIMEOUT = 5000*1000, //!< Timeout in usec.
44 PROCESS_QUERY_INTERVAL = 1000*1000 //!< Running query interval limit in usec.
45 };
46
checkJniException(JNIEnv * env,const char * file,int line)47 static void checkJniException (JNIEnv* env, const char* file, int line)
48 {
49 if (env->ExceptionOccurred())
50 {
51 env->ExceptionDescribe();
52 env->ExceptionClear();
53 throw InternalError("JNI Exception", DE_NULL, file, line);
54 }
55 }
56
57 #define JNI_CHECK(EXPR) do { checkJniException(env, __FILE__, __LINE__); TCU_CHECK_INTERNAL(EXPR); } while (deGetFalse())
58
59 // TestProcess
60
TestProcess(JavaVM * vm,jobject context)61 TestProcess::TestProcess (JavaVM* vm, jobject context)
62 : m_vm (vm)
63 , m_remoteCls (0)
64 , m_remote (0)
65 , m_start (0)
66 , m_kill (0)
67 , m_isRunning (0)
68 , m_launchTime (0)
69 , m_lastQueryTime (0)
70 , m_lastRunningStatus (false)
71 , m_logReader (xs::LOG_BUFFER_BLOCK_SIZE, xs::LOG_BUFFER_NUM_BLOCKS)
72 {
73 DBG_PRINT(("TestProcess::TestProcess(%p, %p)", vm, context));
74
75 JNIEnv* env = getCurrentThreadEnv();
76 jobject remote = 0;
77 jstring logFileName = 0;
78
79 try
80 {
81 jclass remoteCls = 0;
82 jmethodID ctorId = 0;
83
84 remoteCls = env->FindClass("com/drawelements/deqp/testercore/RemoteAPI");
85 JNI_CHECK(remoteCls);
86
87 // Acquire global reference to RemoteAPI class.
88 m_remoteCls = reinterpret_cast<jclass>(env->NewGlobalRef(remoteCls));
89 JNI_CHECK(m_remoteCls);
90 env->DeleteLocalRef(remoteCls);
91 remoteCls = 0;
92
93 ctorId = env->GetMethodID(m_remoteCls, "<init>", "(Landroid/content/Context;Ljava/lang/String;)V");
94 JNI_CHECK(ctorId);
95
96 logFileName = env->NewStringUTF(LOG_FILE_NAME);
97 JNI_CHECK(logFileName);
98
99 // Create RemoteAPI instance.
100 remote = env->NewObject(m_remoteCls, ctorId, context, logFileName);
101 JNI_CHECK(remote);
102
103 env->DeleteLocalRef(logFileName);
104 logFileName = 0;
105
106 // Acquire global reference to remote.
107 m_remote = env->NewGlobalRef(remote);
108 JNI_CHECK(m_remote);
109 env->DeleteLocalRef(remote);
110 remote = 0;
111
112 m_start = env->GetMethodID(m_remoteCls, "start", "(Ljava/lang/String;Ljava/lang/String;Ljava/lang/String;)Z");
113 JNI_CHECK(m_start);
114
115 m_kill = env->GetMethodID(m_remoteCls, "kill", "()Z");
116 JNI_CHECK(m_kill);
117
118 m_isRunning = env->GetMethodID(m_remoteCls, "isRunning", "()Z");
119 JNI_CHECK(m_isRunning);
120 }
121 catch (...)
122 {
123 if (logFileName)
124 env->DeleteLocalRef(logFileName);
125 if (remote)
126 env->DeleteLocalRef(remote);
127 if (m_remoteCls)
128 env->DeleteGlobalRef(reinterpret_cast<jobject>(m_remoteCls));
129 if (m_remote)
130 env->DeleteGlobalRef(m_remote);
131 throw;
132 }
133 }
134
~TestProcess(void)135 TestProcess::~TestProcess (void)
136 {
137 DBG_PRINT(("TestProcess::~TestProcess()"));
138
139 try
140 {
141 JNIEnv* env = getCurrentThreadEnv();
142 env->DeleteGlobalRef(m_remote);
143 env->DeleteGlobalRef(m_remoteCls);
144 }
145 catch (...)
146 {
147 }
148 }
149
start(const char * name,const char * params,const char * workingDir,const char * caseList)150 void TestProcess::start (const char* name, const char* params, const char* workingDir, const char* caseList)
151 {
152 DBG_PRINT(("TestProcess::start(%s, %s, %s, ...)", name, params, workingDir));
153
154 JNIEnv* env = getCurrentThreadEnv();
155 jstring nameStr = 0;
156 jstring paramsStr = 0;
157 jstring caseListStr = 0;
158
159 DE_UNREF(workingDir);
160
161 // Remove old log file if such exists.
162 if (deFileExists(LOG_FILE_NAME))
163 {
164 if (!deDeleteFile(LOG_FILE_NAME) || deFileExists(LOG_FILE_NAME))
165 throw xs::TestProcessException(std::string("Failed to remove '") + LOG_FILE_NAME + "'");
166 }
167
168 try
169 {
170 nameStr = env->NewStringUTF(name);
171 JNI_CHECK(nameStr);
172
173 paramsStr = env->NewStringUTF(params);
174 JNI_CHECK(paramsStr);
175
176 caseListStr = env->NewStringUTF(caseList);
177 JNI_CHECK(caseListStr);
178
179 jboolean res = env->CallBooleanMethod(m_remote, m_start, nameStr, paramsStr, caseListStr);
180 checkJniException(env, __FILE__, __LINE__);
181
182 if (res == JNI_FALSE)
183 throw xs::TestProcessException("Failed to launch activity");
184
185 m_launchTime = deGetMicroseconds();
186 m_lastQueryTime = m_launchTime;
187 m_lastRunningStatus = true;
188 }
189 catch (...)
190 {
191 if (nameStr)
192 env->DeleteLocalRef(nameStr);
193 if (paramsStr)
194 env->DeleteLocalRef(paramsStr);
195 if (caseListStr)
196 env->DeleteLocalRef(caseListStr);
197 throw;
198 }
199
200 env->DeleteLocalRef(nameStr);
201 env->DeleteLocalRef(paramsStr);
202 env->DeleteLocalRef(caseListStr);
203 }
204
terminate(void)205 void TestProcess::terminate (void)
206 {
207 DBG_PRINT(("TestProcess::terminate()"));
208
209 JNIEnv* env = getCurrentThreadEnv();
210 jboolean res = env->CallBooleanMethod(m_remote, m_kill);
211 checkJniException(env, __FILE__, __LINE__);
212 DE_UNREF(res); // Failure to kill process is ignored.
213 }
214
cleanup(void)215 void TestProcess::cleanup (void)
216 {
217 DBG_PRINT(("TestProcess::cleanup()"));
218
219 terminate();
220 m_logReader.stop();
221 }
222
isRunning(void)223 bool TestProcess::isRunning (void)
224 {
225 deUint64 curTime = deGetMicroseconds();
226
227 // On Android process launch is asynchronous so we don't want to poll for process until after some time.
228 if (curTime-m_launchTime < PROCESS_START_TIMEOUT ||
229 curTime-m_lastQueryTime < PROCESS_QUERY_INTERVAL)
230 return m_lastRunningStatus;
231
232 JNIEnv* env = getCurrentThreadEnv();
233 jboolean res = env->CallBooleanMethod(m_remote, m_isRunning);
234 checkJniException(env, __FILE__, __LINE__);
235
236 DBG_PRINT(("TestProcess::isRunning(): %s", res == JNI_TRUE ? "true" : "false"));
237 m_lastQueryTime = curTime;
238 m_lastRunningStatus = res == JNI_TRUE;
239
240 return m_lastRunningStatus;
241 }
242
getCurrentThreadEnv(void)243 JNIEnv* TestProcess::getCurrentThreadEnv (void)
244 {
245 JNIEnv* env = DE_NULL;
246 jint ret = m_vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6);
247
248 if (ret == JNI_OK)
249 return env;
250 else
251 throw InternalError("GetEnv() failed");
252 }
253
readTestLog(deUint8 * dst,int numBytes)254 int TestProcess::readTestLog (deUint8* dst, int numBytes)
255 {
256 if (!m_logReader.isRunning())
257 {
258 if (deGetMicroseconds() - m_launchTime > xs::LOG_FILE_TIMEOUT*1000)
259 {
260 // Timeout, kill process.
261 terminate();
262 DBG_PRINT(("TestProcess:readTestLog(): Log file timeout occurred!"));
263 return 0; // \todo [2013-08-13 pyry] Throw exception?
264 }
265
266 if (!deFileExists(LOG_FILE_NAME))
267 return 0;
268
269 // Start reader.
270 m_logReader.start(LOG_FILE_NAME);
271 }
272
273 DE_ASSERT(m_logReader.isRunning());
274 return m_logReader.read(dst, numBytes);
275 }
276
getExitCode(void) const277 int TestProcess::getExitCode (void) const
278 {
279 return 0;
280 }
281
readInfoLog(deUint8 * dst,int numBytes)282 int TestProcess::readInfoLog (deUint8* dst, int numBytes)
283 {
284 // \todo [2012-11-12 pyry] Read device log.
285 DE_UNREF(dst && numBytes);
286 return 0;
287 }
288
289 // ExecutionServer
290
ExecutionServer(JavaVM * vm,xs::TestProcess * testProcess,deSocketFamily family,int port,RunMode runMode)291 ExecutionServer::ExecutionServer (JavaVM* vm, xs::TestProcess* testProcess, deSocketFamily family, int port, RunMode runMode)
292 : xs::ExecutionServer (testProcess, family, port, runMode)
293 , m_vm (vm)
294 {
295 }
296
createHandler(de::Socket * socket,const de::SocketAddress & clientAddress)297 xs::ConnectionHandler* ExecutionServer::createHandler (de::Socket* socket, const de::SocketAddress& clientAddress)
298 {
299 DE_UNREF(clientAddress);
300 return new ConnectionHandler(m_vm, this, socket);
301 }
302
303 // ConnectionHandler
304
ConnectionHandler(JavaVM * vm,xs::ExecutionServer * server,de::Socket * socket)305 ConnectionHandler::ConnectionHandler (JavaVM* vm, xs::ExecutionServer* server, de::Socket* socket)
306 : xs::ExecutionRequestHandler (server, socket)
307 , m_vm (vm)
308 {
309 }
310
run(void)311 void ConnectionHandler::run (void)
312 {
313 JNIEnv* env = DE_NULL;
314 if (m_vm->AttachCurrentThread(&env, DE_NULL) != 0)
315 {
316 print("AttachCurrentThread() failed");
317 return;
318 }
319
320 xs::ExecutionRequestHandler::run();
321
322 if (m_vm->DetachCurrentThread() != 0)
323 print("DetachCurrentThread() failed");
324 }
325
326 // ServerThread
327
ServerThread(JavaVM * vm,xs::TestProcess * process,deSocketFamily family,int port)328 ServerThread::ServerThread (JavaVM* vm, xs::TestProcess* process, deSocketFamily family, int port)
329 : m_server(vm, process, family, port, xs::ExecutionServer::RUNMODE_FOREVER)
330 {
331 }
332
run(void)333 void ServerThread::run (void)
334 {
335 try
336 {
337 m_server.runServer();
338 }
339 catch (const std::exception& e)
340 {
341 die("ServerThread::run(): %s", e.what());
342 }
343 }
344
stop(void)345 void ServerThread::stop (void)
346 {
347 m_server.stopServer();
348 join();
349 }
350
351 // ExecService
352
ExecService(JavaVM * vm,jobject context,deSocketFamily family,int port)353 ExecService::ExecService (JavaVM* vm, jobject context, deSocketFamily family, int port)
354 : m_process (vm, context)
355 , m_thread (vm, &m_process, family, port)
356 {
357 }
358
~ExecService(void)359 ExecService::~ExecService (void)
360 {
361 }
362
start(void)363 void ExecService::start (void)
364 {
365 m_thread.start();
366 }
367
stop(void)368 void ExecService::stop (void)
369 {
370 m_thread.stop();
371 }
372
373 } // Android
374 } // tcu
375