From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org X-Spam-Level: X-Spam-Status: No, score=-9.9 required=3.0 tests=DKIMWL_WL_HIGH,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS,INCLUDES_PATCH, MAILING_LIST_MULTI,SIGNED_OFF_BY,SPF_HELO_NONE,SPF_PASS,USER_AGENT_GIT autolearn=ham autolearn_force=no version=3.4.0 Received: from mail.kernel.org (mail.kernel.org [198.145.29.99]) by smtp.lore.kernel.org (Postfix) with ESMTP id 2C329C32750 for ; Tue, 13 Aug 2019 09:54:48 +0000 (UTC) Received: from vger.kernel.org (vger.kernel.org [209.132.180.67]) by mail.kernel.org (Postfix) with ESMTP id DDC7920665 for ; Tue, 13 Aug 2019 09:54:47 +0000 (UTC) Authentication-Results: mail.kernel.org; dkim=pass (1024-bit key) header.d=microsoft.com header.i=@microsoft.com header.b="cE25HgrR" Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727881AbfHMJyr (ORCPT ); Tue, 13 Aug 2019 05:54:47 -0400 Received: from mail-eopbgr710125.outbound.protection.outlook.com ([40.107.71.125]:30160 "EHLO NAM05-BY2-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1728322AbfHMJyr (ORCPT ); Tue, 13 Aug 2019 05:54:47 -0400 ARC-Seal: i=1; a=rsa-sha256; s=arcselector9901; d=microsoft.com; cv=none; b=M/kVhudEBJ5lTz6TvYHLplDtDceR2FYcCy0PihLVlGtxN/BGztuQJDFMgrg1B7/U9/MNMpaQIInsAsikg0BUq6cfHgwL4UyCIgvxNxF4hLZtCTmGc2Exov8zPqRairuRzz8muLJtEJG5xOOV8ug+/Dx6y5IgRjhZBnfx4KcIMRwTtwyYpWimNZvKQE6KfdT/qgUqH5I6Oo1P+WoR5tx3fUsHCj1BWQSwYvJIhXTuctpU07fSmducXnDrx1duoy/BlEqfMTtX2sl9hYeDNoq9uNoG+wec5CTr175QMjZ+iM7IkTLgAB6ys3jwY/Ug+A3qVit4bfVWHOrhv8S3lon0iw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=arcselector9901; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=5iXFtSQhwsNKjOHi1mFO6k2zlPkdyjGDHpOgAZeCwQQ=; b=anLAd8unHMfraydAf8BvqISyMUrJhsTUBdWFbXodeDZJd5JwK72To/Mbu7tEmz0XCwn5mAo/doGacsqcVKx14Yniq4t/PkIPR/pK/sl3IZdpW3540uCX0lZ+3v/mPmu8Sk2BmrdnAw8VSkkXjV/bbkiNbcN6tjwIhQPf4MPEUphM4ZVcoZ6RqchaJ2O2pwjAcZTTjgipcIXtWfMGPmbQ34NW0653BE3PmuCag4sf9a3ZMN7BSJDWJgi2UMxazfFCukYbnkJJWa4WNnf0ErvLnKPoJBu+va0GMrt9+1XC210qY7m0zaZ0EnGDo1V4DXQ/Ziqjv6R0y0dO3GlFApPOQg== ARC-Authentication-Results: i=1; mx.microsoft.com 1; spf=pass smtp.mailfrom=microsoft.com; dmarc=pass action=none header.from=microsoft.com; dkim=pass header.d=microsoft.com; arc=none DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=microsoft.com; s=selector1; h=From:Date:Subject:Message-ID:Content-Type:MIME-Version:X-MS-Exchange-SenderADCheck; bh=5iXFtSQhwsNKjOHi1mFO6k2zlPkdyjGDHpOgAZeCwQQ=; b=cE25HgrRVxJtnyc1TxxwMuiBwlmr0g9mgBfD1xbsSxUpPWc+CJxSEYNQbGFosp80wjl6hZ9Tyol9NihTUqiiiHzbIf3QgRCh23IvPCYfd0ZUFJf7mmkkHAco/8AREAbgj0FMayeTdIeeRW/IglqtorMQEDh1yVbDR2CT2ZMGw9o= Received: from DM6PR21MB1401.namprd21.prod.outlook.com (10.255.109.88) by DM6PR21MB1372.namprd21.prod.outlook.com (20.179.50.33) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.20.2157.2; Tue, 13 Aug 2019 09:54:42 +0000 Received: from DM6PR21MB1401.namprd21.prod.outlook.com ([fe80::bd0e:e64e:a357:3759]) by DM6PR21MB1401.namprd21.prod.outlook.com ([fe80::bd0e:e64e:a357:3759%5]) with mapi id 15.20.2199.004; Tue, 13 Aug 2019 09:54:42 +0000 From: Wei Hu To: "b.zolnierkie@samsung.com" , "linux-hyperv@vger.kernel.org" , "dri-devel@lists.freedesktop.org" , "linux-fbdev@vger.kernel.org" , "linux-kernel@vger.kernel.org" , "sashal@kernel.org" , Stephen Hemminger , Haiyang Zhang , KY Srinivasan , Dexuan Cui , Iouri Tarassov , Michael Kelley , Wei Hu Subject: [PATCH] video: hyperv: hyperv_fb: Obtain screen resolution from Hyper-V host Thread-Topic: [PATCH] video: hyperv: hyperv_fb: Obtain screen resolution from Hyper-V host Thread-Index: AQHVUb0g581mp+FsUUiW9Yvczi+TXQ== Date: Tue, 13 Aug 2019 09:54:42 +0000 Message-ID: <20190813095351.1780-1-weh@microsoft.com> Reply-To: Wei Hu Accept-Language: en-US Content-Language: en-US X-MS-Has-Attach: X-MS-TNEF-Correlator: x-clientproxiedby: SG2PR01CA0151.apcprd01.prod.exchangelabs.com (2603:1096:4:8f::31) To DM6PR21MB1401.namprd21.prod.outlook.com (2603:10b6:5:22d::24) authentication-results: spf=none (sender IP is ) smtp.mailfrom=lkmlwhu@microsoft.com; x-ms-exchange-messagesentrepresentingtype: 2 x-mailer: git-send-email 2.20.1 x-originating-ip: [167.220.255.113] x-ms-publictraffictype: Email x-ms-office365-filtering-correlation-id: 473b1f68-6c26-4253-6d3b-08d71fd44358 x-ms-office365-filtering-ht: Tenant x-microsoft-antispam: BCL:0;PCL:0;RULEID:(2390118)(7020095)(4652040)(8989299)(5600158)(711020)(4605104)(1401327)(4618075)(4534185)(4627221)(201703031133081)(201702281549075)(8990200)(2017052603328)(7193020);SRVR:DM6PR21MB1372; x-ms-traffictypediagnostic: DM6PR21MB1372:|DM6PR21MB1372: x-ms-exchange-transport-forked: True x-microsoft-antispam-prvs: x-ms-oob-tlc-oobclassifiers: OLM:5797; x-forefront-prvs: 01283822F8 x-forefront-antispam-report: SFV:NSPM;SFS:(10019020)(4636009)(366004)(376002)(136003)(346002)(396003)(39860400002)(189003)(199004)(3846002)(478600001)(26005)(14454004)(25786009)(8676002)(53936002)(2616005)(476003)(6486002)(66066001)(10090500001)(81166006)(81156014)(52116002)(36756003)(486006)(1511001)(256004)(22452003)(305945005)(316002)(7846003)(6392003)(6436002)(10290500003)(5660300002)(3450700001)(102836004)(8936002)(6512007)(6506007)(2501003)(110136005)(14444005)(186003)(386003)(50226002)(71200400001)(71190400001)(66556008)(2201001)(99286004)(2906002)(7736002)(66446008)(7049001)(66476007)(66946007)(43066004)(64756008)(6116002)(1076003)(921003)(1121003);DIR:OUT;SFP:1102;SCL:1;SRVR:DM6PR21MB1372;H:DM6PR21MB1401.namprd21.prod.outlook.com;FPR:;SPF:None;LANG:en;PTR:InfoNoRecords;MX:1;A:1; received-spf: None (protection.outlook.com: microsoft.com does not designate permitted sender hosts) x-ms-exchange-senderadcheck: 1 x-microsoft-antispam-message-info: Qdk/JVNZvb0FkdJC4v2U6Y3O90xJOhBarTiFLxbAF1S7rUHKKeyfS+Q7Wzpxm9J+32VMw5BcEZ3l1MnSk3K0VGGWplJ14j3HrCOLqNbdHQo+Hy4eED+Roi908uxt2Q+TJt3FUcm33rXCbThhAtVtT7f9CUsijTb1V8fKOCOBmbPvF7lxwEAarIQFY3+f4QLwRRE3NagtcQoZ774+PYVRWXOGELFoPuuUm7uQn2TGt6DV4svqWYJcnK6+/f5A6Q3MO9AlO73x+O6Vt2b/vwKPM8jei24vQ1bEdEMTZfq6u3NNsRGvCNIbksy2dvdnhHp3z3ODIvcuati6n1i6i8ESCsRPb8mjDi+jj94UcGVN9ihlpH5uhFSs7O8BarRWvR4rqxAvdDOifKpaeV4+p/447+gyzWTEgfJqt/UtUbUQI5M= Content-Type: text/plain; charset="iso-8859-1" Content-Transfer-Encoding: quoted-printable MIME-Version: 1.0 X-OriginatorOrg: microsoft.com X-MS-Exchange-CrossTenant-Network-Message-Id: 473b1f68-6c26-4253-6d3b-08d71fd44358 X-MS-Exchange-CrossTenant-originalarrivaltime: 13 Aug 2019 09:54:42.1312 (UTC) X-MS-Exchange-CrossTenant-fromentityheader: Hosted X-MS-Exchange-CrossTenant-id: 72f988bf-86f1-41af-91ab-2d7cd011db47 X-MS-Exchange-CrossTenant-mailboxtype: HOSTED X-MS-Exchange-CrossTenant-userprincipalname: I9fxANhb4cOrzdnoqyU4V6QNIhUFSX7BcgEO2HspiKcvr4CskzMC0BCHhTLCAW1YJmHtXAXNfHaJiND7OgzOsg== X-MS-Exchange-Transport-CrossTenantHeadersStamped: DM6PR21MB1372 Sender: linux-hyperv-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-hyperv@vger.kernel.org Beginning from Windows 10 RS5+, VM screen resolution is obtained from host. The "video=3Dhyperv_fb" boot time option is not needed, but still can be used to overwrite the VM resolution. The VM resolution on the host could be set by executing the powershell "set-vmvideo" command. Signed-off-by: Iouri Tarassov Signed-off-by: Wei Hu --- drivers/video/fbdev/hyperv_fb.c | 136 +++++++++++++++++++++++++++++--- 1 file changed, 125 insertions(+), 11 deletions(-) diff --git a/drivers/video/fbdev/hyperv_fb.c b/drivers/video/fbdev/hyperv_f= b.c index 00f5bdcc6c6f..1042f3311fa2 100644 --- a/drivers/video/fbdev/hyperv_fb.c +++ b/drivers/video/fbdev/hyperv_fb.c @@ -23,6 +23,14 @@ * * Portrait orientation is also supported: * For example: video=3Dhyperv_fb:864x1152 + * + * When a Windows 10 RS5+ host is used, the virtual machine screen + * resolution is obtained from the host. The "video=3Dhyperv_fb" option is + * not needed, but still can be used to overwrite the VM resolution. The + * VM resolution on the host could be set by executing the powershell + * "set-vmvideo" command. For example + * set-vmvideo -vmname name -horizontalresolution:1920 \ + * -verticalresolution:1200 -resolutiontype single */ =20 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt @@ -44,6 +52,7 @@ #define SYNTHVID_VERSION(major, minor) ((minor) << 16 | (major)) #define SYNTHVID_VERSION_WIN7 SYNTHVID_VERSION(3, 0) #define SYNTHVID_VERSION_WIN8 SYNTHVID_VERSION(3, 2) +#define SYNTHVID_VERSION_WIN10 SYNTHVID_VERSION(3, 5) =20 #define SYNTHVID_DEPTH_WIN7 16 #define SYNTHVID_DEPTH_WIN8 32 @@ -82,16 +91,25 @@ enum synthvid_msg_type { SYNTHVID_POINTER_SHAPE =3D 8, SYNTHVID_FEATURE_CHANGE =3D 9, SYNTHVID_DIRT =3D 10, + SYNTHVID_RESOLUTION_REQUEST =3D 13, + SYNTHVID_RESOLUTION_RESPONSE =3D 14, =20 - SYNTHVID_MAX =3D 11 + SYNTHVID_MAX =3D 15 }; =20 +#define SYNTHVID_EDID_BLOCK_SIZE 128 +#define SYNTHVID_MAX_RESOLUTION_COUNT 64 + +struct hvd_screen_info { + u16 width; + u16 height; +} __packed; + struct synthvid_msg_hdr { u32 type; u32 size; /* size of this header + payload after this field*/ } __packed; =20 - struct synthvid_version_req { u32 version; } __packed; @@ -102,6 +120,18 @@ struct synthvid_version_resp { u8 max_video_outputs; } __packed; =20 +struct synthvid_supported_resolution_req { + u8 maximum_resolution_count; +} __packed; + +struct synthvid_supported_resolution_resp { + u8 edid_block[SYNTHVID_EDID_BLOCK_SIZE]; + u8 resolution_count; + u8 default_resolution_index; + u8 is_standard; + struct hvd_screen_info supported_resolution[1]; +} __packed; + struct synthvid_vram_location { u64 user_ctx; u8 is_vram_gpa_specified; @@ -187,6 +217,8 @@ struct synthvid_msg { struct synthvid_pointer_shape ptr_shape; struct synthvid_feature_change feature_chg; struct synthvid_dirt dirt; + struct synthvid_supported_resolution_req resolution_req; + struct synthvid_supported_resolution_resp resolution_resp; }; } __packed; =20 @@ -224,6 +256,8 @@ struct hvfb_par { =20 static uint screen_width =3D HVFB_WIDTH; static uint screen_height =3D HVFB_HEIGHT; +static uint screen_width_max =3D HVFB_WIDTH; +static uint screen_height_max =3D HVFB_HEIGHT; static uint screen_depth; static uint screen_fb_size; =20 @@ -354,6 +388,7 @@ static void synthvid_recv_sub(struct hv_device *hdev) =20 /* Complete the wait event */ if (msg->vid_hdr.type =3D=3D SYNTHVID_VERSION_RESPONSE || + msg->vid_hdr.type =3D=3D SYNTHVID_RESOLUTION_RESPONSE || msg->vid_hdr.type =3D=3D SYNTHVID_VRAM_LOCATION_ACK) { memcpy(par->init_buf, msg, MAX_VMBUS_PKT_SIZE); complete(&par->wait); @@ -428,6 +463,64 @@ static int synthvid_negotiate_ver(struct hv_device *hd= ev, u32 ver) } =20 par->synthvid_version =3D ver; + pr_info("Synthvid Version major %d, minor %d\n", + ver & 0x0000ffff, (ver & 0xffff0000) >> 16); + +out: + return ret; +} + +/* Get current resolution from the host */ +static int synthvid_get_supported_resolution(struct hv_device *hdev) +{ + struct fb_info *info =3D hv_get_drvdata(hdev); + struct hvfb_par *par =3D info->par; + struct synthvid_msg *msg =3D (struct synthvid_msg *)par->init_buf; + int ret =3D 0; + unsigned long t; + u8 index; + int i; + + memset(msg, 0, sizeof(struct synthvid_msg)); + msg->vid_hdr.type =3D SYNTHVID_RESOLUTION_REQUEST; + msg->vid_hdr.size =3D sizeof(struct synthvid_msg_hdr) + + sizeof(struct synthvid_supported_resolution_req); + + msg->resolution_req.maximum_resolution_count =3D + SYNTHVID_MAX_RESOLUTION_COUNT; + synthvid_send(hdev, msg); + + t =3D wait_for_completion_timeout(&par->wait, VSP_TIMEOUT); + if (!t) { + pr_err("Time out on waiting resolution response\n"); + ret =3D -ETIMEDOUT; + goto out; + } + + if (msg->resolution_resp.resolution_count =3D=3D 0) { + pr_err("No supported resolutions\n"); + ret =3D -ENODEV; + goto out; + } + + index =3D msg->resolution_resp.default_resolution_index; + if (index >=3D msg->resolution_resp.resolution_count) { + pr_err("Invalid resolution index: %d\n", index); + ret =3D -ENODEV; + goto out; + } + + for (i =3D 0; i < msg->resolution_resp.resolution_count; i++) { + screen_width_max =3D max_t(unsigned int, screen_width_max, + msg->resolution_resp.supported_resolution[i].width); + screen_height_max =3D max_t(unsigned int, screen_height_max, + msg->resolution_resp.supported_resolution[i].height); + } + + screen_width =3D + msg->resolution_resp.supported_resolution[index].width; + screen_height =3D + msg->resolution_resp.supported_resolution[index].height; =20 out: return ret; @@ -448,11 +541,21 @@ static int synthvid_connect_vsp(struct hv_device *hde= v) } =20 /* Negotiate the protocol version with host */ - if (vmbus_proto_version =3D=3D VERSION_WS2008 || - vmbus_proto_version =3D=3D VERSION_WIN7) + switch (vmbus_proto_version) { + case VERSION_WS2008: + case VERSION_WIN7: ret =3D synthvid_negotiate_ver(hdev, SYNTHVID_VERSION_WIN7); - else + break; + case VERSION_WIN8: + case VERSION_WIN8_1: ret =3D synthvid_negotiate_ver(hdev, SYNTHVID_VERSION_WIN8); + break; + case VERSION_WIN10: + case VERSION_WIN10_V5: + default: + ret =3D synthvid_negotiate_ver(hdev, SYNTHVID_VERSION_WIN10); + break; + } =20 if (ret) { pr_err("Synthetic video device version not accepted\n"); @@ -464,6 +567,12 @@ static int synthvid_connect_vsp(struct hv_device *hdev= ) else screen_depth =3D SYNTHVID_DEPTH_WIN8; =20 + if (par->synthvid_version >=3D SYNTHVID_VERSION_WIN10) { + ret =3D synthvid_get_supported_resolution(hdev); + if (ret) + pr_info("Failed to get supported resolution from host, use default\n"); + } + screen_fb_size =3D hdev->channel->offermsg.offer. mmio_megabytes * 1024 * 1024; =20 @@ -653,6 +762,8 @@ static void hvfb_get_option(struct fb_info *info) } =20 if (x < HVFB_WIDTH_MIN || y < HVFB_HEIGHT_MIN || + (par->synthvid_version >=3D SYNTHVID_VERSION_WIN10 && + (x > screen_width_max || y > screen_height_max)) || (par->synthvid_version =3D=3D SYNTHVID_VERSION_WIN8 && x * y * screen_depth / 8 > SYNTHVID_FB_SIZE_WIN8) || (par->synthvid_version =3D=3D SYNTHVID_VERSION_WIN7 && @@ -689,8 +800,12 @@ static int hvfb_getmem(struct hv_device *hdev, struct = fb_info *info) } =20 if (!(pci_resource_flags(pdev, 0) & IORESOURCE_MEM) || - pci_resource_len(pdev, 0) < screen_fb_size) + pci_resource_len(pdev, 0) < screen_fb_size) { + pr_err("Resource not available or (0x%lx < 0x%lx)\n", + (unsigned long) pci_resource_len(pdev, 0), + (unsigned long) screen_fb_size); goto err1; + } =20 pot_end =3D pci_resource_end(pdev, 0); pot_start =3D pot_end - screen_fb_size + 1; @@ -781,17 +896,16 @@ static int hvfb_probe(struct hv_device *hdev, goto error1; } =20 + hvfb_get_option(info); + pr_info("Screen resolution: %dx%d, Color depth: %d\n", + screen_width, screen_height, screen_depth); + ret =3D hvfb_getmem(hdev, info); if (ret) { pr_err("No memory for framebuffer\n"); goto error2; } =20 - hvfb_get_option(info); - pr_info("Screen resolution: %dx%d, Color depth: %d\n", - screen_width, screen_height, screen_depth); - - /* Set up fb_info */ info->flags =3D FBINFO_DEFAULT; =20 --=20 2.20.1