1In order to buy some performance on the common, uninstrumented, fast path, we replace repeated 2checks for both allocation instrumentation and allocator changes by a single function table 3dispatch, and templatized allocation code that can be used to generate either instrumented 4or uninstrumented versions of allocation routines. 5 6When we call an allocation routine, we always indirect through a thread-local function table that 7either points to instrumented or uninstrumented allocation routines. The instrumented code has a 8`kInstrumented` = true template argument (or `kIsInstrumented` in some places), the uninstrumented 9code has `kInstrumented` = false. 10 11The function table is thread-local. There appears to be no logical necessity for that; it just 12makes it easier to access from compiled Java code. 13 14- The function table is switched out by `InstrumentQuickAllocEntryPoints[Locked]`, and a 15corresponding `UninstrumentQuickAlloc`... function. 16 17- These in turn are called by `SetStatsEnabled()`, `SetAllocationListener()`, et al, which 18require the mutator lock is not held. 19 20- With a started runtime, `SetEntrypointsInstrumented()` calls `ScopedSuspendAll(`) before updating 21 the function table. 22 23Mutual exclusion in the dispatch table is thus ensured by the fact that it is only updated while 24all other threads are suspended, and is only accessed with the mutator lock logically held, 25which inhibits suspension. 26 27To ensure correctness, we thus must: 28 291. Suspend all threads when swapping out the dispatch table, and 302. Make sure that we hold the mutator lock when accessing it. 313. Not trust kInstrumented once we've given up the mutator lock, since it could have changed in the 32 interim. 33 34