GoBackend: integrate into app
Signed-off-by: Jason A. Donenfeld <Jason@zx2c4.com>
This commit is contained in:
parent
b923f7bc57
commit
0ea6f73332
3
.gitmodules
vendored
3
.gitmodules
vendored
@ -4,3 +4,6 @@
|
|||||||
[submodule "app/tools/wireguard"]
|
[submodule "app/tools/wireguard"]
|
||||||
path = app/tools/wireguard
|
path = app/tools/wireguard
|
||||||
url = https://git.zx2c4.com/WireGuard
|
url = https://git.zx2c4.com/WireGuard
|
||||||
|
[submodule "app/tools/wireguard-go"]
|
||||||
|
path = app/tools/wireguard-go
|
||||||
|
url = https://git.zx2c4.com/wireguard-go
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
# Android GUI for [WireGuard](https://www.wireguard.com/)
|
# Android GUI for [WireGuard](https://www.wireguard.com/)
|
||||||
|
|
||||||
##### [Test this app on the Play Store](https://play.google.com/apps/testing/com.wireguard.android).
|
### [Test this app on the Play Store](https://play.google.com/apps/testing/com.wireguard.android).
|
||||||
|
|
||||||
This is a work in progress Android GUI for [WireGuard](https://www.wireguard.com/). The ultimate goal is to [opportunistically use the kernel implementation](https://git.zx2c4.com/android_kernel_wireguard/about/), and fallback to using the non-root userspace implementation. At the time of writing, this only supports using the kernel module, but this should change in the near future.
|
|
||||||
|
|
||||||
|
This is an Android GUI for [WireGuard](https://www.wireguard.com/). It [opportunistically uses the kernel implementation](https://git.zx2c4.com/android_kernel_wireguard/about/), and falls back to using the non-root [userspace implementation](https://git.zx2c4.com/wireguard-go/about/).
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@ import android.os.Looper;
|
|||||||
import android.preference.PreferenceManager;
|
import android.preference.PreferenceManager;
|
||||||
|
|
||||||
import com.wireguard.android.backend.Backend;
|
import com.wireguard.android.backend.Backend;
|
||||||
|
import com.wireguard.android.backend.GoBackend;
|
||||||
import com.wireguard.android.backend.WgQuickBackend;
|
import com.wireguard.android.backend.WgQuickBackend;
|
||||||
import com.wireguard.android.configStore.ConfigStore;
|
import com.wireguard.android.configStore.ConfigStore;
|
||||||
import com.wireguard.android.configStore.FileConfigStore;
|
import com.wireguard.android.configStore.FileConfigStore;
|
||||||
@ -16,6 +17,7 @@ import com.wireguard.android.util.AsyncWorker;
|
|||||||
import com.wireguard.android.util.RootShell;
|
import com.wireguard.android.util.RootShell;
|
||||||
import com.wireguard.android.util.ToolsInstaller;
|
import com.wireguard.android.util.ToolsInstaller;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
import javax.inject.Qualifier;
|
import javax.inject.Qualifier;
|
||||||
@ -56,6 +58,8 @@ public class Application extends android.app.Application {
|
|||||||
ToolsInstaller getToolsInstaller();
|
ToolsInstaller getToolsInstaller();
|
||||||
|
|
||||||
TunnelManager getTunnelManager();
|
TunnelManager getTunnelManager();
|
||||||
|
|
||||||
|
Class getBackendType();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Qualifier
|
@Qualifier
|
||||||
@ -83,7 +87,16 @@ public class Application extends android.app.Application {
|
|||||||
public static Backend getBackend(@ApplicationContext final Context context,
|
public static Backend getBackend(@ApplicationContext final Context context,
|
||||||
final RootShell rootShell,
|
final RootShell rootShell,
|
||||||
final ToolsInstaller toolsInstaller) {
|
final ToolsInstaller toolsInstaller) {
|
||||||
return new WgQuickBackend(context, rootShell, toolsInstaller);
|
if (new File("/sys/module/wireguard").exists())
|
||||||
|
return new WgQuickBackend(context, rootShell, toolsInstaller);
|
||||||
|
else
|
||||||
|
return new GoBackend(context);
|
||||||
|
}
|
||||||
|
|
||||||
|
@ApplicationScope
|
||||||
|
@Provides
|
||||||
|
public static Class getBackendType(final Backend backend) {
|
||||||
|
return backend.getClass();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ApplicationScope
|
@ApplicationScope
|
||||||
|
@ -2,9 +2,12 @@ package com.wireguard.android.activity;
|
|||||||
|
|
||||||
import android.app.Activity;
|
import android.app.Activity;
|
||||||
import android.os.Bundle;
|
import android.os.Bundle;
|
||||||
|
import android.preference.Preference;
|
||||||
import android.preference.PreferenceFragment;
|
import android.preference.PreferenceFragment;
|
||||||
|
|
||||||
|
import com.wireguard.android.Application;
|
||||||
import com.wireguard.android.R;
|
import com.wireguard.android.R;
|
||||||
|
import com.wireguard.android.backend.WgQuickBackend;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Interface for changing application-global persistent settings.
|
* Interface for changing application-global persistent settings.
|
||||||
@ -26,6 +29,11 @@ public class SettingsActivity extends Activity {
|
|||||||
public void onCreate(final Bundle savedInstanceState) {
|
public void onCreate(final Bundle savedInstanceState) {
|
||||||
super.onCreate(savedInstanceState);
|
super.onCreate(savedInstanceState);
|
||||||
addPreferencesFromResource(R.xml.preferences);
|
addPreferencesFromResource(R.xml.preferences);
|
||||||
|
if (Application.getComponent().getBackendType() != WgQuickBackend.class) {
|
||||||
|
final Preference toolsInstaller =
|
||||||
|
getPreferenceManager().findPreference("tools_installer");
|
||||||
|
getPreferenceScreen().removePreference(toolsInstaller);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
125
app/src/main/java/com/wireguard/android/backend/GoBackend.java
Normal file
125
app/src/main/java/com/wireguard/android/backend/GoBackend.java
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
package com.wireguard.android.backend;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.support.v4.util.ArraySet;
|
||||||
|
import android.util.Log;
|
||||||
|
|
||||||
|
import com.wireguard.android.model.Tunnel;
|
||||||
|
import com.wireguard.android.model.Tunnel.State;
|
||||||
|
import com.wireguard.android.model.Tunnel.Statistics;
|
||||||
|
import com.wireguard.config.Config;
|
||||||
|
import com.wireguard.config.Interface;
|
||||||
|
import com.wireguard.config.Peer;
|
||||||
|
import com.wireguard.crypto.KeyEncoding;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Formatter;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
public final class GoBackend implements Backend {
|
||||||
|
private static final String TAG = "WireGuard/" + GoBackend.class.getSimpleName();
|
||||||
|
|
||||||
|
static {
|
||||||
|
System.loadLibrary("wg-go");
|
||||||
|
}
|
||||||
|
|
||||||
|
private final Context context;
|
||||||
|
private Tunnel currentTunnel;
|
||||||
|
|
||||||
|
public GoBackend(final Context context) {
|
||||||
|
this.context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static native int wgGetSocketV4(int handle);
|
||||||
|
|
||||||
|
private static native int wgGetSocketV6(int handle);
|
||||||
|
|
||||||
|
private static native void wgTurnOff(int handle);
|
||||||
|
|
||||||
|
private static native int wgTurnOn(String ifName, int tunFd, String settings);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Config applyConfig(final Tunnel tunnel, final Config config) throws Exception {
|
||||||
|
if (tunnel.getState() == State.UP) {
|
||||||
|
// Restart the tunnel to apply the new config.
|
||||||
|
setStateInternal(tunnel, tunnel.getConfig(), State.DOWN);
|
||||||
|
try {
|
||||||
|
setStateInternal(tunnel, config, State.UP);
|
||||||
|
} catch (final Exception e) {
|
||||||
|
// The new configuration didn't work, so try to go back to the old one.
|
||||||
|
setStateInternal(tunnel, tunnel.getConfig(), State.UP);
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return config;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<String> enumerate() {
|
||||||
|
if (currentTunnel != null) {
|
||||||
|
final Set<String> runningTunnels = new ArraySet<>();
|
||||||
|
runningTunnels.add(currentTunnel.getName());
|
||||||
|
return runningTunnels;
|
||||||
|
}
|
||||||
|
return Collections.emptySet();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public State getState(final Tunnel tunnel) {
|
||||||
|
return currentTunnel == tunnel ? State.UP : State.DOWN;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Statistics getStatistics(final Tunnel tunnel) {
|
||||||
|
return new Statistics();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public State setState(final Tunnel tunnel, State state) throws Exception {
|
||||||
|
final State originalState = getState(tunnel);
|
||||||
|
if (state == State.TOGGLE)
|
||||||
|
state = originalState == State.UP ? State.DOWN : State.UP;
|
||||||
|
if (state == originalState)
|
||||||
|
return originalState;
|
||||||
|
if (state == State.UP && currentTunnel != null)
|
||||||
|
throw new IllegalStateException("Only one userspace tunnel can run at a time");
|
||||||
|
Log.d(TAG, "Changing tunnel " + tunnel.getName() + " to state " + state);
|
||||||
|
setStateInternal(tunnel, tunnel.getConfig(), state);
|
||||||
|
return getState(tunnel);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setStateInternal(final Tunnel tunnel, final Config config, final State state)
|
||||||
|
throws Exception {
|
||||||
|
if (state == State.UP) {
|
||||||
|
// Do something (context.startService()...).
|
||||||
|
currentTunnel = tunnel;
|
||||||
|
|
||||||
|
Formatter fmt = new Formatter(new StringBuilder());
|
||||||
|
final Interface iface = config.getInterface();
|
||||||
|
fmt.format("replace_peers=true\n");
|
||||||
|
if (iface.getPrivateKey() != null)
|
||||||
|
fmt.format("private_key=%s\n", KeyEncoding.keyToHex(KeyEncoding.keyFromBase64(iface.getPrivateKey())));
|
||||||
|
if (iface.getListenPort() != null)
|
||||||
|
fmt.format("listen_port=%d\n", Integer.parseInt(config.getInterface().getListenPort()));
|
||||||
|
for (final Peer peer : config.getPeers()) {
|
||||||
|
if (peer.getPublicKey() != null)
|
||||||
|
fmt.format("public_key=%s\n", KeyEncoding.keyToHex(KeyEncoding.keyFromBase64(peer.getPublicKey())));
|
||||||
|
if (peer.getPreSharedKey() != null)
|
||||||
|
fmt.format("preshared_key=%s\n", KeyEncoding.keyToHex(KeyEncoding.keyFromBase64(peer.getPreSharedKey())));
|
||||||
|
if (peer.getEndpoint() != null)
|
||||||
|
fmt.format("endpoint=%s\n", peer.getEndpoint());
|
||||||
|
if (peer.getPersistentKeepalive() != null)
|
||||||
|
fmt.format("persistent_keepalive_interval=%d\n", Integer.parseInt(peer.getPersistentKeepalive()));
|
||||||
|
if (peer.getAllowedIPs() != null) {
|
||||||
|
for (final String allowedIp : peer.getAllowedIPs().split(" *, *")) {
|
||||||
|
fmt.format("allowed_ip=%s\n", allowedIp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wgTurnOn(tunnel.getName(), -1, fmt.toString());
|
||||||
|
} else {
|
||||||
|
// Do something else.
|
||||||
|
currentTunnel = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -88,8 +88,6 @@ public final class WgQuickBackend implements Backend {
|
|||||||
state = originalState == State.UP ? State.DOWN : State.UP;
|
state = originalState == State.UP ? State.DOWN : State.UP;
|
||||||
if (state == originalState)
|
if (state == originalState)
|
||||||
return originalState;
|
return originalState;
|
||||||
if (state == State.UP && !new File("/sys/module/wireguard").exists())
|
|
||||||
throw new ModuleNotLoadedException("WireGuard module not loaded");
|
|
||||||
Log.d(TAG, "Changing tunnel " + tunnel.getName() + " to state " + state);
|
Log.d(TAG, "Changing tunnel " + tunnel.getName() + " to state " + state);
|
||||||
toolsInstaller.ensureToolsAvailable();
|
toolsInstaller.ensureToolsAvailable();
|
||||||
setStateInternal(tunnel, tunnel.getConfig(), state);
|
setStateInternal(tunnel, tunnel.getConfig(), state);
|
||||||
@ -113,14 +111,4 @@ public final class WgQuickBackend implements Backend {
|
|||||||
if (result != 0)
|
if (result != 0)
|
||||||
throw new Exception("Unable to configure tunnel (wg-quick returned " + result + ')');
|
throw new Exception("Unable to configure tunnel (wg-quick returned " + result + ')');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class ModuleNotLoadedException extends Exception {
|
|
||||||
public ModuleNotLoadedException(final String message, final Throwable cause) {
|
|
||||||
super(message, cause);
|
|
||||||
}
|
|
||||||
|
|
||||||
public ModuleNotLoadedException(final String message) {
|
|
||||||
super(message);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,18 +1,13 @@
|
|||||||
package com.wireguard.android.fragment;
|
package com.wireguard.android.fragment;
|
||||||
|
|
||||||
import android.app.AlertDialog;
|
|
||||||
import android.content.Context;
|
import android.content.Context;
|
||||||
import android.databinding.DataBindingUtil;
|
import android.databinding.DataBindingUtil;
|
||||||
import android.databinding.ViewDataBinding;
|
import android.databinding.ViewDataBinding;
|
||||||
import android.text.Html;
|
|
||||||
import android.text.method.LinkMovementMethod;
|
|
||||||
import android.util.Log;
|
import android.util.Log;
|
||||||
import android.view.View;
|
import android.view.View;
|
||||||
import android.widget.TextView;
|
|
||||||
|
|
||||||
import com.commonsware.cwac.crossport.design.widget.Snackbar;
|
import com.commonsware.cwac.crossport.design.widget.Snackbar;
|
||||||
import com.wireguard.android.R;
|
import com.wireguard.android.R;
|
||||||
import com.wireguard.android.backend.WgQuickBackend;
|
|
||||||
import com.wireguard.android.databinding.TunnelDetailFragmentBinding;
|
import com.wireguard.android.databinding.TunnelDetailFragmentBinding;
|
||||||
import com.wireguard.android.databinding.TunnelListItemBinding;
|
import com.wireguard.android.databinding.TunnelListItemBinding;
|
||||||
import com.wireguard.android.model.Tunnel;
|
import com.wireguard.android.model.Tunnel;
|
||||||
@ -47,26 +42,11 @@ public final class TunnelController {
|
|||||||
if (throwable == null)
|
if (throwable == null)
|
||||||
return;
|
return;
|
||||||
final Context context = view.getContext();
|
final Context context = view.getContext();
|
||||||
if (ExceptionLoggers.unwrap(throwable)
|
final String error = ExceptionLoggers.unwrap(throwable).getMessage();
|
||||||
instanceof WgQuickBackend.ModuleNotLoadedException) {
|
final int messageResId = checked ? R.string.error_up : R.string.error_down;
|
||||||
final String message = context.getString(R.string.not_supported_message);
|
final String message = context.getString(messageResId, error);
|
||||||
final String title = context.getString(R.string.not_supported_title);
|
Snackbar.make(view, message, Snackbar.LENGTH_LONG).show();
|
||||||
final AlertDialog dialog = new AlertDialog.Builder(context)
|
Log.e(TAG, message, throwable);
|
||||||
.setMessage(Html.fromHtml(message))
|
|
||||||
.setPositiveButton(R.string.ok, null)
|
|
||||||
.setTitle(title)
|
|
||||||
.show();
|
|
||||||
// Make links work.
|
|
||||||
((TextView) dialog.findViewById(android.R.id.message))
|
|
||||||
.setMovementMethod(LinkMovementMethod.getInstance());
|
|
||||||
Log.e(TAG, title, throwable);
|
|
||||||
} else {
|
|
||||||
final String error = ExceptionLoggers.unwrap(throwable).getMessage();
|
|
||||||
final int messageResId = checked ? R.string.error_up : R.string.error_down;
|
|
||||||
final String message = context.getString(messageResId, error);
|
|
||||||
Snackbar.make(view, message, Snackbar.LENGTH_LONG).show();
|
|
||||||
Log.e(TAG, message, throwable);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -39,23 +39,6 @@
|
|||||||
<string name="listen_port">Listen port</string>
|
<string name="listen_port">Listen port</string>
|
||||||
<string name="mtu">MTU</string>
|
<string name="mtu">MTU</string>
|
||||||
<string name="name">Name</string>
|
<string name="name">Name</string>
|
||||||
<string name="not_supported_message" tools:ignore="TypographyQuotes"><![CDATA[
|
|
||||||
<p>Your Android device does not <em>currently</em> have the WireGuard kernel module. Please
|
|
||||||
talk to the manufacturer of your Android device or the author of your device’s ROM
|
|
||||||
about including the WireGuard kernel module.</p>
|
|
||||||
|
|
||||||
<p>Fortunately, we are in the process of implementing support for WireGuard in a way that
|
|
||||||
will work on all devices, without any need for the kernel module. This means that while you
|
|
||||||
may not be able to use WireGuard today, you will very likely be able to use WireGuard in
|
|
||||||
several weeks. Things are looking up!</p>
|
|
||||||
|
|
||||||
<p>Sorry for the wait. In the mean time, you may stay up to date on the latest project news
|
|
||||||
by <a href="https://lists.zx2c4.com/mailman/listinfo/wireguard">subscribing to our mailing
|
|
||||||
list</a>. General information about the project is available at
|
|
||||||
<a href="https://www.wireguard.com/">WireGuard.com</a>.</p>
|
|
||||||
]]></string>
|
|
||||||
<string name="not_supported_title">WireGuard not installed</string>
|
|
||||||
<string name="ok">OK</string>
|
|
||||||
<string name="peer">Peer</string>
|
<string name="peer">Peer</string>
|
||||||
<string name="persistent_keepalive">Persistent keepalive</string>
|
<string name="persistent_keepalive">Persistent keepalive</string>
|
||||||
<string name="pre_shared_key">Pre-shared key</string>
|
<string name="pre_shared_key">Pre-shared key</string>
|
||||||
|
@ -5,5 +5,5 @@
|
|||||||
android:key="restore_on_boot"
|
android:key="restore_on_boot"
|
||||||
android:summary="@string/restore_on_boot_summary"
|
android:summary="@string/restore_on_boot_summary"
|
||||||
android:title="@string/restore_on_boot_title" />
|
android:title="@string/restore_on_boot_title" />
|
||||||
<com.wireguard.android.preference.ToolsInstallerPreference />
|
<com.wireguard.android.preference.ToolsInstallerPreference android:key="tools_installer" />
|
||||||
</PreferenceScreen>
|
</PreferenceScreen>
|
||||||
|
@ -10,3 +10,16 @@ target_compile_options(libwg-quick.so PUBLIC -O3 -std=gnu11 -Wall -pedantic -Wno
|
|||||||
file(GLOB WG_SOURCES wireguard/src/tools/*.c libmnl/src/*.c)
|
file(GLOB WG_SOURCES wireguard/src/tools/*.c libmnl/src/*.c)
|
||||||
add_executable(libwg.so ${WG_SOURCES})
|
add_executable(libwg.so ${WG_SOURCES})
|
||||||
target_compile_options(libwg.so PUBLIC "-I${CMAKE_CURRENT_SOURCE_DIR}libmnl/src/" "-I${CMAKE_CURRENT_SOURCE_DIR}/libmnl/include/" "-I${CMAKE_CURRENT_SOURCE_DIR}/wireguard/src/tools/" -O3 -std=gnu11 -D_GNU_SOURCE -DHAVE_VISIBILITY_HIDDEN -DRUNSTATEDIR=\"\\\"/data/data/com.wireguard.android/cache\\\"\" -Wno-pointer-arith -Wno-unused-parameter)
|
target_compile_options(libwg.so PUBLIC "-I${CMAKE_CURRENT_SOURCE_DIR}libmnl/src/" "-I${CMAKE_CURRENT_SOURCE_DIR}/libmnl/include/" "-I${CMAKE_CURRENT_SOURCE_DIR}/wireguard/src/tools/" -O3 -std=gnu11 -D_GNU_SOURCE -DHAVE_VISIBILITY_HIDDEN -DRUNSTATEDIR=\"\\\"/data/data/com.wireguard.android/cache\\\"\" -Wno-pointer-arith -Wno-unused-parameter)
|
||||||
|
|
||||||
|
add_custom_target(libwg-go.so WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}/libwg-go" COMMENT "Building wireguard-go" VERBATIM COMMAND make
|
||||||
|
ANDROID_ARCH_NAME=${ANDROID_ARCH_NAME}
|
||||||
|
ANDROID_C_COMPILER=${ANDROID_C_COMPILER}
|
||||||
|
ANDROID_TOOLCHAIN_ROOT=${ANDROID_TOOLCHAIN_ROOT}
|
||||||
|
ANDROID_LLVM_TRIPLE=${ANDROID_LLVM_TRIPLE}
|
||||||
|
ANDROID_SYSROOT=${ANDROID_SYSROOT}
|
||||||
|
CFLAGS=${CMAKE_C_FLAGS}\ -Wno-unused-command-line-argument
|
||||||
|
LDFLAGS=${CMAKE_SHARED_LINKER_FLAGS}\ -fuse-ld=gold
|
||||||
|
DESTDIR=${CMAKE_LIBRARY_OUTPUT_DIRECTORY}
|
||||||
|
)
|
||||||
|
# Hack to make it actually build as part of the default target
|
||||||
|
add_dependencies(libwg.so libwg-go.so)
|
||||||
|
4
app/tools/libwg-go/.gitignore
vendored
Normal file
4
app/tools/libwg-go/.gitignore
vendored
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
go/
|
||||||
|
*.go
|
||||||
|
libwg-go.h
|
||||||
|
jni.o
|
21
app/tools/libwg-go/Makefile
Normal file
21
app/tools/libwg-go/Makefile
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
containing = $(foreach v,$2,$(if $(findstring $1,$v),$v))
|
||||||
|
FILES := $(wildcard ../wireguard-go/*/*.go) $(wildcard ../wireguard-go/*.go)
|
||||||
|
FILES := $(filter-out %/main.go $(filter-out %_linux.go,$(call containing,_,$(FILES))),$(FILES))
|
||||||
|
|
||||||
|
export GOPATH := $(CURDIR)/go
|
||||||
|
CLANG_FLAGS := --target=$(ANDROID_LLVM_TRIPLE) --gcc-toolchain=$(ANDROID_TOOLCHAIN_ROOT) --sysroot=$(ANDROID_SYSROOT)
|
||||||
|
export CGO_CFLAGS := $(CLANG_FLAGS) $(CFLAGS)
|
||||||
|
export CGO_LDFLAGS := $(CLANG_FLAGS) $(LDFLAGS)
|
||||||
|
export CC := $(ANDROID_C_COMPILER)
|
||||||
|
GO_ARCH_FILTER := case "$(ANDROID_ARCH_NAME)" in x86) echo 386 ;; x86_64) echo amd64 ;; *) echo $(ANDROID_ARCH_NAME) ;; esac
|
||||||
|
export GOARCH := $(shell $(GO_ARCH_FILTER))
|
||||||
|
export GOOS := android
|
||||||
|
export CGO_ENABLED := 1
|
||||||
|
|
||||||
|
$(DESTDIR)/libwg-go.so: $(FILES) api-android.go jni.c
|
||||||
|
find . -name '*.go' -type l -delete
|
||||||
|
find . -type d -empty -delete
|
||||||
|
mkdir -p $(subst ../wireguard-go/,./,$(dir $(FILES)))
|
||||||
|
$(foreach FILE,$(FILES),ln -sfrt $(subst ../wireguard-go/,./,$(dir $(FILE))) $(FILE);)
|
||||||
|
go get -v -d
|
||||||
|
go build -v -o $(DESTDIR)/libwg-go.so -buildmode c-shared
|
111
app/tools/libwg-go/api-android.go
Normal file
111
app/tools/libwg-go/api-android.go
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
// #cgo LDFLAGS: -llog
|
||||||
|
// #include <android/log.h>
|
||||||
|
import "C"
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"math"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AndroidLogger struct {
|
||||||
|
level C.int
|
||||||
|
interfaceName string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (l AndroidLogger) Write(p []byte) (int, error) {
|
||||||
|
C.__android_log_write(l.level, C.CString("WireGuard/GoBackend/"+l.interfaceName), C.CString(string(p)))
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var tunnelHandles map[int32]*Device
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
tunnelHandles = make(map[int32]*Device)
|
||||||
|
}
|
||||||
|
|
||||||
|
//export wgTurnOn
|
||||||
|
func wgTurnOn(ifnameRef string, tun_fd int32, settings string) int32 {
|
||||||
|
interfaceName := string([]byte(ifnameRef))
|
||||||
|
|
||||||
|
logger := &Logger{
|
||||||
|
Debug: log.New(&AndroidLogger{level: C.ANDROID_LOG_DEBUG, interfaceName: interfaceName}, "", 0),
|
||||||
|
Info: log.New(&AndroidLogger{level: C.ANDROID_LOG_INFO, interfaceName: interfaceName}, "", 0),
|
||||||
|
Error: log.New(&AndroidLogger{level: C.ANDROID_LOG_ERROR, interfaceName: interfaceName}, "", 0),
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.Debug.Println("Debug log enabled")
|
||||||
|
|
||||||
|
tun := &NativeTun{
|
||||||
|
fd: os.NewFile(uintptr(tun_fd), ""),
|
||||||
|
events: make(chan TUNEvent, 5),
|
||||||
|
errors: make(chan error, 5),
|
||||||
|
}
|
||||||
|
device := NewDevice(tun, logger)
|
||||||
|
device.tun.mtu = DefaultMTU //TODO: make dynamic
|
||||||
|
|
||||||
|
bufferedSettings := bufio.NewReadWriter(bufio.NewReader(strings.NewReader(settings)), bufio.NewWriter(ioutil.Discard))
|
||||||
|
setError := ipcSetOperation(device, bufferedSettings)
|
||||||
|
if setError != nil {
|
||||||
|
logger.Debug.Println(setError)
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
|
||||||
|
device.Up()
|
||||||
|
logger.Info.Println("Device started")
|
||||||
|
|
||||||
|
var i int32
|
||||||
|
for i = 0; i < math.MaxInt32; i++ {
|
||||||
|
if _, exists := tunnelHandles[i]; !exists {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if i == math.MaxInt32 {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
tunnelHandles[i] = device
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
|
||||||
|
//export wgTurnOff
|
||||||
|
func wgTurnOff(tunnelHandle int32) {
|
||||||
|
device, ok := tunnelHandles[tunnelHandle]
|
||||||
|
if !ok {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
delete(tunnelHandles, tunnelHandle)
|
||||||
|
device.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
//export wgGetSocketV4
|
||||||
|
func wgGetSocketV4(tunnelHandle int32) int32 {
|
||||||
|
device, ok := tunnelHandles[tunnelHandle]
|
||||||
|
if !ok {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
native, ok := device.net.bind.(NativeBind)
|
||||||
|
if !ok {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return int32(native.sock4)
|
||||||
|
}
|
||||||
|
|
||||||
|
//export wgGetSocketV6
|
||||||
|
func wgGetSocketV6(tunnelHandle int32) int32 {
|
||||||
|
device, ok := tunnelHandles[tunnelHandle]
|
||||||
|
if !ok {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
native, ok := device.net.bind.(NativeBind)
|
||||||
|
if !ok {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
return int32(native.sock6)
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {}
|
40
app/tools/libwg-go/jni.c
Normal file
40
app/tools/libwg-go/jni.c
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
#include <jni.h>
|
||||||
|
|
||||||
|
struct go_string { const char *str; long n; };
|
||||||
|
extern int wgTurnOn(struct go_string ifname, int tun_fd, struct go_string settings);
|
||||||
|
extern void wgTurnOff(int handle);
|
||||||
|
extern int wgGetSocketV4(int handle);
|
||||||
|
extern int wgGetSocketV6(int handle);
|
||||||
|
|
||||||
|
JNIEXPORT jint JNICALL Java_com_wireguard_android_backend_GoBackend_wgTurnOn(JNIEnv *env, jclass c, jstring ifname, jint tun_fd, jstring settings)
|
||||||
|
{
|
||||||
|
const char *ifname_str = (*env)->GetStringUTFChars(env, ifname, 0);
|
||||||
|
size_t ifname_len = (*env)->GetStringUTFLength(env, ifname);
|
||||||
|
const char *settings_str = (*env)->GetStringUTFChars(env, settings, 0);
|
||||||
|
size_t settings_len = (*env)->GetStringUTFLength(env, settings);
|
||||||
|
int ret = wgTurnOn((struct go_string){
|
||||||
|
.str = ifname_str,
|
||||||
|
.n = ifname_len
|
||||||
|
}, tun_fd, (struct go_string){
|
||||||
|
.str = settings_str,
|
||||||
|
.n = settings_len
|
||||||
|
});
|
||||||
|
(*env)->ReleaseStringUTFChars(env, ifname, ifname_str);
|
||||||
|
(*env)->ReleaseStringUTFChars(env, settings, settings_str);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT void JNICALL Java_com_wireguard_android_backend_GoBackend_wgTurnOff(JNIEnv *env, jclass c, jint handle)
|
||||||
|
{
|
||||||
|
wgTurnOff(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT jint JNICALL Java_com_wireguard_android_backend_GoBackend_wgGetSocketV4(JNIEnv *env, jclass c, jint handle)
|
||||||
|
{
|
||||||
|
return wgGetSocketV4(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT jint JNICALL Java_com_wireguard_android_backend_GoBackend_wgGetSocketV6(JNIEnv *env, jclass c, jint handle)
|
||||||
|
{
|
||||||
|
return wgGetSocketV6(handle);
|
||||||
|
}
|
1
app/tools/wireguard-go
Submodule
1
app/tools/wireguard-go
Submodule
@ -0,0 +1 @@
|
|||||||
|
Subproject commit 8f1d1b8c54d747309d9fdf06b157823af2a823bd
|
Loading…
Reference in New Issue
Block a user