Add bit-rate command-line option
Add a command-line option (-b/--bit-rate) to customize the video bit-rate.
This commit is contained in:
parent
7fe7bbf58c
commit
6b546a87ab
8 changed files with 79 additions and 15 deletions
|
@ -6,13 +6,15 @@
|
||||||
#include <SDL2/SDL.h>
|
#include <SDL2/SDL.h>
|
||||||
|
|
||||||
#define DEFAULT_LOCAL_PORT 27183
|
#define DEFAULT_LOCAL_PORT 27183
|
||||||
#define DEFAULT_MAX_SIZE 0
|
#define DEFAULT_MAX_SIZE 0 // unlimited
|
||||||
|
#define DEFAULT_BIT_RATE 4000000 // 4Mbps
|
||||||
|
|
||||||
struct args {
|
struct args {
|
||||||
const char *serial;
|
const char *serial;
|
||||||
SDL_bool help;
|
SDL_bool help;
|
||||||
Uint16 port;
|
Uint16 port;
|
||||||
Uint16 max_size;
|
Uint16 max_size;
|
||||||
|
Uint32 bit_rate;
|
||||||
};
|
};
|
||||||
|
|
||||||
static void usage(const char *arg0) {
|
static void usage(const char *arg0) {
|
||||||
|
@ -25,6 +27,11 @@ static void usage(const char *arg0) {
|
||||||
"\n"
|
"\n"
|
||||||
"Options:\n"
|
"Options:\n"
|
||||||
"\n"
|
"\n"
|
||||||
|
" -b, --bit-rate value\n"
|
||||||
|
" Encode the video at the given bit-rate, expressed in bits/s.\n"
|
||||||
|
" Unit suffixes are supported: 'K' (x1000) and 'M' (x1000000).\n"
|
||||||
|
" Default is %d.\n"
|
||||||
|
"\n"
|
||||||
" -h, --help\n"
|
" -h, --help\n"
|
||||||
" Print this help.\n"
|
" Print this help.\n"
|
||||||
"\n"
|
"\n"
|
||||||
|
@ -45,6 +52,7 @@ static void usage(const char *arg0) {
|
||||||
" Ctrl+x: resize window to optimal size (remove black borders)\n"
|
" Ctrl+x: resize window to optimal size (remove black borders)\n"
|
||||||
"\n",
|
"\n",
|
||||||
arg0,
|
arg0,
|
||||||
|
DEFAULT_BIT_RATE,
|
||||||
DEFAULT_MAX_SIZE, DEFAULT_MAX_SIZE ? "" : " (unlimited)",
|
DEFAULT_MAX_SIZE, DEFAULT_MAX_SIZE ? "" : " (unlimited)",
|
||||||
DEFAULT_LOCAL_PORT);
|
DEFAULT_LOCAL_PORT);
|
||||||
}
|
}
|
||||||
|
@ -54,10 +62,11 @@ static int parse_args(struct args *args, int argc, char *argv[]) {
|
||||||
{"help", no_argument, NULL, 'h'},
|
{"help", no_argument, NULL, 'h'},
|
||||||
{"port", required_argument, NULL, 'p'},
|
{"port", required_argument, NULL, 'p'},
|
||||||
{"max-size", required_argument, NULL, 'm'},
|
{"max-size", required_argument, NULL, 'm'},
|
||||||
|
{"bit-rate", required_argument, NULL, 'b'},
|
||||||
{NULL, 0, NULL, 0 },
|
{NULL, 0, NULL, 0 },
|
||||||
};
|
};
|
||||||
int c;
|
int c;
|
||||||
while ((c = getopt_long(argc, argv, "hp:m:", long_options, NULL)) != -1) {
|
while ((c = getopt_long(argc, argv, "hp:m:b:", long_options, NULL)) != -1) {
|
||||||
switch (c) {
|
switch (c) {
|
||||||
case 'h': {
|
case 'h': {
|
||||||
args->help = SDL_TRUE;
|
args->help = SDL_TRUE;
|
||||||
|
@ -99,6 +108,35 @@ static int parse_args(struct args *args, int argc, char *argv[]) {
|
||||||
args->max_size = (Uint16) value;
|
args->max_size = (Uint16) value;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case 'b': {
|
||||||
|
char *endptr;
|
||||||
|
if (*optarg == '\0') {
|
||||||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Bit-rate parameter is empty");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
long value = strtol(optarg, &endptr, 0);
|
||||||
|
int mul = 1;
|
||||||
|
if (*endptr != '\0') {
|
||||||
|
if (optarg == endptr) {
|
||||||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Invalid bit-rate: %s", optarg);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if ((*endptr == 'M' || *endptr == 'm') && endptr[1] == '\0') {
|
||||||
|
mul = 1000000;
|
||||||
|
} else if ((*endptr == 'K' || *endptr == 'k') && endptr[1] == '\0') {
|
||||||
|
mul = 1000;
|
||||||
|
} else {
|
||||||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Invalid bit-rate unit: %s", optarg);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (value < 0 || ((Uint32) -1) / mul < value) {
|
||||||
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION, "Bitrate must be positive and less than 2^32: %s", optarg);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
args->bit_rate = (Uint32) value * mul;
|
||||||
|
break;
|
||||||
|
}
|
||||||
default:
|
default:
|
||||||
// getopt prints the error message on stderr
|
// getopt prints the error message on stderr
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -120,10 +158,11 @@ int main(int argc, char *argv[]) {
|
||||||
int res;
|
int res;
|
||||||
|
|
||||||
struct args args = {
|
struct args args = {
|
||||||
.help = SDL_FALSE,
|
|
||||||
.serial = NULL,
|
.serial = NULL,
|
||||||
.max_size = DEFAULT_MAX_SIZE,
|
.help = SDL_FALSE,
|
||||||
.port = DEFAULT_LOCAL_PORT,
|
.port = DEFAULT_LOCAL_PORT,
|
||||||
|
.max_size = DEFAULT_MAX_SIZE,
|
||||||
|
.bit_rate = DEFAULT_BIT_RATE,
|
||||||
};
|
};
|
||||||
if (parse_args(&args, argc, argv)) {
|
if (parse_args(&args, argc, argv)) {
|
||||||
usage(argv[0]);
|
usage(argv[0]);
|
||||||
|
@ -143,7 +182,7 @@ int main(int argc, char *argv[]) {
|
||||||
|
|
||||||
SDL_LogSetAllPriority(SDL_LOG_PRIORITY_DEBUG);
|
SDL_LogSetAllPriority(SDL_LOG_PRIORITY_DEBUG);
|
||||||
|
|
||||||
res = scrcpy(args.serial, args.port, args.max_size) ? 0 : 1;
|
res = scrcpy(args.serial, args.port, args.max_size, args.bit_rate) ? 0 : 1;
|
||||||
|
|
||||||
avformat_network_deinit(); // ignore failure
|
avformat_network_deinit(); // ignore failure
|
||||||
|
|
||||||
|
|
|
@ -417,7 +417,7 @@ void event_loop(void) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
SDL_bool scrcpy(const char *serial, Uint16 local_port, Uint16 max_size) {
|
SDL_bool scrcpy(const char *serial, Uint16 local_port, Uint16 max_size, Uint32 bit_rate) {
|
||||||
SDL_bool ret = 0;
|
SDL_bool ret = 0;
|
||||||
|
|
||||||
process_t push_proc = push_server(serial);
|
process_t push_proc = push_server(serial);
|
||||||
|
@ -437,7 +437,7 @@ SDL_bool scrcpy(const char *serial, Uint16 local_port, Uint16 max_size) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// server will connect to our socket
|
// server will connect to our socket
|
||||||
process_t server = start_server(serial, max_size);
|
process_t server = start_server(serial, max_size, bit_rate);
|
||||||
if (server == PROCESS_NONE) {
|
if (server == PROCESS_NONE) {
|
||||||
ret = SDL_FALSE;
|
ret = SDL_FALSE;
|
||||||
SDLNet_TCP_Close(server_socket);
|
SDLNet_TCP_Close(server_socket);
|
||||||
|
|
|
@ -3,6 +3,6 @@
|
||||||
|
|
||||||
#include <SDL2/SDL_stdinc.h>
|
#include <SDL2/SDL_stdinc.h>
|
||||||
|
|
||||||
SDL_bool scrcpy(const char *serial, Uint16 local_port, Uint16 max_size);
|
SDL_bool scrcpy(const char *serial, Uint16 local_port, Uint16 max_size, Uint32 bit_rate);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
#include <SDL2/SDL_log.h>
|
#include <SDL2/SDL_log.h>
|
||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
#define SOCKET_NAME "scrcpy"
|
#define SOCKET_NAME "scrcpy"
|
||||||
|
|
||||||
|
@ -21,9 +22,11 @@ process_t disable_tunnel(const char *serial) {
|
||||||
return adb_reverse_remove(serial, SOCKET_NAME);
|
return adb_reverse_remove(serial, SOCKET_NAME);
|
||||||
}
|
}
|
||||||
|
|
||||||
process_t start_server(const char *serial, Uint16 max_size) {
|
process_t start_server(const char *serial, Uint16 max_size, Uint32 bit_rate) {
|
||||||
char max_size_string[6];
|
char max_size_string[6];
|
||||||
sprintf(max_size_string, "%d", max_size);
|
char bit_rate_string[11];
|
||||||
|
sprintf(max_size_string, "%"PRIu16, max_size);
|
||||||
|
sprintf(bit_rate_string, "%"PRIu32, bit_rate);
|
||||||
const char *const cmd[] = {
|
const char *const cmd[] = {
|
||||||
"shell",
|
"shell",
|
||||||
"CLASSPATH=/data/local/tmp/scrcpy.apk",
|
"CLASSPATH=/data/local/tmp/scrcpy.apk",
|
||||||
|
@ -31,6 +34,7 @@ process_t start_server(const char *serial, Uint16 max_size) {
|
||||||
"/", // unused
|
"/", // unused
|
||||||
"com.genymobile.scrcpy.ScrCpyServer",
|
"com.genymobile.scrcpy.ScrCpyServer",
|
||||||
max_size_string,
|
max_size_string,
|
||||||
|
bit_rate_string,
|
||||||
};
|
};
|
||||||
return adb_execute(serial, cmd, sizeof(cmd) / sizeof(cmd[0]));
|
return adb_execute(serial, cmd, sizeof(cmd) / sizeof(cmd[0]));
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,5 +4,5 @@ process_t push_server(const char *serial);
|
||||||
process_t enable_tunnel(const char *serial, Uint16 local_port);
|
process_t enable_tunnel(const char *serial, Uint16 local_port);
|
||||||
process_t disable_tunnel(const char *serial);
|
process_t disable_tunnel(const char *serial);
|
||||||
|
|
||||||
process_t start_server(const char *serial, Uint16 max_size);
|
process_t start_server(const char *serial, Uint16 max_size, Uint32 bit_rate);
|
||||||
void stop_server(process_t server);
|
void stop_server(process_t server);
|
||||||
|
|
|
@ -2,6 +2,7 @@ package com.genymobile.scrcpy;
|
||||||
|
|
||||||
public class Options {
|
public class Options {
|
||||||
private int maxSize;
|
private int maxSize;
|
||||||
|
private int bitRate;
|
||||||
|
|
||||||
public int getMaxSize() {
|
public int getMaxSize() {
|
||||||
return maxSize;
|
return maxSize;
|
||||||
|
@ -10,4 +11,12 @@ public class Options {
|
||||||
public void setMaxSize(int maxSize) {
|
public void setMaxSize(int maxSize) {
|
||||||
this.maxSize = maxSize;
|
this.maxSize = maxSize;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public int getBitRate() {
|
||||||
|
return bitRate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBitRate(int bitRate) {
|
||||||
|
this.bitRate = bitRate;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,7 +7,7 @@ public class ScrCpyServer {
|
||||||
private static void scrcpy(Options options) throws IOException {
|
private static void scrcpy(Options options) throws IOException {
|
||||||
final Device device = new Device(options);
|
final Device device = new Device(options);
|
||||||
try (DesktopConnection connection = DesktopConnection.open(device)) {
|
try (DesktopConnection connection = DesktopConnection.open(device)) {
|
||||||
ScreenEncoder screenEncoder = new ScreenEncoder();
|
ScreenEncoder screenEncoder = new ScreenEncoder(options.getBitRate());
|
||||||
|
|
||||||
// asynchronous
|
// asynchronous
|
||||||
startEventController(device, connection);
|
startEventController(device, connection);
|
||||||
|
@ -36,10 +36,18 @@ public class ScrCpyServer {
|
||||||
|
|
||||||
private static Options createOptions(String... args) {
|
private static Options createOptions(String... args) {
|
||||||
Options options = new Options();
|
Options options = new Options();
|
||||||
if (args.length > 0) {
|
if (args.length < 1) {
|
||||||
|
return options;
|
||||||
|
}
|
||||||
int maxSize = Integer.parseInt(args[0]) & ~7; // multiple of 8
|
int maxSize = Integer.parseInt(args[0]) & ~7; // multiple of 8
|
||||||
options.setMaxSize(maxSize);
|
options.setMaxSize(maxSize);
|
||||||
|
|
||||||
|
if (args.length < 2) {
|
||||||
|
return options;
|
||||||
}
|
}
|
||||||
|
int bitRate = Integer.parseInt(args[1]);
|
||||||
|
options.setBitRate(bitRate);
|
||||||
|
|
||||||
return options;
|
return options;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,10 @@ public class ScreenEncoder implements Device.RotationListener {
|
||||||
this.iFrameInterval = iFrameInterval;
|
this.iFrameInterval = iFrameInterval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public ScreenEncoder(int bitRate) {
|
||||||
|
this(bitRate, DEFAULT_FRAME_RATE, DEFAULT_I_FRAME_INTERVAL);
|
||||||
|
}
|
||||||
|
|
||||||
public ScreenEncoder() {
|
public ScreenEncoder() {
|
||||||
this(DEFAULT_BIT_RATE, DEFAULT_FRAME_RATE, DEFAULT_I_FRAME_INTERVAL);
|
this(DEFAULT_BIT_RATE, DEFAULT_FRAME_RATE, DEFAULT_I_FRAME_INTERVAL);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue