summaryrefslogtreecommitdiffstats
path: root/mojo/runner/android/android_handler.cc
blob: 98313e31b22ddf2a825aa109ed351cf2c4aa3ddc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "mojo/runner/android/android_handler.h"

#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/files/file_path.h"
#include "base/logging.h"
#include "base/scoped_native_library.h"
#include "jni/AndroidHandler_jni.h"
#include "mojo/application/public/cpp/application_impl.h"
#include "mojo/common/data_pipe_utils.h"
#include "mojo/public/c/system/main.h"
#include "mojo/runner/android/run_android_application_function.h"
#include "mojo/runner/host/native_application_support.h"
#include "mojo/util/filename_util.h"
#include "url/gurl.h"

using base::android::AttachCurrentThread;
using base::android::ScopedJavaLocalRef;
using base::android::ConvertJavaStringToUTF8;
using base::android::ConvertUTF8ToJavaString;
using base::android::GetApplicationContext;

namespace mojo {
namespace runner {

namespace {

// This function loads the application library, sets the application context and
// thunks and calls into the application MojoMain. To ensure that the thunks are
// set correctly we keep it in the Mojo shell .so and pass the function pointer
// to the helper libbootstrap.so.
void RunAndroidApplication(JNIEnv* env,
                           jobject j_context,
                           const base::FilePath& app_path,
                           jint j_handle) {
  InterfaceRequest<Application> application_request =
      MakeRequest<Application>(MakeScopedHandle(MessagePipeHandle(j_handle)));

  // Load the library, so that we can set the application context there if
  // needed.
  // TODO(vtl): We'd use a ScopedNativeLibrary, but it doesn't have .get()!
  base::NativeLibrary app_library = LoadNativeApplication(app_path);
  if (!app_library)
    return;

  // Set the application context if needed. Most applications will need to
  // access the Android ApplicationContext in which they are run. If the
  // application library exports the InitApplicationContext function, we will
  // set it there.
  const char* init_application_context_name = "InitApplicationContext";
  typedef void (*InitApplicationContextFn)(
      const base::android::JavaRef<jobject>&);
  InitApplicationContextFn init_application_context =
      reinterpret_cast<InitApplicationContextFn>(
          base::GetFunctionPointerFromNativeLibrary(
              app_library, init_application_context_name));
  if (init_application_context) {
    base::android::ScopedJavaLocalRef<jobject> scoped_context(env, j_context);
    init_application_context(scoped_context);
  }

  // Run the application.
  RunNativeApplication(app_library, application_request.Pass());
  // TODO(vtl): See note about unloading and thread-local destructors above
  // declaration of |LoadNativeApplication()|.
  base::UnloadNativeLibrary(app_library);
}

// Returns true if |url| denotes a cached app. If true |app_dir| is set to the
// path of the directory for the app and |path_to_mojo| the path of the app's
// .mojo file.
bool IsCachedApp(JNIEnv* env,
                 const GURL& url,
                 base::FilePath* app_dir,
                 base::FilePath* path_to_mojo) {
  ScopedJavaLocalRef<jstring> j_cached_apps_dir =
      Java_AndroidHandler_getCachedAppsDir(env, GetApplicationContext());
  const base::FilePath cached_apps_fp(
      ConvertJavaStringToUTF8(env, j_cached_apps_dir.obj()));
  const std::string cached_apps(util::FilePathToFileURL(cached_apps_fp).spec());
  const std::string response_url(GURL(url).spec());
  if (response_url.size() <= cached_apps.size() ||
      cached_apps.compare(0u, cached_apps.size(), response_url, 0u,
                          cached_apps.size()) != 0) {
    return false;
  }

  const std::string mojo_suffix(".mojo");
  // app_rel_path is either something like html_viewer/html_viewer.mojo, or
  // html_viewer.mojo, depending upon whether the app has a package.
  const std::string app_rel_path(response_url.substr(cached_apps.size() + 1));
  const size_t slash_index = app_rel_path.find('/');
  if (slash_index != std::string::npos) {
    const std::string tail =
        app_rel_path.substr(slash_index + 1, std::string::npos);
    const std::string head = app_rel_path.substr(0, slash_index);
    if (head.find('/') != std::string::npos ||
        tail.size() <= mojo_suffix.size() ||
        tail.compare(tail.size() - mojo_suffix.size(), tail.size(),
                     mojo_suffix) != 0) {
      return false;
    }
    *app_dir = cached_apps_fp.Append(head);
    *path_to_mojo = app_dir->Append(tail);
    return true;
  }
  if (app_rel_path.find('/') != std::string::npos ||
      app_rel_path.size() <= mojo_suffix.size() ||
      app_rel_path.compare(app_rel_path.size() - mojo_suffix.size(),
                           mojo_suffix.size(), mojo_suffix) != 0) {
    return false;
  }

  *app_dir = cached_apps_fp.Append(
      app_rel_path.substr(0, app_rel_path.size() - mojo_suffix.size()));
  *path_to_mojo = cached_apps_fp.Append(app_rel_path);
  return true;
}

}  // namespace

AndroidHandler::AndroidHandler() : content_handler_factory_(this) {
}

AndroidHandler::~AndroidHandler() {
}

void AndroidHandler::RunApplication(
    InterfaceRequest<Application> application_request,
    URLResponsePtr response) {
  JNIEnv* env = AttachCurrentThread();
  RunAndroidApplicationFn run_android_application_fn = &RunAndroidApplication;
  if (!response->url.is_null()) {
    base::FilePath internal_app_path;
    base::FilePath path_to_mojo;
    if (IsCachedApp(env, GURL(response->url), &internal_app_path,
                    &path_to_mojo)) {
      ScopedJavaLocalRef<jstring> j_internal_app_path(
          ConvertUTF8ToJavaString(env, internal_app_path.value()));
      ScopedJavaLocalRef<jstring> j_path_to_mojo(
          ConvertUTF8ToJavaString(env, path_to_mojo.value()));
      Java_AndroidHandler_bootstrapCachedApp(
          env, GetApplicationContext(), j_path_to_mojo.obj(),
          j_internal_app_path.obj(),
          application_request.PassMessagePipe().release().value(),
          reinterpret_cast<jlong>(run_android_application_fn));
      return;
    }
  }
  ScopedJavaLocalRef<jstring> j_archive_path =
      Java_AndroidHandler_getNewTempArchivePath(env, GetApplicationContext());
  base::FilePath archive_path(
      ConvertJavaStringToUTF8(env, j_archive_path.obj()));

  common::BlockingCopyToFile(response->body.Pass(), archive_path);
  Java_AndroidHandler_bootstrap(
      env, GetApplicationContext(), j_archive_path.obj(),
      application_request.PassMessagePipe().release().value(),
      reinterpret_cast<jlong>(run_android_application_fn));
}

void AndroidHandler::Initialize(ApplicationImpl* app) {
}

bool AndroidHandler::ConfigureIncomingConnection(
    ApplicationConnection* connection) {
  connection->AddService(&content_handler_factory_);
  return true;
}

bool RegisterAndroidHandlerJni(JNIEnv* env) {
  return RegisterNativesImpl(env);
}

}  // namespace runner
}  // namespace mojo