Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 103dd32a authored by Qasid Ahmad Sadiq's avatar Qasid Ahmad Sadiq
Browse files

Prefetching can be interupted by other service requests.

Slow prefetch requests would block user interactive requests, creating
noticeable sluggishness and unresponsiveness in accessibility services,
especially on the web.

Let's make it so a user interactive requests stops prefetching.
We can't interupt an API call, but we can stop in between API calls.

On the service side, we have to seprate the prefetch callbacks from the
find callback. And we have to make it asynchrnous. It does dispatch into
the main thread, so the AccessibilityCache can remain single threaded.

When the calls are interupted on the application side, returnPendingFindAccessibilityNodeInfosInPrefetch checks the find requests that are waiting in the queue, to see if they can be addressed by the prefetch results. If they can be, we don't have to call into potentially imperformance application code.

Bug:30969887
Test: Performance measurements, tried it out by hand to see if their are any bugs. CTSAccessibility*

Change-Id: Ia8f1152afa3987f262f37ed4583775acdd32db43
parent 312aae75
Loading
Loading
Loading
Loading
+165 −117
Original line number Original line Diff line number Diff line
@@ -113,6 +113,8 @@ public final class AccessibilityInteractionController {


    private AddNodeInfosForViewId mAddNodeInfosForViewId;
    private AddNodeInfosForViewId mAddNodeInfosForViewId;


    private List<Message> mPendingFindNodeByIdMessages;

    @GuardedBy("mLock")
    @GuardedBy("mLock")
    private int mNumActiveRequestPreparers;
    private int mNumActiveRequestPreparers;
    @GuardedBy("mLock")
    @GuardedBy("mLock")
@@ -128,6 +130,7 @@ public final class AccessibilityInteractionController {
        mViewRootImpl = viewRootImpl;
        mViewRootImpl = viewRootImpl;
        mPrefetcher = new AccessibilityNodePrefetcher();
        mPrefetcher = new AccessibilityNodePrefetcher();
        mA11yManager = mViewRootImpl.mContext.getSystemService(AccessibilityManager.class);
        mA11yManager = mViewRootImpl.mContext.getSystemService(AccessibilityManager.class);
        mPendingFindNodeByIdMessages = new ArrayList<>();
    }
    }


    private void scheduleMessage(Message message, int interrogatingPid, long interrogatingTid,
    private void scheduleMessage(Message message, int interrogatingPid, long interrogatingTid,
@@ -177,6 +180,7 @@ public final class AccessibilityInteractionController {
        args.arg4 = arguments;
        args.arg4 = arguments;
        message.obj = args;
        message.obj = args;


        mPendingFindNodeByIdMessages.add(message);
        scheduleMessage(message, interrogatingPid, interrogatingTid, CONSIDER_REQUEST_PREPARERS);
        scheduleMessage(message, interrogatingPid, interrogatingTid, CONSIDER_REQUEST_PREPARERS);
    }
    }


@@ -315,6 +319,8 @@ public final class AccessibilityInteractionController {
    }
    }


    private void findAccessibilityNodeInfoByAccessibilityIdUiThread(Message message) {
    private void findAccessibilityNodeInfoByAccessibilityIdUiThread(Message message) {
        mPendingFindNodeByIdMessages.remove(message);

        final int flags = message.arg1;
        final int flags = message.arg1;


        SomeArgs args = (SomeArgs) message.obj;
        SomeArgs args = (SomeArgs) message.obj;
@@ -329,22 +335,58 @@ public final class AccessibilityInteractionController {


        args.recycle();
        args.recycle();


        List<AccessibilityNodeInfo> infos = mTempAccessibilityNodeInfoList;
        View rootView = null;
        infos.clear();
        AccessibilityNodeInfo rootNode = null;
        try {
        try {
            if (mViewRootImpl.mView == null || mViewRootImpl.mAttachInfo == null) {
            if (mViewRootImpl.mView == null || mViewRootImpl.mAttachInfo == null) {
                return;
                return;
            }
            }
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = flags;
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = flags;
            final View root = findViewByAccessibilityId(accessibilityViewId);
            rootView = findViewByAccessibilityId(accessibilityViewId);
            if (root != null && isShown(root)) {
            if (rootView != null && isShown(rootView)) {
                mPrefetcher.prefetchAccessibilityNodeInfos(
                rootNode = populateAccessibilityNodeInfoForView(
                        root, virtualDescendantId, flags, infos, arguments);
                        rootView, arguments, virtualDescendantId);
            }
            }
        } finally {
        } finally {
            updateInfosForViewportAndReturnFindNodeResult(
            updateInfoForViewportAndReturnFindNodeResult(
                    infos, callback, interactionId, spec, interactiveRegion);
                    rootNode == null ? null : AccessibilityNodeInfo.obtain(rootNode),
                    callback, interactionId, spec, interactiveRegion);
        }
        List<AccessibilityNodeInfo> infos = mTempAccessibilityNodeInfoList;
        infos.clear();
        mPrefetcher.prefetchAccessibilityNodeInfos(
                rootView, rootNode == null ? null : AccessibilityNodeInfo.obtain(rootNode),
                virtualDescendantId, flags, infos);
        mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
        updateInfosForViewPort(infos, spec, interactiveRegion);
        returnPrefetchResult(interactionId, infos, callback);
        returnPendingFindAccessibilityNodeInfosInPrefetch(infos);
    }

    private AccessibilityNodeInfo populateAccessibilityNodeInfoForView(
            View view, Bundle arguments, int virtualViewId) {
        AccessibilityNodeProvider provider = view.getAccessibilityNodeProvider();
        // Determine if we'll be populating extra data
        final String extraDataRequested = (arguments == null) ? null
                : arguments.getString(EXTRA_DATA_REQUESTED_KEY);
        AccessibilityNodeInfo root = null;
        if (provider == null) {
            root = view.createAccessibilityNodeInfo();
            if (root != null) {
                if (extraDataRequested != null) {
                    view.addExtraDataToAccessibilityNodeInfo(root, extraDataRequested, arguments);
                }
            }
        } else {
            root = provider.createAccessibilityNodeInfo(virtualViewId);
            if (root != null) {
                if (extraDataRequested != null) {
                    provider.addExtraDataToAccessibilityNodeInfo(
                            virtualViewId, root, extraDataRequested, arguments);
                }
            }
        }
        }
        return root;
    }
    }


    public void findAccessibilityNodeInfosByViewIdClientThread(long accessibilityNodeId,
    public void findAccessibilityNodeInfosByViewIdClientThread(long accessibilityNodeId,
@@ -402,6 +444,7 @@ public final class AccessibilityInteractionController {
                mAddNodeInfosForViewId.reset();
                mAddNodeInfosForViewId.reset();
            }
            }
        } finally {
        } finally {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
            updateInfosForViewportAndReturnFindNodeResult(
            updateInfosForViewportAndReturnFindNodeResult(
                    infos, callback, interactionId, spec, interactiveRegion);
                    infos, callback, interactionId, spec, interactiveRegion);
        }
        }
@@ -484,6 +527,7 @@ public final class AccessibilityInteractionController {
                }
                }
            }
            }
        } finally {
        } finally {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
            updateInfosForViewportAndReturnFindNodeResult(
            updateInfosForViewportAndReturnFindNodeResult(
                    infos, callback, interactionId, spec, interactiveRegion);
                    infos, callback, interactionId, spec, interactiveRegion);
        }
        }
@@ -575,6 +619,7 @@ public final class AccessibilityInteractionController {
                }
                }
            }
            }
        } finally {
        } finally {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
            updateInfoForViewportAndReturnFindNodeResult(
            updateInfoForViewportAndReturnFindNodeResult(
                    focused, callback, interactionId, spec, interactiveRegion);
                    focused, callback, interactionId, spec, interactiveRegion);
        }
        }
@@ -629,6 +674,7 @@ public final class AccessibilityInteractionController {
                }
                }
            }
            }
        } finally {
        } finally {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
            updateInfoForViewportAndReturnFindNodeResult(
            updateInfoForViewportAndReturnFindNodeResult(
                    next, callback, interactionId, spec, interactiveRegion);
                    next, callback, interactionId, spec, interactiveRegion);
        }
        }
@@ -785,33 +831,6 @@ public final class AccessibilityInteractionController {
        }
        }
    }
    }


    private void applyAppScaleAndMagnificationSpecIfNeeded(List<AccessibilityNodeInfo> infos,
            MagnificationSpec spec) {
        if (infos == null) {
            return;
        }
        final float applicationScale = mViewRootImpl.mAttachInfo.mApplicationScale;
        if (shouldApplyAppScaleAndMagnificationSpec(applicationScale, spec)) {
            final int infoCount = infos.size();
            for (int i = 0; i < infoCount; i++) {
                AccessibilityNodeInfo info = infos.get(i);
                applyAppScaleAndMagnificationSpecIfNeeded(info, spec);
            }
        }
    }

    private void adjustIsVisibleToUserIfNeeded(List<AccessibilityNodeInfo> infos,
            Region interactiveRegion) {
        if (interactiveRegion == null || infos == null) {
            return;
        }
        final int infoCount = infos.size();
        for (int i = 0; i < infoCount; i++) {
            AccessibilityNodeInfo info = infos.get(i);
            adjustIsVisibleToUserIfNeeded(info, interactiveRegion);
        }
    }

    private void adjustIsVisibleToUserIfNeeded(AccessibilityNodeInfo info,
    private void adjustIsVisibleToUserIfNeeded(AccessibilityNodeInfo info,
            Region interactiveRegion) {
            Region interactiveRegion) {
        if (interactiveRegion == null || info == null) {
        if (interactiveRegion == null || info == null) {
@@ -832,17 +851,6 @@ public final class AccessibilityInteractionController {
        return false;
        return false;
    }
    }


    private void adjustBoundsInScreenIfNeeded(List<AccessibilityNodeInfo> infos) {
        if (infos == null || shouldBypassAdjustBoundsInScreen()) {
            return;
        }
        final int infoCount = infos.size();
        for (int i = 0; i < infoCount; i++) {
            final AccessibilityNodeInfo info = infos.get(i);
            adjustBoundsInScreenIfNeeded(info);
        }
    }

    private void adjustBoundsInScreenIfNeeded(AccessibilityNodeInfo info) {
    private void adjustBoundsInScreenIfNeeded(AccessibilityNodeInfo info) {
        if (info == null || shouldBypassAdjustBoundsInScreen()) {
        if (info == null || shouldBypassAdjustBoundsInScreen()) {
            return;
            return;
@@ -890,17 +898,6 @@ public final class AccessibilityInteractionController {
        return screenMatrix == null || screenMatrix.isIdentity();
        return screenMatrix == null || screenMatrix.isIdentity();
    }
    }


    private void associateLeashedParentIfNeeded(List<AccessibilityNodeInfo> infos) {
        if (infos == null || shouldBypassAssociateLeashedParent()) {
            return;
        }
        final int infoCount = infos.size();
        for (int i = 0; i < infoCount; i++) {
            final AccessibilityNodeInfo info = infos.get(i);
            associateLeashedParentIfNeeded(info);
        }
    }

    private void associateLeashedParentIfNeeded(AccessibilityNodeInfo info) {
    private void associateLeashedParentIfNeeded(AccessibilityNodeInfo info) {
        if (info == null || shouldBypassAssociateLeashedParent()) {
        if (info == null || shouldBypassAssociateLeashedParent()) {
            return;
            return;
@@ -974,18 +971,46 @@ public final class AccessibilityInteractionController {
        return (appScale != 1.0f || (spec != null && !spec.isNop()));
        return (appScale != 1.0f || (spec != null && !spec.isNop()));
    }
    }


    private void updateInfosForViewPort(List<AccessibilityNodeInfo> infos, MagnificationSpec spec,
                                        Region interactiveRegion) {
        for (int i = 0; i < infos.size(); i++) {
            updateInfoForViewPort(infos.get(i), spec, interactiveRegion);
        }
    }

    private void updateInfoForViewPort(AccessibilityNodeInfo info, MagnificationSpec spec,
                                       Region interactiveRegion) {
        associateLeashedParentIfNeeded(info);
        applyScreenMatrixIfNeeded(info);
        adjustBoundsInScreenIfNeeded(info);
        // To avoid applyAppScaleAndMagnificationSpecIfNeeded changing the bounds of node,
        // then impact the visibility result, we need to adjust visibility before apply scale.
        adjustIsVisibleToUserIfNeeded(info, interactiveRegion);
        applyAppScaleAndMagnificationSpecIfNeeded(info, spec);
    }

    private void updateInfosForViewportAndReturnFindNodeResult(List<AccessibilityNodeInfo> infos,
    private void updateInfosForViewportAndReturnFindNodeResult(List<AccessibilityNodeInfo> infos,
            IAccessibilityInteractionConnectionCallback callback, int interactionId,
            IAccessibilityInteractionConnectionCallback callback, int interactionId,
            MagnificationSpec spec, Region interactiveRegion) {
            MagnificationSpec spec, Region interactiveRegion) {
        if (infos != null) {
            updateInfosForViewPort(infos, spec, interactiveRegion);
        }
        returnFindNodesResult(infos, callback, interactionId);
    }

    private void returnFindNodeResult(AccessibilityNodeInfo info,
                                      IAccessibilityInteractionConnectionCallback callback,
                                      int interactionId) {
        try {
            callback.setFindAccessibilityNodeInfoResult(info, interactionId);
        } catch (RemoteException re) {
            /* ignore - the other side will time out */
        }
    }

    private void returnFindNodesResult(List<AccessibilityNodeInfo> infos,
            IAccessibilityInteractionConnectionCallback callback, int interactionId) {
        try {
        try {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
            associateLeashedParentIfNeeded(infos);
            applyScreenMatrixIfNeeded(infos);
            adjustBoundsInScreenIfNeeded(infos);
            // To avoid applyAppScaleAndMagnificationSpecIfNeeded changing the bounds of node,
            // then impact the visibility result, we need to adjust visibility before apply scale.
            adjustIsVisibleToUserIfNeeded(infos, interactiveRegion);
            applyAppScaleAndMagnificationSpecIfNeeded(infos, spec);
            callback.setFindAccessibilityNodeInfosResult(infos, interactionId);
            callback.setFindAccessibilityNodeInfosResult(infos, interactionId);
            if (infos != null) {
            if (infos != null) {
                infos.clear();
                infos.clear();
@@ -995,22 +1020,49 @@ public final class AccessibilityInteractionController {
        }
        }
    }
    }


    private void updateInfoForViewportAndReturnFindNodeResult(AccessibilityNodeInfo info,
    private void returnPendingFindAccessibilityNodeInfosInPrefetch(
            IAccessibilityInteractionConnectionCallback callback, int interactionId,
            List<AccessibilityNodeInfo> infos) {
            MagnificationSpec spec, Region interactiveRegion) {
        for (Message pendingMessage : mPendingFindNodeByIdMessages) {
            SomeArgs args = (SomeArgs) pendingMessage.obj;
            final int accessibilityViewId = args.argi1;
            final int virtualDescendantId = args.argi2;
            final int interactionId = args.argi3;
            final IAccessibilityInteractionConnectionCallback callback =
                    (IAccessibilityInteractionConnectionCallback) args.arg1;
            final long nodeId =
                    AccessibilityNodeInfo.makeNodeId(accessibilityViewId, virtualDescendantId);
            for (int i = 0; i < infos.size(); i++) {
                AccessibilityNodeInfo info = infos.get(i);
                if (info.getSourceNodeId() == nodeId) {
                    returnFindNodeResult(
                            AccessibilityNodeInfo.obtain(info), callback, interactionId);
                    mHandler.removeMessages(
                            PrivateHandler.MSG_FIND_ACCESSIBILITY_NODE_INFO_BY_ACCESSIBILITY_ID,
                            pendingMessage.obj);
                    args.recycle();
                    break;
                }
            }
        }
        mPendingFindNodeByIdMessages.clear();
    }

    private void returnPrefetchResult(int interactionId, List<AccessibilityNodeInfo> infos,
                                      IAccessibilityInteractionConnectionCallback callback) {
        if (infos.size() > 0) {
            try {
            try {
            mViewRootImpl.mAttachInfo.mAccessibilityFetchFlags = 0;
                callback.setPrefetchAccessibilityNodeInfoResult(infos, interactionId);
            associateLeashedParentIfNeeded(info);
            applyScreenMatrixIfNeeded(info);
            adjustBoundsInScreenIfNeeded(info);
            // To avoid applyAppScaleAndMagnificationSpecIfNeeded changing the bounds of node,
            // then impact the visibility result, we need to adjust visibility before apply scale.
            adjustIsVisibleToUserIfNeeded(info, interactiveRegion);
            applyAppScaleAndMagnificationSpecIfNeeded(info, spec);
            callback.setFindAccessibilityNodeInfoResult(info, interactionId);
            } catch (RemoteException re) {
            } catch (RemoteException re) {
                /* ignore - the other side will time out */
                /* ignore - other side isn't too bothered if this doesn't arrive */
            }
        }
    }
    }

    private void updateInfoForViewportAndReturnFindNodeResult(AccessibilityNodeInfo info,
            IAccessibilityInteractionConnectionCallback callback, int interactionId,
            MagnificationSpec spec, Region interactiveRegion) {
        updateInfoForViewPort(info, spec, interactiveRegion);
        returnFindNodeResult(info, callback, interactionId);
    }
    }


    private boolean handleClickableSpanActionUiThread(
    private boolean handleClickableSpanActionUiThread(
@@ -1053,20 +1105,11 @@ public final class AccessibilityInteractionController {


        private final ArrayList<View> mTempViewList = new ArrayList<View>();
        private final ArrayList<View> mTempViewList = new ArrayList<View>();


        public void prefetchAccessibilityNodeInfos(View view, int virtualViewId, int fetchFlags,
        public void prefetchAccessibilityNodeInfos(View view, AccessibilityNodeInfo root,
                List<AccessibilityNodeInfo> outInfos, Bundle arguments) {
                int virtualViewId, int fetchFlags, List<AccessibilityNodeInfo> outInfos) {
            if (root != null) {
                AccessibilityNodeProvider provider = view.getAccessibilityNodeProvider();
                AccessibilityNodeProvider provider = view.getAccessibilityNodeProvider();
            // Determine if we'll be populating extra data
            final String extraDataRequested = (arguments == null) ? null
                    : arguments.getString(EXTRA_DATA_REQUESTED_KEY);
                if (provider == null) {
                if (provider == null) {
                AccessibilityNodeInfo root = view.createAccessibilityNodeInfo();
                if (root != null) {
                    if (extraDataRequested != null) {
                        view.addExtraDataToAccessibilityNodeInfo(
                                root, extraDataRequested, arguments);
                    }
                    outInfos.add(root);
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_PREDECESSORS) != 0) {
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_PREDECESSORS) != 0) {
                        prefetchPredecessorsOfRealNode(view, outInfos);
                        prefetchPredecessorsOfRealNode(view, outInfos);
                    }
                    }
@@ -1076,16 +1119,7 @@ public final class AccessibilityInteractionController {
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_DESCENDANTS) != 0) {
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_DESCENDANTS) != 0) {
                        prefetchDescendantsOfRealNode(view, outInfos);
                        prefetchDescendantsOfRealNode(view, outInfos);
                    }
                    }
                }
                } else {
                } else {
                final AccessibilityNodeInfo root =
                        provider.createAccessibilityNodeInfo(virtualViewId);
                if (root != null) {
                    if (extraDataRequested != null) {
                        provider.addExtraDataToAccessibilityNodeInfo(
                                virtualViewId, root, extraDataRequested, arguments);
                    }
                    outInfos.add(root);
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_PREDECESSORS) != 0) {
                    if ((fetchFlags & AccessibilityNodeInfo.FLAG_PREFETCH_PREDECESSORS) != 0) {
                        prefetchPredecessorsOfVirtualNode(root, view, provider, outInfos);
                        prefetchPredecessorsOfVirtualNode(root, view, provider, outInfos);
                    }
                    }
@@ -1096,13 +1130,19 @@ public final class AccessibilityInteractionController {
                        prefetchDescendantsOfVirtualNode(root, provider, outInfos);
                        prefetchDescendantsOfVirtualNode(root, provider, outInfos);
                    }
                    }
                }
                }
            }
                if (ENFORCE_NODE_TREE_CONSISTENT) {
                if (ENFORCE_NODE_TREE_CONSISTENT) {
                enforceNodeTreeConsistent(outInfos);
                    enforceNodeTreeConsistent(root, outInfos);
                }
            }
            }
        }
        }


        private void enforceNodeTreeConsistent(List<AccessibilityNodeInfo> nodes) {
        private boolean shouldStopPrefetching(List prefetchededInfos) {
            return mHandler.hasUserInteractiveMessagesWaiting()
                    || prefetchededInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE;
        }

        private void enforceNodeTreeConsistent(
                AccessibilityNodeInfo root, List<AccessibilityNodeInfo> nodes) {
            LongSparseArray<AccessibilityNodeInfo> nodeMap =
            LongSparseArray<AccessibilityNodeInfo> nodeMap =
                    new LongSparseArray<AccessibilityNodeInfo>();
                    new LongSparseArray<AccessibilityNodeInfo>();
            final int nodeCount = nodes.size();
            final int nodeCount = nodes.size();
@@ -1113,7 +1153,6 @@ public final class AccessibilityInteractionController {


            // If the nodes are a tree it does not matter from
            // If the nodes are a tree it does not matter from
            // which node we start to search for the root.
            // which node we start to search for the root.
            AccessibilityNodeInfo root = nodeMap.valueAt(0);
            AccessibilityNodeInfo parent = root;
            AccessibilityNodeInfo parent = root;
            while (parent != null) {
            while (parent != null) {
                root = parent;
                root = parent;
@@ -1180,9 +1219,11 @@ public final class AccessibilityInteractionController {


        private void prefetchPredecessorsOfRealNode(View view,
        private void prefetchPredecessorsOfRealNode(View view,
                List<AccessibilityNodeInfo> outInfos) {
                List<AccessibilityNodeInfo> outInfos) {
            if (shouldStopPrefetching(outInfos)) {
                return;
            }
            ViewParent parent = view.getParentForAccessibility();
            ViewParent parent = view.getParentForAccessibility();
            while (parent instanceof View
            while (parent instanceof View && !shouldStopPrefetching(outInfos)) {
                    && outInfos.size() < MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                View parentView = (View) parent;
                View parentView = (View) parent;
                AccessibilityNodeInfo info = parentView.createAccessibilityNodeInfo();
                AccessibilityNodeInfo info = parentView.createAccessibilityNodeInfo();
                if (info != null) {
                if (info != null) {
@@ -1194,6 +1235,9 @@ public final class AccessibilityInteractionController {


        private void prefetchSiblingsOfRealNode(View current,
        private void prefetchSiblingsOfRealNode(View current,
                List<AccessibilityNodeInfo> outInfos) {
                List<AccessibilityNodeInfo> outInfos) {
            if (shouldStopPrefetching(outInfos)) {
                return;
            }
            ViewParent parent = current.getParentForAccessibility();
            ViewParent parent = current.getParentForAccessibility();
            if (parent instanceof ViewGroup) {
            if (parent instanceof ViewGroup) {
                ViewGroup parentGroup = (ViewGroup) parent;
                ViewGroup parentGroup = (ViewGroup) parent;
@@ -1203,7 +1247,7 @@ public final class AccessibilityInteractionController {
                    parentGroup.addChildrenForAccessibility(children);
                    parentGroup.addChildrenForAccessibility(children);
                    final int childCount = children.size();
                    final int childCount = children.size();
                    for (int i = 0; i < childCount; i++) {
                    for (int i = 0; i < childCount; i++) {
                        if (outInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                        if (shouldStopPrefetching(outInfos)) {
                            return;
                            return;
                        }
                        }
                        View child = children.get(i);
                        View child = children.get(i);
@@ -1231,7 +1275,7 @@ public final class AccessibilityInteractionController {


        private void prefetchDescendantsOfRealNode(View root,
        private void prefetchDescendantsOfRealNode(View root,
                List<AccessibilityNodeInfo> outInfos) {
                List<AccessibilityNodeInfo> outInfos) {
            if (!(root instanceof ViewGroup)) {
            if (shouldStopPrefetching(outInfos) || !(root instanceof ViewGroup)) {
                return;
                return;
            }
            }
            HashMap<View, AccessibilityNodeInfo> addedChildren =
            HashMap<View, AccessibilityNodeInfo> addedChildren =
@@ -1242,7 +1286,7 @@ public final class AccessibilityInteractionController {
                root.addChildrenForAccessibility(children);
                root.addChildrenForAccessibility(children);
                final int childCount = children.size();
                final int childCount = children.size();
                for (int i = 0; i < childCount; i++) {
                for (int i = 0; i < childCount; i++) {
                    if (outInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                    if (shouldStopPrefetching(outInfos)) {
                        return;
                        return;
                    }
                    }
                    View child = children.get(i);
                    View child = children.get(i);
@@ -1267,7 +1311,7 @@ public final class AccessibilityInteractionController {
            } finally {
            } finally {
                children.clear();
                children.clear();
            }
            }
            if (outInfos.size() < MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
            if (!shouldStopPrefetching(outInfos)) {
                for (Map.Entry<View, AccessibilityNodeInfo> entry : addedChildren.entrySet()) {
                for (Map.Entry<View, AccessibilityNodeInfo> entry : addedChildren.entrySet()) {
                    View addedChild = entry.getKey();
                    View addedChild = entry.getKey();
                    AccessibilityNodeInfo virtualRoot = entry.getValue();
                    AccessibilityNodeInfo virtualRoot = entry.getValue();
@@ -1289,7 +1333,7 @@ public final class AccessibilityInteractionController {
            long parentNodeId = root.getParentNodeId();
            long parentNodeId = root.getParentNodeId();
            int accessibilityViewId = AccessibilityNodeInfo.getAccessibilityViewId(parentNodeId);
            int accessibilityViewId = AccessibilityNodeInfo.getAccessibilityViewId(parentNodeId);
            while (accessibilityViewId != AccessibilityNodeInfo.UNDEFINED_ITEM_ID) {
            while (accessibilityViewId != AccessibilityNodeInfo.UNDEFINED_ITEM_ID) {
                if (outInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                if (shouldStopPrefetching(outInfos)) {
                    return;
                    return;
                }
                }
                final int virtualDescendantId =
                final int virtualDescendantId =
@@ -1334,7 +1378,7 @@ public final class AccessibilityInteractionController {
                if (parent != null) {
                if (parent != null) {
                    final int childCount = parent.getChildCount();
                    final int childCount = parent.getChildCount();
                    for (int i = 0; i < childCount; i++) {
                    for (int i = 0; i < childCount; i++) {
                        if (outInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                        if (shouldStopPrefetching(outInfos)) {
                            return;
                            return;
                        }
                        }
                        final long childNodeId = parent.getChildId(i);
                        final long childNodeId = parent.getChildId(i);
@@ -1359,7 +1403,7 @@ public final class AccessibilityInteractionController {
            final int initialOutInfosSize = outInfos.size();
            final int initialOutInfosSize = outInfos.size();
            final int childCount = root.getChildCount();
            final int childCount = root.getChildCount();
            for (int i = 0; i < childCount; i++) {
            for (int i = 0; i < childCount; i++) {
                if (outInfos.size() >= MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
                if (shouldStopPrefetching(outInfos)) {
                    return;
                    return;
                }
                }
                final long childNodeId = root.getChildId(i);
                final long childNodeId = root.getChildId(i);
@@ -1369,7 +1413,7 @@ public final class AccessibilityInteractionController {
                    outInfos.add(child);
                    outInfos.add(child);
                }
                }
            }
            }
            if (outInfos.size() < MAX_ACCESSIBILITY_NODE_INFO_BATCH_SIZE) {
            if (!shouldStopPrefetching(outInfos)) {
                final int addedChildCount = outInfos.size() - initialOutInfosSize;
                final int addedChildCount = outInfos.size() - initialOutInfosSize;
                for (int i = 0; i < addedChildCount; i++) {
                for (int i = 0; i < addedChildCount; i++) {
                    AccessibilityNodeInfo child = outInfos.get(initialOutInfosSize + i);
                    AccessibilityNodeInfo child = outInfos.get(initialOutInfosSize + i);
@@ -1478,6 +1522,10 @@ public final class AccessibilityInteractionController {
        boolean hasAccessibilityCallback(Message message) {
        boolean hasAccessibilityCallback(Message message) {
            return message.what < FIRST_NO_ACCESSIBILITY_CALLBACK_MSG ? true : false;
            return message.what < FIRST_NO_ACCESSIBILITY_CALLBACK_MSG ? true : false;
        }
        }

        boolean hasUserInteractiveMessagesWaiting() {
            return hasMessagesOrCallbacks();
        }
    }
    }


    private final class AddNodeInfosForViewId implements Predicate<View> {
    private final class AddNodeInfosForViewId implements Predicate<View> {
+47 −9

File changed.

Preview size limit exceeded, changes collapsed.

+9 −0
Original line number Original line Diff line number Diff line
@@ -46,6 +46,15 @@ oneway interface IAccessibilityInteractionConnectionCallback {
    void setFindAccessibilityNodeInfosResult(in List<AccessibilityNodeInfo> infos,
    void setFindAccessibilityNodeInfosResult(in List<AccessibilityNodeInfo> infos,
        int interactionId);
        int interactionId);


    /**
     * Sets the result of a prefetch request that returns {@link AccessibilityNodeInfo}s.
     *
     * @param root The {@link AccessibilityNodeInfo} for which the prefetching is based off of.
     * @param infos The result {@link AccessibilityNodeInfo}s.
     */
    void setPrefetchAccessibilityNodeInfoResult(
        in List<AccessibilityNodeInfo> infos, int interactionId);

    /**
    /**
     * Sets the result of a request to perform an accessibility action.
     * Sets the result of a request to perform an accessibility action.
     *
     *
+3 −6
Original line number Original line Diff line number Diff line
@@ -33,9 +33,6 @@ import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.Mock;


import java.util.Arrays;
import java.util.List;

/**
/**
 * Tests for AccessibilityInteractionClient
 * Tests for AccessibilityInteractionClient
 */
 */
@@ -65,7 +62,7 @@ public class AccessibilityInteractionClientTest {
        final long accessibilityNodeId = 0x4321L;
        final long accessibilityNodeId = 0x4321L;
        AccessibilityNodeInfo nodeFromConnection = AccessibilityNodeInfo.obtain();
        AccessibilityNodeInfo nodeFromConnection = AccessibilityNodeInfo.obtain();
        nodeFromConnection.setSourceNodeId(accessibilityNodeId, windowId);
        nodeFromConnection.setSourceNodeId(accessibilityNodeId, windowId);
        mMockConnection.mInfosToReturn = Arrays.asList(nodeFromConnection);
        mMockConnection.mInfoToReturn = nodeFromConnection;
        AccessibilityInteractionClient client = AccessibilityInteractionClient.getInstance();
        AccessibilityInteractionClient client = AccessibilityInteractionClient.getInstance();
        AccessibilityNodeInfo node = client.findAccessibilityNodeInfoByAccessibilityId(
        AccessibilityNodeInfo node = client.findAccessibilityNodeInfoByAccessibilityId(
                MOCK_CONNECTION_ID, windowId, accessibilityNodeId, true, 0, null);
                MOCK_CONNECTION_ID, windowId, accessibilityNodeId, true, 0, null);
@@ -75,7 +72,7 @@ public class AccessibilityInteractionClientTest {
    }
    }


    private static class MockConnection extends AccessibilityServiceConnectionImpl {
    private static class MockConnection extends AccessibilityServiceConnectionImpl {
        List<AccessibilityNodeInfo> mInfosToReturn;
        AccessibilityNodeInfo mInfoToReturn;


        @Override
        @Override
        public String[] findAccessibilityNodeInfoByAccessibilityId(int accessibilityWindowId,
        public String[] findAccessibilityNodeInfoByAccessibilityId(int accessibilityWindowId,
@@ -83,7 +80,7 @@ public class AccessibilityInteractionClientTest {
                IAccessibilityInteractionConnectionCallback callback, int flags, long threadId,
                IAccessibilityInteractionConnectionCallback callback, int flags, long threadId,
                Bundle arguments) {
                Bundle arguments) {
            try {
            try {
                callback.setFindAccessibilityNodeInfosResult(mInfosToReturn, interactionId);
                callback.setFindAccessibilityNodeInfoResult(mInfoToReturn, interactionId);
            } catch (RemoteException e) {
            } catch (RemoteException e) {
                throw new RuntimeException(e);
                throw new RuntimeException(e);
            }
            }
+96 −89

File changed.

Preview size limit exceeded, changes collapsed.