blob: ff68cf7e8158ebc88968e9d7cf7914fc82425dc6 [file] [log] [blame]
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +01001/*
2 * Copyright (C) 2012 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
17package com.android.server.webkit;
18
Ben Murdochdc00a842014-07-17 14:55:00 +010019import android.content.BroadcastReceiver;
20import android.content.Context;
21import android.content.Intent;
22import android.content.IntentFilter;
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010023import android.os.Binder;
24import android.os.Process;
Primiano Tucci810c0522014-07-25 18:03:16 +010025import android.util.Slog;
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010026import android.webkit.IWebViewUpdateService;
Ben Murdochdc00a842014-07-17 14:55:00 +010027import android.webkit.WebViewFactory;
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010028
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010029import com.android.server.SystemService;
30
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010031/**
32 * Private service to wait for the updatable WebView to be ready for use.
33 * @hide
34 */
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010035public class WebViewUpdateService extends SystemService {
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010036
37 private static final String TAG = "WebViewUpdateService";
Primiano Tuccie76e81a2014-07-29 16:38:33 +010038 private static final int WAIT_TIMEOUT_MS = 5000; // Same as KEY_DISPATCHING_TIMEOUT.
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010039
40 private boolean mRelroReady32Bit = false;
41 private boolean mRelroReady64Bit = false;
42
Ben Murdochdc00a842014-07-17 14:55:00 +010043 private BroadcastReceiver mWebViewUpdatedReceiver;
44
45 public WebViewUpdateService(Context context) {
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010046 super(context);
47 }
48
49 @Override
50 public void onStart() {
Ben Murdochdc00a842014-07-17 14:55:00 +010051 mWebViewUpdatedReceiver = new BroadcastReceiver() {
52 @Override
53 public void onReceive(Context context, Intent intent) {
54 String webviewPackage = "package:" + WebViewFactory.getWebViewPackageName();
55 if (webviewPackage.equals(intent.getDataString())) {
56 onWebViewUpdateInstalled();
57 }
58 }
59 };
60 IntentFilter filter = new IntentFilter();
61 filter.addAction(Intent.ACTION_PACKAGE_REPLACED);
62 filter.addDataScheme("package");
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010063 getContext().registerReceiver(mWebViewUpdatedReceiver, filter);
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010064
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010065 publishBinderService("webviewupdate", new BinderService());
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +010066 }
Ben Murdochdc00a842014-07-17 14:55:00 +010067
68 private void onWebViewUpdateInstalled() {
Primiano Tuccie76e81a2014-07-29 16:38:33 +010069 Slog.d(TAG, "WebView Package updated!");
Ben Murdochdc00a842014-07-17 14:55:00 +010070
71 synchronized (this) {
72 mRelroReady32Bit = false;
73 mRelroReady64Bit = false;
74 }
Ben Murdoch5ced5022014-07-28 15:57:00 +010075 WebViewFactory.onWebViewUpdateInstalled();
Ben Murdochdc00a842014-07-17 14:55:00 +010076 }
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +010077
78 private class BinderService extends IWebViewUpdateService.Stub {
79
80 /**
81 * The shared relro process calls this to notify us that it's done trying to create a relro
82 * file. This method gets called even if the relro creation has failed or the process
83 * crashed.
84 */
85 @Override // Binder call
86 public void notifyRelroCreationCompleted(boolean is64Bit, boolean success) {
87 // Verify that the caller is either the shared relro process (nominal case) or the
88 // system server (only in the case the relro process crashes and we get here via the
89 // crashHandler).
90 if (Binder.getCallingUid() != Process.SHARED_RELRO_UID &&
91 Binder.getCallingUid() != Process.SYSTEM_UID) {
92 return;
93 }
94
95 synchronized (WebViewUpdateService.this) {
96 if (is64Bit) {
97 mRelroReady64Bit = true;
98 } else {
99 mRelroReady32Bit = true;
100 }
101 WebViewUpdateService.this.notifyAll();
102 }
103 }
104
105 /**
106 * WebViewFactory calls this to block WebView loading until the relro file is created.
107 */
108 @Override // Binder call
109 public void waitForRelroCreationCompleted(boolean is64Bit) {
Primiano Tuccie76e81a2014-07-29 16:38:33 +0100110 // The WebViewUpdateService depends on the prepareWebViewInSystemServer call, which
111 // happens later (during the PHASE_ACTIVITY_MANAGER_READY) in SystemServer.java. If
112 // another service there tries to bring up a WebView in the between, the wait below
113 // would deadlock without the check below.
114 if (Binder.getCallingPid() == Process.myPid()) {
115 throw new IllegalStateException("Cannot create a WebView from the SystemServer");
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +0100116 }
117
Primiano Tuccie76e81a2014-07-29 16:38:33 +0100118 final long NS_PER_MS = 1000000;
119 final long timeoutTimeMs = System.nanoTime() / NS_PER_MS + WAIT_TIMEOUT_MS;
120 boolean relroReady = false;
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +0100121 synchronized (WebViewUpdateService.this) {
Primiano Tuccie76e81a2014-07-29 16:38:33 +0100122 while (!relroReady) {
123 final long timeNowMs = System.nanoTime() / NS_PER_MS;
124 if (timeNowMs >= timeoutTimeMs) break;
125 try {
126 WebViewUpdateService.this.wait(timeoutTimeMs - timeNowMs);
127 } catch (InterruptedException e) {}
128 relroReady = (is64Bit ? mRelroReady64Bit : mRelroReady32Bit);
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +0100129 }
130 }
Primiano Tuccie76e81a2014-07-29 16:38:33 +0100131 if (!relroReady) Slog.w(TAG, "creating relro file timed out");
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +0100132 }
Torne (Richard Coles)4dbeb352014-07-29 19:14:24 +0100133 }
134
Torne (Richard Coles)08cfaf62014-05-08 16:07:05 +0100135}