1 /* 2 * Copyright (C) 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.server.am; 18 19 import android.util.ArrayMap; 20 import android.util.TimeUtils; 21 22 import com.android.internal.annotations.GuardedBy; 23 24 import java.io.PrintWriter; 25 import java.util.ArrayList; 26 27 /** 28 * The state info of all content providers in the process. 29 */ 30 final class ProcessProviderRecord { 31 final ProcessRecord mApp; 32 private final ActivityManagerService mService; 33 34 /** 35 * The last time someone else was using a provider in this process. 36 */ 37 private long mLastProviderTime = Long.MIN_VALUE; 38 39 /** 40 * class (String) -> ContentProviderRecord. 41 */ 42 private final ArrayMap<String, ContentProviderRecord> mPubProviders = new ArrayMap<>(); 43 44 /** 45 * All ContentProviderRecord process is using. 46 */ 47 private final ArrayList<ContentProviderConnection> mConProviders = new ArrayList<>(); 48 getLastProviderTime()49 long getLastProviderTime() { 50 return mLastProviderTime; 51 } 52 setLastProviderTime(long lastProviderTime)53 void setLastProviderTime(long lastProviderTime) { 54 mLastProviderTime = lastProviderTime; 55 } 56 hasProvider(String name)57 boolean hasProvider(String name) { 58 return mPubProviders.containsKey(name); 59 } 60 getProvider(String name)61 ContentProviderRecord getProvider(String name) { 62 return mPubProviders.get(name); 63 } 64 numberOfProviders()65 int numberOfProviders() { 66 return mPubProviders.size(); 67 } 68 getProviderAt(int index)69 ContentProviderRecord getProviderAt(int index) { 70 return mPubProviders.valueAt(index); 71 } 72 installProvider(String name, ContentProviderRecord provider)73 void installProvider(String name, ContentProviderRecord provider) { 74 mPubProviders.put(name, provider); 75 } 76 removeProvider(String name)77 void removeProvider(String name) { 78 mPubProviders.remove(name); 79 } 80 ensureProviderCapacity(int capacity)81 void ensureProviderCapacity(int capacity) { 82 mPubProviders.ensureCapacity(capacity); 83 } 84 numberOfProviderConnections()85 int numberOfProviderConnections() { 86 return mConProviders.size(); 87 } 88 getProviderConnectionAt(int index)89 ContentProviderConnection getProviderConnectionAt(int index) { 90 return mConProviders.get(index); 91 } 92 addProviderConnection(ContentProviderConnection connection)93 void addProviderConnection(ContentProviderConnection connection) { 94 mConProviders.add(connection); 95 } 96 removeProviderConnection(ContentProviderConnection connection)97 boolean removeProviderConnection(ContentProviderConnection connection) { 98 return mConProviders.remove(connection); 99 } 100 ProcessProviderRecord(ProcessRecord app)101 ProcessProviderRecord(ProcessRecord app) { 102 mApp = app; 103 mService = app.mService; 104 } 105 106 /** 107 * @return Should the process restart or not. 108 */ 109 @GuardedBy("mService") onCleanupApplicationRecordLocked(boolean allowRestart)110 boolean onCleanupApplicationRecordLocked(boolean allowRestart) { 111 boolean restart = false; 112 // Remove published content providers. 113 for (int i = mPubProviders.size() - 1; i >= 0; i--) { 114 final ContentProviderRecord cpr = mPubProviders.valueAt(i); 115 if (cpr.proc != mApp) { 116 // If the hosting process record isn't really us, bail out 117 continue; 118 } 119 final boolean alwaysRemove = mApp.mErrorState.isBad() || !allowRestart; 120 final boolean inLaunching = mService.mCpHelper 121 .removeDyingProviderLocked(mApp, cpr, alwaysRemove); 122 if (!alwaysRemove && inLaunching && cpr.hasConnectionOrHandle()) { 123 // We left the provider in the launching list, need to 124 // restart it. 125 restart = true; 126 } 127 128 cpr.provider = null; 129 cpr.setProcess(null); 130 } 131 mPubProviders.clear(); 132 133 // Take care of any launching providers waiting for this process. 134 if (mService.mCpHelper.cleanupAppInLaunchingProvidersLocked(mApp, false)) { 135 mService.mProcessList.noteProcessDiedLocked(mApp); 136 restart = true; 137 } 138 139 // Unregister from connected content providers. 140 if (!mConProviders.isEmpty()) { 141 for (int i = mConProviders.size() - 1; i >= 0; i--) { 142 final ContentProviderConnection conn = mConProviders.get(i); 143 conn.provider.connections.remove(conn); 144 mService.stopAssociationLocked(mApp.uid, mApp.processName, conn.provider.uid, 145 conn.provider.appInfo.longVersionCode, conn.provider.name, 146 conn.provider.info.processName); 147 } 148 mConProviders.clear(); 149 } 150 151 return restart; 152 } 153 dump(PrintWriter pw, String prefix, long nowUptime)154 void dump(PrintWriter pw, String prefix, long nowUptime) { 155 if (mLastProviderTime > 0) { 156 pw.print(prefix); pw.print("lastProviderTime="); 157 TimeUtils.formatDuration(mLastProviderTime, nowUptime, pw); 158 pw.println(); 159 } 160 if (mPubProviders.size() > 0) { 161 pw.print(prefix); pw.println("Published Providers:"); 162 for (int i = 0, size = mPubProviders.size(); i < size; i++) { 163 pw.print(prefix); pw.print(" - "); pw.println(mPubProviders.keyAt(i)); 164 pw.print(prefix); pw.print(" -> "); pw.println(mPubProviders.valueAt(i)); 165 } 166 } 167 if (mConProviders.size() > 0) { 168 pw.print(prefix); pw.println("Connected Providers:"); 169 for (int i = 0, size = mConProviders.size(); i < size; i++) { 170 pw.print(prefix); pw.print(" - "); 171 pw.println(mConProviders.get(i).toShortString()); 172 } 173 } 174 } 175 } 176