From nobody Fri Mar 31 06:22:17 2023 X-Original-To: dev-commits-src-all@mlmmj.nyi.freebsd.org Received: from mx1.freebsd.org (mx1.freebsd.org [IPv6:2610:1c1:1:606c::19:1]) by mlmmj.nyi.freebsd.org (Postfix) with ESMTP id 4Pnqvv5TbVz430Gd for ; Fri, 31 Mar 2023 06:22:35 +0000 (UTC) (envelope-from tsoome@me.com) Received: from pv50p00im-zteg10021301.me.com (pv50p00im-zteg10021301.me.com [17.58.6.46]) (using TLSv1.3 with cipher TLS_AES_256_GCM_SHA384 (256/256 bits) key-exchange X25519 server-signature RSA-PSS (4096 bits) server-digest SHA256) (Client did not present a certificate) by mx1.freebsd.org (Postfix) with ESMTPS id 4Pnqvv3128z46Pj for ; Fri, 31 Mar 2023 06:22:35 +0000 (UTC) (envelope-from tsoome@me.com) Authentication-Results: mx1.freebsd.org; none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=me.com; s=1a1hai; t=1680243754; bh=svwnGwAlqrTxylvuWdnlSVe9OIE3t9pX+SrMsttmqzU=; h=From:Message-Id:Content-Type:Mime-Version:Subject:Date:To; b=hmstbAQtpeZYNfp4PdVoumIPU2xAs6hkHYwDDN0D3QGlesxIGoP6oqFQqfrY9yMmJ 2K8HECMifpIsSQzUoL+D21xjc2vai56o1L+2bLtCWGCSJqPbhqoAbaG6snElkhKqu2 CQ8J8WxWVOWG4GRbV7F7FLOZHsjulu3pQZcI6UCxD9V91NlnoI4O3Cn49H3kjDF5CK C2sVEeEOt+uPt02PjL+mXKFZXmjx/pXMkQWtHXhI7XYIZcvW0at0JdLDLgvInvYpIG yTWWS9oZVlO9qI/1QhH1ICAKA3FmI+ddF5Y8ruqdyWoWFYv+tk+Po0ihoSMFavvoac oA57r636lwDyw== Received: from smtpclient.apple (pv50p00im-dlb-asmtp-mailmevip.me.com [17.56.9.10]) by pv50p00im-zteg10021301.me.com (Postfix) with ESMTPSA id A95DC5001F2; Fri, 31 Mar 2023 06:22:30 +0000 (UTC) From: Toomas Soome Message-Id: <256C0F87-8B73-4E5E-9B5E-E9BF00DEC019@me.com> Content-Type: multipart/alternative; boundary="Apple-Mail=_CC2C5268-3B0E-412D-AB43-A76713CFE030" List-Id: Commit messages for all branches of the src repository List-Archive: https://lists.freebsd.org/archives/dev-commits-src-all List-Help: List-Post: List-Subscribe: List-Unsubscribe: Sender: owner-dev-commits-src-all@freebsd.org X-BeenThere: dev-commits-src-all@freebsd.org Mime-Version: 1.0 (Mac OS X Mail 16.0 \(3731.400.51.1.1\)) Subject: Re: git: 927358dd98cb - main - amd64 loader: Use efiserialio for Hyper-V booted systems Date: Fri, 31 Mar 2023 09:22:17 +0300 In-Reply-To: Cc: Gleb Smirnoff , Wei Hu , src-committers , "" , "" , imp@freebsd.org, gallatin@freebsd.org, kevans@freebsd.org To: Warner Losh References: <202303180720.32I7KXOc030612@gitrepo.freebsd.org> X-Mailer: Apple Mail (2.3731.400.51.1.1) X-Proofpoint-ORIG-GUID: T0X7ufiSx51uEnpvnIvdv8y8POAxGlWZ X-Proofpoint-GUID: T0X7ufiSx51uEnpvnIvdv8y8POAxGlWZ X-Proofpoint-Virus-Version: =?UTF-8?Q?vendor=3Dfsecure_engine=3D1.1.170-22c6f66c430a71ce266a39bfe25bc?= =?UTF-8?Q?2903e8d5c8f:6.0.517,18.0.572,17.0.605.474.0000000_definitions?= =?UTF-8?Q?=3D2022-06-21=5F01:2022-06-21=5F01,2020-02-14=5F11,2020-01-23?= =?UTF-8?Q?=5F02_signatures=3D0?= X-Proofpoint-Spam-Details: rule=notspam policy=default score=0 mlxlogscore=999 spamscore=0 clxscore=1011 adultscore=0 mlxscore=0 phishscore=0 bulkscore=0 suspectscore=0 malwarescore=0 classifier=spam adjust=0 reason=mlx scancount=1 engine=8.12.0-2209130000 definitions=main-2303310051 X-Rspamd-Queue-Id: 4Pnqvv3128z46Pj X-Spamd-Bar: ---- X-Spamd-Result: default: False [-4.00 / 15.00]; REPLY(-4.00)[]; ASN(0.00)[asn:714, ipnet:17.58.0.0/20, country:US] X-Rspamd-Pre-Result: action=no action; module=replies; Message is reply to one we originated X-ThisMailContainsUnwantedMimeParts: N --Apple-Mail=_CC2C5268-3B0E-412D-AB43-A76713CFE030 Content-Transfer-Encoding: quoted-printable Content-Type: text/plain; charset=utf-8 > On 31. Mar 2023, at 02:40, Warner Losh wrote: >=20 > Let's back it out. I didn't get a chance to review it and the = duplicated name strikes me as massively unwise. > Of course, the efi serial driver never should have had the name = comconsole in the first place, imho. It was OK > on arm where we couldn't conflict. >=20 > So let's back it out and talk about how we should do this, including = the need to possibly just rename the efi > version of the console driver to something else. IMHO, it never should = have been comconsole in the first > place because it's configured differently than comconsole.... >=20 > Warner ok, lets start from beginning. The loader device naming is total mess = and should not go this way - it is not understandable for users and not = understandable for developers. the config below is perfect example of the situation: First the config does set:=20 >> console=3D"comconsole vidconsole efi=E2=80=9D So, we do try to set serial console, bios video console and efi console. = So we have =E2=80=9Cshared=E2=80=9D config for BIOS and UEFI loader, = which should be totally reasonable from user point of view, but is = technically wrong, because BIOS loader has no idea what is efi console = and EFI loader has no idea what is vidconsole. So this mixture ends up = with message: >> console vidconsole is invalid!" And the user will get the similar error when booting BIOS setup. Then we just set console=3D=E2=80=9Cefi=E2=80=9D. Then there is setting for uart=E2=80=A6=20 With todays UEFI setups, the logic is that you set up your hardware, = that is, you configure serial port attributes from firmware setup, and = the UEFI firmware will populate its internal structures with those = values. Now, first issue is, while developing uefi loader, it was easy path to = reuse bits from BIOS loader. smbios, acpi, and sadly, comconsole. Sadly, = because comconsole is built assuming x86 IO port features which are not = available on other platforms. So yes, bios version of comconsole should not have been used to build = UEFI loader at all. Now the fun part, even as UEFI does provide SIO API to access serial = ports, to read from port, write to port and set attributes, it turns out = there are UEFI implementation(s?), where changing some attributes will = cause system to hung. Which is the reason, why we need to set up = hardware properly from firmware, and our generic config should not try = to enforce some other value(s). Default should pick up values from the = system. Sure, if the machine operator has some reason to change those = values and has tested the change, we should have method to change the = config, but again, defaults should come from system setup. While I see the reasoning to try to create all those different names, = maybe the first thing should be to look on this as innocent end user, = who has no idea about those fine technical details and historical = reasons and is just trying to get system to behave=E2=80=A6. just my 2 cents, toomas >=20 > On Thu, Mar 30, 2023 at 5:31=E2=80=AFPM Gleb Smirnoff = > wrote: >> Wei, Kyle, >>=20 >> this commit hangs loader on real hardware, at least on some >> of it. The loader prints list of consoles and hangs hard: >>=20 >> [Thu Mar 30 20:46:12 2023]^M|^HLoading /boot/loader.conf^M >> ^M/^Hconsole vidconsole is invalid!^M >> ^MAvailable consoles:^M >> ^M efi^M >> ^M comconsole^M >> ^M comconsole^M >> ^M nullconsole^M >> ^M spinconsole^M >>=20 >> Machine is unrecoverable unless you got alternate boot media >> and access to BMC console. >>=20 >> First, please DO NOT MFC this as scheduled. Second, let's try >> to fix it or back it out if we hear from any other CURRENT >> user. >>=20 >> Our configuration isn't special. This is what we got in >> loader.conf, related to consoles: >>=20 >> console=3D"comconsole vidconsole efi" >> comconsole_speed=3D115200 >> comconsole_port=3D0x3e8 >> console=3D"efi" >> hw.uart.console=3D"io:1016,br:115200" >>=20 >> On Sat, Mar 18, 2023 at 07:20:33AM +0000, Wei Hu wrote: >> W> The branch main has been updated by whu: >> W>=20 >> W> URL: = https://cgit.FreeBSD.org/src/commit/?id=3D927358dd98cb902160093e0dc0bac002= d6b43858 = >> W>=20 >> W> commit 927358dd98cb902160093e0dc0bac002d6b43858 >> W> Author: Wei Hu >> W> AuthorDate: 2023-03-14 15:13:46 +0000 >> W> Commit: Wei Hu >> W> CommitDate: 2023-03-18 07:07:35 +0000 >> W>=20 >> W> amd64 loader: Use efiserialio for Hyper-V booted systems >> W> =20 >> W> UEFI provides ConIn/ConOut handles for consoles that it = supports, >> W> which include the text-video and serial ports. When the serial = port >> W> is available, use the UEFI driver instead of direct io-port = accesses >> W> to avoid conflicts between the firmware and direct hardware = access, as >> W> happens on Hyper-V (Azure) setups. >> W> =20 >> W> This change enables efiserialio to be built for efi-amd64 and = has >> W> higher order priority vs comconsole, and only uses efiserialio >> W> if the hypervisor is Hyper-V. When efiserialio successfully >> W> probes, it will set efi_comconsole_avail=3Dtrue which will = prevent >> W> comconsole from probing in this setup. >> W> =20 >> W> Tested on Hyper-V, ESXi and Azure VMs. >> W> =20 >> W> PR: 264267 >> W> Reviewed by: kevans, whu >> W> Tested by: whu >> W> Obtained from: Rubicon Communications, LLC (Netgate) >> W> MFC after: 2 weeks >> W> Sponsored by: Rubicon Communications, LLC (Netgate) >> W> --- >> W> stand/efi/loader/arch/amd64/Makefile.inc | 1 + >> W> stand/efi/loader/bootinfo.c | 11 ++++++-- >> W> stand/efi/loader/conf.c | 6 +++++ >> W> stand/efi/loader/efiserialio.c | 43 = ++++++++++++++++++++++++++++---- >> W> stand/i386/libi386/comconsole.c | 14 +++++++++++ >> W> 5 files changed, 68 insertions(+), 7 deletions(-) >> W>=20 >> W> diff --git a/stand/efi/loader/arch/amd64/Makefile.inc = b/stand/efi/loader/arch/amd64/Makefile.inc >> W> index 0d9e2648cb59..bd89044bd6c7 100644 >> W> --- a/stand/efi/loader/arch/amd64/Makefile.inc >> W> +++ b/stand/efi/loader/arch/amd64/Makefile.inc >> W> @@ -5,6 +5,7 @@ SRCS+=3D amd64_tramp.S \ >> W> elf64_freebsd.c \ >> W> trap.c \ >> W> multiboot2.c \ >> W> + efiserialio.c \ >> W> exc.S >> W> =20 >> W> .PATH: ${BOOTSRC}/i386/libi386 >> W> diff --git a/stand/efi/loader/bootinfo.c = b/stand/efi/loader/bootinfo.c >> W> index 939f2cf4c3fe..d79f59343af1 100644 >> W> --- a/stand/efi/loader/bootinfo.c >> W> +++ b/stand/efi/loader/bootinfo.c >> W> @@ -119,10 +119,17 @@ bi_getboothowto(char *kargs) >> W> if (tmp !=3D NULL) >> W> speed =3D strtol(tmp, NULL, 0); >> W> tmp =3D getenv("efi_com_port"); >> W> - if (tmp =3D=3D NULL) >> W> - tmp =3D getenv("comconsole_port"); >> W> if (tmp !=3D NULL) >> W> port =3D strtol(tmp, NULL, 0); >> W> + if (port <=3D 0) { >> W> + tmp =3D getenv("comconsole_port"); >> W> + if (tmp !=3D NULL) >> W> + port =3D strtol(tmp, NULL, = 0); >> W> + else { >> W> + if (port =3D=3D 0) >> W> + port =3D 0x3f8; >> W> + } >> W> + } >> W> if (speed !=3D -1 && port !=3D -1) { >> W> snprintf(buf, sizeof(buf), = "io:%d,br:%d", port, >> W> speed); >> W> diff --git a/stand/efi/loader/conf.c b/stand/efi/loader/conf.c >> W> index 863c9188c72c..051e1a3381d1 100644 >> W> --- a/stand/efi/loader/conf.c >> W> +++ b/stand/efi/loader/conf.c >> W> @@ -81,6 +81,9 @@ struct netif_driver *netif_drivers[] =3D { >> W> =20 >> W> extern struct console efi_console; >> W> extern struct console comconsole; >> W> +#if defined(__amd64__) >> W> +extern struct console eficomconsole; >> W> +#endif >> W> #if defined(__amd64__) || defined(__i386__) >> W> extern struct console nullconsole; >> W> extern struct console spinconsole; >> W> @@ -88,6 +91,9 @@ extern struct console spinconsole; >> W> =20 >> W> struct console *consoles[] =3D { >> W> &efi_console, >> W> +#if defined(__amd64__) >> W> + &eficomconsole, >> W> +#endif >> W> &comconsole, >> W> #if defined(__amd64__) || defined(__i386__) >> W> &nullconsole, >> W> diff --git a/stand/efi/loader/efiserialio.c = b/stand/efi/loader/efiserialio.c >> W> index 375e679d2590..5fbc700f6ac2 100644 >> W> --- a/stand/efi/loader/efiserialio.c >> W> +++ b/stand/efi/loader/efiserialio.c >> W> @@ -69,6 +69,11 @@ static int comc_speed_set(struct env_var = *, int, const void *); >> W> =20 >> W> static struct serial *comc_port; >> W> extern struct console efi_console; >> W> +bool efi_comconsole_avail =3D false; >> W> + >> W> +#if defined(__amd64__) >> W> +#define comconsole eficomconsole >> W> +#endif >> W> =20 >> W> struct console comconsole =3D { >> W> .c_name =3D "comconsole", >> W> @@ -254,11 +259,22 @@ comc_probe(struct console *sc) >> W> char *env, *buf, *ep; >> W> size_t sz; >> W> =20 >> W> +#if defined(__amd64__) >> W> + /* >> W> + * For x86-64, don't use this driver if not running in = Hyper-V. >> W> + */ >> W> + env =3D getenv("smbios.bios.version"); >> W> + if (env =3D=3D NULL || strncmp(env, "Hyper-V", 7) !=3D 0) { >> W> + return; >> W> + } >> W> +#endif >> W> + >> W> if (comc_port =3D=3D NULL) { >> W> comc_port =3D calloc(1, sizeof (struct serial)); >> W> if (comc_port =3D=3D NULL) >> W> return; >> W> } >> W> + >> W> /* Use defaults from firmware */ >> W> comc_port->databits =3D 8; >> W> comc_port->parity =3D DefaultParity; >> W> @@ -308,6 +324,10 @@ comc_probe(struct console *sc) >> W> comc_port_set, env_nounset); >> W> =20 >> W> env =3D getenv("efi_com_speed"); >> W> + if (env =3D=3D NULL) >> W> + /* fallback to comconsole setting */ >> W> + env =3D getenv("comconsole_speed"); >> W> + >> W> if (comc_parse_intval(env, &val) =3D=3D CMD_OK) >> W> comc_port->baudrate =3D val; >> W> =20 >> W> @@ -318,8 +338,13 @@ comc_probe(struct console *sc) >> W> comc_speed_set, env_nounset); >> W> =20 >> W> comconsole.c_flags =3D 0; >> W> - if (comc_setup()) >> W> + if (comc_setup()) { >> W> sc->c_flags =3D C_PRESENTIN | C_PRESENTOUT; >> W> + efi_comconsole_avail =3D true; >> W> + } else { >> W> + /* disable being seen as "comconsole" */ >> W> + comconsole.c_name =3D "efiserialio"; >> W> + } >> W> } >> W> =20 >> W> static int >> W> @@ -489,6 +514,7 @@ comc_setup(void) >> W> { >> W> EFI_STATUS status; >> W> UINT32 control; >> W> + char *ev; >> W> =20 >> W> /* port is not usable */ >> W> if (comc_port->sio =3D=3D NULL) >> W> @@ -498,10 +524,17 @@ comc_setup(void) >> W> if (EFI_ERROR(status)) >> W> return (false); >> W> =20 >> W> - status =3D comc_port->sio->SetAttributes(comc_port->sio, >> W> - comc_port->baudrate, comc_port->receivefifodepth, >> W> - comc_port->timeout, comc_port->parity, >> W> - comc_port->databits, comc_port->stopbits); >> W> + ev =3D getenv("smbios.bios.version"); >> W> + if (ev !=3D NULL && strncmp(ev, "Hyper-V", 7) =3D=3D 0) { >> W> + status =3D = comc_port->sio->SetAttributes(comc_port->sio, >> W> + 0, 0, 0, DefaultParity, 0, DefaultStopBits); >> W> + } else { >> W> + status =3D = comc_port->sio->SetAttributes(comc_port->sio, >> W> + comc_port->baudrate, comc_port->receivefifodepth, >> W> + comc_port->timeout, comc_port->parity, >> W> + comc_port->databits, comc_port->stopbits); >> W> + } >> W> + >> W> if (EFI_ERROR(status)) >> W> return (false); >> W> =20 >> W> diff --git a/stand/i386/libi386/comconsole.c = b/stand/i386/libi386/comconsole.c >> W> index ed1f1aa08ed7..3fbb6a292c19 100644 >> W> --- a/stand/i386/libi386/comconsole.c >> W> +++ b/stand/i386/libi386/comconsole.c >> W> @@ -85,6 +85,20 @@ comc_probe(struct console *cp) >> W> int speed, port; >> W> uint32_t locator; >> W> =20 >> W> +#if defined(__amd64__) >> W> + extern bool efi_comconsole_avail; >> W> + >> W> + if (efi_comconsole_avail) { >> W> + /* >> W> + * If EFI provides serial I/O, then don't use this = legacy >> W> + * com driver to avoid conflicts with the firmware's = driver. >> W> + * Change c_name so that it cannot be found in the = lookup. >> W> + */ >> W> + comconsole.c_name =3D "xcomconsole"; >> W> + return; >> W> + } >> W> +#endif >> W> + >> W> if (comc_curspeed =3D=3D 0) { >> W> comc_curspeed =3D COMSPEED; >> W> /* >>=20 >> --=20 >> Gleb Smirnoff --Apple-Mail=_CC2C5268-3B0E-412D-AB43-A76713CFE030 Content-Transfer-Encoding: quoted-printable Content-Type: text/html; charset=utf-8

On 31. = Mar 2023, at 02:40, Warner Losh <imp@bsdimp.com> wrote:

Let's back it = out. I didn't get a chance to review it and the duplicated name strikes = me as massively unwise.
Of course, the efi serial driver never = should have had the name comconsole in the first place, imho. It was = OK
on arm where we couldn't = conflict.

So let's back it out and talk about = how we should do this, including the need to possibly just rename the = efi
version of the console driver to something else. IMHO, it = never should have been comconsole in the first
place because = it's configured differently than = comconsole....

Warner

ok, lets start from beginning. The = loader device naming is total mess and should not go this way - it is = not understandable for users and not understandable for = developers.

the config below is perfect example = of the situation:

First the config does = set: 

console=3D"comconsole vidconsole = efi=E2=80=9D

So, we do try = to set serial console, bios video console and efi console. So we have = =E2=80=9Cshared=E2=80=9D config for BIOS and UEFI loader, which should = be totally reasonable from user point of view, but is technically wrong, = because BIOS loader has no idea what is efi console and EFI loader has = no idea what is vidconsole. So this mixture ends up with = message:

console = vidconsole is = invalid!"

And the = user will get the similar error when booting BIOS = setup.

Then we just set = console=3D=E2=80=9Cefi=E2=80=9D.

Then there is = setting for uart=E2=80=A6 

With todays = UEFI setups, the logic is that you set up your hardware, that is, you = configure serial port attributes from firmware setup, and the UEFI = firmware will populate its internal structures with those = values.

Now, first issue is, while developing = uefi loader, it was easy path to reuse bits from BIOS loader. smbios, = acpi, and sadly, comconsole. Sadly, because comconsole is built assuming = x86 IO port features which are not available on other = platforms.

So yes, bios version of comconsole = should not have been used to build UEFI loader at = all.

Now the fun part, even as UEFI does = provide SIO API to access serial ports, to read from port, write to port = and set attributes, it turns out there are UEFI implementation(s?), = where changing some attributes will cause system to hung. Which is the = reason, why we need to set up hardware properly from firmware, and our = generic config should not try to enforce some other value(s). Default = should pick up values from the system. Sure, if the machine operator has = some reason to change those values and has tested the change, we should = have method to change the config, but again, defaults should come from = system setup.

While I see the reasoning to try = to create all those different names, maybe the first thing should be to = look on this as innocent end user, who has no idea about those fine = technical details and historical reasons and is just trying to get = system to behave=E2=80=A6.

just my 2 = cents,
toomas


On Thu, Mar 30, 2023 at 5:31=E2=80=AFPM Gleb = Smirnoff <glebius@freebsd.org> = wrote:
  Wei, Kyle,

this commit hangs loader on real hardware, at least on some
of it.  The loader prints list of consoles and hangs hard:

[Thu Mar 30 20:46:12 2023]^M|^HLoading /boot/loader.conf^M
^M/^Hconsole vidconsole is invalid!^M
^MAvailable consoles:^M
^M    efi^M
^M    comconsole^M
^M    comconsole^M
^M    nullconsole^M
^M    spinconsole^M

Machine is unrecoverable unless you got alternate boot media
and access to BMC console.

First, please DO NOT MFC this as scheduled. Second, let's try
to fix it or back it out if we hear from any other CURRENT
user.

Our configuration isn't special. This is what we got in
loader.conf, related to consoles:

console=3D"comconsole vidconsole efi"
comconsole_speed=3D115200
comconsole_port=3D0x3e8
console=3D"efi"
hw.uart.console=3D"io:1016,br:115200"

On Sat, Mar 18, 2023 at 07:20:33AM +0000, Wei Hu wrote:
W> The branch main has been updated by whu:
W>
W> URL: https://cgit.FreeBSD.org/src/commit/?id=3D927358dd98cb90= 2160093e0dc0bac002d6b43858
W>
W> commit 927358dd98cb902160093e0dc0bac002d6b43858
W> Author:     Wei Hu <whu@FreeBSD.org>
W> AuthorDate: 2023-03-14 15:13:46 +0000
W> Commit:     Wei Hu <whu@FreeBSD.org>
W> CommitDate: 2023-03-18 07:07:35 +0000
W>
W>     amd64 loader: Use efiserialio for Hyper-V = booted systems
W>     
W>     UEFI provides ConIn/ConOut handles for consoles = that it supports,
W>     which include the text-video and serial ports. = When the serial port
W>     is available, use the UEFI driver instead of = direct io-port accesses
W>     to avoid conflicts between the firmware and = direct hardware access, as
W>     happens on Hyper-V (Azure) setups.
W>     
W>     This change enables efiserialio to be built for = efi-amd64 and has
W>     higher order priority vs comconsole, and only = uses efiserialio
W>     if the hypervisor is Hyper-V. When efiserialio = successfully
W>     probes, it will set efi_comconsole_avail=3Dtrue = which will prevent
W>     comconsole from probing in this setup.
W>     
W>     Tested on Hyper-V, ESXi and Azure VMs.
W>     
W>     PR:            =  264267
W>     Reviewed by:    kevans, whu
W>     Tested by:      whu
W>     Obtained from:  Rubicon Communications, = LLC (Netgate)
W>     MFC after:      2 weeks
W>     Sponsored by:   Rubicon = Communications, LLC (Netgate)
W> ---
W>  stand/efi/loader/arch/amd64/Makefile.inc |  1 +
W>  stand/efi/loader/bootinfo.c        =       | 11 ++++++--
W>  stand/efi/loader/conf.c          =         |  6 +++++
W>  stand/efi/loader/efiserialio.c        =    | 43 ++++++++++++++++++++++++++++----
W>  stand/i386/libi386/comconsole.c        =   | 14 +++++++++++
W>  5 files changed, 68 insertions(+), 7 deletions(-)
W>
W> diff --git a/stand/efi/loader/arch/amd64/Makefile.inc = b/stand/efi/loader/arch/amd64/Makefile.inc
W> index 0d9e2648cb59..bd89044bd6c7 100644
W> --- a/stand/efi/loader/arch/amd64/Makefile.inc
W> +++ b/stand/efi/loader/arch/amd64/Makefile.inc
W> @@ -5,6 +5,7 @@ SRCS+=3D       amd64_tramp.S = \
W>      elf64_freebsd.c \
W>      trap.c \
W>      multiboot2.c \
W> +    efiserialio.c \
W>      exc.S
W> 
W>  .PATH:      ${BOOTSRC}/i386/libi386
W> diff --git a/stand/efi/loader/bootinfo.c = b/stand/efi/loader/bootinfo.c
W> index 939f2cf4c3fe..d79f59343af1 100644
W> --- a/stand/efi/loader/bootinfo.c
W> +++ b/stand/efi/loader/bootinfo.c
W> @@ -119,10 +119,17 @@ bi_getboothowto(char *kargs)
W>                  =     if (tmp !=3D NULL)
W>                  =             speed =3D strtol(tmp, NULL, = 0);
W>                  =     tmp =3D getenv("efi_com_port");
W> -                  =   if (tmp =3D=3D NULL)
W> -                  =           tmp =3D = getenv("comconsole_port");
W>                  =     if (tmp !=3D NULL)
W>                  =             port =3D strtol(tmp, NULL, = 0);
W> +                  =   if (port <=3D 0) {
W> +                  =           tmp =3D = getenv("comconsole_port");
W> +                  =           if (tmp !=3D NULL)
W> +                  =                   port =3D = strtol(tmp, NULL, 0);
W> +                  =           else {
W> +                  =                   if (port = =3D=3D 0)
W> +                  =                     =       port =3D 0x3f8;
W> +                  =           }
W> +                  =   }
W>                  =     if (speed !=3D -1 && port !=3D -1) {
W>                  =             snprintf(buf, sizeof(buf), = "io:%d,br:%d", port,
W>                  =                 speed);
W> diff --git a/stand/efi/loader/conf.c b/stand/efi/loader/conf.c
W> index 863c9188c72c..051e1a3381d1 100644
W> --- a/stand/efi/loader/conf.c
W> +++ b/stand/efi/loader/conf.c
W> @@ -81,6 +81,9 @@ struct netif_driver *netif_drivers[] =3D {
W> 
W>  extern struct console efi_console;
W>  extern struct console comconsole;
W> +#if defined(__amd64__)
W> +extern struct console eficomconsole;
W> +#endif
W>  #if defined(__amd64__) || defined(__i386__)
W>  extern struct console nullconsole;
W>  extern struct console spinconsole;
W> @@ -88,6 +91,9 @@ extern struct console spinconsole;
W> 
W>  struct console *consoles[] =3D {
W>      &efi_console,
W> +#if defined(__amd64__)
W> +    &eficomconsole,
W> +#endif
W>      &comconsole,
W>  #if defined(__amd64__) || defined(__i386__)
W>      &nullconsole,
W> diff --git a/stand/efi/loader/efiserialio.c = b/stand/efi/loader/efiserialio.c
W> index 375e679d2590..5fbc700f6ac2 100644
W> --- a/stand/efi/loader/efiserialio.c
W> +++ b/stand/efi/loader/efiserialio.c
W> @@ -69,6 +69,11 @@ static int        = comc_speed_set(struct env_var *, int, const void *);
W> 
W>  static struct serial        = *comc_port;
W>  extern struct console efi_console;
W> +bool efi_comconsole_avail =3D false;
W> +
W> +#if defined(__amd64__)
W> +#define comconsole eficomconsole
W> +#endif
W> 
W>  struct console comconsole =3D {
W>      .c_name =3D "comconsole",
W> @@ -254,11 +259,22 @@ comc_probe(struct console *sc)
W>      char *env, *buf, *ep;
W>      size_t sz;
W> 
W> +#if defined(__amd64__)
W> +    /*
W> +     * For x86-64, don't use this driver if not = running in Hyper-V.
W> +     */
W> +    env =3D getenv("smbios.bios.version");
W> +    if (env =3D=3D NULL || strncmp(env, "Hyper-V", 7) = !=3D 0) {
W> +            return;
W> +    }
W> +#endif
W> +
W>      if (comc_port =3D=3D NULL) {
W>              comc_port =3D = calloc(1, sizeof (struct serial));
W>              if (comc_port =3D=3D= NULL)
W>                  =     return;
W>      }
W> +
W>      /* Use defaults from firmware */
W>      comc_port->databits =3D 8;
W>      comc_port->parity =3D DefaultParity;
W> @@ -308,6 +324,10 @@ comc_probe(struct console *sc)
W>          comc_port_set, env_nounset);
W> 
W>      env =3D getenv("efi_com_speed");
W> +    if (env =3D=3D NULL)
W> +            /* fallback to = comconsole setting */
W> +            env =3D = getenv("comconsole_speed");
W> +
W>      if (comc_parse_intval(env, &val) =3D=3D = CMD_OK)
W>              = comc_port->baudrate =3D val;
W> 
W> @@ -318,8 +338,13 @@ comc_probe(struct console *sc)
W>          comc_speed_set, = env_nounset);
W> 
W>      comconsole.c_flags =3D 0;
W> -    if (comc_setup())
W> +    if (comc_setup()) {
W>              sc->c_flags =3D = C_PRESENTIN | C_PRESENTOUT;
W> +            efi_comconsole_avail =3D = true;
W> +    } else {
W> +            /* disable being seen = as "comconsole" */
W> +            comconsole.c_name =3D = "efiserialio";
W> +    }
W>  }
W> 
W>  static int
W> @@ -489,6 +514,7 @@ comc_setup(void)
W>  {
W>      EFI_STATUS status;
W>      UINT32 control;
W> +    char *ev;
W> 
W>      /* port is not usable */
W>      if (comc_port->sio =3D=3D NULL)
W> @@ -498,10 +524,17 @@ comc_setup(void)
W>      if (EFI_ERROR(status))
W>              return = (false);
W> 
W> -    status =3D = comc_port->sio->SetAttributes(comc_port->sio,
W> -        comc_port->baudrate, = comc_port->receivefifodepth,
W> -        comc_port->timeout, = comc_port->parity,
W> -        comc_port->databits, = comc_port->stopbits);
W> +    ev =3D getenv("smbios.bios.version");
W> +    if (ev !=3D NULL && strncmp(ev, "Hyper-V", = 7) =3D=3D 0) {
W> +            status =3D = comc_port->sio->SetAttributes(comc_port->sio,
W> +                0, 0, 0, = DefaultParity, 0, DefaultStopBits);
W> +    } else {
W> +            status =3D = comc_port->sio->SetAttributes(comc_port->sio,
W> +                = comc_port->baudrate, comc_port->receivefifodepth,
W> +                = comc_port->timeout, comc_port->parity,
W> +                = comc_port->databits, comc_port->stopbits);
W> +    }
W> +
W>      if (EFI_ERROR(status))
W>              return = (false);
W> 
W> diff --git a/stand/i386/libi386/comconsole.c = b/stand/i386/libi386/comconsole.c
W> index ed1f1aa08ed7..3fbb6a292c19 100644
W> --- a/stand/i386/libi386/comconsole.c
W> +++ b/stand/i386/libi386/comconsole.c
W> @@ -85,6 +85,20 @@ comc_probe(struct console *cp)
W>      int speed, port;
W>      uint32_t locator;
W> 
W> +#if defined(__amd64__)
W> +    extern bool efi_comconsole_avail;
W> +
W> +    if (efi_comconsole_avail) {
W> +            /*
W> +             * If EFI provides = serial I/O, then don't use this legacy
W> +             * com driver to = avoid conflicts with the firmware's driver.
W> +             * Change c_name = so that it cannot be found in the lookup.
W> +             */
W> +            comconsole.c_name =3D = "xcomconsole";
W> +            return;
W> +    }
W> +#endif
W> +
W>      if (comc_curspeed =3D=3D 0) {
W>              comc_curspeed =3D = COMSPEED;
W>              /*

--
Gleb Smirnoff

= --Apple-Mail=_CC2C5268-3B0E-412D-AB43-A76713CFE030--