Home | History | Annotate | Download | only in pm
      1 /*
      2  * Copyright (C) 2015 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 package com.android.car.pm;
     17 
     18 import android.car.content.pm.CarAppBlockingPolicy;
     19 import android.car.content.pm.ICarAppBlockingPolicy;
     20 import android.car.content.pm.ICarAppBlockingPolicySetter;
     21 import android.content.ComponentName;
     22 import android.content.Context;
     23 import android.content.Intent;
     24 import android.content.ServiceConnection;
     25 import android.content.pm.ServiceInfo;
     26 import android.os.Handler;
     27 import android.os.IBinder;
     28 import android.os.RemoteException;
     29 import android.os.UserHandle;
     30 import android.util.Log;
     31 
     32 import com.android.car.CarLog;
     33 import com.android.internal.annotations.GuardedBy;
     34 
     35 public class AppBlockingPolicyProxy implements ServiceConnection {
     36 
     37     private final CarPackageManagerService mService;
     38     private final Context mContext;
     39     private final ServiceInfo mServiceInfo;
     40     private final ICarAppBlockingPolicySetterImpl mSetter;
     41 
     42     @GuardedBy("this")
     43     private ICarAppBlockingPolicy mPolicyService = null;
     44 
     45     /**
     46      * policy not set within this time after binding will be treated as failure and will be
     47      * ignored.
     48      */
     49     private static final long TIMEOUT_MS = 5000;
     50     private static final int MAX_CRASH_RETRY = 2;
     51     @GuardedBy("this")
     52     private int mCrashCount = 0;
     53     @GuardedBy("this")
     54     private boolean mBound = false;
     55 
     56     private final Handler mHandler;
     57     private final Runnable mTimeoutRunnable = new Runnable() {
     58         @Override
     59         public void run() {
     60             Log.w(CarLog.TAG_PACKAGE, "Timeout for policy setting for service:" + mServiceInfo);
     61             disconnect();
     62             mService.onPolicyConnectionFailure(AppBlockingPolicyProxy.this);
     63         }
     64     };
     65 
     66     public AppBlockingPolicyProxy(CarPackageManagerService service, Context context,
     67             ServiceInfo serviceInfo) {
     68         mService = service;
     69         mContext = context;
     70         mServiceInfo = serviceInfo;
     71         mSetter = new ICarAppBlockingPolicySetterImpl();
     72         mHandler = new Handler(mService.getLooper());
     73     }
     74 
     75     public String getPackageName() {
     76         return mServiceInfo.packageName;
     77     }
     78 
     79     public void connect() {
     80         Intent intent = new Intent();
     81         intent.setClassName(mServiceInfo.packageName, mServiceInfo.name);
     82         mContext.bindServiceAsUser(intent, this, Context.BIND_AUTO_CREATE | Context.BIND_IMPORTANT,
     83                 UserHandle.SYSTEM);
     84         synchronized (this) {
     85             mBound = true;
     86         }
     87         mHandler.postDelayed(mTimeoutRunnable, TIMEOUT_MS);
     88     }
     89 
     90     public void disconnect() {
     91         synchronized (this) {
     92             if (!mBound) {
     93                 return;
     94             }
     95             mBound = false;
     96             mPolicyService = null;
     97         }
     98         mHandler.removeCallbacks(mTimeoutRunnable);
     99         try {
    100             mContext.unbindService(this);
    101         } catch (IllegalArgumentException e) {
    102             Log.w(CarLog.TAG_PACKAGE, "unbind", e);
    103         }
    104     }
    105 
    106     @Override
    107     public void onServiceConnected(ComponentName name, IBinder service) {
    108         ICarAppBlockingPolicy policy = null;
    109         boolean failed = false;
    110         synchronized (this) {
    111             mPolicyService = ICarAppBlockingPolicy.Stub.asInterface(service);
    112             policy = mPolicyService;
    113             if (policy == null) {
    114                 failed = true;
    115             }
    116         }
    117         if (failed) {
    118             Log.w(CarLog.TAG_PACKAGE, "Policy service connected with null binder:" + name);
    119             mService.onPolicyConnectionFailure(this);
    120             return;
    121         }
    122         try {
    123             mPolicyService.setAppBlockingPolicySetter(mSetter);
    124         } catch (RemoteException e) {
    125             // let retry handle this
    126         }
    127     }
    128 
    129     @Override
    130     public void onServiceDisconnected(ComponentName name) {
    131         boolean failed = false;
    132         synchronized (this) {
    133             mCrashCount++;
    134             if (mCrashCount > MAX_CRASH_RETRY) {
    135                 mPolicyService = null;
    136                 failed = true;
    137             }
    138         }
    139         if (failed) {
    140             Log.w(CarLog.TAG_PACKAGE, "Policy service keep crashing, giving up:" + name);
    141             mService.onPolicyConnectionFailure(this);
    142         }
    143     }
    144 
    145     @Override
    146     public String toString() {
    147         return "AppBlockingPolicyProxy [mServiceInfo=" + mServiceInfo + ", mCrashCount="
    148                 + mCrashCount + "]";
    149     }
    150 
    151     private class ICarAppBlockingPolicySetterImpl extends ICarAppBlockingPolicySetter.Stub {
    152 
    153         @Override
    154         public void setAppBlockingPolicy(CarAppBlockingPolicy policy) {
    155             mHandler.removeCallbacks(mTimeoutRunnable);
    156             if (policy == null) {
    157                 Log.w(CarLog.TAG_PACKAGE, "setAppBlockingPolicy null policy from policy service:" +
    158                         mServiceInfo);
    159             }
    160             mService.onPolicyConnectionAndSet(AppBlockingPolicyProxy.this, policy);
    161         }
    162     }
    163 }
    164