2007-01-16 00:42:10 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# domUloader.py
|
|
|
|
"""Loader for kernel and (optional) ramdisk from domU filesystem
|
2007-02-11 11:48:10 +01:00
|
|
|
|
|
|
|
Given a physical disk (or disk image) for a domU and the path of a kernel and
|
|
|
|
optional ramdisk, copies the kernel and ramdisk from the domU disk to a
|
|
|
|
temporary location in dom0.
|
|
|
|
|
|
|
|
The --entry parameter specifies the location of the kernel (and optional
|
|
|
|
ramdisk) within the domU filesystem. dev is the disk as seen by domU.
|
|
|
|
Filenames are relative to that filesystem.
|
|
|
|
|
|
|
|
The disk is passed as the last parameter. It must be a block device or raw
|
|
|
|
disk image. More complex disk images (QCOW, VMDK, etc) must already be
|
|
|
|
configured via blktap and presented as a block device.
|
|
|
|
|
|
|
|
The script writes an sxpr specifying the locations of the copied kernel and
|
|
|
|
ramdisk into the file specified by --output (default is stdout).
|
|
|
|
|
|
|
|
Limitations:
|
|
|
|
- It is assumed both kernel and ramdisk are on the same filesystem.
|
|
|
|
- domUs might use LVM; the script currently does not have support for setting
|
|
|
|
up LVM mappings for domUs; it's not trivial and we might risk namespace
|
|
|
|
conflicts. If you want to use LVM inside domUs, set up a small non-LVM boot
|
|
|
|
partition and specify it in bootentry.
|
2007-01-16 00:42:10 +01:00
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
The script uses kpartx (multipath-tools) to create mappings for devices that
|
|
|
|
are exported as whole disk devices that are partitioned.
|
2007-01-16 00:42:10 +01:00
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
(c) 01/2006 Novell Inc
|
|
|
|
License: GNU GPL
|
|
|
|
Author: Kurt Garloff <garloff@suse.de>
|
2007-01-16 00:42:10 +01:00
|
|
|
"""
|
|
|
|
|
|
|
|
import os, sys, getopt
|
|
|
|
from stat import *
|
|
|
|
from xen.xend import sxp
|
|
|
|
import tempfile
|
|
|
|
import time
|
|
|
|
|
|
|
|
# Global options
|
|
|
|
quiet = False
|
|
|
|
verbose = False
|
2007-02-11 11:48:10 +01:00
|
|
|
dryrun = False
|
2007-01-16 00:42:10 +01:00
|
|
|
tmpdir = '/var/lib/xen/tmp'
|
2007-03-06 09:12:51 +01:00
|
|
|
in_args = ''
|
2007-11-19 18:19:02 +01:00
|
|
|
# kpartx, left to its own devices, does not consistently pick the
|
|
|
|
# same partition separator. Explicitly specify it.
|
|
|
|
kpartx_args = '-p -part'
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
# Helper functions
|
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
def error(s):
|
|
|
|
print >> sys.stderr, "domUloader error: %s" % s
|
|
|
|
|
|
|
|
def verbose_print(s):
|
|
|
|
if verbose:
|
|
|
|
print >> sys.stderr, "domUloader: %s" % s
|
|
|
|
|
2007-01-16 00:42:10 +01:00
|
|
|
def traildigits(strg):
|
2007-02-11 11:48:10 +01:00
|
|
|
"""Return the trailing digits, used to split the partition number off"""
|
2007-01-16 00:42:10 +01:00
|
|
|
idx = len(strg)-1
|
|
|
|
while strg[idx].isdigit():
|
|
|
|
if len == 0:
|
|
|
|
return strg
|
|
|
|
idx -= 1
|
|
|
|
return strg[idx+1:]
|
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
def getWholedisk(part):
|
|
|
|
while len(part) and part[len(part)-1].isdigit():
|
|
|
|
part = part[:-1]
|
|
|
|
return part
|
2007-01-16 00:42:10 +01:00
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
#def isWholedisk(domUname):
|
|
|
|
# """Determines whether dev is a wholedisk dev"""
|
|
|
|
# return not domUname[-1:].isdigit()
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
|
|
|
|
class Wholedisk:
|
2007-02-11 11:48:10 +01:00
|
|
|
"Class representing a whole disk that may have partitions"
|
|
|
|
def __init__(self, vdev, pdev):
|
2007-01-16 00:42:10 +01:00
|
|
|
"c'tor: set up"
|
2007-11-19 18:19:02 +01:00
|
|
|
# Initialize object; will not raise:
|
2007-02-11 11:48:10 +01:00
|
|
|
self.ldev = None
|
|
|
|
self.vdev = vdev
|
|
|
|
self.pdev = pdev
|
|
|
|
self.mapped = 0
|
|
|
|
self.partitions = []
|
2007-07-29 10:57:36 +02:00
|
|
|
self.pcount = 0
|
2007-11-19 18:19:02 +01:00
|
|
|
# Finish initialization; may raise:
|
2007-07-29 10:57:36 +02:00
|
|
|
self.is_blk = (S_ISBLK(os.stat(pdev)[ST_MODE]))
|
2007-02-11 11:48:10 +01:00
|
|
|
self.pcount = self.scanpartitions()
|
|
|
|
|
|
|
|
def physdev(self):
|
|
|
|
"""Gets the physical device used to access the device from dom0"""
|
|
|
|
if self.ldev:
|
|
|
|
return self.ldev
|
|
|
|
return self.pdev
|
|
|
|
|
|
|
|
def findPart(self, vdev):
|
|
|
|
"Find device dev in list of partitions"
|
|
|
|
if len(vdev) > 5 and vdev[:5] == "/dev/":
|
|
|
|
vdev = vdev[5:]
|
|
|
|
for part in self.partitions:
|
|
|
|
if vdev == part.vdev:
|
|
|
|
return part
|
|
|
|
return None
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
def loopsetup(self):
|
2007-02-11 11:48:10 +01:00
|
|
|
"""Sets up the loop mapping for a disk image.
|
|
|
|
|
|
|
|
Will raise if no loopbacks are available.
|
|
|
|
"""
|
|
|
|
if not self.is_blk and not self.ldev:
|
2007-01-16 00:42:10 +01:00
|
|
|
# Loops through all loopback devices, attempting to
|
|
|
|
# find a free one to set up. Don't scan for free and
|
|
|
|
# then try to set it up as a separate step - too racy!
|
|
|
|
i = 0
|
|
|
|
while True:
|
|
|
|
ldev = '/dev/loop%i' % (i)
|
|
|
|
if not os.path.exists(ldev):
|
|
|
|
break
|
|
|
|
i += 1
|
2007-05-21 22:28:46 +02:00
|
|
|
fd = os.popen("losetup %s '%s' 2> /dev/null" % (ldev, self.pdev))
|
2007-01-16 00:42:10 +01:00
|
|
|
if not fd.close():
|
2007-05-21 22:28:46 +02:00
|
|
|
verbose_print("losetup %s '%s'" % (ldev, self.pdev))
|
2007-02-11 11:48:10 +01:00
|
|
|
self.ldev = ldev
|
2007-01-16 00:42:10 +01:00
|
|
|
break
|
2007-02-11 11:48:10 +01:00
|
|
|
if not self.ldev:
|
|
|
|
raise RuntimeError("No free loop device found")
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
def loopclean(self):
|
2007-02-11 11:48:10 +01:00
|
|
|
"""Delete the loop mapping.
|
|
|
|
|
|
|
|
Will never raise.
|
|
|
|
"""
|
|
|
|
if self.ldev:
|
|
|
|
verbose_print("losetup -d %s" % self.ldev)
|
2007-01-16 00:42:10 +01:00
|
|
|
# Even seemingly innocent queries like "losetup /dev/loop0"
|
|
|
|
# can temporarily block the loopback and cause transient
|
|
|
|
# failures deleting the loopback, hence the retry logic.
|
|
|
|
retries = 10
|
|
|
|
while retries:
|
2007-02-11 11:48:10 +01:00
|
|
|
fd = os.popen("losetup -d %s" % self.ldev)
|
2007-01-16 00:42:10 +01:00
|
|
|
if not fd.close():
|
2007-02-11 11:48:10 +01:00
|
|
|
self.ldev = None
|
2007-01-16 00:42:10 +01:00
|
|
|
break
|
|
|
|
else:
|
2009-01-09 02:09:16 +01:00
|
|
|
# Mappings may not have been deleted due to race
|
|
|
|
# between udev and dm - see bnc#379032. Causes
|
|
|
|
# loop devices to leak. Call kpartx -d again
|
|
|
|
os.system("kpartx %s -d '%s'" % (kpartx_args, self.physdev()))
|
2007-01-16 00:42:10 +01:00
|
|
|
time.sleep(0.1)
|
|
|
|
retries -= 1
|
|
|
|
|
|
|
|
def scanpartitions(self):
|
|
|
|
"""Scan device for partitions (kpartx -l) and set up data structures,
|
|
|
|
Returns number of partitions found."""
|
|
|
|
self.loopsetup()
|
|
|
|
# TODO: We could use fdisk -l instead and look at the type of
|
|
|
|
# partitions; this way we could also detect LVM and support it.
|
2007-11-19 18:19:02 +01:00
|
|
|
fd = os.popen("kpartx %s -l '%s'" % (kpartx_args, self.physdev()))
|
2007-01-16 00:42:10 +01:00
|
|
|
pcount = 0
|
|
|
|
for line in fd.readlines():
|
|
|
|
line = line.strip()
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print("kpartx -l: %s" % (line,))
|
2009-05-04 18:38:09 +02:00
|
|
|
(pname, params) = line.split(' : ')
|
2007-05-21 22:28:46 +02:00
|
|
|
pname = pname.strip()
|
|
|
|
pno = int(traildigits(pname))
|
2007-01-16 00:42:10 +01:00
|
|
|
#if pname.rfind('/') != -1:
|
|
|
|
# pname = pname[pname.rfind('/')+1:]
|
2007-02-11 11:48:10 +01:00
|
|
|
#pname = self.pdev[:self.pdev.rfind('/')] + '/' + pname
|
2007-01-16 00:42:10 +01:00
|
|
|
pname = "/dev/mapper/" + pname
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print("Found partition: vdev %s, pdev %s" % ('%s%i' % (self.vdev, pno), pname))
|
|
|
|
self.partitions.append(Partition(self, '%s%i' % (self.vdev, pno), pname))
|
2007-01-16 00:42:10 +01:00
|
|
|
pcount += 1
|
|
|
|
fd.close()
|
|
|
|
if not pcount:
|
2007-02-11 11:48:10 +01:00
|
|
|
if self.ldev:
|
2007-01-16 00:42:10 +01:00
|
|
|
ref = self
|
|
|
|
else:
|
|
|
|
ref = None
|
2007-02-11 11:48:10 +01:00
|
|
|
self.partitions.append(Partition(ref, self.vdev, self.pdev))
|
2007-01-16 00:42:10 +01:00
|
|
|
return pcount
|
|
|
|
|
|
|
|
def activatepartitions(self):
|
|
|
|
"Set up loop mapping and device-mapper mappings"
|
|
|
|
if not self.mapped:
|
|
|
|
self.loopsetup()
|
|
|
|
if self.pcount:
|
2007-11-19 18:19:02 +01:00
|
|
|
verbose_print("kpartx %s -a '%s'" % (kpartx_args, self.physdev()))
|
|
|
|
fd = os.popen("kpartx %s -a '%s'" % (kpartx_args, self.physdev()))
|
2007-01-16 00:42:10 +01:00
|
|
|
fd.close()
|
|
|
|
self.mapped += 1
|
|
|
|
|
|
|
|
def deactivatepartitions(self):
|
2007-02-11 11:48:10 +01:00
|
|
|
"""Remove device-mapper mappings and loop mapping.
|
|
|
|
|
|
|
|
Will never raise.
|
|
|
|
"""
|
2007-01-16 00:42:10 +01:00
|
|
|
if not self.mapped:
|
|
|
|
return
|
|
|
|
self.mapped -= 1
|
|
|
|
if not self.mapped:
|
|
|
|
if self.pcount:
|
2007-11-19 18:19:02 +01:00
|
|
|
verbose_print("kpartx %s -d '%s'" % (kpartx_args, self.physdev()))
|
|
|
|
fd = os.popen("kpartx %s -d '%s'" % (kpartx_args, self.physdev()))
|
2007-01-16 00:42:10 +01:00
|
|
|
fd.close()
|
|
|
|
self.loopclean()
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
"d'tor: clean up"
|
|
|
|
self.deactivatepartitions()
|
|
|
|
self.loopclean()
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
"string representation for debugging"
|
2007-02-11 11:48:10 +01:00
|
|
|
strg = "[" + self.vdev + "," + self.pdev + ","
|
|
|
|
if self.ldev:
|
|
|
|
strg += self.ldev
|
2007-01-16 00:42:10 +01:00
|
|
|
strg += "," + str(self.pcount) + ",mapped %ix]" % self.mapped
|
|
|
|
return strg
|
|
|
|
|
|
|
|
class Partition:
|
|
|
|
"""Class representing a domU filesystem (partition) that can be
|
|
|
|
mounted in dom0"""
|
2007-02-11 11:48:10 +01:00
|
|
|
def __init__(self, whole = None, vdev = None, pdev = None):
|
2007-01-16 00:42:10 +01:00
|
|
|
"c'tor: setup"
|
2007-02-11 11:48:10 +01:00
|
|
|
self.wholedisk = whole
|
|
|
|
self.vdev = vdev
|
|
|
|
self.pdev = pdev
|
2007-01-16 00:42:10 +01:00
|
|
|
self.mountpoint = None
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
"d'tor: cleanup"
|
|
|
|
if self.mountpoint:
|
|
|
|
self.umount()
|
|
|
|
# Not needed: Refcounting will take care of it.
|
|
|
|
#if self.wholedisk:
|
|
|
|
# self.wholedisk.deactivatepartitions()
|
|
|
|
|
|
|
|
def __repr__(self):
|
|
|
|
"string representation for debugging"
|
2007-02-11 11:48:10 +01:00
|
|
|
strg = "[" + self.vdev + "," + self.pdev + ","
|
2007-01-16 00:42:10 +01:00
|
|
|
if self.mountpoint:
|
|
|
|
strg += "mounted on " + self.mountpoint + ","
|
|
|
|
else:
|
|
|
|
strg += "not mounted,"
|
|
|
|
if self.wholedisk:
|
|
|
|
return strg + self.wholedisk.__repr__() + "]"
|
|
|
|
else:
|
|
|
|
return strg + "]"
|
|
|
|
|
|
|
|
def mount(self, fstype = None, options = "ro"):
|
|
|
|
"mount filesystem, sets self.mountpoint"
|
|
|
|
if self.mountpoint:
|
|
|
|
return
|
|
|
|
if self.wholedisk:
|
|
|
|
self.wholedisk.activatepartitions()
|
2007-02-11 11:48:10 +01:00
|
|
|
mtpt = tempfile.mkdtemp(prefix = "%s." % self.vdev, dir = tmpdir)
|
2007-01-16 00:42:10 +01:00
|
|
|
mopts = ""
|
|
|
|
if fstype:
|
|
|
|
mopts += " -t %s" % fstype
|
2007-07-29 10:57:36 +02:00
|
|
|
if options:
|
|
|
|
mopts += " -o %s" % options
|
2007-05-21 22:28:46 +02:00
|
|
|
verbose_print("mount %s '%s' %s" % (mopts, self.pdev, mtpt))
|
|
|
|
fd = os.popen("mount %s '%s' %s" % (mopts, self.pdev, mtpt))
|
2007-01-16 00:42:10 +01:00
|
|
|
err = fd.close()
|
|
|
|
if err:
|
2007-07-29 10:57:36 +02:00
|
|
|
try:
|
|
|
|
os.rmdir(mtpt)
|
|
|
|
except:
|
|
|
|
pass
|
2007-05-21 22:28:46 +02:00
|
|
|
raise RuntimeError("Error %i from mount %s '%s' on %s" % \
|
2007-02-11 11:48:10 +01:00
|
|
|
(err, mopts, self.pdev, mtpt))
|
2007-01-16 00:42:10 +01:00
|
|
|
self.mountpoint = mtpt
|
|
|
|
|
|
|
|
def umount(self):
|
2007-02-11 11:48:10 +01:00
|
|
|
"""umount filesystem at self.mountpoint"""
|
2007-01-16 00:42:10 +01:00
|
|
|
if not self.mountpoint:
|
|
|
|
return
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print("umount %s" % self.mountpoint)
|
2007-01-16 00:42:10 +01:00
|
|
|
fd = os.popen("umount %s" % self.mountpoint)
|
|
|
|
err = fd.close()
|
2007-02-11 11:48:10 +01:00
|
|
|
try:
|
|
|
|
os.rmdir(self.mountpoint)
|
|
|
|
except:
|
|
|
|
pass
|
2007-01-16 00:42:10 +01:00
|
|
|
if err:
|
2007-02-11 11:48:10 +01:00
|
|
|
error("Error %i from umount %s" % (err, self.mountpoint))
|
|
|
|
else:
|
|
|
|
self.mountpoint = None
|
2007-01-16 00:42:10 +01:00
|
|
|
if self.wholedisk:
|
|
|
|
self.wholedisk.deactivatepartitions()
|
|
|
|
|
|
|
|
def parseEntry(entry):
|
2007-02-11 11:48:10 +01:00
|
|
|
"disects bootentry and returns vdev, kernel, ramdisk"
|
|
|
|
def bad():
|
|
|
|
raise RuntimeError, "Malformed --entry"
|
2007-01-16 00:42:10 +01:00
|
|
|
fsspl = entry.split(':')
|
2007-02-11 11:48:10 +01:00
|
|
|
if len(fsspl) != 2:
|
|
|
|
bad()
|
|
|
|
vdev = fsspl[0]
|
|
|
|
entry = fsspl[1]
|
2007-01-16 00:42:10 +01:00
|
|
|
enspl = entry.split(',')
|
2007-02-11 11:48:10 +01:00
|
|
|
if len(enspl) not in (1, 2):
|
|
|
|
bad()
|
2007-01-16 00:42:10 +01:00
|
|
|
# Prepend '/' if missing
|
|
|
|
kernel = enspl[0]
|
2007-02-11 11:48:10 +01:00
|
|
|
if kernel == '':
|
|
|
|
bad()
|
2007-01-16 00:42:10 +01:00
|
|
|
if kernel[0] != '/':
|
|
|
|
kernel = '/' + kernel
|
2007-02-11 11:48:10 +01:00
|
|
|
ramdisk = None
|
2007-01-16 00:42:10 +01:00
|
|
|
if len(enspl) > 1:
|
2007-02-11 11:48:10 +01:00
|
|
|
ramdisk = enspl[1]
|
|
|
|
if ramdisk != '' and ramdisk[0] != '/':
|
|
|
|
ramdisk = '/' + ramdisk
|
|
|
|
return vdev, kernel, ramdisk
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
def copyFile(src, dst):
|
|
|
|
"Wrapper for shutil.filecopy"
|
|
|
|
import shutil
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print("cp %s %s" % (src, dst))
|
2007-01-16 00:42:10 +01:00
|
|
|
stat = os.stat(src)
|
|
|
|
if stat.st_size > 16*1024*1024:
|
2007-02-11 11:48:10 +01:00
|
|
|
raise RuntimeError("Too large file %s (%s larger than 16MB)" \
|
2007-01-16 00:42:10 +01:00
|
|
|
% (src, stat.st_size))
|
|
|
|
try:
|
|
|
|
shutil.copyfile(src, dst)
|
|
|
|
except:
|
|
|
|
os.unlink(dst)
|
|
|
|
raise()
|
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
def copyKernelAndRamdisk(disk, vdev, kernel, ramdisk):
|
|
|
|
"""Finds vdev in list of partitions, mounts the partition, copies
|
|
|
|
kernel [and ramdisk] off to dom0 files, umounts the parition again,
|
2007-01-16 00:42:10 +01:00
|
|
|
and returns sxpr pointing to these copies."""
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print("copyKernelAndRamdisk(%s, %s, %s, %s)" % (disk, vdev, kernel, ramdisk))
|
2007-01-16 00:42:10 +01:00
|
|
|
if dryrun:
|
2007-02-11 11:48:10 +01:00
|
|
|
return "linux (kernel kernel.dummy) (ramdisk ramdisk.dummy)"
|
|
|
|
part = disk.findPart(vdev)
|
2007-01-16 00:42:10 +01:00
|
|
|
if not part:
|
2007-02-11 11:48:10 +01:00
|
|
|
raise RuntimeError("Partition '%s' does not exist" % vdev)
|
2007-01-16 00:42:10 +01:00
|
|
|
part.mount()
|
|
|
|
try:
|
2007-02-11 11:48:10 +01:00
|
|
|
(fd, knm) = tempfile.mkstemp(prefix = "kernel.", dir = tmpdir)
|
2007-01-16 00:42:10 +01:00
|
|
|
os.close(fd)
|
|
|
|
copyFile(part.mountpoint + kernel, knm)
|
|
|
|
except:
|
|
|
|
os.unlink(knm)
|
2007-02-11 11:48:10 +01:00
|
|
|
part.umount()
|
2007-01-16 00:42:10 +01:00
|
|
|
raise
|
|
|
|
if not quiet:
|
2007-02-11 11:48:10 +01:00
|
|
|
print "Copy kernel %s from %s to %s for booting" % (kernel, vdev, knm)
|
2007-01-16 00:42:10 +01:00
|
|
|
sxpr = "linux (kernel %s)" % knm
|
2007-02-11 11:48:10 +01:00
|
|
|
if ramdisk:
|
2007-01-16 00:42:10 +01:00
|
|
|
try:
|
2007-02-11 11:48:10 +01:00
|
|
|
(fd, inm) = tempfile.mkstemp(prefix = "ramdisk.", dir = tmpdir)
|
2007-01-16 00:42:10 +01:00
|
|
|
os.close(fd)
|
2007-02-11 11:48:10 +01:00
|
|
|
copyFile(part.mountpoint + ramdisk, inm)
|
2007-01-16 00:42:10 +01:00
|
|
|
except:
|
|
|
|
os.unlink(knm)
|
|
|
|
os.unlink(inm)
|
2007-02-11 11:48:10 +01:00
|
|
|
part.umount()
|
2007-01-16 00:42:10 +01:00
|
|
|
raise
|
|
|
|
sxpr += "(ramdisk %s)" % inm
|
|
|
|
part.umount()
|
|
|
|
return sxpr
|
|
|
|
|
|
|
|
def main(argv):
|
|
|
|
"Main routine: Parses options etc."
|
2007-03-06 09:12:51 +01:00
|
|
|
global quiet, dryrun, verbose, tmpdir, in_args
|
2007-01-16 00:42:10 +01:00
|
|
|
def usage():
|
|
|
|
"Help output (usage info)"
|
|
|
|
global verbose, quiet, dryrun
|
2007-02-11 11:48:10 +01:00
|
|
|
print >> sys.stderr, "domUloader usage: domUloader [--output=fd] [--quiet] [--dryrun] [--verbose]\n" +\
|
2007-03-06 09:12:51 +01:00
|
|
|
"[--args] [--help] --entry=dev:kernel[,ramdisk] physdisk [virtdisk]\n"
|
2007-01-16 00:42:10 +01:00
|
|
|
print >> sys.stderr, __doc__
|
|
|
|
|
|
|
|
try:
|
|
|
|
(optlist, args) = getopt.gnu_getopt(argv, 'qvh', \
|
2008-04-26 11:41:49 +02:00
|
|
|
('entry=', 'output=', 'tmpdir=', 'args=', 'kernel=', 'ramdisk=', 'help', 'quiet', 'dryrun', 'verbose'))
|
2007-01-16 00:42:10 +01:00
|
|
|
except:
|
|
|
|
usage()
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
entry = None
|
|
|
|
output = None
|
2007-02-11 11:48:10 +01:00
|
|
|
pdisk = None
|
|
|
|
vdisk = None
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
for (opt, oarg) in optlist:
|
|
|
|
if opt in ('-h', '--help'):
|
|
|
|
usage()
|
2007-02-11 11:48:10 +01:00
|
|
|
sys.exit(1)
|
2007-01-16 00:42:10 +01:00
|
|
|
elif opt in ('-q', '--quiet'):
|
|
|
|
quiet = True
|
|
|
|
elif opt in ('-n', '--dryrun'):
|
|
|
|
dryrun = True
|
|
|
|
elif opt in ('-v', '--verbose'):
|
|
|
|
verbose = True
|
|
|
|
elif opt == '--output':
|
|
|
|
output = oarg
|
|
|
|
elif opt == '--entry':
|
|
|
|
entry = oarg
|
|
|
|
elif opt == '--tmpdir':
|
|
|
|
tmpdir = oarg
|
2007-03-06 09:12:51 +01:00
|
|
|
elif opt == '--args':
|
|
|
|
in_args = oarg
|
2007-01-16 00:42:10 +01:00
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
verbose_print(str(argv))
|
|
|
|
|
|
|
|
if args:
|
|
|
|
if len(args) == 2:
|
|
|
|
pdisk = args[1]
|
|
|
|
elif len(args) == 3:
|
|
|
|
pdisk = args[1]
|
|
|
|
vdisk = args[2]
|
|
|
|
|
|
|
|
if not entry or not pdisk:
|
2007-01-16 00:42:10 +01:00
|
|
|
usage()
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
if output is None or output == "-":
|
|
|
|
fd = sys.stdout.fileno()
|
|
|
|
else:
|
|
|
|
fd = os.open(output, os.O_WRONLY)
|
|
|
|
|
|
|
|
if not os.access(tmpdir, os.X_OK):
|
|
|
|
os.mkdir(tmpdir)
|
|
|
|
os.chmod(tmpdir, 0750)
|
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
vdev, kernel, ramdisk = parseEntry(entry)
|
|
|
|
if not vdisk:
|
|
|
|
vdisk = getWholedisk(vdev)
|
|
|
|
verbose_print("vdisk not specified; guessing '%s' based on '%s'" % (vdisk, vdev))
|
|
|
|
if not vdev.startswith(vdisk):
|
|
|
|
error("Virtual disk '%s' does not match entry '%s'" % (vdisk, entry))
|
|
|
|
sys.exit(1)
|
|
|
|
disk = Wholedisk(vdisk, pdisk)
|
2007-01-16 00:42:10 +01:00
|
|
|
|
2007-02-11 11:48:10 +01:00
|
|
|
r = 0
|
|
|
|
try:
|
|
|
|
sxpr = copyKernelAndRamdisk(disk, vdev, kernel, ramdisk)
|
2007-03-06 09:12:51 +01:00
|
|
|
if in_args:
|
2007-04-26 01:53:07 +02:00
|
|
|
sxpr += "(args '%s')" % in_args
|
2007-02-11 11:48:10 +01:00
|
|
|
os.write(fd, sxpr)
|
|
|
|
except Exception, e:
|
|
|
|
error(str(e))
|
|
|
|
r = 1
|
|
|
|
|
|
|
|
for part in disk.partitions:
|
|
|
|
part.wholedisk = None
|
|
|
|
del disk
|
|
|
|
|
|
|
|
return r
|
2007-01-16 00:42:10 +01:00
|
|
|
|
|
|
|
# Call main if called (and not imported)
|
|
|
|
if __name__ == "__main__":
|
2007-02-11 11:48:10 +01:00
|
|
|
r = 1
|
|
|
|
try:
|
|
|
|
r = main(sys.argv)
|
|
|
|
except Exception, e:
|
|
|
|
error(str(e))
|
|
|
|
sys.exit(r)
|