/* * Copyright © 2017-2023 WireGuard LLC. All Rights Reserved. * SPDX-License-Identifier: Apache-2.0 */ package com.wireguard.android.util; import android.content.Context; import android.os.Build; import android.util.Log; import com.wireguard.util.NonNullForAll; import java.io.File; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.Arrays; import java.util.Collection; import java.util.HashSet; import java.util.zip.ZipEntry; import java.util.zip.ZipFile; import androidx.annotation.RestrictTo; import androidx.annotation.RestrictTo.Scope; @NonNullForAll @RestrictTo(Scope.LIBRARY_GROUP) public final class SharedLibraryLoader { private static final String TAG = "WireGuard/SharedLibraryLoader"; private SharedLibraryLoader() { } public static boolean extractLibrary(final Context context, final String libName, final File destination) throws IOException { final Collection apks = new HashSet<>(); if (context.getApplicationInfo().sourceDir != null) apks.add(context.getApplicationInfo().sourceDir); if (context.getApplicationInfo().splitSourceDirs != null) apks.addAll(Arrays.asList(context.getApplicationInfo().splitSourceDirs)); for (final String abi : Build.SUPPORTED_ABIS) { for (final String apk : apks) { try (final ZipFile zipFile = new ZipFile(new File(apk), ZipFile.OPEN_READ)) { final String mappedLibName = System.mapLibraryName(libName); final String libZipPath = "lib" + File.separatorChar + abi + File.separatorChar + mappedLibName; final ZipEntry zipEntry = zipFile.getEntry(libZipPath); if (zipEntry == null) continue; Log.d(TAG, "Extracting apk:/" + libZipPath + " to " + destination.getAbsolutePath()); try (final FileOutputStream out = new FileOutputStream(destination); final InputStream in = zipFile.getInputStream(zipEntry)) { int len; final byte[] buffer = new byte[1024 * 32]; while ((len = in.read(buffer)) != -1) { out.write(buffer, 0, len); } out.getFD().sync(); } } return true; } } return false; } public static void loadSharedLibrary(final Context context, final String libName) { Throwable noAbiException; try { System.loadLibrary(libName); return; } catch (final UnsatisfiedLinkError e) { Log.d(TAG, "Failed to load library normally, so attempting to extract from apk", e); noAbiException = e; } File f = null; try { f = File.createTempFile("lib", ".so", context.getCodeCacheDir()); if (extractLibrary(context, libName, f)) { System.load(f.getAbsolutePath()); return; } } catch (final Exception e) { Log.d(TAG, "Failed to load library apk:/" + libName, e); noAbiException = e; } finally { if (f != null) // noinspection ResultOfMethodCallIgnored f.delete(); } if (noAbiException instanceof RuntimeException) throw (RuntimeException) noAbiException; throw new RuntimeException(noAbiException); } }