From ba127f57ae143a0c7fad5a46b327c04eb2254737 Mon Sep 17 00:00:00 2001 From: Hung-ying Tyan Date: Sat, 18 Jul 2009 02:55:56 +0800 Subject: Improve VpnSettings start-up time. * Changes + add orientation and keyboard config change in the manifest file. + remove state saving code + improve status checking time by doing it in one server connection. --- .../android/settings/vpn/AuthenticationActor.java | 22 ++- src/com/android/settings/vpn/VpnSettings.java | 158 +++++++++++---------- 2 files changed, 90 insertions(+), 90 deletions(-) (limited to 'src/com') diff --git a/src/com/android/settings/vpn/AuthenticationActor.java b/src/com/android/settings/vpn/AuthenticationActor.java index a05cfa7..0228934 100644 --- a/src/com/android/settings/vpn/AuthenticationActor.java +++ b/src/com/android/settings/vpn/AuthenticationActor.java @@ -26,6 +26,7 @@ import android.net.vpn.IVpnService; import android.net.vpn.VpnManager; import android.net.vpn.VpnProfile; import android.net.vpn.VpnState; +import android.os.ConditionVariable; import android.os.IBinder; import android.os.RemoteException; import android.text.TextUtils; @@ -42,7 +43,6 @@ import java.io.IOException; */ public class AuthenticationActor implements VpnProfileActor { private static final String TAG = AuthenticationActor.class.getName(); - private static final int ONE_SECOND = 1000; // ms private Context mContext; private VpnProfile mProfile; @@ -175,27 +175,31 @@ public class AuthenticationActor implements VpnProfileActor { //@Override public void checkStatus() { + final ConditionVariable cv = new ConditionVariable(); + cv.close(); ServiceConnection c = new ServiceConnection() { public synchronized void onServiceConnected(ComponentName className, IBinder service) { + cv.open(); try { IVpnService.Stub.asInterface(service).checkStatus(mProfile); } catch (RemoteException e) { Log.e(TAG, "checkStatus()", e); broadcastConnectivity(VpnState.IDLE); } finally { - notify(); + mContext.unbindService(this); } } public void onServiceDisconnected(ComponentName className) { - // do nothing + cv.open(); + broadcastConnectivity(VpnState.IDLE); + mContext.unbindService(this); } }; if (bindService(c)) { // wait for a second, let status propagate - wait(c, ONE_SECOND); - mContext.unbindService(c); + if (!cv.block(1000)) broadcastConnectivity(VpnState.IDLE); } } @@ -211,14 +215,6 @@ public class AuthenticationActor implements VpnProfileActor { mVpnManager.broadcastConnectivity(mProfile.getName(), s, errorCode); } - private void wait(Object o, int ms) { - synchronized (o) { - try { - o.wait(ms); - } catch (Exception e) {} - } - } - private void setSavedUsername(String name) throws IOException { if (!name.equals(mProfile.getSavedUsername())) { mProfile.setSavedUsername(name); diff --git a/src/com/android/settings/vpn/VpnSettings.java b/src/com/android/settings/vpn/VpnSettings.java index 1446164..c41a105 100644 --- a/src/com/android/settings/vpn/VpnSettings.java +++ b/src/com/android/settings/vpn/VpnSettings.java @@ -21,10 +21,13 @@ import com.android.settings.SecuritySettings; import android.app.AlertDialog; import android.app.Dialog; +import android.content.ComponentName; import android.content.BroadcastReceiver; import android.content.Context; import android.content.DialogInterface; import android.content.Intent; +import android.content.ServiceConnection; +import android.net.vpn.IVpnService; import android.net.vpn.L2tpIpsecPskProfile; import android.net.vpn.L2tpProfile; import android.net.vpn.VpnManager; @@ -33,8 +36,10 @@ import android.net.vpn.VpnState; import android.net.vpn.VpnType; import android.os.Bundle; import android.os.ConditionVariable; +import android.os.IBinder; import android.os.Parcel; import android.os.Parcelable; +import android.os.RemoteException; import android.preference.Preference; import android.preference.PreferenceActivity; import android.preference.PreferenceCategory; @@ -86,8 +91,6 @@ public class VpnSettings extends PreferenceActivity implements private static final String PROFILES_ROOT = VpnManager.PROFILES_PATH + "/"; private static final String PROFILE_OBJ_FILE = ".pobj"; - private static final String STATE_ACTIVE_ACTOR = "active_actor"; - private static final int REQUEST_ADD_OR_EDIT_PROFILE = 1; private static final int REQUEST_SELECT_VPN_TYPE = 2; @@ -113,8 +116,6 @@ public class VpnSettings extends PreferenceActivity implements private Map mVpnPreferenceMap; private List mVpnProfileList; - private int mIndexOfEditedProfile = -1; - // profile engaged in a connection private VpnProfile mActiveProfile; @@ -159,28 +160,13 @@ public class VpnSettings extends PreferenceActivity implements // listen to vpn connectivity event mVpnManager.registerConnectivityReceiver(mConnectivityReceiver); - String profileName = (savedInstanceState == null) - ? null - : savedInstanceState.getString(STATE_ACTIVE_ACTOR); retrieveVpnListFromStorage(); - if (!TextUtils.isEmpty(profileName)) { - mActiveProfile = mVpnPreferenceMap.get(profileName).mProfile; - mConnectingActor = getActor(mActiveProfile); - } else { - checkVpnConnectionStatusInBackground(); - } - } - - @Override - public void onPause() { - super.onPause(); - mStatusChecker.onPause(); + checkVpnConnectionStatusInBackground(); } @Override public void onResume() { super.onResume(); - mStatusChecker.onResume(); if ((mUnlockAction != null) && isKeystoreUnlocked()) { Runnable action = mUnlockAction; @@ -190,15 +176,6 @@ public class VpnSettings extends PreferenceActivity implements } @Override - protected synchronized void onSaveInstanceState(Bundle outState) { - if (mConnectingActor == null) return; - - Log.d(TAG, " ~~~~~ save connecting actor"); - outState.putString(STATE_ACTIVE_ACTOR, - mConnectingActor.getProfile().getName()); - } - - @Override protected void onDestroy() { super.onDestroy(); unregisterForContextMenu(getListView()); @@ -278,8 +255,6 @@ public class VpnSettings extends PreferenceActivity implements public void onClick(DialogInterface dialog, int w) { VpnProfile p = mConnectingActor.getProfile(); onIdle(); - mIndexOfEditedProfile = - mVpnProfileList.indexOf(p); startVpnEditor(p); } }) @@ -325,7 +300,6 @@ public class VpnSettings extends PreferenceActivity implements return true; case CONTEXT_MENU_EDIT_ID: - mIndexOfEditedProfile = position; startVpnEditor(p); return true; @@ -340,9 +314,6 @@ public class VpnSettings extends PreferenceActivity implements @Override protected void onActivityResult(final int requestCode, final int resultCode, final Intent data) { - final int index = mIndexOfEditedProfile; - mIndexOfEditedProfile = -1; - if ((resultCode == RESULT_CANCELED) || (data == null)) { Log.d(TAG, "no result returned by editor"); return; @@ -358,6 +329,7 @@ public class VpnSettings extends PreferenceActivity implements return; } + int index = getProfileIndexFromId(p.getId()); if (checkDuplicateName(p, index)) { final VpnProfile profile = p; Util.showErrorMessage(this, String.format( @@ -374,7 +346,6 @@ public class VpnSettings extends PreferenceActivity implements if (needKeystoreToSave(p)) { Runnable action = new Runnable() { public void run() { - mIndexOfEditedProfile = index; onActivityResult(requestCode, resultCode, data); } }; @@ -382,7 +353,7 @@ public class VpnSettings extends PreferenceActivity implements } try { - if ((index < 0) || (index >= mVpnProfileList.size())) { + if (index < 0) { addProfile(p); Util.showShortToastMessage(this, String.format( getString(R.string.vpn_profile_added), p.getName())); @@ -440,6 +411,18 @@ public class VpnSettings extends PreferenceActivity implements } } + private int getProfileIndexFromId(String id) { + int index = 0; + for (VpnProfile p : mVpnProfileList) { + if (p.getId().equals(id)) { + return index; + } else { + index++; + } + } + return -1; + } + // Replaces the profile at index in mVpnProfileList with p. // Returns true if p's name is a duplicate. private boolean checkDuplicateName(VpnProfile p, int index) { @@ -517,11 +500,16 @@ public class VpnSettings extends PreferenceActivity implements disableProfilePreferencesIfOneActive(); } + private VpnPreference addPreferenceFor(VpnProfile p) { + return addPreferenceFor(p, true); + } + // Adds a preference in mVpnListContainer - private void addPreferenceFor(VpnProfile p) { + private VpnPreference addPreferenceFor( + VpnProfile p, boolean addToContainer) { VpnPreference pref = new VpnPreference(this, p); mVpnPreferenceMap.put(p.getName(), pref); - mVpnListContainer.addPreference(pref); + if (addToContainer) mVpnListContainer.addPreference(pref); pref.setOnPreferenceClickListener( new Preference.OnPreferenceClickListener() { @@ -530,6 +518,7 @@ public class VpnSettings extends PreferenceActivity implements return true; } }); + return pref; } // index: index to mVpnProfileList @@ -790,12 +779,18 @@ public class VpnSettings extends PreferenceActivity implements return false; } }); - for (VpnProfile p : mVpnProfileList) addPreferenceFor(p); + for (VpnProfile p : mVpnProfileList) { + Preference pref = addPreferenceFor(p, false); + } disableProfilePreferencesIfOneActive(); } private void checkVpnConnectionStatusInBackground() { - mStatusChecker.check(mVpnProfileList); + new Thread(new Runnable() { + public void run() { + mStatusChecker.check(mVpnProfileList); + } + }).start(); } // A sanity check. Returns true if the profile directory name and profile ID @@ -931,49 +926,58 @@ public class VpnSettings extends PreferenceActivity implements // managing status check in a background thread private class StatusChecker { - private Set mQueue = new HashSet(); - private boolean mPaused = true; - private ConditionVariable mThreadCv = new ConditionVariable(); - - void onPause() { - mPaused = true; - mThreadCv.block(); // until the checking thread is over - } + private List mList; + + synchronized void check(final List list) { + final ConditionVariable cv = new ConditionVariable(); + cv.close(); + mVpnManager.startVpnService(); + ServiceConnection c = new ServiceConnection() { + public synchronized void onServiceConnected( + ComponentName className, IBinder binder) { + cv.open(); + + IVpnService service = IVpnService.Stub.asInterface(binder); + for (VpnProfile p : list) { + try { + service.checkStatus(p); + } catch (Throwable e) { + Log.e(TAG, " --- checkStatus(): " + p.getName(), e); + changeState(p, VpnState.IDLE); + } + } + VpnSettings.this.unbindService(this); + showPreferences(); + } - synchronized void onResume() { - start(); - } + public void onServiceDisconnected(ComponentName className) { + cv.open(); - synchronized void check(List list) { - boolean started = !mQueue.isEmpty(); - for (VpnProfile p : list) { - if (!mQueue.contains(p)) mQueue.add(p); + setDefaultState(list); + VpnSettings.this.unbindService(this); + showPreferences(); + } + }; + if (mVpnManager.bindVpnService(c)) { + if (!cv.block(1000)) { + Log.d(TAG, "checkStatus() bindService failed"); + setDefaultState(list); + } + } else { + setDefaultState(list); } - if (!started) start(); } - private synchronized VpnProfile next() { - if (mPaused || mQueue.isEmpty()) return null; - Iterator i = mQueue.iterator(); - VpnProfile p = i.next(); - i.remove(); - return p; + private void showPreferences() { + for (VpnProfile p : mVpnProfileList) { + VpnPreference pref = mVpnPreferenceMap.get(p.getName()); + mVpnListContainer.addPreference(pref); + } } - private synchronized void start() { - if (!mPaused) return; - mPaused = false; - mThreadCv.close(); - new Thread(new Runnable() { - public void run() { - while (true) { - VpnProfile p = next(); - if (p == null) break; - getActor(p).checkStatus(); - } - mThreadCv.open(); - } - }).start(); + private void setDefaultState(List list) { + for (VpnProfile p : list) changeState(p, VpnState.IDLE); + showPreferences(); } } } -- cgit v1.1