util: allow specifying both src and dst pid in virNetlinkCommand

Until now, virNetlinkCommand has assumed that the nl_pid in the source
address of outgoing netlink messages should always be the return value
of getpid(). In most cases it actually doesn't matter, but in the case
of communication with lldpad, lldpad saves this info and later uses it
to send netlink messages back to libvirt. A recent patch to fix Bug
816465 changed the order of the universe such that the netlink event
service socket is no longer bound with nl_pid == getpid(), so lldpad
could no longer send unsolicited messages to libvirtd. Adding src_pid
as an argument to virNetlinkCommand() is the first step in notifying
lldpad of the proper address of the netlink event service socket.
(cherry picked from commit cca7bb1fb583459824f7a42be0406d0833a80593)
This commit is contained in:
Laine Stump 2012-05-04 12:58:36 -04:00 committed by Cole Robinson
parent 3cc52164b1
commit 443e37da42
5 changed files with 15 additions and 14 deletions

View File

@ -1288,7 +1288,7 @@ virNetDevLinkDump(const char *ifname, int ifindex,
} }
} }
if (virNetlinkCommand(nl_msg, recvbuf, &recvbuflen, pid) < 0) if (virNetlinkCommand(nl_msg, recvbuf, &recvbuflen, 0, pid) < 0)
goto cleanup; goto cleanup;
if (recvbuflen < NLMSG_LENGTH(0) || *recvbuf == NULL) if (recvbuflen < NLMSG_LENGTH(0) || *recvbuf == NULL)
@ -1416,7 +1416,7 @@ virNetDevSetVfConfig(const char *ifname, int ifindex, int vf,
} }
} }
if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, pid) < 0) if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0, pid) < 0)
goto cleanup; goto cleanup;
if (recvbuflen < NLMSG_LENGTH(0) || recvbuf == NULL) if (recvbuflen < NLMSG_LENGTH(0) || recvbuf == NULL)

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (C) 2010-2011 Red Hat, Inc. * Copyright (C) 2010-2012 Red Hat, Inc.
* Copyright (C) 2010-2012 IBM Corporation * Copyright (C) 2010-2012 IBM Corporation
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
@ -154,7 +154,7 @@ virNetDevMacVLanCreate(const char *ifname,
nla_nest_end(nl_msg, linkinfo); nla_nest_end(nl_msg, linkinfo);
if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0) < 0) { if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0, 0) < 0) {
goto cleanup; goto cleanup;
} }
@ -242,7 +242,7 @@ int virNetDevMacVLanDelete(const char *ifname)
if (nla_put(nl_msg, IFLA_IFNAME, strlen(ifname)+1, ifname) < 0) if (nla_put(nl_msg, IFLA_IFNAME, strlen(ifname)+1, ifname) < 0)
goto buffer_too_small; goto buffer_too_small;
if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0) < 0) { if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0, 0) < 0) {
goto cleanup; goto cleanup;
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (C) 2009-2011 Red Hat, Inc. * Copyright (C) 2009-2012 Red Hat, Inc.
* *
* This library is free software; you can redistribute it and/or * This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public * modify it under the terms of the GNU Lesser General Public
@ -395,7 +395,7 @@ virNetDevVPortProfileOpSetLink(const char *ifname, int ifindex,
goto cleanup; goto cleanup;
} }
if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, pid) < 0) if (virNetlinkCommand(nl_msg, &recvbuf, &recvbuflen, 0, pid) < 0)
goto cleanup; goto cleanup;
if (recvbuflen < NLMSG_LENGTH(0) || recvbuf == NULL) if (recvbuflen < NLMSG_LENGTH(0) || recvbuf == NULL)

View File

@ -158,12 +158,12 @@ virNetlinkShutdown(void)
*/ */
int virNetlinkCommand(struct nl_msg *nl_msg, int virNetlinkCommand(struct nl_msg *nl_msg,
unsigned char **respbuf, unsigned int *respbuflen, unsigned char **respbuf, unsigned int *respbuflen,
int nl_pid) uint32_t src_pid, uint32_t dst_pid)
{ {
int rc = 0; int rc = 0;
struct sockaddr_nl nladdr = { struct sockaddr_nl nladdr = {
.nl_family = AF_NETLINK, .nl_family = AF_NETLINK,
.nl_pid = nl_pid, .nl_pid = dst_pid,
.nl_groups = 0, .nl_groups = 0,
}; };
ssize_t nbytes; ssize_t nbytes;
@ -191,7 +191,7 @@ int virNetlinkCommand(struct nl_msg *nl_msg,
nlmsg_set_dst(nl_msg, &nladdr); nlmsg_set_dst(nl_msg, &nladdr);
nlmsg->nlmsg_pid = getpid(); nlmsg->nlmsg_pid = src_pid ? src_pid : getpid();
nbytes = nl_send_auto_complete(nlhandle, nl_msg); nbytes = nl_send_auto_complete(nlhandle, nl_msg);
if (nbytes < 0) { if (nbytes < 0) {
@ -604,7 +604,8 @@ virNetlinkShutdown(void)
int virNetlinkCommand(struct nl_msg *nl_msg ATTRIBUTE_UNUSED, int virNetlinkCommand(struct nl_msg *nl_msg ATTRIBUTE_UNUSED,
unsigned char **respbuf ATTRIBUTE_UNUSED, unsigned char **respbuf ATTRIBUTE_UNUSED,
unsigned int *respbuflen ATTRIBUTE_UNUSED, unsigned int *respbuflen ATTRIBUTE_UNUSED,
int nl_pid ATTRIBUTE_UNUSED) uint32_t src_pid ATTRIBUTE_UNUSED,
uint32_t dst_pid ATTRIBUTE_UNUSED)
{ {
netlinkError(VIR_ERR_INTERNAL_ERROR, "%s", _(unsupported)); netlinkError(VIR_ERR_INTERNAL_ERROR, "%s", _(unsupported));
return -1; return -1;

View File

@ -40,7 +40,7 @@ void virNetlinkShutdown(void);
int virNetlinkCommand(struct nl_msg *nl_msg, int virNetlinkCommand(struct nl_msg *nl_msg,
unsigned char **respbuf, unsigned int *respbuflen, unsigned char **respbuf, unsigned int *respbuflen,
int nl_pid); uint32_t src_port, uint32_t dst_port);
typedef void (*virNetlinkEventHandleCallback)(unsigned char *msg, int length, struct sockaddr_nl *peer, bool *handled, void *opaque); typedef void (*virNetlinkEventHandleCallback)(unsigned char *msg, int length, struct sockaddr_nl *peer, bool *handled, void *opaque);