Received: by 2002:a05:6a10:8c0a:0:0:0:0 with SMTP id go10csp7952017pxb; Fri, 19 Feb 2021 03:46:32 -0800 (PST) X-Google-Smtp-Source: ABdhPJz8pQ/wYtO2zCCOy/g5dzl7Y8P3MbzcTsjQuyF+4+Sl6R+MXuW9ZfLmAZCFIbYCe0vY42Bv X-Received: by 2002:a05:6402:b86:: with SMTP id cf6mr8968746edb.66.1613735192754; Fri, 19 Feb 2021 03:46:32 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1613735192; cv=none; d=google.com; s=arc-20160816; b=jbiBrBXZzxa59OIXuSXxmk/Lc1+/gGztyMyCL1Tt4tjkgAs23GxwiOV78bX+63n6Hg DwgfXBSInDW/Ewf0DcLa8dGcO3g3cCuyAznJ8DAovriL/YXcFyJY2WfH6vYbo2z5L4W3 ZumK5ohQ+6HhRToCMByOKqSpQasyZhvnjnmTlnEDUtTr35jlFLmb5Qrz8e7V9NccfNyX SuEI+qvkgeU7YfwPY3MighyTaf76FUWPxlDn+NalMX9aZBVgXjsRgEpwwFgzR9xE6iZJ TKG1sPDT7bwvTIIJflyhUq/o3jb845ZBKcq7U3dsIsMhGo6UMmqeSkj5jSa9i4TfTNti Hu6Q== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:references:in-reply-to:message-id:date:subject :cc:to:from; bh=cfpgWYWsvYHxC4vJncF0fC64KxUzp4usmDC3KDvPqtw=; b=pNc+7c9TS0sNkHCMCAnaBBEYGivRAsUMeBhATwqsgUGuvPsayPCbsTaCFYEr5aymwG p5bD2F/DfQ+sTzrpXStPek5EzcDoGVr5VCJl4lBbd/s7px5Wa/2+xKqtS/n09wZJlkRT +gs8GF6qAwLxfe6oSQcZhBMBiSDLnf/gtlUdc3WVmtoLdObSpPLtp7Af+bb3P/B8t79t cT6I4gAs4ZGO0JCZ9iNYYpcN34n8fBnzNAyM8HrHJYdqGvmGvWwONfVQQWxCEvkkFGC9 +8aMpMzKhYANVwERRRU3R4O1YBOG+3HvMG/++Q14rbC+pR72ym8nIPNUuv0bXPvLyiJ/ o6MQ== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id ly13si104871ejb.361.2021.02.19.03.46.07; Fri, 19 Feb 2021 03:46:32 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=fail (p=NONE sp=NONE dis=NONE) header.from=arm.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S230484AbhBSLmg (ORCPT + 99 others); Fri, 19 Feb 2021 06:42:36 -0500 Received: from foss.arm.com ([217.140.110.172]:34270 "EHLO foss.arm.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S230439AbhBSLku (ORCPT ); Fri, 19 Feb 2021 06:40:50 -0500 Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id 832A41435; Fri, 19 Feb 2021 03:39:52 -0800 (PST) Received: from lakrids.cambridge.arm.com (usa-sjc-imap-foss1.foss.arm.com [10.121.207.14]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPA id 4D48B3F694; Fri, 19 Feb 2021 03:39:51 -0800 (PST) From: Mark Rutland To: linux-arm-kernel@lists.infradead.org, linux-kernel@vger.kernel.org Cc: catalin.marinas@arm.com, james.morse@arm.com, marcan@marcan.st, mark.rutland@arm.com, maz@kernel.org, tglx@linutronix.de, will@kernel.org Subject: [PATCH 8/8] arm64: irq: allow FIQs to be handled Date: Fri, 19 Feb 2021 11:39:04 +0000 Message-Id: <20210219113904.41736-9-mark.rutland@arm.com> X-Mailer: git-send-email 2.11.0 In-Reply-To: <20210219113904.41736-1-mark.rutland@arm.com> References: <20210219113904.41736-1-mark.rutland@arm.com> Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On contemporary platforms we don't use FIQ, and treat any stray FIQ as a fatal event. However, some platforms have an interrupt controller wired to FIQ, and need to handle FIQ as part of regular operation. So that we can support both cases dynamically, this patch updates the FIQ exception handling code to operate the same way as the IRQ handling code, with its own handle_arch_fiq handler. Where an FIQ handler is not registered, an unexpected FIQ exception will trigger the default FIQ handler, which will panic() as today. Where a FIQ handler is registered, handling of the FIQ is deferred to that handler. Signed-off-by: Mark Rutland Cc: Catalin Marinas Cc: Hector Martin Cc: James Morse Cc: Marc Zyngier Cc: Thomas Gleixner Cc: Will Deacon --- arch/arm64/include/asm/irq.h | 1 + arch/arm64/kernel/entry.S | 26 ++++++++++++++++++++++---- arch/arm64/kernel/irq.c | 15 +++++++++++++++ 3 files changed, 38 insertions(+), 4 deletions(-) diff --git a/arch/arm64/include/asm/irq.h b/arch/arm64/include/asm/irq.h index 8391c6f6f746..fac08e18bcd5 100644 --- a/arch/arm64/include/asm/irq.h +++ b/arch/arm64/include/asm/irq.h @@ -10,6 +10,7 @@ struct pt_regs; int set_handle_irq(void (*handle_irq)(struct pt_regs *)); #define set_handle_irq set_handle_irq +int set_handle_fiq(void (*handle_fiq)(struct pt_regs *)); static inline int nr_legacy_irqs(void) { diff --git a/arch/arm64/kernel/entry.S b/arch/arm64/kernel/entry.S index 0474cca9f1a9..a8290bd87a49 100644 --- a/arch/arm64/kernel/entry.S +++ b/arch/arm64/kernel/entry.S @@ -586,23 +586,23 @@ SYM_CODE_START(vectors) kernel_ventry 1, sync // Synchronous EL1h kernel_ventry 1, irq // IRQ EL1h - kernel_ventry 1, fiq_invalid // FIQ EL1h + kernel_ventry 1, fiq // FIQ EL1h kernel_ventry 1, error // Error EL1h kernel_ventry 0, sync // Synchronous 64-bit EL0 kernel_ventry 0, irq // IRQ 64-bit EL0 - kernel_ventry 0, fiq_invalid // FIQ 64-bit EL0 + kernel_ventry 0, fiq // FIQ 64-bit EL0 kernel_ventry 0, error // Error 64-bit EL0 #ifdef CONFIG_COMPAT kernel_ventry 0, sync_compat, 32 // Synchronous 32-bit EL0 kernel_ventry 0, irq_compat, 32 // IRQ 32-bit EL0 - kernel_ventry 0, fiq_invalid_compat, 32 // FIQ 32-bit EL0 + kernel_ventry 0, fiq_compat, 32 // FIQ 32-bit EL0 kernel_ventry 0, error_compat, 32 // Error 32-bit EL0 #else kernel_ventry 0, sync_invalid, 32 // Synchronous 32-bit EL0 kernel_ventry 0, irq_invalid, 32 // IRQ 32-bit EL0 - kernel_ventry 0, fiq_invalid, 32 // FIQ 32-bit EL0 + kernel_ventry 0, fiq, 32 // FIQ 32-bit EL0 kernel_ventry 0, error_invalid, 32 // Error 32-bit EL0 #endif SYM_CODE_END(vectors) @@ -703,6 +703,12 @@ SYM_CODE_START_LOCAL_NOALIGN(el1_irq) kernel_exit 1 SYM_CODE_END(el1_irq) +SYM_CODE_START_LOCAL_NOALIGN(el1_fiq) + kernel_entry 1 + el1_interrupt_handler handle_arch_fiq + kernel_exit 1 +SYM_CODE_END(el1_fiq) + /* * EL0 mode handlers. */ @@ -729,6 +735,11 @@ SYM_CODE_START_LOCAL_NOALIGN(el0_irq_compat) b el0_irq_naked SYM_CODE_END(el0_irq_compat) +SYM_CODE_START_LOCAL_NOALIGN(el0_fiq_compat) + kernel_entry 0, 32 + b el0_fiq_naked +SYM_CODE_END(el0_fiq_compat) + SYM_CODE_START_LOCAL_NOALIGN(el0_error_compat) kernel_entry 0, 32 b el0_error_naked @@ -743,6 +754,13 @@ el0_irq_naked: b ret_to_user SYM_CODE_END(el0_irq) +SYM_CODE_START_LOCAL_NOALIGN(el0_fiq) + kernel_entry 0 +el0_fiq_naked: + el0_interrupt_handler handle_arch_fiq + b ret_to_user +SYM_CODE_END(el0_fiq) + SYM_CODE_START_LOCAL(el1_error) kernel_entry 1 mrs x1, esr_el1 diff --git a/arch/arm64/kernel/irq.c b/arch/arm64/kernel/irq.c index 00bcf37aa0ea..bc3215dedf47 100644 --- a/arch/arm64/kernel/irq.c +++ b/arch/arm64/kernel/irq.c @@ -76,7 +76,13 @@ void default_handle_irq(struct pt_regs *regs) panic("IRQ taken without a registered IRQ controller\n"); } +void default_handle_fiq(struct pt_regs *regs) +{ + panic("FIQ taken without a registered FIQ controller\n"); +} + void (*handle_arch_irq)(struct pt_regs *) __ro_after_init = default_handle_irq; +void (*handle_arch_fiq)(struct pt_regs *) __ro_after_init = default_handle_fiq; int __init set_handle_irq(void (*handle_irq)(struct pt_regs *)) { @@ -87,6 +93,15 @@ int __init set_handle_irq(void (*handle_irq)(struct pt_regs *)) return 0; } +int __init set_handle_fiq(void (*handle_fiq)(struct pt_regs *)) +{ + if (handle_arch_fiq != default_handle_fiq) + return -EBUSY; + + handle_arch_fiq = handle_fiq; + return 0; +} + void __init init_IRQ(void) { init_irq_stacks(); -- 2.11.0