From patchwork Tue Jan 5 10:07:20 2016 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Vincent Wan X-Patchwork-Id: 59165 Delivered-To: patch@linaro.org Received: by 10.112.130.2 with SMTP id oa2csp5709998lbb; Mon, 4 Jan 2016 18:29:08 -0800 (PST) X-Received: by 10.66.227.102 with SMTP id rz6mr129922152pac.4.1451960948029; Mon, 04 Jan 2016 18:29:08 -0800 (PST) Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id uv10si36348326pac.62.2016.01.04.18.29.07; Mon, 04 Jan 2016 18:29:07 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1754011AbcAEC3G (ORCPT + 29 others); Mon, 4 Jan 2016 21:29:06 -0500 Received: from mail-by2on0079.outbound.protection.outlook.com ([207.46.100.79]:32352 "EHLO na01-by2-obe.outbound.protection.outlook.com" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1753841AbcAEC27 (ORCPT ); Mon, 4 Jan 2016 21:28:59 -0500 Received: from BY2PR12CA0010.namprd12.prod.outlook.com (10.160.121.20) by DM3PR1201MB1087.namprd12.prod.outlook.com (10.164.198.11) with Microsoft SMTP Server (TLS) id 15.1.361.13; Tue, 5 Jan 2016 02:28:56 +0000 Received: from BY2NAM03FT019.eop-NAM03.prod.protection.outlook.com (2a01:111:f400:7e4a::207) by BY2PR12CA0010.outlook.office365.com (2a01:111:e400:2c84::20) with Microsoft SMTP Server (TLS) id 15.1.361.13 via Frontend Transport; Tue, 5 Jan 2016 02:28:56 +0000 Authentication-Results: spf=none (sender IP is 165.204.84.221) smtp.mailfrom=amd.com; 8bytes.org; dkim=none (message not signed) header.d=none; 8bytes.org; dmarc=permerror action=none header.from=amd.com; Received-SPF: None (protection.outlook.com: amd.com does not designate permitted sender hosts) Received: from atltwp01.amd.com (165.204.84.221) by BY2NAM03FT019.mail.protection.outlook.com (10.152.84.221) with Microsoft SMTP Server id 15.1.355.15 via Frontend Transport; Tue, 5 Jan 2016 02:28:54 +0000 X-WSS-ID: 0O0GK85-07-1AB-02 X-M-MSG: Received: from satlvexedge02.amd.com (satlvexedge02.amd.com [10.177.96.29]) (using TLSv1 with cipher ECDHE-RSA-AES256-SHA (256/256 bits)) (No client certificate requested) by atltwp01.amd.com (Axway MailGate 5.3.1) with ESMTPS id 2778B12C08D1; Mon, 4 Jan 2016 21:28:53 -0500 (EST) Received: from SATLEXDAG05.amd.com (10.181.40.11) by SATLVEXEDGE02.amd.com (10.177.96.29) with Microsoft SMTP Server (TLS) id 14.3.195.1; Mon, 4 Jan 2016 20:29:37 -0600 Received: from SCYBEXDAG04.amd.com (10.34.11.14) by satlexdag05.amd.com (10.181.40.11) with Microsoft SMTP Server (TLS) id 14.3.266.1; Mon, 4 Jan 2016 21:28:53 -0500 Received: from viwan-emmc.amd.com (10.237.75.243) by SCYBEXDAG04.amd.com (10.34.11.14) with Microsoft SMTP Server id 14.3.266.1; Tue, 5 Jan 2016 10:28:48 +0800 From: Wan Zongshun To: Joerg Roedel , CC: Suravee Suthikulpanit , Borislav Petkov , Ray Huang , , , Wan Zongshun Subject: [PATCH 2/6] iommu/amd: Use the most comprehensive IVHD type that the driver can support Date: Tue, 5 Jan 2016 05:07:20 -0500 Message-ID: <1451988444-4694-3-git-send-email-vincent.wan@amd.com> X-Mailer: git-send-email 1.9.1 In-Reply-To: <1451988444-4694-1-git-send-email-vincent.wan@amd.com> References: <1451988444-4694-1-git-send-email-vincent.wan@amd.com> MIME-Version: 1.0 X-EOPAttributedMessage: 0 X-Forefront-Antispam-Report: CIP:165.204.84.221; CTRY:US; IPV:NLI; EFV:NLI; SFV:NSPM; SFS:(10009020)(6009001)(2980300002)(428002)(199003)(189002)(586003)(53416004)(50226001)(92566002)(19580405001)(229853001)(50986999)(4326007)(77096005)(11100500001)(33646002)(19580395003)(5003940100001)(47776003)(36756003)(76176999)(189998001)(1096002)(106466001)(50466002)(97736004)(48376002)(86362001)(5003600100002)(87936001)(101416001)(5008740100001)(5001770100001)(2950100001)(105586002)(1220700001); DIR:OUT; SFP:1101; SCL:1; SRVR:DM3PR1201MB1087; H:atltwp01.amd.com; FPR:; SPF:None; PTR:InfoDomainNonexistent; A:1; MX:1; LANG:en; X-Microsoft-Exchange-Diagnostics: 1; DM3PR1201MB1087; 2:f4UJ1mwU39MbufAbdunr2pLtxbBDptjowJp5VQB36sLzN0rr9dHETRQCDKKCyfIqh2M72/YPSsoK9uD4Hhde5ig24/wR5Fi6AUODWX+epppFpwbjP08keexph7shBc2VhpHOYbP2wqSLcuhOTy4TfQ==; 3:I1GAmU+FxAhqqbpKMwQ27WZZXnOaayO/2x7smvDKLbdgZn1PFXrPvGf7wTX3VDycvFb4mq85NT0n1VzZu0Vdmy0Kv8D3BKWbtYYHkEP50Pxl2gzq1aqhWnwGVvAkVTOArcdbb54SGI8kJdCO8STCNvSfWeTUMBnpg1p+BrmTMdkz08cP/y+BEgePx6v9JivbfMvGN5ZhYbwMWNmcVDf2bON+plm+gMy0FANfbQ+RleU=; 25:TrKwuf5Tvm30VTj0ArBPpWJ3k3dhX11BKsgOhKZQcmmUImZy9Hpm6gsPIBqWxOqgA0cbT5cPM79ybuAiyLyI0Wb82DoYJIv+B7kj+J+kca30abv8QHru1GtV3HuoUE3+jOb+F7OgIHwxJEZFURrZMy6QtgEprJ6T8m5eMavZmOjxgsaPAsFIGcf+q/YrZ8zAnOgLrAQr5bmVf4/wc23qdhAzZ8u5mcKUH1RjHatoheEhI7hcs2QvtopBsXb/F2+MMqfOpOcnK3XWXijMWXgCIQ== X-Microsoft-Antispam: UriScan:;BCL:0;PCL:0;RULEID:;SRVR:DM3PR1201MB1087; X-Microsoft-Exchange-Diagnostics: 1; DM3PR1201MB1087; 20:EBpTJNCGbUexjSvW764h10CSAFiTH93pB9NwwWADhAbRb9hrrLisG8RCxknU40hw0bvEXG2D2f1awNhIUeuw6FNOsqNDn4IsT5ujHr//y+8b00olEfr64Fm9Rb14IQnRpMiKjJWJNFd4D6AmJudQT+UxV9Ga52SmrBBAVLz5zv9Y6/PGZU10Do305GfghxU2EgYut3hQ6z3OgP7DGvrJTHvh4zGPI2kAhUtVTtk+Hyd47A9aE9mBkJ0kT47tkjYmQegnbuwr695bU0agShW+6DQWdpxu8t0GHTP2vW/l+vyofu0s8RfoGuPqBjqsz4i2mRSy+xFK5SquiNcZaf8gb0HyghwIKz7XGZ1Zyyr76otNCY6R0sKkZM+wFo1e7UMdXGoFgYscq6zTIZw1L3eJIUeiYijKo8Zj9V613LIAF8TITVL2FctJ/GVJhlgST2l8IMk10Rfvq4FTsSNAscbeMiAe/WvsDzKyQ9g2uwcUutQlYFzfiOd0Zxnhiwv2vn+w; 4:os/Ib6uMSjxVbUTJW77HPO1XCfZAFKYlx5tPPK4Ip59jr9fR3ljWjSSGn1age8wU2JR6vOYX6oDao+Cu8bXbgTkl97i7RPta3x6wK7k4IfQxkQXEqncvB5ImaFuISduTSNbVPuhH9Qn2FFWdglgwEQ5Gn7RDJGSiaDV74dCMBRu5Y+GefkfJJYgiWReYPWQIyoYqhZ6gNAcPEAqKgM9i6poHGygXX4X3IgZ9izMjWaySgo4op17MHOoIjn1sJ9yfMw1LABx2esImpte/TrtkzB+iS6Xd7ZoDMrSwz2zZmKbF3T59vM3mocdc/KcFuPWcAhuMdxojKZvZjBWAWT9LWtmr5s8pk2b5R4lzQ4Uo32Z89ofx24UiwutFHlpQ90FnyewHz7vSx8+J2haGdySnhi5H2/CTdENeyyvQpcDmMWI= X-Microsoft-Antispam-PRVS: X-Exchange-Antispam-Report-Test: UriScan:(767451399110); X-Exchange-Antispam-Report-CFA-Test: BCL:0; PCL:0; RULEID:(601004)(2401047)(8121501046)(520078)(5005006)(10201501046)(3002001); SRVR:DM3PR1201MB1087; BCL:0; PCL:0; RULEID:; SRVR:DM3PR1201MB1087; X-Forefront-PRVS: 0812095267 X-Microsoft-Exchange-Diagnostics: =?us-ascii?Q?1; DM3PR1201MB1087; 23:aTRkNnucXEH9nfh2y4KCb44cn2iRBJYqGWYrIn6?= =?us-ascii?Q?2Gds/cpz7o114Nguzi5uZn3d6OawYCJqTTgXv39oDstE4BnnOUzyW45IBMD1?= =?us-ascii?Q?HqAkf1ikA7kNFVwCImUm3v7YWRweRx07RYeYVEdNYU1MpLyjipoVIxyq3dqR?= =?us-ascii?Q?/f0Cj6qGRcWYVjiCQz4Hk7ZFF7bX5/Z7OVUzouHl7Fr35Hk4qH5iCeHeZiCP?= =?us-ascii?Q?YBQCureqpKNIVKH5XED6DBfZGESKmTpRokavqYulTzqMKMpxiO3DZy4whx4u?= =?us-ascii?Q?Ai4jeYeMxQnMcd+7BaaHGMcOsALqL/PmOC0I29QUolK4nLZl/99ZnDsG2oEn?= =?us-ascii?Q?wQ2NeflPoRyGssWgFL5kHey6+07qDWNN8xa4+lX4sjgY8wwvVrEj6nTaVChr?= =?us-ascii?Q?RzgohwOdUvu7Pa54DPaUr/z+oUKsRi78Z71YlKB072f9hcrpbFWakxrzFMVG?= =?us-ascii?Q?DQ/qTwQM9KrnidH/qyg/56PL5mkdNmDLckwfqbnOcdwojQAb6mRya1koPd1F?= =?us-ascii?Q?Q1DQ5FkOU9AAmJD5xd85V/ewvQ5fjynQaD0XB7Ge1jrnsN7bXUKRz8VeAz3y?= =?us-ascii?Q?QxLyG+ENxa5mw/UJMYF7y/oaZckE9aGb9gOudhPPOnxPW4bGBYl63c0XF3u2?= =?us-ascii?Q?VM7gCkT8n8zIWifWM7hqnJxhQ7TDNUx52plUjgDN/i4NhaPaDYkzHIa9ZmYG?= =?us-ascii?Q?/UfYGzEEk0+UPohsk26eZZnOaxPx7JGvH2U7RyBhbiT53X4SVfDPk+7v7cNl?= =?us-ascii?Q?/8Dl6InPIdy0N2FSlXXoSc+GSrecoZ3K7m5Qg2Xkxwboh38rB1izVyPkNJcg?= =?us-ascii?Q?DUx4AkGtbAvuGQ/TXa8cL1oUfhYJSqvWxvnyAdCw2lrZ6OeLB2BievX/OuYU?= =?us-ascii?Q?6pVFKpNNISKcDDhqBEe5wvYOwxojaZjot3xsaVrgaRaQFZ63tikGUtxde9Df?= =?us-ascii?Q?Dr8buSEzabFDgyWzI/LkFoy6ymQs23afsNdcwknaQRo0UvAdzBjIC5FOwRbu?= =?us-ascii?Q?GGAI=3D?= X-Microsoft-Exchange-Diagnostics: 1; DM3PR1201MB1087; 5:l7ex/QKH9FxHmb/c3kkkzG13L6hX0JkqnBnI6TkQQ3rOsxQ3cPxW4+opyGdazmKc7Zdd3+BSE8kDKueR/CDtUMr6dUV0rh+IY0UHzRk5rkjtk1Zoz/XEZ85PuEog0arTCrFp4+M+spuvfW7NI1S8jw==; 24:mIjBbw0ye0Ii3vpDH4e+p2auncb+te/CNIxB+zyrwtUvk3L+MvxfvEvT9FFMnQwIZk54iqPUtkx0/5b1nEQT+2uxX981KWdB+o0fK8nlm9c=; 20:4cs88CJg+UwTFLDERjIoRllN1F30hyNzMWJ333JjmY2WeZVKPRr4yBhkoaMWmddTWkbPr8KGCubF/hKOgycjLsepdbNM1JRgGZnZ+y/dQW3JGDFZ477oDGzg88fTxLYWhyBGqZvvJXKlobipq8pzqP6YsWaLC8e8pRjJLNi0RB33os8UeM+94n1HxRsrzjzSu8EID331aS6PihjeHKkrEqOV9y67APnT2lEg1lXjV74W6QQCZv0d5EIv2vNZhoAl SpamDiagnosticOutput: 1:23 SpamDiagnosticMetadata: NSPM X-OriginatorOrg: amd.com X-MS-Exchange-CrossTenant-OriginalArrivalTime: 05 Jan 2016 02:28:54.4817 (UTC) X-MS-Exchange-CrossTenant-Id: 3dd8961f-e488-4e60-8e11-a82d994e183d X-MS-Exchange-CrossTenant-OriginalAttributedTenantConnectingIp: TenantId=3dd8961f-e488-4e60-8e11-a82d994e183d; Ip=[165.204.84.221]; Helo=[atltwp01.amd.com] X-MS-Exchange-CrossTenant-FromEntityHeader: HybridOnPrem X-MS-Exchange-Transport-CrossTenantHeadersStamped: DM3PR1201MB1087 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Suravee Suthikulpanit The IVRS in more recent AMD system usually contains multiple IVHD block types (e.g. 0x10, 0x11, and 0x40) for each IOMMU. The newer IVHD types provide more information (e.g. new features specified in the IOMMU spec), while maintain compatibility with the older IVHD type. Having multiple IVHD type allows older IOMMU drivers to still function (e.g. using the older IVHD type 0x10) while the newer IOMMU driver can use the newer IVHD types (e.g. 0x11 and 0x40). Therefore, the IOMMU driver should only make use of the newest IVHD type that it can support. This patch adds new logic to determine the highest level of IVHD type it can support, and use it throughout the to initialize the driver. This requires adding another pass to the IVRS parsing to determine appropriate IVHD type (see function get_highest_supported_ivhd_type()) before parsing the contents. [Vincent: fix the build error of IVHD_DEV_ACPI_HID flag not found] Signed-off-by: Wan Zongshun Signed-off-by: Suravee Suthikulpanit --- drivers/iommu/amd_iommu_init.c | 107 ++++++++++++++++++++++++++++++----------- 1 file changed, 78 insertions(+), 29 deletions(-) -- 1.9.1 -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majordomo@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/ diff --git a/drivers/iommu/amd_iommu_init.c b/drivers/iommu/amd_iommu_init.c index 2ff7000..146f3ee 100644 --- a/drivers/iommu/amd_iommu_init.c +++ b/drivers/iommu/amd_iommu_init.c @@ -44,7 +44,7 @@ */ #define IVRS_HEADER_LENGTH 48 -#define ACPI_IVHD_TYPE 0x10 +#define ACPI_IVHD_TYPE_MAX_SUPPORTED 0x40 #define ACPI_IVMD_TYPE_ALL 0x20 #define ACPI_IVMD_TYPE 0x21 #define ACPI_IVMD_TYPE_RANGE 0x22 @@ -58,6 +58,7 @@ #define IVHD_DEV_EXT_SELECT 0x46 #define IVHD_DEV_EXT_SELECT_RANGE 0x47 #define IVHD_DEV_SPECIAL 0x48 +#define IVHD_DEV_ACPI_HID 0xf0 #define IVHD_SPECIAL_IOAPIC 1 #define IVHD_SPECIAL_HPET 2 @@ -137,6 +138,7 @@ bool amd_iommu_irq_remap __read_mostly; static bool amd_iommu_detected; static bool __initdata amd_iommu_disabled; +static int amd_iommu_target_ivhd_type; u16 amd_iommu_last_bdf; /* largest PCI device id we have to handle */ @@ -424,7 +426,15 @@ static inline u32 get_ivhd_header_size(struct ivhd_header *h) */ static inline int ivhd_entry_length(u8 *ivhd) { - return 0x04 << (*ivhd >> 6); + u32 type = ((struct ivhd_entry *)ivhd)->type; + + if (type < 0x80) { + return 0x04 << (*ivhd >> 6); + } else if (type == IVHD_DEV_ACPI_HID) { + /* For ACPI_HID, offset 21 is uid len */ + return *((u8 *)ivhd + 21) + 22; + } + return 0; } /* @@ -470,6 +480,22 @@ static int __init find_last_devid_from_ivhd(struct ivhd_header *h) return 0; } +static int __init check_ivrs_checksum(struct acpi_table_header *table) +{ + int i; + u8 checksum = 0, *p = (u8 *)table; + + for (i = 0; i < table->length; ++i) + checksum += p[i]; + if (checksum != 0) { + /* ACPI table corrupt */ + pr_err(FW_BUG "AMD-Vi: IVRS invalid checksum\n"); + return -ENODEV; + } + + return 0; +} + /* * Iterate over all IVHD entries in the ACPI table and find the highest device * id which we need to handle. This is the first of three functions which parse @@ -477,31 +503,19 @@ static int __init find_last_devid_from_ivhd(struct ivhd_header *h) */ static int __init find_last_devid_acpi(struct acpi_table_header *table) { - int i; - u8 checksum = 0, *p = (u8 *)table, *end = (u8 *)table; + u8 *p = (u8 *)table, *end = (u8 *)table; struct ivhd_header *h; - /* - * Validate checksum here so we don't need to do it when - * we actually parse the table - */ - for (i = 0; i < table->length; ++i) - checksum += p[i]; - if (checksum != 0) - /* ACPI table corrupt */ - return -ENODEV; - p += IVRS_HEADER_LENGTH; end += table->length; while (p < end) { h = (struct ivhd_header *)p; - switch (h->type) { - case ACPI_IVHD_TYPE: - find_last_devid_from_ivhd(h); - break; - default: - break; + if (h->type == amd_iommu_target_ivhd_type) { + int ret = find_last_devid_from_ivhd(h); + + if (ret) + return ret; } p += h->length; } @@ -1118,6 +1132,32 @@ static int __init init_iommu_one(struct amd_iommu *iommu, struct ivhd_header *h) return 0; } +/** + * get_highest_supported_ivhd_type - Look up the appropriate IVHD type + * @ivrs Pointer to the IVRS header + * + * This function search through all IVDB of the maximum supported IVHD + */ +static u8 get_highest_supported_ivhd_type(struct acpi_table_header *ivrs) +{ + u8 *base = (u8 *)ivrs; + struct ivhd_header *ivhd = (struct ivhd_header *) + (base + IVRS_HEADER_LENGTH); + u8 last_type = ivhd->type; + u16 devid = ivhd->devid; + + while (((u8 *)ivhd - base < ivrs->length) && + (ivhd->type <= ACPI_IVHD_TYPE_MAX_SUPPORTED)) { + u8 *p = (u8 *) ivhd; + + if (ivhd->devid == devid) + last_type = ivhd->type; + ivhd = (struct ivhd_header *)(p + ivhd->length); + } + + return last_type; +} + /* * Iterates over all IOMMU entries in the ACPI table, allocates the * IOMMU structure and initializes it with init_iommu_one() @@ -1134,8 +1174,7 @@ static int __init init_iommu_all(struct acpi_table_header *table) while (p < end) { h = (struct ivhd_header *)p; - switch (*p) { - case ACPI_IVHD_TYPE: + if (*p == amd_iommu_target_ivhd_type) { DUMP_printk("device: %02x:%02x.%01x cap: %04x " "seg: %d flags: %01x info %04x\n", @@ -1152,9 +1191,6 @@ static int __init init_iommu_all(struct acpi_table_header *table) ret = init_iommu_one(iommu, h); if (ret) return ret; - break; - default: - break; } p += h->length; @@ -1818,18 +1854,20 @@ static void __init free_dma_resources(void) * remapping setup code. * * This function basically parses the ACPI table for AMD IOMMU (IVRS) - * three times: + * four times: * - * 1 pass) Find the highest PCI device id the driver has to handle. + * 1 pass) Discover the most comprehensive IVHD type to use. + * + * 2 pass) Find the highest PCI device id the driver has to handle. * Upon this information the size of the data structures is * determined that needs to be allocated. * - * 2 pass) Initialize the data structures just allocated with the + * 3 pass) Initialize the data structures just allocated with the * information in the ACPI table about available AMD IOMMUs * in the system. It also maps the PCI devices in the * system to specific IOMMUs * - * 3 pass) After the basic data structures are allocated and + * 4 pass) After the basic data structures are allocated and * initialized we update them with information about memory * remapping requirements parsed out of the ACPI table in * this last pass. @@ -1857,6 +1895,17 @@ static int __init early_amd_iommu_init(void) } /* + * Validate checksum here so we don't need to do it when + * we actually parse the table + */ + ret = check_ivrs_checksum(ivrs_base); + if (ret) + return ret; + + amd_iommu_target_ivhd_type = get_highest_supported_ivhd_type(ivrs_base); + DUMP_printk("Using IVHD type %#x\n", amd_iommu_target_ivhd_type); + + /* * First parse ACPI tables to find the largest Bus/Dev/Func * we need to handle. Upon this information the shared data * structures for the IOMMUs in the system will be allocated