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

Commit f1178698 authored by Lee Shombert's avatar Lee Shombert
Browse files

New watchables for PackageManagerService

Bug: 161323622

Add WatchedArrayList, WatchedArraySet, and WatchedSparseIntArray to
the server utilities.  Test code is added to the frameworks test.

The watched "collections" offer some consistent APIs.  Every
collection includes a copy constructor for itself and for the
unwatched type.  Every collection offers a copyIn() from, and
copyOut() to, the unwatched type, to support runtime conversion.

Add the Watched annotation interface.  This annotation marks
attributes that are watched for change detection; generally, any
change invalidates a snapshot.  It is also generally true that Watched
attributes must be included in the snapshot.

A bug in Snapshots.java is also fixed.

Test: atest
 * FrameworksServicesTests:WatcherTest

Change-Id: I9b6ad0b3f6a901e77c342c5a3e45c86bed510db6
parent 677c97dd
Loading
Loading
Loading
Loading
+2 −2
Original line number Diff line number Diff line
@@ -443,7 +443,7 @@ public class AppsFilter implements Watchable, Snappable {
        }
        final StateProvider stateProvider = command -> {
            synchronized (injector.getLock()) {
                command.currentState(injector.getSettings().getPackagesLocked().untrackedMap(),
                command.currentState(injector.getSettings().getPackagesLocked().untrackedStorage(),
                        injector.getUserManagerInternal().getUserInfos());
            }
        };
@@ -979,7 +979,7 @@ public class AppsFilter implements Watchable, Snappable {
    @Nullable
    SparseArray<int[]> getVisibilityAllowList(PackageSetting setting, int[] users,
            WatchedArrayMap<String, PackageSetting> existingSettings) {
        return getVisibilityAllowList(setting, users, existingSettings.untrackedMap());
        return getVisibilityAllowList(setting, users, existingSettings.untrackedStorage());
    }

    /**
+1 −1
Original line number Diff line number Diff line
@@ -471,7 +471,7 @@ public final class Settings implements Watchable, Snappable {
    private final File mSystemDir;

    public final KeySetManagerService mKeySetManagerService =
            new KeySetManagerService(mPackages.untrackedMap());
            new KeySetManagerService(mPackages.untrackedStorage());

    /** Settings and other information about permissions */
    final LegacyPermissionSettings mPermissions;
+1 −1
Original line number Diff line number Diff line
@@ -100,7 +100,7 @@ public class WatchableImpl implements Watchable {

    /**
     * Freeze the {@link Watchable}.
     **/
     */
    public void seal() {
        synchronized (mObservers) {
            mSealed = true;
+32 −0
Original line number Diff line number Diff line
/*
 * Copyright (C) 2020 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.server.utils;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * Annotation type to mark an attribute that is monitored for change detection and
 * snapshot creation.
 * TODO(b/176923052) Automate validation of @Watchable attributes.
 */
@Target({ ElementType.FIELD })
@Retention(RetentionPolicy.CLASS)
public @interface Watched {
}
+416 −0
Original line number Diff line number Diff line
/*
 * Copyright (C) 2020 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.server.utils;

import android.annotation.NonNull;
import android.annotation.Nullable;

import java.util.ArrayList;
import java.util.Collection;

/**
 * WatchedArrayMap is an {@link android.util.ArrayMap} that can report changes to itself.  If its
 * values are {@link Watchable} then the WatchedArrayMap will also report changes to the values.
 * A {@link Watchable} is notified only once, no matter how many times it is stored in the array.
 * @param <E> The element type, stored in the array.
 */
public class WatchedArrayList<E> extends WatchableImpl
        implements Snappable {

    // The storage
    private final ArrayList<E> mStorage;

    // If true, the array is watching its children
    private volatile boolean mWatching = false;

    // The local observer
    private final Watcher mObserver = new Watcher() {
            @Override
            public void onChange(@Nullable Watchable what) {
                WatchedArrayList.this.dispatchChange(what);
            }
        };

    /**
     * A convenience function called when the elements are added to or removed from the storage.
     * The watchable is always {@link this}.
     */
    private void onChanged() {
        dispatchChange(this);
    }

    /**
     * A convenience function.  Register the object if it is {@link Watchable} and if the
     * array is currently watching.  Note that the watching flag must be true if this
     * function is to succeed.  Also note that if this is called with the same object
     * twice, <this> is only registered once.
     */
    private void registerChild(Object o) {
        if (mWatching && o instanceof Watchable) {
            ((Watchable) o).registerObserver(mObserver);
        }
    }

    /**
     * A convenience function.  Unregister the object if it is {@link Watchable} and if the
     * array is currently watching.  This unconditionally removes the object from the
     * registered list.
     */
    private void unregisterChild(Object o) {
        if (mWatching && o instanceof Watchable) {
            ((Watchable) o).unregisterObserver(mObserver);
        }
    }

    /**
     * A convenience function.  Unregister the object if it is {@link Watchable}, if the
     * array is currently watching, and if there are no other instances of this object in
     * the storage.  Note that the watching flag must be true if this function is to
     * succeed.  The object must already have been removed from the storage before this
     * method is called.
     */
    private void unregisterChildIf(Object o) {
        if (mWatching && o instanceof Watchable) {
            if (!mStorage.contains(o)) {
                ((Watchable) o).unregisterObserver(mObserver);
            }
        }
    }

    /**
     * Register a {@link Watcher} with the array.  If this is the first Watcher than any
     * array values that are {@link Watchable} are registered to the array itself.
     */
    @Override
    public void registerObserver(@NonNull Watcher observer) {
        super.registerObserver(observer);
        if (registeredObserverCount() == 1) {
            // The watching flag must be set true before any children are registered.
            mWatching = true;
            final int end = mStorage.size();
            for (int i = 0; i < end; i++) {
                registerChild(mStorage.get(i));
            }
        }
    }

    /**
     * Unregister a {@link Watcher} from the array.  If this is the last Watcher than any
     * array values that are {@link Watchable} are unregistered to the array itself.
     */
    @Override
    public void unregisterObserver(@NonNull Watcher observer) {
        super.unregisterObserver(observer);
        if (registeredObserverCount() == 0) {
            final int end = mStorage.size();
            for (int i = 0; i < end; i++) {
                unregisterChild(mStorage.get(i));
            }
            // The watching flag must be true while children are unregistered.
            mWatching = false;
        }
    }

    /**
     * Create a new empty {@link WatchedArrayList}.  The default capacity of an array map
     * is 0, and will grow once items are added to it.
     */
    public WatchedArrayList() {
        this(0);
    }

    /**
     * Create a new {@link WatchedArrayList} with a given initial capacity.
     */
    public WatchedArrayList(int capacity) {
        mStorage = new ArrayList<E>(capacity);
    }

    /**
     * Create a new {@link WatchedArrayList} with the content of the collection.
     */
    public WatchedArrayList(@Nullable Collection<? extends E> c) {
        mStorage = new ArrayList<E>();
        if (c != null) {
            // There is no need to register children because the WatchedArrayList starts
            // life unobserved.
            mStorage.addAll(c);
        }
    }

    /**
     * Create a {@link WatchedArrayList} from an {@link ArrayList}
     */
    public WatchedArrayList(@NonNull ArrayList<E> c) {
        mStorage = new ArrayList<>(c);
    }

    /**
     * Create a {@link WatchedArrayList} from an {@link WatchedArrayList}
     */
    public WatchedArrayList(@NonNull WatchedArrayList<E> c) {
        mStorage = new ArrayList<>(c.mStorage);
    }

    /**
     * Make <this> a copy of src.  Any data in <this> is discarded.
     */
    public void copyFrom(@NonNull ArrayList<E> src) {
        clear();
        final int end = src.size();
        mStorage.ensureCapacity(end);
        for (int i = 0; i < end; i++) {
            add(src.get(i));
        }
    }

    /**
     * Make dst a copy of <this>.  Any previous data in dst is discarded.
     */
    public void copyTo(@NonNull ArrayList<E> dst) {
        dst.clear();
        final int end = size();
        dst.ensureCapacity(end);
        for (int i = 0; i < end; i++) {
            dst.add(get(i));
        }
    }

    /**
     * Return the underlying storage.  This breaks the wrapper but is necessary when
     * passing the array to distant methods.
     */
    public ArrayList<E> untrackedStorage() {
        return mStorage;
    }

    /**
     * Append the specified element to the end of the list
     */
    public boolean add(E value) {
        final boolean result = mStorage.add(value);
        registerChild(value);
        onChanged();
        return result;
    }

    /**
     * Insert the element into the list
     */
    public void add(int index, E value) {
        mStorage.add(index, value);
        registerChild(value);
        onChanged();
    }

    /**
     * Append the elements of the collection to the list.
     */
    public boolean addAll(Collection<? extends E> c) {
        if (c.size() > 0) {
            for (E e: c) {
                mStorage.add(e);
            }
            onChanged();
            return true;
        } else {
            return false;
        }
    }

    /**
     * Insert the elements of the collection into the list at the index.
     */
    public boolean addAll(int index, Collection<? extends E> c) {
        if (c.size() > 0) {
            for (E e: c) {
                mStorage.add(index++, e);
            }
            onChanged();
            return true;
        } else {
            return false;
        }
    }


    /**
     * Remove all elements from the list.
     */
    public void clear() {
        // The storage cannot be simply cleared.  Each element in the storage must be
        // unregistered.  Deregistration is only needed if the array is actually
        // watching.
        if (mWatching) {
            final int end = mStorage.size();
            for (int i = 0; i < end; i++) {
                unregisterChild(mStorage.get(i));
            }
        }
        mStorage.clear();
        onChanged();
    }

    /**
     * Return true if the object is in the array.
     */
    public boolean contains(Object o) {
        return mStorage.contains(o);
    }

    /**
     * Ensure capacity.
     */
    public void ensureCapacity(int min) {
        mStorage.ensureCapacity(min);
    }

    /**
     * Retrieve the element at the specified index.
     */
    public E get(int index) {
        return mStorage.get(index);
    }

    /**
     * Return the index of the object.  -1 is returned if the object is not in the list.
     */
    public int indexOf(Object o) {
        return mStorage.indexOf(o);
    }

    /**
     * True if the list has no elements
     */
    public boolean isEmpty() {
        return mStorage.isEmpty();
    }

    /**
     * Return the index of the last occurrence of the object.
     */
    public int lastIndexOf(Object o) {
        return mStorage.lastIndexOf(o);
    }

    /**
     * Remove and return the element at the specified position.
     */
    public E remove(int index) {
        final E result = mStorage.remove(index);
        unregisterChildIf(result);
        onChanged();
        return result;
    }

    /**
     * Remove the first occurrence of the object in the list.  Return true if the object
     * was actually in the list and false otherwise.
     */
    public boolean remove(Object o) {
        if (mStorage.remove(o)) {
            unregisterChildIf(o);
            onChanged();
            return true;
        }
        return false;
    }

    /**
     * Replace the object at the index.
     */
    public E set(int index, E value) {
        final E result = mStorage.set(index, value);
        if (value != result) {
            unregisterChildIf(result);
            registerChild(value);
            onChanged();
        }
        return result;
    }

    /**
     * Return the number of elements in the list.
     */
    public int size() {
        return mStorage.size();
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public boolean equals(@Nullable Object o) {
        if (o instanceof WatchedArrayList) {
            WatchedArrayList w = (WatchedArrayList) o;
            return mStorage.equals(w.mStorage);
        } else {
            return false;
        }
    }

    /**
     * {@inheritDoc}
     */
    @Override
    public int hashCode() {
        return mStorage.hashCode();
    }

    /**
     * Create a copy of the array.  If the element is a subclass of Snapper then the copy
     * contains snapshots of the elements.  Otherwise the copy contains references to the
     * elements.  The returned snapshot is immutable.
     * @return A new array whose elements are the elements of <this>.
     */
    public WatchedArrayList<E> snapshot() {
        WatchedArrayList<E> l = new WatchedArrayList<>(size());
        snapshot(l, this);
        return l;
    }

    /**
     * Make <this> a snapshot of the argument.  Note that <this> is immutable when the
     * method returns.  <this> must be empty when the function is called.
     * @param r The source array, which is copied into <this>
     */
    public void snapshot(@NonNull WatchedArrayList<E> r) {
        snapshot(this, r);
    }

    /**
     * Make the destination a copy of the source.  If the element is a subclass of Snapper then the
     * copy contains snapshots of the elements.  Otherwise the copy contains references to the
     * elements.  The destination must be initially empty.  Upon return, the destination is
     * immutable.
     * @param dst The destination array.  It must be empty.
     * @param src The source array.  It is not modified.
     */
    public static <E> void snapshot(@NonNull WatchedArrayList<E> dst,
            @NonNull WatchedArrayList<E> src) {
        if (dst.size() != 0) {
            throw new IllegalArgumentException("snapshot destination is not empty");
        }
        final int end = src.size();
        dst.mStorage.ensureCapacity(end);
        for (int i = 0; i < end; i++) {
            final E val = Snapshots.maybeSnapshot(src.get(i));
            dst.add(i, val);
        }
        dst.seal();
    }
}
Loading