-
Notifications
You must be signed in to change notification settings - Fork 143
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Basic watchdog that triggers thread dumps on puck jank #2558
Draft
jush
wants to merge
1
commit into
main
Choose a base branch
from
rs/thread_dump_missing_animation
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Draft
Changes from all commits
Commits
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,8 +1,11 @@ | ||
package com.mapbox.maps.testapp.examples | ||
|
||
import android.animation.Animator | ||
import android.animation.ValueAnimator | ||
import android.os.Bundle | ||
import android.os.Handler | ||
import android.os.Looper | ||
import android.util.Log | ||
import androidx.appcompat.app.AppCompatActivity | ||
import androidx.interpolator.view.animation.FastOutSlowInInterpolator | ||
import com.mapbox.geojson.Point | ||
|
@@ -33,6 +36,28 @@ class LocationComponentAnimationActivity : AppCompatActivity() { | |
private inner class FakeLocationProvider : LocationProvider { | ||
|
||
private var locationConsumer: LocationConsumer? = null | ||
private val listeners = | ||
object : ValueAnimator.AnimatorUpdateListener, Animator.AnimatorListener { | ||
override fun onAnimationUpdate(animation: ValueAnimator) { | ||
Watchdog.reschedule() | ||
} | ||
|
||
override fun onAnimationStart(animation: Animator) { | ||
Watchdog.reschedule() | ||
} | ||
|
||
override fun onAnimationEnd(animation: Animator) { | ||
Watchdog.stop() | ||
animation.removeListener(this) | ||
(animation as ValueAnimator).removeUpdateListener(this) | ||
} | ||
|
||
override fun onAnimationCancel(animation: Animator) { | ||
} | ||
|
||
override fun onAnimationRepeat(animation: Animator) { | ||
} | ||
} | ||
|
||
private fun emitFakeLocations() { | ||
// after several first emits we update puck animator options | ||
|
@@ -73,7 +98,10 @@ class LocationComponentAnimationActivity : AppCompatActivity() { | |
POINT_LNG + delta, | ||
POINT_LAT + delta | ||
) | ||
) | ||
) { | ||
addUpdateListener([email protected]) | ||
addListener([email protected]) | ||
} | ||
} | ||
} | ||
locationConsumer?.onBearingUpdated(BEARING + delta * 10000.0 * 5) | ||
|
@@ -88,11 +116,20 @@ class LocationComponentAnimationActivity : AppCompatActivity() { | |
override fun registerLocationConsumer(locationConsumer: LocationConsumer) { | ||
this.locationConsumer = locationConsumer | ||
emitFakeLocations() | ||
Watchdog.enabled = true | ||
// Fake a busy main thread after 15s | ||
handler.postDelayed({ | ||
Log.d("TAG", "emitFakeLocations: Blocking main thread") | ||
// Simulate main thread busy for few milliseconds | ||
Thread.sleep(150) | ||
Log.d("TAG", "emitFakeLocations: Finished blocking main thread") | ||
}, 15_000L) | ||
} | ||
|
||
override fun unRegisterLocationConsumer(locationConsumer: LocationConsumer) { | ||
this.locationConsumer = null | ||
handler.removeCallbacksAndMessages(null) | ||
Watchdog.enabled = false | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Remember to disable it to stop its thread and free other resources. |
||
} | ||
} | ||
|
||
|
93 changes: 93 additions & 0 deletions
93
app/src/main/java/com/mapbox/maps/testapp/examples/Watchdog.kt
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
package com.mapbox.maps.testapp.examples | ||
|
||
import android.os.Handler | ||
import android.os.HandlerThread | ||
import android.os.Message | ||
import android.os.Process | ||
import android.util.Log | ||
import com.mapbox.maps.testapp.examples.Watchdog.TIME_TO_FIRST_TRIGGER | ||
import com.mapbox.maps.testapp.examples.Watchdog.TIME_TO_SUBSEQUENT_TRIGGER | ||
import com.mapbox.maps.testapp.examples.Watchdog.reschedule | ||
import com.mapbox.maps.testapp.examples.Watchdog.stop | ||
|
||
/** | ||
* A simple watchdog that will trigger a [Process.SIGNAL_QUIT] signal if [reschedule] is not called | ||
* within [TIME_TO_FIRST_TRIGGER] milliseconds and continues to do so every | ||
* [TIME_TO_SUBSEQUENT_TRIGGER] until [reschedule] or [stop] is called. | ||
*/ | ||
object Watchdog { | ||
var enabled = false | ||
set(value) { | ||
if (field == value) return | ||
field = value | ||
if (value) { | ||
// Start the watchdog thread when enabled to avoid unnecessary overhead | ||
watchdogHandlerThread = HandlerThread(TAG).apply { start() } | ||
watchdogHandler = Handler(watchdogHandlerThread!!.looper) | ||
} else { | ||
// Stop the watchdog thread and free properties when disabled to avoid unnecessary overhead | ||
stop() | ||
watchdogHandlerThread?.quit() | ||
watchdogHandler = null | ||
watchdogHandlerThread = null | ||
} | ||
} | ||
|
||
private var watchdogHandlerThread: HandlerThread? = null | ||
private var watchdogHandler: Handler? = null | ||
private var currentCounter = 0 | ||
|
||
private val quitSignalTask: () -> Unit = { | ||
Log.w(TAG, "(${currentCounter++}) Task not rescheduled on time. Triggering SIGNAL_QUIT.") | ||
// Send a quit signal to the current process to write a thread dump to `/data/anr/`. | ||
Process.sendSignal(Process.myPid(), Process.SIGNAL_QUIT) | ||
scheduleQuitSignalTask() | ||
} | ||
|
||
private fun scheduleQuitSignalTask() { | ||
if (currentCounter >= MAX_CONSECUTIVE_TRIGGERS) { | ||
Log.w( | ||
TAG, | ||
"Max consecutive triggers ($currentCounter) reached. Not scheduling another trigger." | ||
) | ||
return | ||
} | ||
if (enabled) { | ||
watchdogHandler?.let { | ||
it.sendMessageDelayed(Message.obtain(it, quitSignalTask), TIME_TO_SUBSEQUENT_TRIGGER) | ||
} | ||
} | ||
} | ||
|
||
fun reschedule() { | ||
if (enabled) { | ||
stop() | ||
watchdogHandler?.let { | ||
it.sendMessageDelayed(Message.obtain(it, quitSignalTask), TIME_TO_FIRST_TRIGGER) | ||
} | ||
} | ||
} | ||
|
||
fun stop() { | ||
// Cancel all pending tasks | ||
watchdogHandler?.removeCallbacksAndMessages(null) | ||
currentCounter = 0 | ||
} | ||
|
||
private const val TAG = "Watchdog" | ||
|
||
/** | ||
* The amount of time that need to pass before the watchdog triggers the first time. | ||
* That is, if [reschedule] is not called within this time, the watchdog task will trigger. | ||
* Unit is milliseconds. | ||
*/ | ||
private const val TIME_TO_FIRST_TRIGGER: Long = 50L | ||
|
||
/** | ||
* The amount of time that the task will wait before running again. | ||
* Unit is milliseconds. | ||
*/ | ||
private const val TIME_TO_SUBSEQUENT_TRIGGER: Long = 100L | ||
|
||
private const val MAX_CONSECUTIVE_TRIGGERS = 5 | ||
} |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Before using the
Watchdog
you must enable it.