Add --replace flag

This allows replacing a running xdpw instance. xdg-desktop-portal
itself has this feature.

This is useful when developing, to stop the currently running
system instance.
This commit is contained in:
Simon Ser 2021-02-08 16:57:54 +01:00 committed by danshick
parent c6d52b4412
commit 5401e7f9f2

View file

@ -24,6 +24,7 @@ static int xdpw_usage(FILE* stream, int rc) {
" QUIET, ERROR, WARN, INFO, DEBUG, TRACE\n" " QUIET, ERROR, WARN, INFO, DEBUG, TRACE\n"
" -o, --output=<name> Select output to capture.\n" " -o, --output=<name> Select output to capture.\n"
" metadata (performs no conversion).\n" " metadata (performs no conversion).\n"
" -r, --replace Replace a running instance.\n"
" -h, --help Get help (this text).\n" " -h, --help Get help (this text).\n"
"\n"; "\n";
@ -31,14 +32,22 @@ static int xdpw_usage(FILE* stream, int rc) {
return rc; return rc;
} }
static int handle_name_lost(sd_bus_message *m, void *userdata, sd_bus_error *ret_error) {
logprint(INFO, "dbus: lost name, closing connection");
sd_bus_close(sd_bus_message_get_bus(m));
return 1;
}
int main(int argc, char *argv[]) { int main(int argc, char *argv[]) {
const char* output_name = NULL; const char* output_name = NULL;
enum LOGLEVEL loglevel = ERROR; enum LOGLEVEL loglevel = ERROR;
bool replace = false;
static const char* shortopts = "l:o:h"; static const char* shortopts = "l:o:rh";
static const struct option longopts[] = { static const struct option longopts[] = {
{ "loglevel", required_argument, NULL, 'l' }, { "loglevel", required_argument, NULL, 'l' },
{ "output", required_argument, NULL, 'o' }, { "output", required_argument, NULL, 'o' },
{ "replace", no_argument, NULL, 'r' },
{ "help", no_argument, NULL, 'h' }, { "help", no_argument, NULL, 'h' },
{ NULL, 0, NULL, 0 } { NULL, 0, NULL, 0 }
}; };
@ -55,6 +64,9 @@ int main(int argc, char *argv[]) {
case 'o': case 'o':
output_name = optarg; output_name = optarg;
break; break;
case 'r':
replace = true;
break;
case 'h': case 'h':
return xdpw_usage(stdout, EXIT_SUCCESS); return xdpw_usage(stdout, EXIT_SUCCESS);
default: default:
@ -110,12 +122,41 @@ int main(int argc, char *argv[]) {
goto error; goto error;
} }
ret = sd_bus_request_name(bus, service_name, 0); uint64_t flags = SD_BUS_NAME_ALLOW_REPLACEMENT;
if (replace) {
flags |= SD_BUS_NAME_REPLACE_EXISTING;
}
ret = sd_bus_request_name(bus, service_name, flags);
if (ret < 0) { if (ret < 0) {
logprint(ERROR, "dbus: failed to acquire service name: %s", strerror(-ret)); logprint(ERROR, "dbus: failed to acquire service name: %s", strerror(-ret));
goto error; goto error;
} }
const char *unique_name;
ret = sd_bus_get_unique_name(bus, &unique_name);
if (ret < 0) {
logprint(ERROR, "dbus: failed to get unique bus name: %s", strerror(-ret));
goto error;
}
static char match[1024];
snprintf(match, sizeof(match), "sender='org.freedesktop.DBus',"
"type='signal',"
"interface='org.freedesktop.DBus',"
"member='NameOwnerChanged',"
"path='/org/freedesktop/DBus',"
"arg0='%s',"
"arg1='%s'",
service_name, unique_name);
sd_bus_slot *slot;
ret = sd_bus_add_match(bus, &slot, match, handle_name_lost, NULL);
if (ret < 0) {
logprint(ERROR, "dbus: failed to add NameOwnerChanged signal match: %s", strerror(-ret));
goto error;
}
struct pollfd pollfds[] = { struct pollfd pollfds[] = {
[EVENT_LOOP_DBUS] = { [EVENT_LOOP_DBUS] = {
.fd = sd_bus_get_fd(state.bus), .fd = sd_bus_get_fd(state.bus),