| #!/usr/bin/python -u | 
 | # | 
 | # Copyright 2016 IBM Corporation | 
 | # | 
 | # Licensed under the Apache License, Version 2.0 (the "License"); | 
 | # you may not use this file except in compliance with the License. | 
 | # You may obtain a copy of the License at | 
 | # | 
 | #   http://www.apache.org/licenses/LICENSE-2.0 | 
 | # | 
 | # Unless required by applicable law or agreed to in writing, software | 
 | # distributed under the License is distributed on an "AS IS" BASIS, | 
 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | 
 | # implied. | 
 | # See the License for the specific language governing permissions and | 
 | # limitations under the License. | 
 |  | 
 | import sys | 
 | import os | 
 | import dbus | 
 | import uuid | 
 | import argparse | 
 | import subprocess | 
 | import obmc.mapper | 
 | import shutil | 
 |  | 
 | INV_INTF_NAME = 'xyz.openbmc_project.Inventory.Item.NetworkInterface' | 
 | NET_DBUS_NAME = 'org.openbmc.NetworkManager' | 
 | NET_OBJ_NAME = '/org/openbmc/NetworkManager/Interface' | 
 | CHS_DBUS_NAME = 'org.openbmc.control.Chassis' | 
 | CHS_INTF_NAME = 'xyz.openbmc_project.Common.UUID' | 
 | CHS_OBJ_NAME = '/org/openbmc/control/chassis0' | 
 | PROP_INTF_NAME = 'org.freedesktop.DBus.Properties' | 
 | INVENTORY_ROOT = '/xyz/openbmc_project/inventory' | 
 | NETWORK_ROOT = '/xyz/openbmc_project/network' | 
 | ETHERNET_INTF_NAME = 'xyz.openbmc_project.Network.EthernetInterface' | 
 | MAC_INTF_NAME = 'xyz.openbmc_project.Network.MACAddress' | 
 |  | 
 | FRUS = {} | 
 |  | 
 | # IEEE 802 MAC address mask for locally administered. | 
 | # This means the admin has set the MAC and is no longer | 
 | # the unique number set by the device manufacturer. | 
 | MAC_LOCALLY_ADMIN_MASK = 0x20000000000 | 
 |  | 
 |  | 
 | # Get inventory MACAddress value. | 
 | def get_bmc_mac_address(bus, prop): | 
 |     mapper = obmc.mapper.Mapper(bus) | 
 |  | 
 |     # Get the inventory subtree, limited | 
 |     # to objects that implement NetworkInterface. | 
 |     for path, info in \ | 
 |         mapper.get_subtree( | 
 |             path=INVENTORY_ROOT, | 
 |             interfaces=[INV_INTF_NAME]).iteritems(): | 
 |  | 
 |             # Find a NetworkInterface with 'bmc' in the path. | 
 |             if 'bmc' not in path: | 
 |                 continue | 
 |  | 
 |             # Only expecting a single service to implement | 
 |             # NetworkInterface.  Get the service connection | 
 |             # from the mapper response | 
 |             conn = info.keys()[0] | 
 |  | 
 |             # Get the inventory object implementing NetworkInterface. | 
 |             obj = bus.get_object(conn, path) | 
 |  | 
 |             # Get the MAC address | 
 |             mproxy = obj.get_dbus_method('Get', PROP_INTF_NAME) | 
 |             return mproxy(INV_INTF_NAME, prop) | 
 |  | 
 |  | 
 | # Get Network Interface object. | 
 | def get_network_interface_object(bus): | 
 |     mapper = obmc.mapper.Mapper(bus) | 
 |  | 
 |     # Get the network subtree, limited | 
 |     # to objects that implements EthernetInterface. | 
 |     for path, info in \ | 
 |         mapper.get_subtree( | 
 |             path=NETWORK_ROOT, | 
 |             interfaces=[ETHERNET_INTF_NAME]).iteritems(): | 
 |  | 
 |         # Find the one which is having physical interface,it may happen | 
 |         # that vlan interface is there and we want the physical | 
 |         # interface here. | 
 |         if path.split('/')[-1].find('_') < 0: | 
 |             service = info.keys()[0] | 
 |             net_obj = bus.get_object(service, path) | 
 |             return net_obj | 
 |  | 
 |  | 
 | # Get inventory UUID value. | 
 | def get_uuid(bus, prop): | 
 |     mapper = obmc.mapper.Mapper(bus) | 
 |  | 
 |     # Get the inventory subtree, limited | 
 |     # to objects that implement UUID. | 
 |     resp = mapper.get_subtree( | 
 |         path=INVENTORY_ROOT, | 
 |         interfaces=[CHS_INTF_NAME]) | 
 |  | 
 |     # Only expecting a single object to implement UUID. | 
 |     try: | 
 |         path, info = resp.items()[0] | 
 |     except IndexError as e: | 
 |         return None | 
 |  | 
 |     # Only expecting a single service to implement | 
 |     # UUID.  Get the service connection | 
 |     # from the mapper response | 
 |     conn = info.keys()[0] | 
 |  | 
 |     # Get the inventory object implementing UUID. | 
 |     obj = bus.get_object(conn, path) | 
 |  | 
 |     # Get the uuid | 
 |     mproxy = obj.get_dbus_method('Get', PROP_INTF_NAME) | 
 |     return mproxy(CHS_INTF_NAME, prop) | 
 |  | 
 |  | 
 | # Get the value of the mac on the system (from u-boot) without ':' separators | 
 | def get_sys_mac(obj): | 
 |     sys_mac = '' | 
 |     try: | 
 |         sys_mac = subprocess.check_output(["fw_printenv", "-n", "ethaddr"]) | 
 |     except Exception: | 
 |         # Handle when mac does not exist in u-boot | 
 |         return sys_mac | 
 |     return sys_mac | 
 |  | 
 |  | 
 | # Replace the value of the system mac with the value of the inventory | 
 | # MAC if the system MAC is not locally administered because this means | 
 | # the system admin has purposely set the MAC | 
 | def sync_mac(obj, inv_mac, sys_mac): | 
 |     if sys_mac: | 
 |         # Convert sys MAC to int to perform bitwise '&' | 
 |         sys_mac = sys_mac.replace(":", "") | 
 |         int_sys_mac = int(sys_mac, 16) | 
 |     else: | 
 |         # Set mac to 0 for when u-boot mac is not present | 
 |         int_sys_mac = 0 | 
 |     if not int_sys_mac & MAC_LOCALLY_ADMIN_MASK: | 
 |         # Sys MAC is not locally administered, go replace it with inv value | 
 |         intf = dbus.Interface(obj, dbus.PROPERTIES_IFACE) | 
 |         intf.Set(MAC_INTF_NAME, "MACAddress", inv_mac) | 
 |  | 
 |  | 
 | # Set sys uuid, this reboots the BMC for the value to take effect | 
 | def set_sys_uuid(uuid): | 
 |     rc = subprocess.call(["fw_setenv", "uuid", uuid]) | 
 |     if rc == 0: | 
 |         print "Rebooting BMC to set uuid" | 
 |         # TODO Uncomment once sync from u-boot to /etc/machine-id is in place | 
 |         # Issue openbmc/openbmc#479 | 
 |         # rc = subprocess.call(["reboot"]) | 
 |     else: | 
 |         print "Error setting uuid" | 
 |  | 
 |  | 
 | if __name__ == '__main__': | 
 |     arg = argparse.ArgumentParser() | 
 |     arg.add_argument('-p') | 
 |     arg.add_argument('-s') | 
 |  | 
 |     opt = arg.parse_args() | 
 |     prop_name = opt.p | 
 |     sync_type = opt.s | 
 |  | 
 |     bus = dbus.SystemBus() | 
 |     if sync_type == "mac": | 
 |         inv_mac = get_bmc_mac_address(bus, prop_name) | 
 |         if not inv_mac: | 
 |             sys.exit(1) | 
 |         net_obj = get_network_interface_object(bus) | 
 |         if not net_obj: | 
 |             print "Unable to get the network object" | 
 |             sys.exit(1) | 
 |         sys_mac = get_sys_mac(net_obj) | 
 |         if inv_mac != sys_mac: | 
 |             print "Inventory MAC=%s,System MAC=%s" % (inv_mac, sys_mac) | 
 |             sync_mac(net_obj, inv_mac, sys_mac) | 
 |     elif sync_type == "uuid": | 
 |             inv_uuid = get_uuid(bus, prop_name) | 
 |             if inv_uuid: | 
 |                 inv_uuid = uuid.UUID(inv_uuid) | 
 |                 chs_obj = bus.get_object(CHS_DBUS_NAME, CHS_OBJ_NAME) | 
 |                 chs_uuid = get_sys_uuid(chs_obj) | 
 |                 if inv_uuid != sys_uuid: | 
 |                     set_sys_uuid(inv_uuid) | 
 |  | 
 | # vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4 |