Introduce visibleOnHandlerThread
This is a helper function that lets a caller make sure the side effects of a piece of code are visible both on a handler thread and in the caller thread before continuing. Also move HandlerUtilsTests to the tests directory of frameworks/libs/net from the NetworkStack where it used to be Test: NetworkStaticLibTests Change-Id: Icf1bd8bbe1216777c52b73cfbd0a46b647b72260
This commit is contained in:
@@ -0,0 +1,76 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.testutils
|
||||||
|
|
||||||
|
import android.os.Handler
|
||||||
|
import android.os.HandlerThread
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
import org.junit.Test
|
||||||
|
import org.junit.runner.RunWith
|
||||||
|
import org.junit.runners.JUnit4
|
||||||
|
|
||||||
|
private const val ATTEMPTS = 50 // Causes testWaitForIdle to take about 150ms on aosp_crosshatch-eng
|
||||||
|
private const val TIMEOUT_MS = 200
|
||||||
|
|
||||||
|
@RunWith(JUnit4::class)
|
||||||
|
class HandlerUtilsTest {
|
||||||
|
@Test
|
||||||
|
fun testWaitForIdle() {
|
||||||
|
val handlerThread = HandlerThread("testHandler").apply { start() }
|
||||||
|
|
||||||
|
// Tests that waitForIdle can be called many times without ill impact if the service is
|
||||||
|
// already idle.
|
||||||
|
repeat(ATTEMPTS) {
|
||||||
|
handlerThread.waitForIdle(TIMEOUT_MS)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tests that calling waitForIdle waits for messages to be processed. Use both an
|
||||||
|
// inline runnable that's instantiated at each loop run and a runnable that's instantiated
|
||||||
|
// once for all.
|
||||||
|
val tempRunnable = object : Runnable {
|
||||||
|
// Use StringBuilder preferentially to StringBuffer because StringBuilder is NOT
|
||||||
|
// thread-safe. It's part of the point that both runnables run on the same thread
|
||||||
|
// so if anything is wrong in that space it's better to opportunistically use a class
|
||||||
|
// where things might go wrong, even if there is no guarantee of failure.
|
||||||
|
var memory = StringBuilder()
|
||||||
|
override fun run() {
|
||||||
|
memory.append("b")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
repeat(ATTEMPTS) { i ->
|
||||||
|
handlerThread.threadHandler.post { tempRunnable.memory.append("a"); }
|
||||||
|
handlerThread.threadHandler.post(tempRunnable)
|
||||||
|
handlerThread.waitForIdle(TIMEOUT_MS)
|
||||||
|
assertEquals(tempRunnable.memory.toString(), "ab".repeat(i + 1))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Statistical test : even if visibleOnHandlerThread doesn't work this is likely to succeed,
|
||||||
|
// but it will be at least flaky.
|
||||||
|
@Test
|
||||||
|
fun testVisibleOnHandlerThread() {
|
||||||
|
val handlerThread = HandlerThread("testHandler").apply { start() }
|
||||||
|
val handler = Handler(handlerThread.looper)
|
||||||
|
|
||||||
|
repeat(ATTEMPTS) { attempt ->
|
||||||
|
var x = -10
|
||||||
|
visibleOnHandlerThread(handler) { x = attempt }
|
||||||
|
assertEquals(attempt, x)
|
||||||
|
handler.post { assertEquals(attempt, x) }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -21,9 +21,14 @@ package com.android.testutils
|
|||||||
import android.os.ConditionVariable
|
import android.os.ConditionVariable
|
||||||
import android.os.Handler
|
import android.os.Handler
|
||||||
import android.os.HandlerThread
|
import android.os.HandlerThread
|
||||||
|
import android.util.Log
|
||||||
|
import com.android.testutils.FunctionalUtils.ThrowingRunnable
|
||||||
|
import java.lang.Exception
|
||||||
import java.util.concurrent.Executor
|
import java.util.concurrent.Executor
|
||||||
import kotlin.test.fail
|
import kotlin.test.fail
|
||||||
|
|
||||||
|
private const val TAG = "HandlerUtils"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Block until the specified Handler or HandlerThread becomes idle, or until timeoutMs has passed.
|
* Block until the specified Handler or HandlerThread becomes idle, or until timeoutMs has passed.
|
||||||
*/
|
*/
|
||||||
@@ -48,3 +53,28 @@ fun waitForIdleSerialExecutor(executor: Executor, timeoutMs: Long) {
|
|||||||
fail("Executor did not become idle after ${timeoutMs}ms")
|
fail("Executor did not become idle after ${timeoutMs}ms")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Executes a block of code, making its side effects visible on the caller and the handler thread
|
||||||
|
*
|
||||||
|
* After this function returns, the side effects of the passed block of code are guaranteed to be
|
||||||
|
* observed both on the thread running the handler and on the thread running this method.
|
||||||
|
* To achieve this, this method runs the passed block on the handler and blocks this thread
|
||||||
|
* until it's executed, so keep in mind this method will block, (including, if the handler isn't
|
||||||
|
* running, blocking forever).
|
||||||
|
*/
|
||||||
|
fun visibleOnHandlerThread(handler: Handler, r: ThrowingRunnable) {
|
||||||
|
val cv = ConditionVariable()
|
||||||
|
handler.post {
|
||||||
|
try {
|
||||||
|
r.run()
|
||||||
|
} catch (exception: Exception) {
|
||||||
|
Log.e(TAG, "visibleOnHandlerThread caught exception", exception)
|
||||||
|
}
|
||||||
|
cv.open()
|
||||||
|
}
|
||||||
|
// After block() returns, the handler thread has seen the change (since it ran it)
|
||||||
|
// and this thread also has seen the change (since cv.open() happens-before cv.block()
|
||||||
|
// returns).
|
||||||
|
cv.block()
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user