summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorChris Lumens <clumens@redhat.com>2009-08-03 13:58:28 -0400
committerChris Lumens <clumens@redhat.com>2009-08-04 10:41:16 -0400
commit4b954d1232992b33b7491a89689737842fe926c2 (patch)
treec3de693aa504b4d484f178a9baa147f60620689d
parenta2bbd30d46dbb68d6fa51bc8a55eec2f28976963 (diff)
downloadanaconda-4b954d1232992b33b7491a89689737842fe926c2.tar.gz
anaconda-4b954d1232992b33b7491a89689737842fe926c2.tar.xz
anaconda-4b954d1232992b33b7491a89689737842fe926c2.zip
Move basic udev methods out of the storage module (#514592).
The udev code can be used for more than just storage - it can be used for networking as well. Moving these methods out makes that a little more clear.
-rw-r--r--baseudev.py136
-rw-r--r--storage/udev.py117
2 files changed, 153 insertions, 100 deletions
diff --git a/baseudev.py b/baseudev.py
new file mode 100644
index 000000000..f441bceaa
--- /dev/null
+++ b/baseudev.py
@@ -0,0 +1,136 @@
+# udev.py
+# Python module for querying the udev database for device information.
+#
+# Copyright (C) 2009 Red Hat, Inc.
+#
+# This copyrighted material is made available to anyone wishing to use,
+# modify, copy, or redistribute it subject to the terms and conditions of
+# the GNU General Public License v.2, or (at your option) any later version.
+# This program is distributed in the hope that it will be useful, but WITHOUT
+# ANY WARRANTY expressed or implied, including the implied warranties of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
+# Public License for more details. You should have received a copy of the
+# GNU General Public License along with this program; if not, write to the
+# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+# 02110-1301, USA. Any Red Hat trademarks that are incorporated in the
+# source code or documentation are not subject to the GNU General Public
+# License and may only be used or replicated with the express permission of
+# Red Hat, Inc.
+#
+# Red Hat Author(s): Dave Lehman <dlehman@redhat.com>
+# Chris Lumens <clumens@redhat.com>
+#
+
+import iutil
+import os
+
+import logging
+log = logging.getLogger("storage")
+
+def udev_enumerate_devices(deviceClass="block"):
+ top_dir = "/sys/class/%s" % deviceClass
+ devices = []
+ for dev_name in os.listdir(top_dir):
+ full_path = os.path.join(top_dir, dev_name)
+ link_ref = os.readlink(full_path)
+ real_path = os.path.join(top_dir, link_ref)
+ sysfs_path = os.path.normpath(real_path)
+ devices.append(sysfs_path)
+ return devices
+
+def udev_get_device(sysfs_path):
+ if not os.path.exists(sysfs_path):
+ log.debug("%s does not exist" % sysfs_path)
+ return None
+
+ db_entry = sysfs_path[4:].replace("/", "\\x2f")
+ db_root = "/dev/.udev/db"
+ db_path = os.path.normpath("%s/%s" % (db_root, db_entry))
+ if not os.access(db_path, os.R_OK):
+ log.debug("db entry %s does not exist" % db_path)
+ return None
+
+ entry = open(db_path).read()
+ dev = udev_parse_entry(entry)
+ if dev.has_key("name"):
+ # XXX why do we do this? is /sys going to move during installation?
+ dev['sysfs_path'] = sysfs_path[4:] # strip off the leading '/sys'
+ dev = udev_parse_uevent_file(dev)
+
+ # now add in the contents of the uevent file since they're handy
+ return dev
+
+def udev_get_devices(deviceClass="block"):
+ udev_settle(timeout=30)
+ entries = []
+ for path in udev_enumerate_devices(deviceClass):
+ entry = udev_get_device(path)
+ if entry:
+ entries.append(entry)
+ return entries
+
+def udev_parse_entry(buf):
+ dev = {}
+
+ for line in buf.splitlines():
+ line.strip()
+ (tag, sep, val) = line.partition(":")
+ if not sep:
+ continue
+
+ if tag == "N":
+ dev['name'] = val
+ elif tag == "S":
+ if dev.has_key('symlinks'):
+ dev['symlinks'].append(val)
+ else:
+ dev['symlinks'] = [val]
+ elif tag == "E":
+ if val.count("=") > 1 and val.count(" ") > 0:
+ # eg: LVM2_LV_NAME when querying the VG for its LVs
+ vars = val.split()
+ vals = []
+ var_name = None
+ for (index, subval) in enumerate(vars):
+ (var_name, sep, var_val) = subval.partition("=")
+ if sep:
+ vals.append(var_val)
+
+ dev[var_name] = vals
+ else:
+ (var_name, sep, var_val) = val.partition("=")
+ if not sep:
+ continue
+
+ dev[var_name] = var_val
+
+ return dev
+
+def udev_parse_uevent_file(dev):
+ path = os.path.normpath("/sys/%s/uevent" % dev['sysfs_path'])
+ if not os.access(path, os.R_OK):
+ return dev
+
+ with open(path) as f:
+ for line in f.readlines():
+ (key, equals, value) = line.strip().partition("=")
+ if not equals:
+ continue
+
+ dev[key] = value
+
+ return dev
+
+def udev_settle(timeout=None):
+ argv = ["settle"]
+ if timeout:
+ argv.append("--timeout=%d" % int(timeout))
+
+ iutil.execWithRedirect("udevadm", argv, stderr="/dev/null", searchPath=1)
+
+def udev_trigger(subsystem=None):
+ argv = ["trigger"]
+ if subsystem:
+ argv.append("--subsystem-match=%s" % subsystem)
+
+ iutil.execWithRedirect("udevadm", argv, stderr="/dev/null", searchPath=1)
diff --git a/storage/udev.py b/storage/udev.py
index 051dc21e9..450b54aa6 100644
--- a/storage/udev.py
+++ b/storage/udev.py
@@ -25,6 +25,7 @@ import stat
import iutil
from errors import *
+from baseudev import *
import logging
log = logging.getLogger("storage")
@@ -56,7 +57,7 @@ def udev_resolve_devspec(devspec):
def udev_get_block_devices():
udev_settle(timeout=30)
entries = []
- for path in enumerate_block_devices():
+ for path in udev_enumerate_block_devices():
entry = udev_get_block_device(path)
if entry:
entries.append(entry)
@@ -87,112 +88,28 @@ def __is_blacklisted_blockdev(dev_name):
return False
-def enumerate_block_devices():
- top_dir = "/sys/class/block"
- devices = []
- for dev_name in os.listdir(top_dir):
- if __is_blacklisted_blockdev(dev_name):
- continue
- full_path = os.path.join(top_dir, dev_name)
- link_ref = os.readlink(full_path)
- real_path = os.path.join(top_dir, link_ref)
- sysfs_path = os.path.normpath(real_path)
- devices.append(sysfs_path)
- return devices
-
-def udev_get_block_device(sysfs_path, requireName=True):
- if not os.path.exists(sysfs_path):
- log.debug("%s does not exist" % sysfs_path)
- return None
-
- db_entry = sysfs_path[4:].replace("/", "\\x2f")
- db_root = "/dev/.udev/db"
- db_path = os.path.normpath("%s/%s" % (db_root, db_entry))
- if not os.access(db_path, os.R_OK):
- log.debug("db entry %s does not exist" % db_path)
- return None
+def udev_enumerate_block_devices():
+ import os.path
- entry = open(db_path).read()
- dev = udev_parse_block_entry(entry)
- if requireName and dev.has_key("name"):
- # XXX why do we do this? is /sys going to move during installation?
- dev['sysfs_path'] = sysfs_path[4:] # strip off the leading '/sys'
- dev = udev_parse_uevent_file(dev)
+ return filter(lambda d: not __is_blacklisted_blockdev(os.path.basename(d)),
+ udev_enumerate_devices(deviceClass="block"))
- # now add in the contents of the uevent file since they're handy
- return dev
-
-def udev_parse_uevent_file(dev):
- path = os.path.normpath("/sys/%s/uevent" % dev['sysfs_path'])
- if not os.access(path, os.R_OK):
+def udev_get_block_device(sysfs_path):
+ dev = udev_get_device(sysfs_path)
+ if not dev or not dev.has_key("name"):
+ return {"name": None, "symlinks": []}
+ else:
return dev
- with open(path) as f:
- for line in f.readlines():
- (key, equals, value) = line.strip().partition("=")
- if not equals:
- continue
-
- dev[key] = value
-
- return dev
-
def udev_parse_block_entry(buf):
- dev = {}
-
- for line in buf.splitlines():
- line.strip()
- (tag, sep, val) = line.partition(":")
- if not sep:
- continue
-
- if tag == "N":
- dev['name'] = val
- elif tag == "S":
- if dev.has_key('symlinks'):
- dev['symlinks'].append(val)
- else:
- dev['symlinks'] = [val]
- elif tag == "E":
- if val.count("=") > 1 and val.count(" ") > 0:
- # eg: LVM2_LV_NAME when querying the VG for its LVs
- vars = val.split()
- vals = []
- var_name = None
- for (index, subval) in enumerate(vars):
- (var_name, sep, var_val) = subval.partition("=")
- if sep:
- vals.append(var_val)
-
- dev[var_name] = vals
- else:
- (var_name, sep, var_val) = val.partition("=")
- if not sep:
- continue
-
- # Skip splitting for any keys matching MODEL or VENDOR, since
- # those values could include embedded, unquoted spaces.
- if var_val.count(" ") and var_name.find("MODEL") == -1 and var_name.find("VENDOR") == -1:
- # eg: DEVLINKS
- var_val = var_val.split()
-
- dev[var_name] = var_val
-
- return dev
+ dev = udev_parse_entry(buf)
-def udev_settle(timeout=None):
- argv = ["settle"]
- if timeout:
- argv.append("--timeout=%d" % int(timeout))
+ for (key, value) in dev.iteritems():
+ if value.count(" "):
+ # eg: DEVLINKS
+ dev.update({key: value.split()})
- iutil.execWithRedirect("udevadm", argv, stderr="/dev/null", searchPath=1)
-
-def udev_trigger(subsystem=None):
- argv = ["trigger"]
- if subsystem:
- argv.append("--subsystem-match=%s" % subsystem)
-
- iutil.execWithRedirect("udevadm", argv, stderr="/dev/null", searchPath=1)
+ return dev
# These are functions for retrieving specific pieces of information from