最近在维护Neptune框架的时候,遇到了一个问题,这里简单的记录下问题原因和解决方案。
这是爱奇艺开源的两个插件化框架之一,一个是Neptune,一个是Qigsaw。
Neptune is a flexible, powerful and lightweight plugin framework for Android.
Neptune早期借鉴了百度的插件化框架,后续借鉴了滴滴推出的插件化框架方案VirtualAPK,慢慢发展成为一套较为稳定的方案。
目前服务于爱奇艺和随刻两个App中,爱奇艺主App目前有25+插件,随刻有10+。
[entrypoint_utils-inl.h:97] Inlined method resolution crossed dex file boundary: from java.lang.String com.xxx.xxx.xxx.a(android.content.Context, org.json.JSONObject) in /data/user/0/com.qiyi.video/app_pluginapp/...
这是一个跨Dex调用的常见问题:
Google在Android P中添加了新的检测项,对国内大多数应用造成了严重影响:在调用resolve inline method时,如果检测到caller与callee处于不同的dex file,会主动发起abort(inline不允许跨dex文件),导致应用出现闪退等异常问题。
art/runtime/entrypoints/entrypoint_utils-inl.h
让我们看看相关的源码:
namespace art {
inline ArtMethod* GetResolvedMethod(ArtMethod* outer_method,
const CodeInfo& code_info,
const BitTableRange& inline_infos)
REQUIRES_SHARED(Locks::mutator_lock_) {
DCHECK(!outer_method->IsObsolete());
// This method is being used by artQuickResolutionTrampoline, before it sets up
// the passed parameters in a GC friendly way. Therefore we must never be
// suspended while executing it.
ScopedAssertNoThreadSuspension sants(__FUNCTION__);
{
InlineInfo inline_info = inline_infos.back();
if (inline_info.EncodesArtMethod()) {
return inline_info.GetArtMethod();
}
uint32_t method_index = code_info.GetMethodIndexOf(inline_info);
if (inline_info.GetDexPc() == static_cast(-1)) {
// "charAt" special case. It is the only non-leaf method we inline across dex files.
ArtMethod* inlined_method = jni::DecodeArtMethod(WellKnownClasses::java_lang_String_charAt);
DCHECK_EQ(inlined_method->GetDexMethodIndex(), method_index);
return inlined_method;
}
}
// Find which method did the call in the inlining hierarchy.
ClassLinker* class_linker = Runtime::Current()->GetClassLinker();
ArtMethod* method = outer_method;
for (InlineInfo inline_info : inline_infos) {
DCHECK(!inline_info.EncodesArtMethod());
DCHECK_NE(inline_info.GetDexPc(), static_cast(-1));
uint32_t method_index = code_info.GetMethodIndexOf(inline_info);
ArtMethod* inlined_method = class_linker->LookupResolvedMethod(method_index,
method->GetDexCache(),
method->GetClassLoader());
if (UNLIKELY(inlined_method == nullptr)) {
LOG(FATAL) << "Could not find an inlined method from an .oat file: "
<< method->GetDexFile()->PrettyMethod(method_index) << " . "
<< "This must be due to duplicate classes or playing wrongly with class loaders";
UNREACHABLE();
}
DCHECK(!inlined_method->IsRuntimeMethod());
if (UNLIKELY(inlined_method->GetDexFile() != method->GetDexFile())) {
// TODO: We could permit inlining within a multi-dex oat file and the boot image,
// even going back from boot image methods to the same oat file. However, this is
// not currently implemented in the compiler. Therefore crossing dex file boundary
// indicates that the inlined definition is not the same as the one used at runtime.
bool target_sdk_at_least_p =
IsSdkVersionSetAndAtLeast(Runtime::Current()->GetTargetSdkVersion(), SdkVersion::kP);
// 这里便是抛异常的地方
LOG(target_sdk_at_least_p ? FATAL : WARNING)
<< "Inlined method resolution crossed dex file boundary: from "
<< method->PrettyMethod()
<< " in " << method->GetDexFile()->GetLocation() << "/"
<< static_cast(method->GetDexFile())
<< " to " << inlined_method->PrettyMethod()
<< " in " << inlined_method->GetDexFile()->GetLocation() << "/"
<< static_cast(inlined_method->GetDexFile()) << ". "
<< "This must be due to duplicate classes or playing wrongly with class loaders. "
<< "The runtime is in an unsafe state.";
}
method = inlined_method;
}
return method;
}
...
}
在这部分代码中关键注释如下:
TODO: We could permit inlining within a multi-dex oat file and the boot image, even going back from boot image methods to the same oat file. However, this is not currently implemented in the compiler. Therefore crossing dex file boundary indicates that the inlined definition is not the same as the one used at runtime.
这里根据Neptune的实现来分析解决方案。
SdkLibrary/src/org/qiyi/pluginlibrary/loader/PluginClassLoader.java
/**
* 插件的DexClassLoader,用来做一些"更高级"的特性,
* 比如添加插件依赖,支持multidex
*/
public class PluginClassLoader extends DexClassLoader {
// 插件的包名
private String pkgName;
// 依赖的插件的ClassLoader
private List<DexClassLoader> dependencies;
public PluginClassLoader(PluginPackageInfo packageInfo, String dexPath, String optimizedDirectory,
String librarySearchPath, ClassLoader parent) {
super(dexPath, optimizedDirectory, librarySearchPath, parent);
this.pkgName = packageInfo.getPackageName();
this.dependencies = new ArrayList<>();
MultiDex.install(packageInfo, dexPath, this);
}
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
// 根据Java ClassLoader的双亲委托模型,执行到此在parent ClassLoader中没有找到
// 类似的,我们优先在依赖的插件ClassLoader中查找
for (ClassLoader classLoader : dependencies) {
try {
Class<?> c = classLoader.loadClass(name);
if (c != null) {
// find class in the dependency
return c;
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found ini dependency class loader
}
}
// If still not found, find in this class loader
try {
return super.findClass(name);
} catch (ClassNotFoundException e) {
StringBuilder sb = new StringBuilder("tried ClassLoaders ");
if (dependencies.isEmpty()) {
sb.append("none;");
} else {
for (DexClassLoader dependency : dependencies) {
sb.append(dependency.toString());
sb.append(";");
}
}
throw new ClassNotFoundException(sb.toString(), e);
}
}
/**
* 获取插件ClassLoader对应的插件包名
*/
public String getPackageName() {
return pkgName;
}
/**
* 添加依赖的插件ClassLoader
*/
public void addDependency(DexClassLoader classLoader) {
dependencies.add(classLoader);
}
@Override
public String toString() {
String self = super.toString();
StringBuilder deps = new StringBuilder();
for (ClassLoader classLoader : dependencies) {
deps.append(classLoader.toString());
}
String parent = getParent().toString();
return "self: " + self
+ "; deps: size=" + dependencies.size()
+ ", content=" + deps
+ "; parent: " + parent;
}
}
可以看到,Neptune插件化框架的ClassLoader并没有打破双亲委派模型,优先从宿主中找相关的类,如果找不到再从插件中查找。
插件宿主的dex和插件dex中存在重复类,系统后台优化inline编译后,便会出现此问题。
由 Classloader A 加载的 class1 调用一个由 Classloader B 加载的 class2里的某个 inline 方法,将导致应用闪退。
public class PluginClassLoader extends DexClassLoader {
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
// 优先加载插件Dex,解决Android9之后虚拟机oat后禁止跨Dex调用的问题
try {
Class clazz = super.findClass(name);
if (clazz == null) {
return super.loadClass(name);
}
return clazz;
} catch (ClassNotFoundException e) {
return super.loadClass(name);
}
}
}
通告 | Android P新增检测项 应用热修复受重大影响