diff options
| author | David Sommerseth <davids@redhat.com> | 2010-07-30 19:30:30 +0200 |
|---|---|---|
| committer | David Sommerseth <davids@redhat.com> | 2010-07-30 19:30:30 +0200 |
| commit | e9aa46ab32a45bd7fc0ad32573d1db84f5049554 (patch) | |
| tree | a736740bf5c38a84cf0f7868d97bfc02fdf5aaed /python-ethtool/etherinfo_obj.c | |
| parent | 1d4b0d894dd833cd9ac9e62cbdc400f24a11e32b (diff) | |
| download | python-ethtool-e9aa46ab32a45bd7fc0ad32573d1db84f5049554.tar.gz python-ethtool-e9aa46ab32a45bd7fc0ad32573d1db84f5049554.tar.xz python-ethtool-e9aa46ab32a45bd7fc0ad32573d1db84f5049554.zip | |
Improved IPv6 support
As the IPv6 protocol allows a single device to have more than one IPv6 address,
the previous implementation did not provide all IPv6 information. It would reject
all except the last parsed IPv6 address.
NOTE: This implementation will break the previous API.
This change removes the ethtool.etherinfo.ipv6_address and
ethtool.etherinfo.ipv6_netmask members. A new member is added,
ethtool.etherinfo.ipv6_addresses (in plural). This contains a tupple list
containing of ethtool.etherinfo_ipv6addr objects, one object for each configured
IPv6 address on the device. These objects have the following members available:
.address - The IPv6 address
.netmask - The IPv6 netmask (in bit notation)
.scope - A string with the IPv6 address scope
Example code:
import ethtool
devs = ethtool.get_interfaces_info('eth0')
for ip6 in devs[0].ipv6_addresses:
print "[%s] %s/%i" % (ip6.scope, ip6.address, ip6.netmask)
Signed-off-by: David Sommerseth <davids@redhat.com>
Diffstat (limited to 'python-ethtool/etherinfo_obj.c')
| -rw-r--r-- | python-ethtool/etherinfo_obj.c | 60 |
1 files changed, 38 insertions, 22 deletions
diff --git a/python-ethtool/etherinfo_obj.c b/python-ethtool/etherinfo_obj.c index 814da78..384e915 100644 --- a/python-ethtool/etherinfo_obj.c +++ b/python-ethtool/etherinfo_obj.c @@ -10,9 +10,11 @@ #include <Python.h> #include "structmember.h" +#include <netlink/route/rtnl.h> #include "etherinfo_struct.h" #include "etherinfo.h" +extern PyTypeObject ethtool_etherinfoIPv6Type; /** * ethtool.etherinfo deallocator - cleans up when a object is deleted @@ -72,15 +74,6 @@ int _ethtool_etherinfo_init(etherinfo_py *self, PyObject *args, PyObject *kwds) } /** - * NULL safe PyString_FromString() wrapper. If input string is NULL, None will be returned - * - * @param str Input C string (char *) - * - * @return Returns a PyObject with either the input string wrapped up, or a Python None value. - */ -#define RETURN_STRING(str) (str ? PyString_FromString(str) : Py_None); - -/** * ethtool.etherinfo function for retrieving data from a Python object. * * @param self @@ -112,12 +105,29 @@ PyObject *_ethtool_etherinfo_getter(etherinfo_py *self, PyObject *attr_o) } else if( strcmp(attr, "ipv4_broadcast") == 0 ) { get_etherinfo(self->data->ethinfo, self->data->nlc, NLQRY_ADDR); ret = RETURN_STRING(self->data->ethinfo->ipv4_broadcast); - } else if( strcmp(attr, "ipv6_address") == 0 ) { - get_etherinfo(self->data->ethinfo, self->data->nlc, NLQRY_ADDR); - ret = RETURN_STRING(self->data->ethinfo->ipv6_address); - } else if( strcmp(attr, "ipv6_netmask") == 0 ) { + } else if( strcmp(attr, "ipv6_addresses") == 0 ) { + struct ipv6address *ipv6 = NULL; + int i = 0; + ret = PyTuple_New(1); + get_etherinfo(self->data->ethinfo, self->data->nlc, NLQRY_ADDR); - ret = PyInt_FromLong(self->data->ethinfo->ipv6_netmask); + ipv6 = self->data->ethinfo->ipv6_addresses; + while( ipv6 ) { + PyObject *ipv6_pyobj = NULL, *ipv6_pydata = NULL, *args = NULL; + struct ipv6address *next = ipv6->next; + + ipv6->next = NULL; + ipv6_pydata = PyCObject_FromVoidPtr(ipv6, NULL); + args = PyTuple_New(1); + PyTuple_SetItem(args, 0, ipv6_pydata); + ipv6_pyobj = PyObject_CallObject((PyObject *)ðtool_etherinfoIPv6Type, args); + if( ipv6_pyobj ) { + PyTuple_SetItem(ret, i++, ipv6_pyobj); + _PyTuple_Resize(&ret, i+1); + } + ipv6 = next; + } + _PyTuple_Resize(&ret, i); } else { ret = PyObject_GenericGetAttr((PyObject *)self, attr_o); } @@ -180,11 +190,19 @@ PyObject *_ethtool_etherinfo_str(etherinfo_py *self) PyString_Concat(&ret, tmp); } - if( self->data->ethinfo->ipv6_address ) { - PyObject *tmp = PyString_FromFormat("\tIPv6 address: %s/%i\n", - self->data->ethinfo->ipv6_address, - self->data->ethinfo->ipv6_netmask); + if( self->data->ethinfo->ipv6_addresses ) { + struct ipv6address *ipv6 = self->data->ethinfo->ipv6_addresses; + PyObject *tmp = PyString_FromFormat("\tIPv6 addresses:\n"); PyString_Concat(&ret, tmp); + + for( ; ipv6; ipv6 = ipv6->next) { + char scope[66]; + + rtnl_scope2str(ipv6->scope, scope, 64); + PyObject *addr = PyString_FromFormat("\t [%s] %s/%i\n", + scope, ipv6->address, ipv6->netmask); + PyString_Concat(&ret, addr); + } } return ret; } @@ -214,10 +232,8 @@ static PyMemberDef _ethtool_etherinfo_members[] = { "IPv4 netmask in bits"}, {"ipv4_broadcast", T_OBJECT_EX, offsetof(etherinfo_py, data), 0, "IPv4 broadcast address"}, - {"ipv6_address", T_OBJECT_EX, offsetof(etherinfo_py, data), 0, - "IPv6 address"}, - {"ipv6_netmask", T_INT, offsetof(etherinfo_py, data), 0, - "IPv6 netmask in bits"}, + {"ipv6_addresses", T_OBJECT_EX, offsetof(etherinfo_py, data), 0, + "Returns a list of associated IPv6 addresses"}, {NULL} /* End of member list */ }; |
