Received: by 2002:a25:6193:0:0:0:0:0 with SMTP id v141csp5027751ybb; Tue, 24 Mar 2020 09:36:32 -0700 (PDT) X-Google-Smtp-Source: ADFU+vv7q0RHvnQpudgHromXnrVTL/cXoyw/jH4XaofatuYQmyfKUyc4dBTHr+wW0nLMKGTl5m5d X-Received: by 2002:aca:5508:: with SMTP id j8mr4134549oib.71.1585067792392; Tue, 24 Mar 2020 09:36:32 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1585067792; cv=none; d=google.com; s=arc-20160816; b=krVxzIVoFbioUlgxA5LaiySw37UE7PwL8LftBUSqTTkGhEbzd+e/Rb9C1eguguFZd6 pdk3/CkHYseiWVP4JbKlVheKWenW6aScUXDMdOVhjXW0mvreM2NMm/+HeUbP4UgI+HNs xPoi4KaU+XKHr5S66nRs3VZEJeoH2TrD6fCvOMh+l6/bi72yB3c7lu8bwGzMhKqUQP3A GfEcu7OLV+cr2Z8l28MvBVTwTXrrMD0ENJeQSbcSxUFyQswqkH2R8iGlQ0vYtPM/x/As ZjlMO673HKsXHM32qjL1e+L7dLnb36/xJ7wVcUydl86xAWVoOSNJT9Y/VyU62UZdIOI1 AIWw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-disposition:mime-version:references:message-id:subject:cc :to:from:date:dkim-signature; bh=vff69t2B7SQ1Pw7zvoBcJmdMnx/6FGItZ4cwDATRo48=; b=LAEn8A6aEjNlHg7Z93FlfWDSFjrqoLCTJ02Nm4uwmGlABZ59yqHTyhmYJE3jca2Tix oiTKwAerwUCajPCXUrFMCbSCzUfltjOn1FpUKK+NZjnGJzaBYpTPey4dwPSg2GYEqRg1 pOqfaFIO9tyF5yeybVI3vPH11c4Mjla5+Gl8GRhq1adD6QguPdXOKtiLXS8mwhVKtS1f DAUmx/WK+Z8ipbQFjXq6Wgrhjb0rjjJevws78wy8rwX/5lChRaMcnBpyHukx+R5kDCHs /mayM6aqSKSl8e2qFW7y/G+u3IJNr4+QdB5Avuj7UrBe6rXvVM2gQ9fUIAeRD3oRtLkJ LR8A== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=mkn9ysj5; 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; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id g144si9404263oib.40.2020.03.24.09.36.18; Tue, 24 Mar 2020 09:36:32 -0700 (PDT) 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=@linaro.org header.s=google header.b=mkn9ysj5; 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; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727907AbgCXQfZ (ORCPT + 99 others); Tue, 24 Mar 2020 12:35:25 -0400 Received: from mail-pl1-f194.google.com ([209.85.214.194]:40855 "EHLO mail-pl1-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727257AbgCXQfZ (ORCPT ); Tue, 24 Mar 2020 12:35:25 -0400 Received: by mail-pl1-f194.google.com with SMTP id h11so7600278plk.7 for ; Tue, 24 Mar 2020 09:35:23 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=date:from:to:cc:subject:message-id:references:mime-version :content-disposition:in-reply-to:user-agent; bh=vff69t2B7SQ1Pw7zvoBcJmdMnx/6FGItZ4cwDATRo48=; b=mkn9ysj5s7yf+bDP9/trY8miOqmNkIxwScT2coC6IYNMr9DD4KWk/LUKGbvrGGXCWQ j6Az2+CId0v7n+xaGEnNGMtqRFjBNcw30gHjVhE/K4icX8YjXyJrH2o4t90KaPh6lKJ7 BO9fSrUyh0NKWp5lz+OmBz3JA//lD9MaCkeVDjQKlcgVTFebRpK8SZGPsGOYLrUJZAoN 4lXWl/EzdUHU0vcN0dgSO5zI4pLV0FsBdscOE+/wNl+Ds/TDJoxOvOwKTu3/XWSZzjnB UNWtZU23ZIuZmHtTBJQz7d6fOvIEwKD3+j75aye6JFWrl28Mbfg+9fwsNZf1N08y8eh7 IseQ== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:references :mime-version:content-disposition:in-reply-to:user-agent; bh=vff69t2B7SQ1Pw7zvoBcJmdMnx/6FGItZ4cwDATRo48=; b=Xo6P0cxh+GIk2+1JPlV4nrxLFjfSC3wqmsmqJipxFQRcRTiEMGwi06YHZQLhUjieu2 53WDkkOr9/0Qsgz58y4WxE6gDvyruHl8gkbveRzoheh3xc24Tivqnq6/ieUxYBMW04Wf 3sfYUcrqmhxOhHwtfY/AzdGY7UnFKj3b/lQwZhmtL6J5/VqcDrlQiuJON+1v8UZYfv7K buDMpKHgQ5CF7a4uLpTpT9pzjahNAx9m6kce2hlgFmc4Y246R9LBZwLZZDl2sUjGOZWP 0uJIQVYLNRPzhuZes/YgMp4n+6zP8I+eG37wo39JfbyaU0063zbK0se2KewzP89x+ENF j71A== X-Gm-Message-State: ANhLgQ3XUXPSK5i9OJPYkJ4Ed6CSX0bcvsiVLWcLYUoteIpzlf3AvQ4l w2JaAzVBOg7Jhv6xLA26BmlICQ== X-Received: by 2002:a17:902:b60f:: with SMTP id b15mr13456088pls.14.1585067722880; Tue, 24 Mar 2020 09:35:22 -0700 (PDT) Received: from xps15 (S0106002369de4dac.cg.shawcable.net. [68.147.8.254]) by smtp.gmail.com with ESMTPSA id h4sm16588503pfg.177.2020.03.24.09.35.21 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 24 Mar 2020 09:35:22 -0700 (PDT) Date: Tue, 24 Mar 2020 10:35:20 -0600 From: Mathieu Poirier To: Tero Kristo Cc: bjorn.andersson@linaro.org, ohad@wizery.com, linux-remoteproc@vger.kernel.org, linux-kernel@vger.kernel.org, s-anna@ti.com, afd@ti.com Subject: Re: [PATCHv9 14/15] remoteproc/omap: Add watchdog functionality for remote processors Message-ID: <20200324163520.GA17283@xps15> References: <20200324110035.29907-1-t-kristo@ti.com> <20200324110035.29907-15-t-kristo@ti.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <20200324110035.29907-15-t-kristo@ti.com> User-Agent: Mutt/1.10.1 (2018-07-13) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Tue, Mar 24, 2020 at 01:00:34PM +0200, Tero Kristo wrote: > From: Suman Anna > > Remote processors can be stuck in a loop, and may not be recoverable > if they do not have a built-in watchdog. The watchdog implementation > for OMAP remote processors uses external gptimers that can be used > to interrupt both the Linux host as well as the remote processor. > > Each remote processor is responsible for refreshing the timer during > normal behavior - during OS task scheduling or entering the idle loop > properly. During a watchdog condition (executing a tight loop causing > no scheduling), the host processor gets interrupts and schedules a > recovery for the corresponding remote processor. The remote processor > may also get interrupted to be able to print a back trace. > > A menuconfig option has also been added to enable/disable the Watchdog > functionality, with the default as disabled. > > Signed-off-by: Suman Anna > Signed-off-by: Tero Kristo > Reviewed-by: Andrew F. Davis > --- > v9: > * split out the counting of watchdog timers to separate func, and added this behind the config flag > drivers/remoteproc/Kconfig | 12 ++ > drivers/remoteproc/omap_remoteproc.c | 166 +++++++++++++++++++++++++-- > 2 files changed, 167 insertions(+), 11 deletions(-) > > diff --git a/drivers/remoteproc/Kconfig b/drivers/remoteproc/Kconfig > index b52abc2268cc..5f33358eb2f1 100644 > --- a/drivers/remoteproc/Kconfig > +++ b/drivers/remoteproc/Kconfig > @@ -52,6 +52,18 @@ config OMAP_REMOTEPROC > It's safe to say N here if you're not interested in multimedia > offloading or just want a bare minimum kernel. > > +config OMAP_REMOTEPROC_WATCHDOG > + bool "OMAP remoteproc watchdog timer" > + depends on OMAP_REMOTEPROC > + default n > + help > + Say Y here to enable watchdog timer for remote processors. > + > + This option controls the watchdog functionality for the remote > + processors in OMAP. Dedicated OMAP DMTimers are used by the remote > + processors and triggers the timer interrupt upon a watchdog > + detection. > + > config WKUP_M3_RPROC > tristate "AMx3xx Wakeup M3 remoteproc support" > depends on SOC_AM33XX || SOC_AM43XX > diff --git a/drivers/remoteproc/omap_remoteproc.c b/drivers/remoteproc/omap_remoteproc.c > index cee6c33869b3..cdb546f7232e 100644 > --- a/drivers/remoteproc/omap_remoteproc.c > +++ b/drivers/remoteproc/omap_remoteproc.c > @@ -24,6 +24,7 @@ > #include > #include > #include > +#include > #include > #include > #include > @@ -72,10 +73,12 @@ struct omap_rproc_mem { > * struct omap_rproc_timer - data structure for a timer used by a omap rproc > * @odt: timer pointer > * @timer_ops: OMAP dmtimer ops for @odt timer > + * @irq: timer irq > */ > struct omap_rproc_timer { > struct omap_dm_timer *odt; > const struct omap_dm_timer_ops *timer_ops; > + int irq; > }; > > /** > @@ -86,6 +89,7 @@ struct omap_rproc_timer { > * @mem: internal memory regions data > * @num_mems: number of internal memory regions > * @num_timers: number of rproc timer(s) > + * @num_wd_timers: number of rproc watchdog timers > * @timers: timer(s) info used by rproc > * @autosuspend_delay: auto-suspend delay value to be used for runtime pm > * @need_resume: if true a resume is needed in the system resume callback > @@ -102,6 +106,7 @@ struct omap_rproc { > struct omap_rproc_mem *mem; > int num_mems; > int num_timers; > + int num_wd_timers; > struct omap_rproc_timer *timers; > int autosuspend_delay; > bool need_resume; > @@ -219,6 +224,79 @@ static inline int omap_rproc_release_timer(struct omap_rproc_timer *timer) > return timer->timer_ops->free(timer->odt); > } > > +/** > + * omap_rproc_get_timer_irq() - get the irq for a timer > + * @timer: handle to a OMAP rproc timer > + * > + * This function is used to get the irq associated with a watchdog timer. The > + * function is called by the OMAP remoteproc driver to register a interrupt > + * handler to handle watchdog events on the remote processor. > + * > + * Return: irq id on success, otherwise a failure as returned by DMTimer ops > + */ > +static inline int omap_rproc_get_timer_irq(struct omap_rproc_timer *timer) > +{ > + return timer->timer_ops->get_irq(timer->odt); > +} > + > +/** > + * omap_rproc_ack_timer_irq() - acknowledge a timer irq > + * @timer: handle to a OMAP rproc timer > + * > + * This function is used to clear the irq associated with a watchdog timer. The > + * The function is called by the OMAP remoteproc upon a watchdog event on the > + * remote processor to clear the interrupt status of the watchdog timer. > + */ > +static inline void omap_rproc_ack_timer_irq(struct omap_rproc_timer *timer) > +{ > + timer->timer_ops->write_status(timer->odt, OMAP_TIMER_INT_OVERFLOW); > +} > + > +/** > + * omap_rproc_watchdog_isr() - Watchdog ISR handler for remoteproc device > + * @irq: IRQ number associated with a watchdog timer > + * @data: IRQ handler data > + * > + * This ISR routine executes the required necessary low-level code to > + * acknowledge a watchdog timer interrupt. There can be multiple watchdog > + * timers associated with a rproc (like IPUs which have 2 watchdog timers, > + * one per Cortex M3/M4 core), so a lookup has to be performed to identify > + * the timer to acknowledge its interrupt. > + * > + * The function also invokes rproc_report_crash to report the watchdog event > + * to the remoteproc driver core, to trigger a recovery. > + * > + * Return: IRQ_HANDLED on success, otherwise IRQ_NONE > + */ > +static irqreturn_t omap_rproc_watchdog_isr(int irq, void *data) > +{ > + struct rproc *rproc = data; > + struct omap_rproc *oproc = rproc->priv; > + struct device *dev = rproc->dev.parent; > + struct omap_rproc_timer *timers = oproc->timers; > + struct omap_rproc_timer *wd_timer = NULL; > + int num_timers = oproc->num_timers + oproc->num_wd_timers; > + int i; > + > + for (i = oproc->num_timers; i < num_timers; i++) { > + if (timers[i].irq > 0 && irq == timers[i].irq) { > + wd_timer = &timers[i]; > + break; > + } > + } > + > + if (!wd_timer) { > + dev_err(dev, "invalid timer\n"); > + return IRQ_NONE; > + } > + > + omap_rproc_ack_timer_irq(wd_timer); > + > + rproc_report_crash(rproc, RPROC_WATCHDOG); > + > + return IRQ_HANDLED; > +} > + > /** > * omap_rproc_enable_timers() - enable the timers for a remoteproc > * @rproc: handle of a remote processor > @@ -242,19 +320,26 @@ static int omap_rproc_enable_timers(struct rproc *rproc, bool configure) > struct omap_rproc_timer *timers = oproc->timers; > struct device *dev = rproc->dev.parent; > struct device_node *np = NULL; > + int num_timers = oproc->num_timers + oproc->num_wd_timers; > > - if (!oproc->num_timers) > + if (!num_timers) > return 0; > > if (!configure) > goto start_timers; > > - for (i = 0; i < oproc->num_timers; i++) { > - np = of_parse_phandle(dev->of_node, "ti,timers", i); > + for (i = 0; i < num_timers; i++) { > + if (i < oproc->num_timers) > + np = of_parse_phandle(dev->of_node, "ti,timers", i); > + else > + np = of_parse_phandle(dev->of_node, > + "ti,watchdog-timers", > + (i - oproc->num_timers)); > if (!np) { > ret = -ENXIO; > dev_err(dev, "device node lookup for timer at index %d failed: %d\n", > - i, ret); > + i < oproc->num_timers ? i : > + i - oproc->num_timers, ret); > goto free_timers; > } > > @@ -277,12 +362,14 @@ static int omap_rproc_enable_timers(struct rproc *rproc, bool configure) > if (!timer_ops || !timer_ops->request_by_node || > !timer_ops->set_source || !timer_ops->set_load || > !timer_ops->free || !timer_ops->start || > - !timer_ops->stop) { > + !timer_ops->stop || !timer_ops->get_irq || > + !timer_ops->write_status) { > ret = -EINVAL; > dev_err(dev, "device does not have required timer ops\n"); > goto put_node; > } > > + timers[i].irq = -1; > timers[i].timer_ops = timer_ops; > ret = omap_rproc_request_timer(dev, np, &timers[i]); > if (ret) { > @@ -291,10 +378,33 @@ static int omap_rproc_enable_timers(struct rproc *rproc, bool configure) > goto put_node; > } > of_node_put(np); > + > + if (i >= oproc->num_timers) { > + timers[i].irq = omap_rproc_get_timer_irq(&timers[i]); > + if (timers[i].irq < 0) { > + dev_err(dev, "get_irq for timer %p failed: %d\n", > + np, timers[i].irq); > + ret = -EBUSY; > + goto free_timers; > + } > + > + ret = request_irq(timers[i].irq, > + omap_rproc_watchdog_isr, IRQF_SHARED, > + "rproc-wdt", rproc); > + if (ret) { > + dev_err(dev, "error requesting irq for timer %p\n", > + np); > + omap_rproc_release_timer(&timers[i]); > + timers[i].odt = NULL; > + timers[i].timer_ops = NULL; > + timers[i].irq = -1; > + goto free_timers; > + } > + } > } > > start_timers: > - for (i = 0; i < oproc->num_timers; i++) { > + for (i = 0; i < num_timers; i++) { > ret = omap_rproc_start_timer(&timers[i]); > if (ret) { > dev_err(dev, "start timer %p failed failed: %d\n", np, > @@ -316,9 +426,12 @@ static int omap_rproc_enable_timers(struct rproc *rproc, bool configure) > of_node_put(np); > free_timers: > while (i--) { > + if (i >= oproc->num_timers) > + free_irq(timers[i].irq, rproc); > omap_rproc_release_timer(&timers[i]); > timers[i].odt = NULL; > timers[i].timer_ops = NULL; > + timers[i].irq = -1; > } > > return ret; > @@ -341,16 +454,20 @@ static int omap_rproc_disable_timers(struct rproc *rproc, bool configure) > int i; > struct omap_rproc *oproc = rproc->priv; > struct omap_rproc_timer *timers = oproc->timers; > + int num_timers = oproc->num_timers + oproc->num_wd_timers; > > - if (!oproc->num_timers) > + if (!num_timers) > return 0; > > - for (i = 0; i < oproc->num_timers; i++) { > + for (i = 0; i < num_timers; i++) { > omap_rproc_stop_timer(&timers[i]); > if (configure) { > + if (i >= oproc->num_timers) > + free_irq(timers[i].irq, rproc); > omap_rproc_release_timer(&timers[i]); > timers[i].odt = NULL; > timers[i].timer_ops = NULL; > + timers[i].irq = -1; > } > } > > @@ -1104,12 +1221,35 @@ static int omap_rproc_of_get_internal_memories(struct platform_device *pdev, > return 0; > } > > +#ifdef CONFIG_OMAP_REMOTEPROC_WATCHDOG > +static int omap_rproc_count_wdog_timers(struct device *dev) > +{ > + struct device_node *np = dev->of_node; > + int ret; > + > + ret = of_count_phandle_with_args(np, "ti,watchdog-timers", NULL); > + if (ret <= 0) { > + dev_dbg(dev, "device does not have watchdog timers, status = %d\n", > + ret); > + ret = 0; > + } > + > + return ret; > +} > +#else > +static int omap_rproc_count_wdog_timers(struct device *dev) > +{ > + return 0; > +} > +#endif > + > static int omap_rproc_of_get_timers(struct platform_device *pdev, > struct rproc *rproc) > { > struct device_node *np = pdev->dev.of_node; > struct omap_rproc *oproc = rproc->priv; > struct device *dev = &pdev->dev; > + int num_timers; > > /* > * Timer nodes are directly used in client nodes as phandles, so > @@ -1122,14 +1262,18 @@ static int omap_rproc_of_get_timers(struct platform_device *pdev, > oproc->num_timers = 0; > } > > - if (oproc->num_timers) { > - oproc->timers = devm_kcalloc(dev, oproc->num_timers, > + oproc->num_wd_timers = omap_rproc_count_wdog_timers(dev); > + > + num_timers = oproc->num_timers + oproc->num_wd_timers; > + if (num_timers) { > + oproc->timers = devm_kcalloc(dev, num_timers, > sizeof(*oproc->timers), > GFP_KERNEL); > if (!oproc->timers) > return -ENOMEM; > > - dev_dbg(dev, "device has %d tick timers\n", oproc->num_timers); > + dev_dbg(dev, "device has %d tick timers and %d watchdog timers\n", > + oproc->num_timers, oproc->num_wd_timers); > } Acked-by: Mathieu Poirier > > return 0; > -- > 2.17.1 > > -- > Texas Instruments Finland Oy, Porkkalankatu 22, 00180 Helsinki. Y-tunnus/Business ID: 0615521-4. Kotipaikka/Domicile: Helsinki