Received: by 2002:ac0:8c9a:0:0:0:0:0 with SMTP id r26csp1549178ima; Sat, 2 Feb 2019 03:01:54 -0800 (PST) X-Google-Smtp-Source: ALg8bN7BerjnVaeifeM56N/bQ1yxZ1sD57ZHjHipClXv/MdmzvEVmPwj2e+h7BsWDztm7X3UvNgu X-Received: by 2002:a17:902:f20d:: with SMTP id gn13mr41711423plb.11.1549105313989; Sat, 02 Feb 2019 03:01:53 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1549105313; cv=none; d=google.com; s=arc-20160816; b=r32/A7l0qwzeTwCX+wXIXstEErIHjUFypf1FC1FqI0jpO7GiDxKQjgX/bwz19+fi2S dFlQ5UjCPJkMia6qBZl+iAxW6BFQRzzNYxLV7eWlfIOsI6ZzQSZWrlsNnzLJ+XqYS4lP 9eGs+vOC+Nqf4M3yEGno3pHbuw9oWrF4IqIiEGuPmOEYGgZiiW/AEOtTSNnjBFRDJxV8 kKajuhVGhjJ3BrH14F7DiksjhXdj5I9cB8sAhzVJi+Y5fndZksiEzEkW9QUOifrKRteE VximKRpeAaEik+iwsqtFJmA3o5X/Y3s5d5VSeM3z4SoZSEfKrxxZ5xMgE/IswhzqQg2k 4yfQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:references:in-reply-to:message-id:date :subject:cc:to:from:dkim-signature; bh=Cevim4k8pWlCJnXmyRrMHpWJAEi7p1qF41QwSdAouEA=; b=FzBDuV5Gxpt4rgWTVv9UIgzXi6r/88zLm+9QR+74sR+4UW8Q/mDKxNTXP+XhtJvA5K FpTZoE0J753RzrI20s1pFNDWj3uhRvxoI7wifrdMhawuwaHEumNlMrG7k1RuPbHb/7Mh iSPiGLbj5uqsub8pArKTNmm8CQggBopFiKydaYbIhOeYl78seNt0ZnVVgUFh32FtIzUE LSbWX1+sFlq0AF8KLm40CunBz9jqrqWPnZgRXnFVX+c5KS0IxebWJVkaHGwGdfDo4Smv iLOJsccK9AVKltanca7ZHkV2gqG3qtu1+mgBMh6QzFSqNbDLu7sY/kKTkf/taTNrhcvQ +9aQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=fail header.i=@aussec.com header.s=2016 header.b=lFKxdEoo; 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=fail (p=NONE sp=NONE dis=NONE) header.from=aussec.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id t12si10098672plr.311.2019.02.02.03.01.38; Sat, 02 Feb 2019 03:01:53 -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=fail header.i=@aussec.com header.s=2016 header.b=lFKxdEoo; 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=fail (p=NONE sp=NONE dis=NONE) header.from=aussec.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727906AbfBBK75 (ORCPT + 99 others); Sat, 2 Feb 2019 05:59:57 -0500 Received: from csm1.csm-office.com.au ([165.228.118.109]:40246 "EHLO sleepy.aussec.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1727316AbfBBK74 (ORCPT ); Sat, 2 Feb 2019 05:59:56 -0500 X-Virus-Scanned: amavisd-new at aussec.com Received: from localhost.localdomain ([1.128.141.87]) (authenticated bits=0) by sleepy.aussec.com (8.15.2/8.15.2) with ESMTPSA id x12AwsVA016455 (version=TLSv1.2 cipher=DHE-RSA-AES128-SHA bits=128 verify=NO); Sat, 2 Feb 2019 21:59:26 +1100 DKIM-Signature: v=1; a=rsa-sha256; c=simple/simple; d=aussec.com; s=2016; t=1549105167; bh=+yolntkn3hWaRnlN58LhzDMoL6RUSitiNpb2lt3NkT4=; h=From:To:Cc:Subject:Date:In-Reply-To:References; b=lFKxdEooLjxn9oJ+49WWkumloYNi4jBjbC8fvuZ9K/V2zdeC+sQYFOZCOZveD+1Ql xj2j8UAw8PSNQJzNaYP0n5DISXj5jO2SZm+apzOtntE1/puarjN8Y6vI5KUVKrhqbi 8FmkvwzKKxasJkG3nfU/SrFHJFDl9vtEllgzQDqDKeKPMh+G1IIOOMfVZ/z/xGFxpj 5UtqPMHSlejW3QEVdr/UiPBrsPAjnzVHobf9RZTSXnjISAmjmOGLpI9GauJcCbtier 3Ij0i5l30sxICpdEnWp4VVIemtsIyUDqd0yiVKIwjCiYmX8u14GIuG8mB0f1uC41vZ 9MdaQANYBPSQQ== From: Tom Burkart To: Linux kernel mailing list Cc: Tom Burkart , Rodolfo Giometti , Ricardo Martins , James Nuss , Lukas Senger Subject: [PATCH v15 3/3] pps: pps-gpio pps-echo implementation Date: Sat, 2 Feb 2019 21:57:24 +1100 Message-Id: <20190202105724.4743-4-tom@aussec.com> X-Mailer: git-send-email 2.12.3 In-Reply-To: <20190202105724.4743-3-tom@aussec.com> References: <20190202105724.4743-1-tom@aussec.com> <20190202105724.4743-2-tom@aussec.com> <20190202105724.4743-3-tom@aussec.com> Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org This patch implements the pps echo functionality for pps-gpio, that sysfs claims is available already. Configuration is done via device tree bindings. This patch was originally written by Lukas Senger as part of a masters thesis project and modified for inclusion into the linux kernel by Tom Burkart. Signed-off-by: Lukas Senger Signed-off-by: Tom Burkart --- drivers/pps/clients/pps-gpio.c | 88 ++++++++++++++++++++++++++++++++++++++++-- include/linux/pps-gpio.h | 2 + 2 files changed, 87 insertions(+), 3 deletions(-) diff --git a/drivers/pps/clients/pps-gpio.c b/drivers/pps/clients/pps-gpio.c index 4e5e9229814b..4b6418039387 100644 --- a/drivers/pps/clients/pps-gpio.c +++ b/drivers/pps/clients/pps-gpio.c @@ -35,6 +35,8 @@ #include #include #include +#include +#include /* Info for each registered platform device */ struct pps_gpio_device_data { @@ -42,8 +44,12 @@ struct pps_gpio_device_data { struct pps_device *pps; /* PPS source device */ struct pps_source_info info; /* PPS source information */ struct gpio_desc *gpio_pin; /* GPIO port descriptors */ + struct gpio_desc *echo_pin; + struct timer_list echo_timer; /* timer to reset echo active state */ bool assert_falling_edge; bool capture_clear; + unsigned int echo_active_ms; /* PPS echo active duration */ + unsigned long echo_timeout; /* timer timeout value in jiffies */ }; /* @@ -64,19 +70,56 @@ static irqreturn_t pps_gpio_irq_handler(int irq, void *data) rising_edge = gpiod_get_value(info->gpio_pin); if ((rising_edge && !info->assert_falling_edge) || (!rising_edge && info->assert_falling_edge)) - pps_event(info->pps, &ts, PPS_CAPTUREASSERT, NULL); + pps_event(info->pps, &ts, PPS_CAPTUREASSERT, data); else if (info->capture_clear && ((rising_edge && info->assert_falling_edge) || (!rising_edge && !info->assert_falling_edge))) - pps_event(info->pps, &ts, PPS_CAPTURECLEAR, NULL); + pps_event(info->pps, &ts, PPS_CAPTURECLEAR, data); return IRQ_HANDLED; } +/* This function will only be called when an ECHO GPIO is defined */ +static void pps_gpio_echo(struct pps_device *pps, int event, void *data) +{ + /* add_timer() needs to write into info->echo_timer */ + struct pps_gpio_device_data *info = data; + + switch (event) { + case PPS_CAPTUREASSERT: + if (pps->params.mode & PPS_ECHOASSERT) + gpiod_set_value(info->echo_pin, 1); + break; + + case PPS_CAPTURECLEAR: + if (pps->params.mode & PPS_ECHOCLEAR) + gpiod_set_value(info->echo_pin, 1); + break; + } + + /* fire the timer */ + if (info->pps->params.mode & (PPS_ECHOASSERT | PPS_ECHOCLEAR)) { + info->echo_timer.expires = jiffies + info->echo_timeout; + add_timer(&info->echo_timer); + } +} + +/* Timer callback to reset the echo pin to the inactive state */ +static void pps_gpio_echo_timer_callback(struct timer_list *t) +{ + const struct pps_gpio_device_data *info; + + info = from_timer(info, t, echo_timer); + + gpiod_set_value(info->echo_pin, 0); +} + static int pps_gpio_setup(struct platform_device *pdev) { struct pps_gpio_device_data *data = platform_get_drvdata(pdev); struct device_node *np = pdev->dev.of_node; + int ret; + u32 value; data->gpio_pin = devm_gpiod_get(&pdev->dev, NULL, /* request "gpios" */ @@ -87,6 +130,33 @@ static int pps_gpio_setup(struct platform_device *pdev) return PTR_ERR(data->gpio_pin); } + data->echo_pin = devm_gpiod_get_optional(&pdev->dev, + "echo", + GPIOD_OUT_LOW); + if (data->echo_pin) { + if (IS_ERR(data->echo_pin)) { + dev_err(&pdev->dev, "failed to request ECHO GPIO\n"); + return PTR_ERR(data->echo_pin); + } + + ret = of_property_read_u32(np, + "echo-active-ms", + &value); + if (ret) { + dev_err(&pdev->dev, + "failed to get echo-active-ms from OF\n"); + return ret; + } + data->echo_active_ms = value; + /* sanity check on echo_active_ms */ + if (!data->echo_active_ms || data->echo_active_ms > 999) { + dev_err(&pdev->dev, + "echo-active-ms: %u - bad value from OF\n", + data->echo_active_ms); + return -EINVAL; + } + } + if (of_property_read_bool(np, "assert-falling-edge")) data->assert_falling_edge = true; return 0; @@ -122,9 +192,11 @@ static int pps_gpio_probe(struct platform_device *pdev) /* GPIO setup */ if (pdata) { data->gpio_pin = pdata->gpio_pin; + data->echo_pin = pdata->echo_pin; data->assert_falling_edge = pdata->assert_falling_edge; data->capture_clear = pdata->capture_clear; + data->echo_active_ms = pdata->echo_active_ms; } else { ret = pps_gpio_setup(pdev); if (ret) @@ -148,6 +220,11 @@ static int pps_gpio_probe(struct platform_device *pdev) data->info.owner = THIS_MODULE; snprintf(data->info.name, PPS_MAX_NAME_LEN - 1, "%s.%d", pdev->name, pdev->id); + if (data->echo_pin) { + data->info.echo = pps_gpio_echo; + data->echo_timeout = msecs_to_jiffies(data->echo_active_ms); + timer_setup(&data->echo_timer, pps_gpio_echo_timer_callback, 0); + } /* register PPS source */ pps_default_params = PPS_CAPTUREASSERT | PPS_OFFSETASSERT; @@ -180,6 +257,11 @@ static int pps_gpio_remove(struct platform_device *pdev) struct pps_gpio_device_data *data = platform_get_drvdata(pdev); pps_unregister_source(data->pps); + if (data->echo_pin) { + del_timer_sync(&data->echo_timer); + /* reset echo pin in any case */ + gpiod_set_value(data->echo_pin, 0); + } dev_info(&pdev->dev, "removed IRQ %d as PPS source\n", data->irq); return 0; } @@ -204,4 +286,4 @@ MODULE_AUTHOR("Ricardo Martins "); MODULE_AUTHOR("James Nuss "); MODULE_DESCRIPTION("Use GPIO pin as PPS source"); MODULE_LICENSE("GPL"); -MODULE_VERSION("1.1.0"); +MODULE_VERSION("1.2.0"); diff --git a/include/linux/pps-gpio.h b/include/linux/pps-gpio.h index f028d2cda6f5..44171e6b7197 100644 --- a/include/linux/pps-gpio.h +++ b/include/linux/pps-gpio.h @@ -24,8 +24,10 @@ struct pps_gpio_platform_data { struct gpio_desc *gpio_pin; + struct gpio_desc *echo_pin; bool assert_falling_edge; bool capture_clear; + unsigned int echo_active_ms; }; #endif /* _PPS_GPIO_H */ -- 2.12.3