Received: by 2002:a25:1506:0:0:0:0:0 with SMTP id 6csp1618135ybv; Fri, 21 Feb 2020 00:23:19 -0800 (PST) X-Google-Smtp-Source: APXvYqy1RbMv5HEy2JOZy7OtQPMREGnHR7pJh/Tzrpmaqza4RdxXLKy+FYlXssTA1uazqqnOo5kE X-Received: by 2002:a9d:de9:: with SMTP id 96mr27256105ots.222.1582273399828; Fri, 21 Feb 2020 00:23:19 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1582273399; cv=none; d=google.com; s=arc-20160816; b=Ybrtg3FgkWtATHT7/5wwlvFN7+wMQO8AphFEdcx4mA4bmvnYlzPAVDmlxpbXRekKlW 4O51EINFB0+aUe/D/xMx51xXvs2qOIknIfVCiGfL48rIuPt44FX+a3yEcwIvgcX92l/S /CCUWA8BVVsr1fbjd09hykFOJW6w0gphvcXxM+sd8PXzcCxqmEMJHge4dD9zBnc35T2r z+asRr66NCEZ3m4UHnX29p3hh1xDP4oizIJ6p0/mz+N3VM8y8U+HRAaJCln7RLEpXPf1 TbCjMjnvAnqums89hyT9RvQxECYVukCEcVq4BgdzNfkzK8B8uITGpef58CiOnvzT6XLF UCgQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :user-agent:references:in-reply-to:message-id:date:subject:cc:to :from:dkim-signature; bh=QVoRRodgYTNQ599OWoIS48rNbT/ZJe5MGIIzYj///ho=; b=UWgCS8I8c67He4bMtKuPTNB9KrMnAJH8FARc2sXyYXWMgShhpKezKYGpVvCbJ8oj3r nTsrOFxTXA7oRneQ8rDdSy6SImhac/yWvhlLNVv69C82CkB0al25T7mAmw2YJNLd03P0 xDsZHbQUbotOqc68k6PyOl/QHg5Eu1C5NyCrJGcIe72IJXP7HlCHD/eSgBDYZDMfHTg2 a5wZtQM0Ght4ZCXizEwHyScJjOZV0nPhsfEINjt5Ay0g/M+a6U5qMoZrlF9+oExuaSpN gQSHsjgBKjVzr2wcAulnPg+KLqA6bFqweeS1KRrx23vgudOR3/p1y5SUWt0CDDxLJ5J6 s/Xw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@kernel.org header.s=default header.b=hlFrtRHM; 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 Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id v9si443529oie.22.2020.02.21.00.23.07; Fri, 21 Feb 2020 00:23:19 -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; dkim=pass header.i=@kernel.org header.s=default header.b=hlFrtRHM; 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 S2388308AbgBUIW2 (ORCPT + 99 others); Fri, 21 Feb 2020 03:22:28 -0500 Received: from mail.kernel.org ([198.145.29.99]:34022 "EHLO mail.kernel.org" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S2388299AbgBUIW1 (ORCPT ); Fri, 21 Feb 2020 03:22:27 -0500 Received: from localhost (83-86-89-107.cable.dynamic.v4.ziggo.nl [83.86.89.107]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by mail.kernel.org (Postfix) with ESMTPSA id 8066724676; Fri, 21 Feb 2020 08:22:25 +0000 (UTC) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=kernel.org; s=default; t=1582273346; bh=8NIPlXE5fhwx6ut/fi8+s25a+Xp8Zth5z/XDUDUewjw=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=hlFrtRHMD1pxeQkpMH+r5MUHieKjpavNRPpZ4nQENI1e9D4ww9eLlKNpI3ulRCMCl wJHZO/uIiqJZzeQ00zsFf7As8CCmY5MFVyW7ReBdfNZQjeYM6hUr/82c33lVSP8nBt YZMjjA/qJWR+kKyS8fU41dt6y46q4nTKtyvfQPnc= From: Greg Kroah-Hartman To: linux-kernel@vger.kernel.org Cc: Greg Kroah-Hartman , stable@vger.kernel.org, Pankaj Bansal , Hanjun Guo , Lorenzo Pieralisi , Will Deacon , Sudeep Holla , Catalin Marinas , Robin Murphy , Sasha Levin Subject: [PATCH 4.19 137/191] ACPI/IORT: Fix Number of IDs handling in iort_id_map() Date: Fri, 21 Feb 2020 08:41:50 +0100 Message-Id: <20200221072307.150803032@linuxfoundation.org> X-Mailer: git-send-email 2.25.1 In-Reply-To: <20200221072250.732482588@linuxfoundation.org> References: <20200221072250.732482588@linuxfoundation.org> User-Agent: quilt/0.66 MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Hanjun Guo [ Upstream commit 3c23b83a88d00383e1d498cfa515249aa2fe0238 ] The IORT specification [0] (Section 3, table 4, page 9) defines the 'Number of IDs' as 'The number of IDs in the range minus one'. However, the IORT ID mapping function iort_id_map() treats the 'Number of IDs' field as if it were the full IDs mapping count, with the following check in place to detect out of boundary input IDs: InputID >= Input base + Number of IDs This check is flawed in that it considers the 'Number of IDs' field as the full number of IDs mapping and disregards the 'minus one' from the IDs count. The correct check in iort_id_map() should be implemented as: InputID > Input base + Number of IDs this implements the specification correctly but unfortunately it breaks existing firmwares that erroneously set the 'Number of IDs' as the full IDs mapping count rather than IDs mapping count minus one. e.g. PCI hostbridge mapping entry 1: Input base: 0x1000 ID Count: 0x100 Output base: 0x1000 Output reference: 0xC4 //ITS reference PCI hostbridge mapping entry 2: Input base: 0x1100 ID Count: 0x100 Output base: 0x2000 Output reference: 0xD4 //ITS reference Two mapping entries which the second entry's Input base = the first entry's Input base + ID count, so for InputID 0x1100 and with the correct InputID check in place in iort_id_map() the kernel would map the InputID to ITS 0xC4 not 0xD4 as it would be expected. Therefore, to keep supporting existing flawed firmwares, introduce a workaround that instructs the kernel to use the old InputID range check logic in iort_id_map(), so that we can support both firmwares written with the flawed 'Number of IDs' logic and the correct one as defined in the specifications. [0]: http://infocenter.arm.com/help/topic/com.arm.doc.den0049d/DEN0049D_IO_Remapping_Table.pdf Reported-by: Pankaj Bansal Link: https://lore.kernel.org/linux-acpi/20191215203303.29811-1-pankaj.bansal@nxp.com/ Signed-off-by: Hanjun Guo Signed-off-by: Lorenzo Pieralisi Cc: Pankaj Bansal Cc: Will Deacon Cc: Sudeep Holla Cc: Catalin Marinas Cc: Robin Murphy Signed-off-by: Will Deacon Signed-off-by: Sasha Levin --- drivers/acpi/arm64/iort.c | 57 +++++++++++++++++++++++++++++++++++++-- 1 file changed, 55 insertions(+), 2 deletions(-) diff --git a/drivers/acpi/arm64/iort.c b/drivers/acpi/arm64/iort.c index e11b5da6f828f..7d86468300b78 100644 --- a/drivers/acpi/arm64/iort.c +++ b/drivers/acpi/arm64/iort.c @@ -306,6 +306,59 @@ out: return status; } +struct iort_workaround_oem_info { + char oem_id[ACPI_OEM_ID_SIZE + 1]; + char oem_table_id[ACPI_OEM_TABLE_ID_SIZE + 1]; + u32 oem_revision; +}; + +static bool apply_id_count_workaround; + +static struct iort_workaround_oem_info wa_info[] __initdata = { + { + .oem_id = "HISI ", + .oem_table_id = "HIP07 ", + .oem_revision = 0, + }, { + .oem_id = "HISI ", + .oem_table_id = "HIP08 ", + .oem_revision = 0, + } +}; + +static void __init +iort_check_id_count_workaround(struct acpi_table_header *tbl) +{ + int i; + + for (i = 0; i < ARRAY_SIZE(wa_info); i++) { + if (!memcmp(wa_info[i].oem_id, tbl->oem_id, ACPI_OEM_ID_SIZE) && + !memcmp(wa_info[i].oem_table_id, tbl->oem_table_id, ACPI_OEM_TABLE_ID_SIZE) && + wa_info[i].oem_revision == tbl->oem_revision) { + apply_id_count_workaround = true; + pr_warn(FW_BUG "ID count for ID mapping entry is wrong, applying workaround\n"); + break; + } + } +} + +static inline u32 iort_get_map_max(struct acpi_iort_id_mapping *map) +{ + u32 map_max = map->input_base + map->id_count; + + /* + * The IORT specification revision D (Section 3, table 4, page 9) says + * Number of IDs = The number of IDs in the range minus one, but the + * IORT code ignored the "minus one", and some firmware did that too, + * so apply a workaround here to keep compatible with both the spec + * compliant and non-spec compliant firmwares. + */ + if (apply_id_count_workaround) + map_max--; + + return map_max; +} + static int iort_id_map(struct acpi_iort_id_mapping *map, u8 type, u32 rid_in, u32 *rid_out) { @@ -322,8 +375,7 @@ static int iort_id_map(struct acpi_iort_id_mapping *map, u8 type, u32 rid_in, return -ENXIO; } - if (rid_in < map->input_base || - (rid_in >= map->input_base + map->id_count)) + if (rid_in < map->input_base || rid_in > iort_get_map_max(map)) return -ENXIO; *rid_out = map->output_base + (rid_in - map->input_base); @@ -1542,5 +1594,6 @@ void __init acpi_iort_init(void) return; } + iort_check_id_count_workaround(iort_table); iort_init_platform_devices(); } -- 2.20.1