monitor: Enable adding an inherited fd to an fd set
qmp_add_fd() gets an fd that was received over a socket with SCM_RIGHTS and adds it to an fd set. This patch adds support that will enable adding an fd that was inherited on the command line to an fd set. Note: All of the code added to monitor_fdset_add_fd(), with the exception of the error path for non-valid fdset-id, is code motion from qmp_add_fd(). Signed-off-by: Corey Bryant <coreyb@linux.vnet.ibm.com> Reviewed-by: Eric Blake <eblake@redhat.com> Signed-off-by: Kevin Wolf <kwolf@redhat.com>
This commit is contained in:
parent
9ac54af0c3
commit
e446f70d54
157
monitor.c
157
monitor.c
@ -2135,8 +2135,6 @@ AddfdInfo *qmp_add_fd(bool has_fdset_id, int64_t fdset_id, bool has_opaque,
|
|||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
Monitor *mon = cur_mon;
|
Monitor *mon = cur_mon;
|
||||||
MonFdset *mon_fdset = NULL;
|
|
||||||
MonFdsetFd *mon_fdset_fd;
|
|
||||||
AddfdInfo *fdinfo;
|
AddfdInfo *fdinfo;
|
||||||
|
|
||||||
fd = qemu_chr_fe_get_msgfd(mon->chr);
|
fd = qemu_chr_fe_get_msgfd(mon->chr);
|
||||||
@ -2145,78 +2143,12 @@ AddfdInfo *qmp_add_fd(bool has_fdset_id, int64_t fdset_id, bool has_opaque,
|
|||||||
goto error;
|
goto error;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (has_fdset_id) {
|
fdinfo = monitor_fdset_add_fd(fd, has_fdset_id, fdset_id,
|
||||||
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
has_opaque, opaque, errp);
|
||||||
/* Break if match found or match impossible due to ordering by ID */
|
if (fdinfo) {
|
||||||
if (fdset_id <= mon_fdset->id) {
|
return fdinfo;
|
||||||
if (fdset_id < mon_fdset->id) {
|
|
||||||
mon_fdset = NULL;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (mon_fdset == NULL) {
|
|
||||||
int64_t fdset_id_prev = -1;
|
|
||||||
MonFdset *mon_fdset_cur = QLIST_FIRST(&mon_fdsets);
|
|
||||||
|
|
||||||
if (has_fdset_id) {
|
|
||||||
if (fdset_id < 0) {
|
|
||||||
error_set(errp, QERR_INVALID_PARAMETER_VALUE, "fdset-id",
|
|
||||||
"a non-negative value");
|
|
||||||
goto error;
|
|
||||||
}
|
|
||||||
/* Use specified fdset ID */
|
|
||||||
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
|
||||||
mon_fdset_cur = mon_fdset;
|
|
||||||
if (fdset_id < mon_fdset_cur->id) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
/* Use first available fdset ID */
|
|
||||||
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
|
||||||
mon_fdset_cur = mon_fdset;
|
|
||||||
if (fdset_id_prev == mon_fdset_cur->id - 1) {
|
|
||||||
fdset_id_prev = mon_fdset_cur->id;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mon_fdset = g_malloc0(sizeof(*mon_fdset));
|
|
||||||
if (has_fdset_id) {
|
|
||||||
mon_fdset->id = fdset_id;
|
|
||||||
} else {
|
|
||||||
mon_fdset->id = fdset_id_prev + 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* The fdset list is ordered by fdset ID */
|
|
||||||
if (!mon_fdset_cur) {
|
|
||||||
QLIST_INSERT_HEAD(&mon_fdsets, mon_fdset, next);
|
|
||||||
} else if (mon_fdset->id < mon_fdset_cur->id) {
|
|
||||||
QLIST_INSERT_BEFORE(mon_fdset_cur, mon_fdset, next);
|
|
||||||
} else {
|
|
||||||
QLIST_INSERT_AFTER(mon_fdset_cur, mon_fdset, next);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mon_fdset_fd = g_malloc0(sizeof(*mon_fdset_fd));
|
|
||||||
mon_fdset_fd->fd = fd;
|
|
||||||
mon_fdset_fd->removed = false;
|
|
||||||
if (has_opaque) {
|
|
||||||
mon_fdset_fd->opaque = g_strdup(opaque);
|
|
||||||
}
|
|
||||||
QLIST_INSERT_HEAD(&mon_fdset->fds, mon_fdset_fd, next);
|
|
||||||
|
|
||||||
fdinfo = g_malloc0(sizeof(*fdinfo));
|
|
||||||
fdinfo->fdset_id = mon_fdset->id;
|
|
||||||
fdinfo->fd = mon_fdset_fd->fd;
|
|
||||||
|
|
||||||
return fdinfo;
|
|
||||||
|
|
||||||
error:
|
error:
|
||||||
if (fd != -1) {
|
if (fd != -1) {
|
||||||
close(fd);
|
close(fd);
|
||||||
@ -2301,6 +2233,87 @@ FdsetInfoList *qmp_query_fdsets(Error **errp)
|
|||||||
return fdset_list;
|
return fdset_list;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
AddfdInfo *monitor_fdset_add_fd(int fd, bool has_fdset_id, int64_t fdset_id,
|
||||||
|
bool has_opaque, const char *opaque,
|
||||||
|
Error **errp)
|
||||||
|
{
|
||||||
|
MonFdset *mon_fdset = NULL;
|
||||||
|
MonFdsetFd *mon_fdset_fd;
|
||||||
|
AddfdInfo *fdinfo;
|
||||||
|
|
||||||
|
if (has_fdset_id) {
|
||||||
|
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
||||||
|
/* Break if match found or match impossible due to ordering by ID */
|
||||||
|
if (fdset_id <= mon_fdset->id) {
|
||||||
|
if (fdset_id < mon_fdset->id) {
|
||||||
|
mon_fdset = NULL;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mon_fdset == NULL) {
|
||||||
|
int64_t fdset_id_prev = -1;
|
||||||
|
MonFdset *mon_fdset_cur = QLIST_FIRST(&mon_fdsets);
|
||||||
|
|
||||||
|
if (has_fdset_id) {
|
||||||
|
if (fdset_id < 0) {
|
||||||
|
error_set(errp, QERR_INVALID_PARAMETER_VALUE, "fdset-id",
|
||||||
|
"a non-negative value");
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
/* Use specified fdset ID */
|
||||||
|
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
||||||
|
mon_fdset_cur = mon_fdset;
|
||||||
|
if (fdset_id < mon_fdset_cur->id) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
/* Use first available fdset ID */
|
||||||
|
QLIST_FOREACH(mon_fdset, &mon_fdsets, next) {
|
||||||
|
mon_fdset_cur = mon_fdset;
|
||||||
|
if (fdset_id_prev == mon_fdset_cur->id - 1) {
|
||||||
|
fdset_id_prev = mon_fdset_cur->id;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mon_fdset = g_malloc0(sizeof(*mon_fdset));
|
||||||
|
if (has_fdset_id) {
|
||||||
|
mon_fdset->id = fdset_id;
|
||||||
|
} else {
|
||||||
|
mon_fdset->id = fdset_id_prev + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* The fdset list is ordered by fdset ID */
|
||||||
|
if (!mon_fdset_cur) {
|
||||||
|
QLIST_INSERT_HEAD(&mon_fdsets, mon_fdset, next);
|
||||||
|
} else if (mon_fdset->id < mon_fdset_cur->id) {
|
||||||
|
QLIST_INSERT_BEFORE(mon_fdset_cur, mon_fdset, next);
|
||||||
|
} else {
|
||||||
|
QLIST_INSERT_AFTER(mon_fdset_cur, mon_fdset, next);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mon_fdset_fd = g_malloc0(sizeof(*mon_fdset_fd));
|
||||||
|
mon_fdset_fd->fd = fd;
|
||||||
|
mon_fdset_fd->removed = false;
|
||||||
|
if (has_opaque) {
|
||||||
|
mon_fdset_fd->opaque = g_strdup(opaque);
|
||||||
|
}
|
||||||
|
QLIST_INSERT_HEAD(&mon_fdset->fds, mon_fdset_fd, next);
|
||||||
|
|
||||||
|
fdinfo = g_malloc0(sizeof(*fdinfo));
|
||||||
|
fdinfo->fdset_id = mon_fdset->id;
|
||||||
|
fdinfo->fd = mon_fdset_fd->fd;
|
||||||
|
|
||||||
|
return fdinfo;
|
||||||
|
}
|
||||||
|
|
||||||
int monitor_fdset_get_fd(int64_t fdset_id, int flags)
|
int monitor_fdset_get_fd(int64_t fdset_id, int flags)
|
||||||
{
|
{
|
||||||
#ifndef _WIN32
|
#ifndef _WIN32
|
||||||
|
@ -90,6 +90,9 @@ int qmp_qom_set(Monitor *mon, const QDict *qdict, QObject **ret);
|
|||||||
|
|
||||||
int qmp_qom_get(Monitor *mon, const QDict *qdict, QObject **ret);
|
int qmp_qom_get(Monitor *mon, const QDict *qdict, QObject **ret);
|
||||||
|
|
||||||
|
AddfdInfo *monitor_fdset_add_fd(int fd, bool has_fdset_id, int64_t fdset_id,
|
||||||
|
bool has_opaque, const char *opaque,
|
||||||
|
Error **errp);
|
||||||
int monitor_fdset_get_fd(int64_t fdset_id, int flags);
|
int monitor_fdset_get_fd(int64_t fdset_id, int flags);
|
||||||
int monitor_fdset_dup_fd_add(int64_t fdset_id, int dup_fd);
|
int monitor_fdset_dup_fd_add(int64_t fdset_id, int dup_fd);
|
||||||
int monitor_fdset_dup_fd_remove(int dup_fd);
|
int monitor_fdset_dup_fd_remove(int dup_fd);
|
||||||
|
Loading…
Reference in New Issue
Block a user