/* * Copyright (C) 2012 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.webview.chromium; import android.content.pm.PackageManager; import android.content.res.Resources; import android.app.ActivityManager; import android.app.ActivityThread; import android.content.ComponentCallbacks2; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.net.Uri; import android.os.Build; import android.os.FileUtils; import android.os.Looper; import android.os.StrictMode; import android.os.SystemProperties; import android.os.Trace; import android.util.Log; import android.webkit.CookieManager; import android.webkit.GeolocationPermissions; import android.webkit.WebIconDatabase; import android.webkit.WebStorage; import android.webkit.WebView; import android.webkit.WebViewDatabase; import android.webkit.WebViewFactory; import android.webkit.WebViewFactoryProvider; import android.webkit.WebViewProvider; import org.chromium.android_webview.AwBrowserContext; import org.chromium.android_webview.AwBrowserProcess; import org.chromium.android_webview.AwContents; import org.chromium.android_webview.AwContentsStatics; import org.chromium.android_webview.AwCookieManager; import org.chromium.android_webview.AwDevToolsServer; import org.chromium.android_webview.AwFormDatabase; import org.chromium.android_webview.AwGeolocationPermissions; import org.chromium.android_webview.AwQuotaManagerBridge; import org.chromium.android_webview.AwResource; import org.chromium.android_webview.AwSettings; import org.chromium.base.CommandLine; import org.chromium.base.MemoryPressureListener; import org.chromium.base.PathService; import org.chromium.base.PathUtils; import org.chromium.base.ThreadUtils; import org.chromium.base.TraceEvent; import org.chromium.base.library_loader.LibraryLoader; import org.chromium.base.library_loader.ProcessInitException; import org.chromium.content.app.ContentMain; import org.chromium.content.browser.ContentViewStatics; import org.chromium.content.browser.ResourceExtractor; import java.io.File; import java.lang.ref.WeakReference; import java.util.ArrayList; public class WebViewChromiumFactoryProvider implements WebViewFactoryProvider { private static final String TAG = "WebViewChromiumFactoryProvider"; private static final String CHROMIUM_PREFS_NAME = "WebViewChromiumPrefs"; private static final String VERSION_CODE_PREF = "lastVersionCodeUsed"; private static final String COMMAND_LINE_FILE = "/data/local/tmp/webview-command-line"; // Guards accees to the other members, and is notifyAll() signalled on the UI thread // when the chromium process has been started. private final Object mLock = new Object(); // Initialization guarded by mLock. private AwBrowserContext mBrowserContext; private Statics mStaticMethods; private GeolocationPermissionsAdapter mGeolocationPermissions; private CookieManagerAdapter mCookieManager; private WebIconDatabaseAdapter mWebIconDatabase; private WebStorageAdapter mWebStorage; private WebViewDatabaseAdapter mWebViewDatabase; private AwDevToolsServer mDevToolsServer; private ArrayList> mWebViewsToStart = new ArrayList>(); // Read/write protected by mLock. private boolean mStarted; private SharedPreferences mWebViewPrefs; public WebViewChromiumFactoryProvider() { ThreadUtils.setWillOverrideUiThread(); // Load chromium library. Trace.traceBegin(Trace.TRACE_TAG_WEBVIEW, "AwBrowserProcess.loadLibrary()"); AwBrowserProcess.loadLibrary(); Trace.traceEnd(Trace.TRACE_TAG_WEBVIEW); // Load glue-layer support library. System.loadLibrary("webviewchromium_plat_support"); // TODO: temporary try/catch while framework builds catch up with WebView builds. // Remove this. try { // Use shared preference to check for package downgrade. mWebViewPrefs = ActivityThread.currentApplication().getSharedPreferences( CHROMIUM_PREFS_NAME, Context.MODE_PRIVATE); int lastVersion = mWebViewPrefs.getInt(VERSION_CODE_PREF, 0); int currentVersion = WebViewFactory.getLoadedPackageInfo().versionCode; if (lastVersion > currentVersion) { // The WebView package has been downgraded since we last ran in this application. // Delete the WebView data directory's contents. String dataDir = PathUtils.getDataDirectory(ActivityThread.currentApplication()); Log.i(TAG, "WebView package downgraded from " + lastVersion + " to " + currentVersion + "; deleting contents of " + dataDir); FileUtils.deleteContents(new File(dataDir)); } if (lastVersion != currentVersion) { mWebViewPrefs.edit().putInt(VERSION_CODE_PREF, currentVersion).apply(); } } catch (NoSuchMethodError e) { Log.w(TAG, "Not doing version downgrade check as framework is too old."); } // Now safe to use WebView data directory. } private void initPlatSupportLibrary() { DrawGLFunctor.setChromiumAwDrawGLFunction(AwContents.getAwDrawGLFunction()); AwContents.setAwDrawSWFunctionTable(GraphicsUtils.getDrawSWFunctionTable()); AwContents.setAwDrawGLFunctionTable(GraphicsUtils.getDrawGLFunctionTable()); } private static void initTraceEvent() { syncATraceState(); SystemProperties.addChangeCallback(new Runnable() { @Override public void run() { syncATraceState(); } }); } private static void syncATraceState() { long enabledFlags = SystemProperties.getLong("debug.atrace.tags.enableflags", 0); TraceEvent.setATraceEnabled((enabledFlags & Trace.TRACE_TAG_WEBVIEW) != 0); } private void ensureChromiumStartedLocked(boolean onMainThread) { assert Thread.holdsLock(mLock); if (mStarted) { // Early-out for the common case. return; } Looper looper = !onMainThread ? Looper.myLooper() : Looper.getMainLooper(); Log.v(TAG, "Binding Chromium to " + (Looper.getMainLooper().equals(looper) ? "main":"background") + " looper " + looper); ThreadUtils.setUiThread(looper); if (ThreadUtils.runningOnUiThread()) { startChromiumLocked(); return; } // We must post to the UI thread to cover the case that the user has invoked Chromium // startup by using the (thread-safe) CookieManager rather than creating a WebView. ThreadUtils.postOnUiThread(new Runnable() { @Override public void run() { synchronized (mLock) { startChromiumLocked(); } } }); while (!mStarted) { try { // Important: wait() releases |mLock| the UI thread can take it :-) mLock.wait(); } catch (InterruptedException e) { // Keep trying... eventually the UI thread will process the task we sent it. } } } private void startChromiumLocked() { assert Thread.holdsLock(mLock) && ThreadUtils.runningOnUiThread(); // The post-condition of this method is everything is ready, so notify now to cover all // return paths. (Other threads will not wake-up until we release |mLock|, whatever). mLock.notifyAll(); if (mStarted) { return; } if (Build.IS_DEBUGGABLE) { // Suppress the StrictMode violation as this codepath is only hit on debugglable builds. StrictMode.ThreadPolicy oldPolicy = StrictMode.allowThreadDiskReads(); CommandLine.initFromFile(COMMAND_LINE_FILE); StrictMode.setThreadPolicy(oldPolicy); } else { CommandLine.init(null); } CommandLine cl = CommandLine.getInstance(); // TODO: currently in a relase build the DCHECKs only log. We either need to insall // a report handler with SetLogReportHandler to make them assert, or else compile // them out of the build altogether (b/8284203). Either way, so long they're // compiled in, we may as unconditionally enable them here. cl.appendSwitch("enable-dcheck"); // TODO: Remove when GL is supported by default in the upstream code. if (!cl.hasSwitch("disable-webview-gl-mode")) { cl.appendSwitch("testing-webview-gl-mode"); } // We don't need to extract any paks because for WebView, they are // in the system image. ResourceExtractor.setMandatoryPaksToExtract(""); try { LibraryLoader.ensureInitialized(); } catch(ProcessInitException e) { throw new RuntimeException("Error initializing WebView library", e); } PathService.override(PathService.DIR_MODULE, "/system/lib/"); // TODO: DIR_RESOURCE_PAKS_ANDROID needs to live somewhere sensible, // inlined here for simplicity setting up the HTMLViewer demo. Unfortunately // it can't go into base.PathService, as the native constant it refers to // lives in the ui/ layer. See ui/base/ui_base_paths.h final int DIR_RESOURCE_PAKS_ANDROID = 3003; PathService.override(DIR_RESOURCE_PAKS_ANDROID, "/system/framework/webview/paks"); // Make sure that ResourceProvider is initialized before starting the browser process. setUpResources(ActivityThread.currentApplication()); initPlatSupportLibrary(); AwBrowserProcess.start(ActivityThread.currentApplication()); if (Build.IS_DEBUGGABLE) { setWebContentsDebuggingEnabled(true); } initTraceEvent(); mStarted = true; for (WeakReference wvc : mWebViewsToStart) { WebViewChromium w = wvc.get(); if (w != null) { w.startYourEngine(); } } mWebViewsToStart.clear(); mWebViewsToStart = null; } boolean hasStarted() { return mStarted; } void startYourEngines(boolean onMainThread) { synchronized (mLock) { ensureChromiumStartedLocked(onMainThread); } } AwBrowserContext getBrowserContext() { synchronized (mLock) { return getBrowserContextLocked(); } } private AwBrowserContext getBrowserContextLocked() { assert Thread.holdsLock(mLock); assert mStarted; if (mBrowserContext == null) { mBrowserContext = new AwBrowserContext(mWebViewPrefs); } return mBrowserContext; } private void setWebContentsDebuggingEnabled(boolean enable) { if (Looper.myLooper() != ThreadUtils.getUiThreadLooper()) { throw new RuntimeException( "Toggling of Web Contents Debugging must be done on the UI thread"); } if (mDevToolsServer == null) { if (!enable) return; mDevToolsServer = new AwDevToolsServer(); } mDevToolsServer.setRemoteDebuggingEnabled(enable); } private void setUpResources(Context ctx) { ResourceRewriter.rewriteRValues(ctx); AwResource.setResources(ctx.getResources()); AwResource.setErrorPageResources(android.R.raw.loaderror, android.R.raw.nodomain); AwResource.setConfigKeySystemUuidMapping( android.R.array.config_keySystemUuidMapping); } @Override public Statics getStatics() { synchronized (mLock) { if (mStaticMethods == null) { // TODO: Optimization potential: most these methods only need the native library // loaded and initialized, not the entire browser process started. // See also http://b/7009882 ensureChromiumStartedLocked(true); mStaticMethods = new WebViewFactoryProvider.Statics() { @Override public String findAddress(String addr) { return ContentViewStatics.findAddress(addr); } @Override public String getDefaultUserAgent(Context context) { return AwSettings.getDefaultUserAgent(); } @Override public void setWebContentsDebuggingEnabled(boolean enable) { // Web Contents debugging is always enabled on debug builds. if (!Build.IS_DEBUGGABLE) { WebViewChromiumFactoryProvider.this. setWebContentsDebuggingEnabled(enable); } } // TODO enable after L release to AOSP //@Override public void clearClientCertPreferences(Runnable onCleared) { AwContentsStatics.clearClientCertPreferences(onCleared); } @Override public void freeMemoryForTests() { if (ActivityManager.isRunningInTestHarness()) { MemoryPressureListener.maybeNotifyMemoryPresure( ComponentCallbacks2.TRIM_MEMORY_COMPLETE); } } // TODO: Add @Override. public void enableSlowWholeDocumentDraw() { WebViewChromium.enableSlowWholeDocumentDraw(); } @Override public Uri[] parseFileChooserResult(int resultCode, Intent intent) { return FileChooserParamsAdapter.parseFileChooserResult(resultCode, intent); } }; } } return mStaticMethods; } @Override public WebViewProvider createWebView(WebView webView, WebView.PrivateAccess privateAccess) { WebViewChromium wvc = new WebViewChromium(this, webView, privateAccess); synchronized (mLock) { if (mWebViewsToStart != null) { mWebViewsToStart.add(new WeakReference(wvc)); } } return wvc; } @Override public GeolocationPermissions getGeolocationPermissions() { synchronized (mLock) { if (mGeolocationPermissions == null) { ensureChromiumStartedLocked(true); mGeolocationPermissions = new GeolocationPermissionsAdapter( getBrowserContextLocked().getGeolocationPermissions()); } } return mGeolocationPermissions; } @Override public CookieManager getCookieManager() { synchronized (mLock) { if (mCookieManager == null) { if (!mStarted) { // We can use CookieManager without starting Chromium; the native code // will bring up just the parts it needs to make this work on a temporary // basis until Chromium is started for real. The temporary cookie manager // needs the application context to have been set. ContentMain.initApplicationContext(ActivityThread.currentApplication()); } mCookieManager = new CookieManagerAdapter(new AwCookieManager()); } } return mCookieManager; } @Override public WebIconDatabase getWebIconDatabase() { synchronized (mLock) { if (mWebIconDatabase == null) { ensureChromiumStartedLocked(true); mWebIconDatabase = new WebIconDatabaseAdapter(); } } return mWebIconDatabase; } @Override public WebStorage getWebStorage() { synchronized (mLock) { if (mWebStorage == null) { ensureChromiumStartedLocked(true); mWebStorage = new WebStorageAdapter(AwQuotaManagerBridge.getInstance()); } } return mWebStorage; } @Override public WebViewDatabase getWebViewDatabase(Context context) { synchronized (mLock) { if (mWebViewDatabase == null) { ensureChromiumStartedLocked(true); AwBrowserContext browserContext = getBrowserContextLocked(); mWebViewDatabase = new WebViewDatabaseAdapter( browserContext.getFormDatabase(), browserContext.getHttpAuthDatabase(context)); } } return mWebViewDatabase; } }