mirror of
https://github.com/Genymobile/scrcpy
synced 2024-11-15 06:12:53 +00:00
179 lines
4.4 KiB
C
179 lines
4.4 KiB
C
|
#include "file_pusher.h"
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
#include "adb.h"
|
||
|
#include "util/log.h"
|
||
|
#include "util/process_intr.h"
|
||
|
|
||
|
#define DEFAULT_PUSH_TARGET "/sdcard/Download/"
|
||
|
|
||
|
static void
|
||
|
sc_file_pusher_request_destroy(struct sc_file_pusher_request *req) {
|
||
|
free(req->file);
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
sc_file_pusher_init(struct sc_file_pusher *fp, const char *serial,
|
||
|
const char *push_target) {
|
||
|
assert(serial);
|
||
|
|
||
|
cbuf_init(&fp->queue);
|
||
|
|
||
|
bool ok = sc_mutex_init(&fp->mutex);
|
||
|
if (!ok) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
ok = sc_cond_init(&fp->event_cond);
|
||
|
if (!ok) {
|
||
|
sc_mutex_destroy(&fp->mutex);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
ok = sc_intr_init(&fp->intr);
|
||
|
if (!ok) {
|
||
|
sc_cond_destroy(&fp->event_cond);
|
||
|
sc_mutex_destroy(&fp->mutex);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
fp->serial = strdup(serial);
|
||
|
if (!fp->serial) {
|
||
|
LOG_OOM();
|
||
|
sc_intr_destroy(&fp->intr);
|
||
|
sc_cond_destroy(&fp->event_cond);
|
||
|
sc_mutex_destroy(&fp->mutex);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// lazy initialization
|
||
|
fp->initialized = false;
|
||
|
|
||
|
fp->stopped = false;
|
||
|
|
||
|
fp->push_target = push_target ? push_target : DEFAULT_PUSH_TARGET;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
sc_file_pusher_destroy(struct sc_file_pusher *fp) {
|
||
|
sc_cond_destroy(&fp->event_cond);
|
||
|
sc_mutex_destroy(&fp->mutex);
|
||
|
sc_intr_destroy(&fp->intr);
|
||
|
free(fp->serial);
|
||
|
|
||
|
struct sc_file_pusher_request req;
|
||
|
while (cbuf_take(&fp->queue, &req)) {
|
||
|
sc_file_pusher_request_destroy(&req);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
sc_file_pusher_request(struct sc_file_pusher *fp,
|
||
|
enum sc_file_pusher_action action, char *file) {
|
||
|
// start file_pusher if it's used for the first time
|
||
|
if (!fp->initialized) {
|
||
|
if (!sc_file_pusher_start(fp)) {
|
||
|
return false;
|
||
|
}
|
||
|
fp->initialized = true;
|
||
|
}
|
||
|
|
||
|
LOGI("Request to %s %s", action == SC_FILE_PUSHER_ACTION_INSTALL_APK
|
||
|
? "install" : "push",
|
||
|
file);
|
||
|
struct sc_file_pusher_request req = {
|
||
|
.action = action,
|
||
|
.file = file,
|
||
|
};
|
||
|
|
||
|
sc_mutex_lock(&fp->mutex);
|
||
|
bool was_empty = cbuf_is_empty(&fp->queue);
|
||
|
bool res = cbuf_push(&fp->queue, req);
|
||
|
if (was_empty) {
|
||
|
sc_cond_signal(&fp->event_cond);
|
||
|
}
|
||
|
sc_mutex_unlock(&fp->mutex);
|
||
|
return res;
|
||
|
}
|
||
|
|
||
|
static int
|
||
|
run_file_pusher(void *data) {
|
||
|
struct sc_file_pusher *fp = data;
|
||
|
struct sc_intr *intr = &fp->intr;
|
||
|
|
||
|
const char *serial = fp->serial;
|
||
|
assert(serial);
|
||
|
|
||
|
const char *push_target = fp->push_target;
|
||
|
assert(push_target);
|
||
|
|
||
|
for (;;) {
|
||
|
sc_mutex_lock(&fp->mutex);
|
||
|
while (!fp->stopped && cbuf_is_empty(&fp->queue)) {
|
||
|
sc_cond_wait(&fp->event_cond, &fp->mutex);
|
||
|
}
|
||
|
if (fp->stopped) {
|
||
|
// stop immediately, do not process further events
|
||
|
sc_mutex_unlock(&fp->mutex);
|
||
|
break;
|
||
|
}
|
||
|
struct sc_file_pusher_request req;
|
||
|
bool non_empty = cbuf_take(&fp->queue, &req);
|
||
|
assert(non_empty);
|
||
|
(void) non_empty;
|
||
|
sc_mutex_unlock(&fp->mutex);
|
||
|
|
||
|
if (req.action == SC_FILE_PUSHER_ACTION_INSTALL_APK) {
|
||
|
LOGI("Installing %s...", req.file);
|
||
|
bool ok = adb_install(intr, serial, req.file, 0);
|
||
|
if (ok) {
|
||
|
LOGI("%s successfully installed", req.file);
|
||
|
} else {
|
||
|
LOGE("Failed to install %s", req.file);
|
||
|
}
|
||
|
} else {
|
||
|
LOGI("Pushing %s...", req.file);
|
||
|
bool ok = adb_push(intr, serial, req.file, push_target, 0);
|
||
|
if (ok) {
|
||
|
LOGI("%s successfully pushed to %s", req.file, push_target);
|
||
|
} else {
|
||
|
LOGE("Failed to push %s to %s", req.file, push_target);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sc_file_pusher_request_destroy(&req);
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
bool
|
||
|
sc_file_pusher_start(struct sc_file_pusher *fp) {
|
||
|
LOGD("Starting file_pusher thread");
|
||
|
|
||
|
bool ok = sc_thread_create(&fp->thread, run_file_pusher, "scrcpy-file", fp);
|
||
|
if (!ok) {
|
||
|
LOGC("Could not start file_pusher thread");
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
sc_file_pusher_stop(struct sc_file_pusher *fp) {
|
||
|
sc_mutex_lock(&fp->mutex);
|
||
|
fp->stopped = true;
|
||
|
sc_cond_signal(&fp->event_cond);
|
||
|
sc_intr_interrupt(&fp->intr);
|
||
|
sc_mutex_unlock(&fp->mutex);
|
||
|
}
|
||
|
|
||
|
void
|
||
|
sc_file_pusher_join(struct sc_file_pusher *fp) {
|
||
|
sc_thread_join(&fp->thread, NULL);
|
||
|
}
|