mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 1 | __author__ = """Copyright Martin J. Bligh, 2006, |
| 2 | Copyright IBM Corp. 2006, Ryan Harper <ryanh@us.ibm.com>""" |
| 3 | |
mbligh | 02ff2d5 | 2008-06-03 15:00:21 +0000 | [diff] [blame] | 4 | import os, shutil, copy, pickle, re, glob |
mbligh | c61fb36 | 2008-06-05 16:22:15 +0000 | [diff] [blame] | 5 | from autotest_lib.client.bin import kernel, kernel_config, os_dep, test |
mbligh | 34d0117 | 2008-06-05 16:26:31 +0000 | [diff] [blame^] | 6 | from autotest_lib.client.bin import autotest_utils |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 7 | |
| 8 | |
| 9 | class xen(kernel.kernel): |
| 10 | |
| 11 | def log(self, msg): |
| 12 | print msg |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 13 | self.logfile.write('%s\n' % msg) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 14 | |
| 15 | |
| 16 | def __init__(self, job, base_tree, results_dir, tmp_dir, build_dir, \ |
| 17 | leave = False, kjob = None): |
| 18 | # call base-class |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 19 | kernel.kernel.__init__(self, job, base_tree, results_dir, \ |
| 20 | tmp_dir, build_dir, leave) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 21 | self.kjob = kjob |
| 22 | |
| 23 | |
| 24 | def config(self, config_file, config_list = None): |
mbligh | 642b03e | 2008-01-14 16:53:15 +0000 | [diff] [blame] | 25 | raise NotImplementedError('config() not implemented for xen') |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 26 | |
| 27 | |
| 28 | def build(self, make_opts = '', logfile = '', extraversion='autotest'): |
| 29 | """build xen |
| 30 | |
| 31 | make_opts |
| 32 | additional options to make, if any |
| 33 | """ |
| 34 | self.log('running build') |
| 35 | os_dep.commands('gcc', 'make') |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 36 | # build xen with extraversion flag |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 37 | os.environ['XEN_EXTRAVERSION'] = '-unstable-%s'% extraversion |
| 38 | if logfile == '': |
| 39 | logfile = os.path.join(self.log_dir, 'xen_build') |
| 40 | os.chdir(self.build_dir) |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 41 | self.log('log_dir: %s ' % os.path.join(self.log_dir, 'stdout')) |
mbligh | 44da937 | 2007-11-05 23:30:07 +0000 | [diff] [blame] | 42 | self.job.stdout.tee_redirect(logfile + '.stdout') |
| 43 | self.job.stderr.tee_redirect(logfile + '.stderr') |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 44 | |
| 45 | # build xen hypervisor and user-space tools |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 46 | targets = ['xen', 'tools'] |
mbligh | 34d0117 | 2008-06-05 16:26:31 +0000 | [diff] [blame^] | 47 | threads = 2 * autotest_utils.count_cpus() |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 48 | for t in targets: |
| 49 | build_string = 'make -j %d %s %s' % (threads, make_opts, t) |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 50 | self.log('build_string: %s' % build_string) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 51 | system(build_string) |
| 52 | |
| 53 | # make a kernel job out of the kernel from the xen src if one isn't provided |
| 54 | if self.kjob == None: |
| 55 | # get xen kernel tree ready |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 56 | self.log("prep-ing xen'ified kernel source tree") |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 57 | system('make prep-kernels') |
| 58 | |
| 59 | v = self.get_xen_kernel_build_ver() |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 60 | self.log('building xen kernel version: %s' % v) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 61 | |
| 62 | # build xen-ified kernel in xen tree |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 63 | kernel_base_tree = os.path.join(self.build_dir, \ |
| 64 | 'linux-%s' % self.get_xen_kernel_build_ver()) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 65 | |
| 66 | self.log('kernel_base_tree = %s' % kernel_base_tree) |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 67 | # fix up XENGUEST value in EXTRAVERSION; we can't have |
| 68 | # files with '$(XENGEUST)' in the name, =( |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 69 | self.fix_up_xen_kernel_makefile(kernel_base_tree) |
| 70 | |
| 71 | # make the kernel job |
| 72 | self.kjob = self.job.kernel(kernel_base_tree) |
| 73 | |
| 74 | # hardcoding dom0 config (no modules for testing, yay!) |
| 75 | # FIXME: probe host to determine which config to pick |
| 76 | c = self.build_dir + '/buildconfigs/linux-defconfig_xen0_x86_32' |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 77 | self.log('using kernel config: %s ' % c) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 78 | self.kjob.config(c) |
| 79 | |
| 80 | # Xen's kernel tree sucks; doesn't use bzImage, but vmlinux |
| 81 | self.kjob.set_build_target('vmlinuz') |
| 82 | |
| 83 | # also, the vmlinuz is not out in arch/*/boot, ARGH! more hackery |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 84 | self.kjob.set_build_image(self.job.tmpdir + '/build/linux/vmlinuz') |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 85 | |
| 86 | self.kjob.build() |
| 87 | |
| 88 | self.job.stdout.restore() |
| 89 | self.job.stderr.restore() |
| 90 | |
| 91 | xen_version = self.get_xen_build_ver() |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 92 | self.log('BUILD VERSION: Xen: %s Kernel:%s' % \ |
| 93 | (xen_version, self.kjob.get_kernel_build_ver())) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 94 | |
| 95 | |
| 96 | def build_timed(self, *args, **kwds): |
mbligh | 642b03e | 2008-01-14 16:53:15 +0000 | [diff] [blame] | 97 | raise NotImplementedError('build_timed() not implemented') |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 98 | |
| 99 | |
| 100 | def install(self, tag='', prefix = '/', extraversion='autotest'): |
| 101 | """make install in the kernel tree""" |
| 102 | self.log('Installing ...') |
| 103 | |
| 104 | os.chdir(self.build_dir) |
| 105 | |
| 106 | if not os.path.isdir(prefix): |
| 107 | os.mkdir(prefix) |
| 108 | self.boot_dir = os.path.join(prefix, 'boot') |
| 109 | if not os.path.isdir(self.boot_dir): |
| 110 | os.mkdir(self.boot_dir) |
| 111 | |
| 112 | # remember what we are going to install |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 113 | xen_version = '%s-%s' % (self.get_xen_build_ver(), extraversion) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 114 | self.xen_image = self.boot_dir + '/xen-' + xen_version + '.gz' |
| 115 | self.xen_syms = self.boot_dir + '/xen-syms-' + xen_version |
| 116 | |
| 117 | self.log('Installing Xen ...') |
| 118 | os.environ['XEN_EXTRAVERSION'] = '-unstable-%s'% extraversion |
| 119 | |
| 120 | # install xen |
| 121 | system('make DESTDIR=%s -C xen install' % prefix) |
| 122 | |
| 123 | # install tools |
| 124 | system('make DESTDIR=%s -C tools install' % prefix) |
| 125 | |
| 126 | # install kernel |
| 127 | ktag = self.kjob.get_kernel_build_ver() |
| 128 | kprefix = prefix |
| 129 | self.kjob.install(tag=ktag, prefix=kprefix) |
| 130 | |
| 131 | |
| 132 | def add_to_bootloader(self, tag='autotest', args=''): |
| 133 | """ add this kernel to bootloader, taking an |
| 134 | optional parameter of space separated parameters |
| 135 | e.g.: kernel.add_to_bootloader('mykernel', 'ro acpi=off') |
| 136 | """ |
| 137 | |
| 138 | # turn on xen mode |
| 139 | self.job.bootloader.enable_xen_mode() |
| 140 | |
| 141 | # remove existing entry if present |
| 142 | self.job.bootloader.remove_kernel(tag) |
| 143 | |
| 144 | # add xen and xen kernel |
| 145 | self.job.bootloader.add_kernel(self.kjob.image, tag, \ |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 146 | self.kjob.initrd, self.xen_image) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 147 | |
| 148 | # if no args passed, populate from /proc/cmdline |
| 149 | if not args: |
| 150 | args = open('/proc/cmdline', 'r').readline().strip() |
| 151 | |
| 152 | # add args to entry one at a time |
| 153 | for a in args.split(' '): |
| 154 | self.job.bootloader.add_args(tag, a) |
| 155 | |
| 156 | # turn off xen mode |
| 157 | self.job.bootloader.disable_xen_mode() |
| 158 | |
| 159 | |
| 160 | def get_xen_kernel_build_ver(self): |
| 161 | """Check xen buildconfig for current kernel version""" |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 162 | version = patchlevel = sublevel = '' |
| 163 | extraversion = localversion = '' |
| 164 | |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 165 | version_file = self.build_dir + '/buildconfigs/mk.linux-2.6-xen' |
| 166 | |
| 167 | for line in open(version_file, 'r').readlines(): |
| 168 | if line.startswith('LINUX_VER'): |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 169 | start = line.index('=') + 1 |
| 170 | version = line[start:].strip() + "-xen" |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 171 | break |
| 172 | |
| 173 | return version |
| 174 | |
| 175 | |
| 176 | def fix_up_xen_kernel_makefile(self, kernel_dir): |
| 177 | """Fix up broken EXTRAVERSION in xen-ified Linux kernel Makefile""" |
| 178 | xenguest = '' |
| 179 | makefile = kernel_dir + '/Makefile' |
| 180 | |
| 181 | for line in open(makefile, 'r').readlines(): |
| 182 | if line.startswith('XENGUEST'): |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 183 | start = line.index('=') + 1 |
| 184 | xenguest = line[start:].strip() |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 185 | break; |
| 186 | |
| 187 | # change out $XENGUEST in EXTRAVERSION line |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 188 | system('sed -i.old "s,\$(XENGUEST),%s," %s' % \ |
| 189 | (xenguest, makefile)) |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 190 | |
| 191 | |
| 192 | def get_xen_build_ver(self): |
| 193 | """Check Makefile and .config to return kernel version""" |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 194 | version = patchlevel = sublevel = '' |
| 195 | extraversion = localversion = '' |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 196 | |
| 197 | for line in open(self.build_dir + '/xen/Makefile', 'r').readlines(): |
| 198 | if line.startswith('export XEN_VERSION'): |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 199 | start = line.index('=') + 1 |
| 200 | version = line[start:].strip() |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 201 | if line.startswith('export XEN_SUBVERSION'): |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 202 | start = line.index('=') + 1 |
| 203 | sublevel = line[start:].strip() |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 204 | if line.startswith('export XEN_EXTRAVERSION'): |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 205 | start = line.index('=') + 1 |
| 206 | extraversion = line[start:].strip() |
mbligh | 8baa2ea | 2006-12-17 23:01:24 +0000 | [diff] [blame] | 207 | |
mbligh | f25cde1 | 2006-12-17 23:13:48 +0000 | [diff] [blame] | 208 | return "%s.%s%s" % (version, sublevel, extraversion) |