From 70eec57765280032e08a1d738402926b14a851bf Mon Sep 17 00:00:00 2001 From: "Richard W.M. Jones" Date: Thu, 16 May 2024 13:27:49 +0100 Subject: [PATCH] convert: windows: Install blnsvr from virtio-win Also update the common module to get these two commits for the implementation: Richard W.M. Jones (2): mlcustomize: Add virt-customize --inject-blnsvr generated files mlcustomize: Add Inject_virtio_win.inject_blnsvr implementation Fixes: https://issues.redhat.com/browse/RHEL-36591 --- common | 2 +- convert/convert_windows.ml | 6 ++++++ 2 files changed, 7 insertions(+), 1 deletion(-) Submodule common 93a7f3af..830cbdcf: diff --git a/common/mlcustomize/customize-options.pod b/common/mlcustomize/customize-options.pod index ff93630d..b2ac5752 100644 --- a/common/mlcustomize/customize-options.pod +++ b/common/mlcustomize/customize-options.pod @@ -193,6 +193,18 @@ L. Set the hostname of the guest to C. You can use a dotted hostname.domainname (FQDN) if you want. +=item B<--inject-blnsvr> METHOD + +Inject the Balloon Server (F) into a Windows guest. +This operation also injects a firstboot script so that the Balloon +Server is installed when the guest boots. + +The parameter is the same as used by the I<--inject-virtio-win> operation. + +Note that to do a full conversion of a Windows guest from a +foreign hypervisor like VMware (which involves many other operations) +you should use the L tool instead of this. + =item B<--inject-qemu-ga> METHOD Inject the QEMU Guest Agent into a Windows guest. The guest diff --git a/common/mlcustomize/customize-synopsis.pod b/common/mlcustomize/customize-synopsis.pod index bb0ce125..957de8cf 100644 --- a/common/mlcustomize/customize-synopsis.pod +++ b/common/mlcustomize/customize-synopsis.pod @@ -3,16 +3,16 @@ [--copy SOURCE:DEST] [--copy-in LOCALPATH:REMOTEDIR] [--delete PATH] [--edit FILE:EXPR] [--firstboot SCRIPT] [--firstboot-command 'CMD+ARGS'] [--firstboot-install PKG,PKG..] - [--hostname HOSTNAME] [--inject-qemu-ga METHOD] - [--inject-virtio-win METHOD] [--install PKG,PKG..] - [--link TARGET:LINK[:LINK..]] [--mkdir DIR] [--move SOURCE:DEST] - [--password USER:SELECTOR] [--root-password SELECTOR] - [--run SCRIPT] [--run-command 'CMD+ARGS'] [--scrub FILE] - [--sm-attach SELECTOR] [--sm-register] [--sm-remove] - [--sm-unregister] [--ssh-inject USER[:SELECTOR]] - [--tar-in TARFILE:REMOTEDIR] [--timezone TIMEZONE] [--touch FILE] - [--truncate FILE] [--truncate-recursive PATH] - [--uninstall PKG,PKG..] [--update] [--upload FILE:DEST] - [--write FILE:CONTENT] [--no-logfile] + [--hostname HOSTNAME] [--inject-blnsvr METHOD] + [--inject-qemu-ga METHOD] [--inject-virtio-win METHOD] + [--install PKG,PKG..] [--link TARGET:LINK[:LINK..]] [--mkdir DIR] + [--move SOURCE:DEST] [--password USER:SELECTOR] + [--root-password SELECTOR] [--run SCRIPT] + [--run-command 'CMD+ARGS'] [--scrub FILE] [--sm-attach SELECTOR] + [--sm-register] [--sm-remove] [--sm-unregister] + [--ssh-inject USER[:SELECTOR]] [--tar-in TARFILE:REMOTEDIR] + [--timezone TIMEZONE] [--touch FILE] [--truncate FILE] + [--truncate-recursive PATH] [--uninstall PKG,PKG..] [--update] + [--upload FILE:DEST] [--write FILE:CONTENT] [--no-logfile] [--password-crypto md5|sha256|sha512] [--no-selinux-relabel] [--selinux-relabel] [--sm-credentials SELECTOR] diff --git a/common/mlcustomize/customize_cmdline.ml b/common/mlcustomize/customize_cmdline.ml index 48ee3344..c4d6a77d 100644 --- a/common/mlcustomize/customize_cmdline.ml +++ b/common/mlcustomize/customize_cmdline.ml @@ -61,6 +61,8 @@ and op = [ (* --firstboot-install PKG,PKG.. *) | `Hostname of string (* --hostname HOSTNAME *) + | `InjectBalloonServer of string + (* --inject-blnsvr METHOD *) | `InjectQemuGA of string (* --inject-qemu-ga METHOD *) | `InjectVirtioWin of string @@ -286,6 +288,12 @@ let rec argspec () = s_"Set the hostname" ), Some "HOSTNAME", "Set the hostname of the guest to C. You can use a\ndotted hostname.domainname (FQDN) if you want."; + ( + [ L"inject-blnsvr" ], + Getopt.String (s_"METHOD", fun s -> List.push_front (`InjectBalloonServer s) ops), + s_"Inject the Balloon Server into a Windows guest" + ), + Some "METHOD", "Inject the Balloon Server (F) into a Windows guest.\nThis operation also injects a firstboot script so that the Balloon\nServer is installed when the guest boots.\n\nThe parameter is the same as used by the I<--inject-virtio-win> operation.\n\nNote that to do a full conversion of a Windows guest from a\nforeign hypervisor like VMware (which involves many other operations)\nyou should use the L tool instead of this."; ( [ L"inject-qemu-ga" ], Getopt.String (s_"METHOD", fun s -> List.push_front (`InjectQemuGA s) ops), diff --git a/common/mlcustomize/customize_cmdline.mli b/common/mlcustomize/customize_cmdline.mli index 51a156ea..ee62961a 100644 --- a/common/mlcustomize/customize_cmdline.mli +++ b/common/mlcustomize/customize_cmdline.mli @@ -53,6 +53,8 @@ and op = [ (* --firstboot-install PKG,PKG.. *) | `Hostname of string (* --hostname HOSTNAME *) + | `InjectBalloonServer of string + (* --inject-blnsvr METHOD *) | `InjectQemuGA of string (* --inject-qemu-ga METHOD *) | `InjectVirtioWin of string diff --git a/common/mlcustomize/inject_virtio_win.ml b/common/mlcustomize/inject_virtio_win.ml index 2a30b200..4e0ed0e0 100644 --- a/common/mlcustomize/inject_virtio_win.ml +++ b/common/mlcustomize/inject_virtio_win.ml @@ -24,6 +24,8 @@ open Common_gettext.Gettext open Regedit +let re_blnsvr = PCRE.compile ~caseless:true "\\bblnsvr\\.exe$" + type t = { g : Guestfs.guestfs; (** guestfs handle *) @@ -267,6 +269,18 @@ and inject_qemu_ga t = configure_qemu_ga t msi_files; msi_files <> [] (* return true if we found some qemu-ga MSI files *) +and inject_blnsvr t = + let files = copy_blnsvr t in + match files with + | [] -> false (* Didn't find or install anything. *) + + (* We usually find blnsvr.exe in two locations (drivers/by-os and + * drivers/by-driver). Pick the first. + *) + | blnsvr :: _ -> + configure_blnsvr t blnsvr; + true + and add_guestor_to_registry t ((g, root) as reg) drv_name drv_pciid = let ddb_node = g#hivex_node_get_child root "DriverDatabase" in @@ -351,6 +365,11 @@ and copy_qemu_ga t = (fun () -> error (f_"root directory ‘/’ is missing from the virtio-win directory or ISO.\n\nThis should not happen and may indicate that virtio-win or virt-v2v is broken in some way. Please report this as a bug with a full debug log.")) +and copy_blnsvr t = + copy_from_virtio_win t "/" "/" (virtio_iso_path_matches_blnsvr t) + (fun () -> + error (f_"root directory ‘/’ is missing from the virtio-win directory or ISO.\n\nThis should not happen and may indicate that virtio-win or virt-v2v is broken in some way. Please report this as a bug with a full debug log.")) + (* Copy all files from virtio_win directory/ISO located in [srcdir] * subdirectory and all its subdirectories to the [destdir]. The directory * hierarchy is not preserved, meaning all files will be directly in [destdir]. @@ -513,6 +532,10 @@ and virtio_iso_path_matches_qemu_ga t path = | ("x86_64", "rhev-qga64.msi") -> true | _ -> false +(* Find blnsvr for the current Windows version. *) +and virtio_iso_path_matches_blnsvr t path = + virtio_iso_path_matches_guest_os t path && PCRE.matches re_blnsvr path + (* Look up in libosinfo for the OS, and copy all the locally * available files specified as drivers for that OS to the [destdir]. * @@ -586,3 +609,10 @@ and configure_qemu_ga t files = Firstboot.add_firstboot_powershell t.g t.root (sprintf "install-%s.ps1" msi_path) !psh_script; ) files + +and configure_blnsvr t blnsvr = + let cmd = sprintf "\ + @echo off\n\ + echo Installing %s\n\ + c:\\%s -i\n" blnsvr blnsvr in + Firstboot.add_firstboot_script t.g t.root (sprintf "install-%s" blnsvr) cmd diff --git a/common/mlcustomize/inject_virtio_win.mli b/common/mlcustomize/inject_virtio_win.mli index d14f0497..d273c4dd 100644 --- a/common/mlcustomize/inject_virtio_win.mli +++ b/common/mlcustomize/inject_virtio_win.mli @@ -93,3 +93,11 @@ val inject_qemu_ga : t -> bool the MSI(s). Returns [true] iff we were able to inject qemu-ga. *) + +val inject_blnsvr : t -> bool +(** Inject the Balloon Server ([blnsvr.exe]) into a Windows guest. + + A firstboot script is also injected which should install + the server by running [blnsvr -i]. + + Returns [true] iff we were able to inject the Balloon Server. *) diff --git a/convert/convert_windows.ml b/convert/convert_windows.ml index 34cf341b..2d6e2059 100644 --- a/convert/convert_windows.ml +++ b/convert/convert_windows.ml @@ -357,6 +357,12 @@ let convert (g : G.guestfs) _ inspect i_firmware block_driver _ static_ips = may want to install the guest agent manually after \ conversion."); + (* Install Balloon Server unconditionally and warn if missing *) + if not (Inject_virtio_win.inject_blnsvr virtio_win) then + warning (f_"Balloon Server (blnsvr.exe) not found on tools \ + ISO/directory. You may want to install this component \ + manually after conversion."); + unconfigure_xenpv (); unconfigure_prltools (); unconfigure_vmwaretools ()