mirror of https://github.com/Genymobile/scrcpy
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
8.9 KiB
Java
249 lines
8.9 KiB
Java
package com.genymobile.scrcpy;
|
|
|
|
import android.graphics.Rect;
|
|
import android.media.MediaCodec;
|
|
import android.media.MediaCodecInfo;
|
|
import android.os.Build;
|
|
|
|
import java.util.Locale;
|
|
import java.util.ArrayList;
|
|
import java.io.IOException;
|
|
|
|
public final class Server {
|
|
|
|
// public static final String SERVER_PATH;
|
|
|
|
// static {
|
|
// String[] classPaths = System.getProperty("java.class.path").split(File.pathSeparator);
|
|
// // By convention, scrcpy is always executed with the absolute path of scrcpy-server.jar as the first item in the classpath
|
|
// SERVER_PATH = classPaths[0];
|
|
// }
|
|
|
|
private static class Completion {
|
|
private int running;
|
|
private boolean fatalError;
|
|
|
|
Completion(int running) {
|
|
this.running = running;
|
|
}
|
|
|
|
synchronized void addCompleted(boolean fatalError) {
|
|
--running;
|
|
if (fatalError) {
|
|
this.fatalError = true;
|
|
}
|
|
if (running == 0 || this.fatalError) {
|
|
notify();
|
|
}
|
|
}
|
|
|
|
synchronized void await() {
|
|
try {
|
|
while (running > 0 && !fatalError) {
|
|
wait();
|
|
}
|
|
} catch (InterruptedException e) {
|
|
// ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
private Server() {
|
|
// not instantiable
|
|
}
|
|
|
|
private static void parseArguments(Options options, VideoSettings videoSettings, String... args) {
|
|
if (args.length < 1) {
|
|
throw new IllegalArgumentException("Missing client version");
|
|
}
|
|
|
|
String clientVersion = args[0];
|
|
if (!clientVersion.equals(BuildConfig.VERSION_NAME)) {
|
|
throw new IllegalArgumentException(
|
|
"The server version (" + BuildConfig.VERSION_NAME + ") does not match the client " + "(" + clientVersion + ")");
|
|
}
|
|
|
|
if (args[1].toLowerCase().equals("web")) {
|
|
options.setServerType(Options.TYPE_WEB_SOCKET);
|
|
if (args.length > 2) {
|
|
Ln.Level level = Ln.Level.valueOf(args[2].toUpperCase(Locale.ENGLISH));
|
|
options.setLogLevel(level);
|
|
}
|
|
if (args.length > 3) {
|
|
int portNumber = Integer.parseInt(args[3]);
|
|
options.setPortNumber(portNumber);
|
|
}
|
|
if (args.length > 4) {
|
|
boolean listenOnAllInterfaces = Boolean.parseBoolean(args[4]);
|
|
options.setListenOnAllInterfaces(listenOnAllInterfaces);
|
|
}
|
|
return;
|
|
}
|
|
|
|
final int expectedParameters = 16;
|
|
if (args.length != expectedParameters) {
|
|
throw new IllegalArgumentException("Expecting " + expectedParameters + " parameters");
|
|
}
|
|
|
|
Ln.Level level = Ln.Level.valueOf(args[1].toUpperCase(Locale.ENGLISH));
|
|
options.setLogLevel(level);
|
|
|
|
int maxSize = Integer.parseInt(args[2]);
|
|
if (maxSize != 0) {
|
|
videoSettings.setBounds(maxSize, maxSize);
|
|
}
|
|
|
|
int bitRate = Integer.parseInt(args[3]);
|
|
videoSettings.setBitRate(bitRate);
|
|
|
|
int maxFps = Integer.parseInt(args[4]);
|
|
videoSettings.setMaxFps(maxFps);
|
|
|
|
int lockedVideoOrientation = Integer.parseInt(args[5]);
|
|
videoSettings.setLockedVideoOrientation(lockedVideoOrientation);
|
|
|
|
// use "adb forward" instead of "adb tunnel"? (so the server must listen)
|
|
boolean tunnelForward = Boolean.parseBoolean(args[6]);
|
|
options.setTunnelForward(tunnelForward);
|
|
|
|
Rect crop = parseCrop(args[7]);
|
|
videoSettings.setCrop(crop);
|
|
|
|
boolean sendFrameMeta = Boolean.parseBoolean(args[8]);
|
|
videoSettings.setSendFrameMeta(sendFrameMeta);
|
|
|
|
boolean control = Boolean.parseBoolean(args[9]);
|
|
options.setControl(control);
|
|
|
|
int displayId = Integer.parseInt(args[10]);
|
|
videoSettings.setDisplayId(displayId);
|
|
|
|
boolean showTouches = Boolean.parseBoolean(args[11]);
|
|
options.setShowTouches(showTouches);
|
|
|
|
boolean stayAwake = Boolean.parseBoolean(args[12]);
|
|
options.setStayAwake(stayAwake);
|
|
|
|
String codecOptions = args[13];
|
|
options.setCodecOptions(codecOptions);
|
|
videoSettings.setCodecOptions(codecOptions);
|
|
|
|
String encoderName = "-".equals(args[14]) ? null : args[14];
|
|
videoSettings.setEncoderName(encoderName);
|
|
|
|
boolean powerOffScreenOnClose = Boolean.parseBoolean(args[15]);
|
|
options.setPowerOffScreenOnClose(powerOffScreenOnClose);
|
|
}
|
|
|
|
private static Rect parseCrop(String crop) {
|
|
if ("-".equals(crop)) {
|
|
return null;
|
|
}
|
|
// input format: "width:height:x:y"
|
|
String[] tokens = crop.split(":");
|
|
if (tokens.length != 4) {
|
|
throw new IllegalArgumentException("Crop must contains 4 values separated by colons: \"" + crop + "\"");
|
|
}
|
|
int width = Integer.parseInt(tokens[0]);
|
|
int height = Integer.parseInt(tokens[1]);
|
|
int x = Integer.parseInt(tokens[2]);
|
|
int y = Integer.parseInt(tokens[3]);
|
|
return new Rect(x, y, x + width, y + height);
|
|
}
|
|
|
|
private static void suggestFix(Throwable e) {
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
|
|
if (e instanceof MediaCodec.CodecException) {
|
|
MediaCodec.CodecException mce = (MediaCodec.CodecException) e;
|
|
if (mce.getErrorCode() == 0xfffffc0e) {
|
|
Ln.e("The hardware encoder is not able to encode at the given definition.");
|
|
Ln.e("Try with a lower definition:");
|
|
Ln.e(" scrcpy -m 1024");
|
|
}
|
|
}
|
|
}
|
|
if (e instanceof InvalidDisplayIdException) {
|
|
InvalidDisplayIdException idie = (InvalidDisplayIdException) e;
|
|
int[] displayIds = idie.getAvailableDisplayIds();
|
|
if (displayIds != null && displayIds.length > 0) {
|
|
Ln.e("Try to use one of the available display ids:");
|
|
for (int id : displayIds) {
|
|
Ln.e(" scrcpy --display " + id);
|
|
}
|
|
}
|
|
} else if (e instanceof InvalidEncoderException) {
|
|
InvalidEncoderException iee = (InvalidEncoderException) e;
|
|
MediaCodecInfo[] encoders = iee.getAvailableEncoders();
|
|
if (encoders != null && encoders.length > 0) {
|
|
Ln.e("Try to use one of the available encoders:");
|
|
for (MediaCodecInfo encoder : encoders) {
|
|
Ln.e(" scrcpy --encoder '" + encoder.getName() + "'");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public static void main(String... args) {
|
|
int status = 0;
|
|
try {
|
|
internalMain(args);
|
|
} catch (Throwable t) {
|
|
Ln.e(t.getMessage(), t);
|
|
status = 1;
|
|
} finally {
|
|
// By default, the Java process exits when all non-daemon threads are terminated.
|
|
// The Android SDK might start some non-daemon threads internally, preventing the scrcpy server to exit.
|
|
// So force the process to exit explicitly.
|
|
System.exit(status);
|
|
}
|
|
}
|
|
|
|
private static void internalMain(String... args) throws Exception {
|
|
Thread.setDefaultUncaughtExceptionHandler((t, e) -> {
|
|
Ln.e("Exception on thread " + t, e);
|
|
});
|
|
|
|
Options options = new Options();
|
|
VideoSettings videoSettings = new VideoSettings();
|
|
parseArguments(options, videoSettings, args);
|
|
|
|
Ln.disableSystemStreams();
|
|
Ln.initLogLevel(options.getLogLevel());
|
|
|
|
Ln.i("Device: [" + Build.MANUFACTURER + "] " + Build.BRAND + " " + Build.MODEL + " (Android " + Build.VERSION.RELEASE + ")");
|
|
|
|
if (options.getList()) {
|
|
if (options.getCleanup()) {
|
|
CleanUp.unlinkSelf();
|
|
}
|
|
|
|
if (options.getListEncoders()) {
|
|
Ln.i(LogUtils.buildVideoEncoderListMessage());
|
|
Ln.i(LogUtils.buildAudioEncoderListMessage());
|
|
}
|
|
if (options.getListDisplays()) {
|
|
Ln.i(LogUtils.buildDisplayListMessage());
|
|
}
|
|
if (options.getListCameras() || options.getListCameraSizes()) {
|
|
Workarounds.apply(false, true);
|
|
Ln.i(LogUtils.buildCameraListMessage(options.getListCameraSizes()));
|
|
}
|
|
// Just print the requested data, do not mirror
|
|
return;
|
|
}
|
|
|
|
try {
|
|
Ln.i("scrcpy server type is " + options.getServerType());
|
|
if (options.getServerType() == Options.TYPE_LOCAL_SOCKET) {
|
|
new DesktopConnection(options, videoSettings);
|
|
} else if (options.getServerType() == Options.TYPE_WEB_SOCKET) {
|
|
WSServer wsServer = new WSServer(options);
|
|
wsServer.setReuseAddr(true);
|
|
wsServer.run();
|
|
}
|
|
} catch (Exception e) {
|
|
// Do not print stack trace, a user-friendly error-message has already been logged
|
|
}
|
|
}
|
|
}
|