Received: by 2002:a25:ad19:0:0:0:0:0 with SMTP id y25csp6680515ybi; Sun, 21 Jul 2019 23:33:22 -0700 (PDT) X-Google-Smtp-Source: APXvYqx1Wa1DbZkYCsisA8gbMz5/DV4jZIjcsp1PY5ONhhPqtdZZOFiHw0GdO+E/GNhFeVilz5yU X-Received: by 2002:a17:902:543:: with SMTP id 61mr73575705plf.20.1563777202307; Sun, 21 Jul 2019 23:33:22 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1563777202; cv=none; d=google.com; s=arc-20160816; b=imeHVEdqZv7sk5ENlENR7Sv4LyIAHsui7fQM1bBO/Lcn9Zl6KNehvAo0SqOYJp9Y1y Sf29HYeoI8jOIZKcVutZjpzsu3FVrQwvXCTwp46pkCx3gErvud0Lf6v5nYqrIksS1aUm 8l9k3LH+w9w0W72QrI3D9o0wWPPezzs4YhB8e3xI/PVNdnqOnG66wX+qhq/YwzKX+A2D 8IyXVTC75HZlCfjpeCPUTZIn0Rj2V2Ql3LKfzUMLbLmcOVSUVw6VdRBL4zgYrPYaLm5z TO6csVnyLTJsxKxsBaJ5kNpwQxck/go0aFoxAJ5Fo4emZAF5dH5tplPMsa37Ml6aHgk2 jTWQ== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:message-id:date:subject:cc:to:from :dkim-signature; bh=9VtjEpgMaECiPt5rwojRl0NUGWfR4eW7OxT6f8ANPYE=; b=CgSYlPQHagBHBw1EfmwnfyrRrKxsSnR14zf/QS49cn7SWJDVuJ3qtcCIzNk1tSv7Cc Do7vGRTmn3uykEH5vPs6QxWieF6bSeMWeilPD1Aj2KCBWSTtvEcyG7CVkNWJhvhrA7y1 82AwaHWW5to4lyRpxw5DlDChBK3KXaw0M2kFeO/eAXjSGUxkBz9kdnRVEuYrxdU8RWWQ QLNd+/ZlKkjRMLmRFCW7lMZKEQxLNoexWcPJwLLzUpfZUHRxb3p3Yq5ewt7vGcrle6gt BNunYeowXXKm2pvBHpYYGyq+6iYtouZ5WdyojPXrzSXQThx7/d++5LaxyxWA675lbh0i qOuA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=Hna7Lx0O; 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=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id a15si11489104pfg.87.2019.07.21.23.33.06; Sun, 21 Jul 2019 23:33:22 -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=@gmail.com header.s=20161025 header.b=Hna7Lx0O; 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=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727605AbfGVGWO (ORCPT + 99 others); Mon, 22 Jul 2019 02:22:14 -0400 Received: from mail-pl1-f194.google.com ([209.85.214.194]:36734 "EHLO mail-pl1-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1725920AbfGVGWN (ORCPT ); Mon, 22 Jul 2019 02:22:13 -0400 Received: by mail-pl1-f194.google.com with SMTP id k8so18720476plt.3; Sun, 21 Jul 2019 23:22:12 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=from:to:cc:subject:date:message-id; bh=9VtjEpgMaECiPt5rwojRl0NUGWfR4eW7OxT6f8ANPYE=; b=Hna7Lx0O+0X1F1xnoEfIzDmUVonQkB+cixCUA9Yx9lEnuV2QMOTO9GEEg5+qZzOQSp Dw/pXWdUktMyzqABXt/W7UaUs9Qvlo8PaaFsiFZB4A9YcYKaGhvn9ClyOhPm5A4UZpzp c5QSqwIStm2uUuG6ltM9PxUBZqA4J420bo9BBEj/PFkhrpzJrdWmWVyviGjcop0ymHDH rCCYdffC7i6CzFmqG43pBY3dJO8HWyTJx/qIEcwbW0I6HhyZuCy0NA6+EeeB5nH7hNSk DWlYsclrMj+4GyTw8w4WNZCaC68YntCn+XFQhLBJdaLwJJr0azGmeDIEp8kg8Hr0Ig0F Xpfg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id; bh=9VtjEpgMaECiPt5rwojRl0NUGWfR4eW7OxT6f8ANPYE=; b=Z638P7kNNh3n2+d4DaiE7r98FeT/f5bMewbVyQgddUlEzBETkAWOeHS+D2FGglhuLa 9AQDU1vpbn/46HnunO49nsmsbZl+nfGfhpz95jrSGBXnAbMmMpp1lvB5/4oIjfuUaDHM oXWHSW/2p0Q3hG1DX5vtLnagFWSNCb1qd962dWGgJyt8tRqusZ/9aBpt/YG/5d4/ZPvx J7xXNxyJwpzDNeRdlt3E6yGiPWKX1djD70H3uwjR9L2cjL0rUjoWjTIXjXP/5XLxSWP0 wR29dxeOkMMTuWH/JR19lifOYl2mXan4Rm1HUYaI/4AiY7F+UocmkjCUrOAqWteQnvtM dgEw== X-Gm-Message-State: APjAAAUR+Xb5WVx3UNA0v7KLknvr0Vb9/jdyVNGqghIlyGZLPUb9BaWL zN84iGY/pTBugMxjkhj3XEZrZ5ca X-Received: by 2002:a17:902:aa88:: with SMTP id d8mr69856599plr.274.1563776532427; Sun, 21 Jul 2019 23:22:12 -0700 (PDT) Received: from localhost (59-120-186-245.HINET-IP.hinet.net. [59.120.186.245]) by smtp.gmail.com with ESMTPSA id s66sm40277381pfs.8.2019.07.21.23.22.11 (version=TLS1_2 cipher=ECDHE-RSA-AES128-GCM-SHA256 bits=128/128); Sun, 21 Jul 2019 23:22:11 -0700 (PDT) From: "Ji-Ze Hong (Peter Hong)" X-Google-Original-From: "Ji-Ze Hong (Peter Hong)" To: wg@grandegger.com, mkl@pengutronix.de, peter_hong@fintek.com.tw Cc: davem@davemloft.net, f.suligoi@asem.it, linux-kernel@vger.kernel.org, linux-can@vger.kernel.org, netdev@vger.kernel.org, "Ji-Ze Hong (Peter Hong)" Subject: [PATCH V2 1/1] can: sja1000: f81601: add Fintek F81601 support Date: Mon, 22 Jul 2019 14:22:01 +0800 Message-Id: <1563776521-28317-1-git-send-email-hpeter+linux_kernel@gmail.com> X-Mailer: git-send-email 2.7.4 Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org This patch add support for Fintek PCIE to 2 CAN controller support Signed-off-by: Ji-Ze Hong (Peter Hong) --- Changelog: v2: 1: Fix comment on the spinlock with write access. 2: Use ARRAY_SIZE instead of F81601_PCI_MAX_CHAN. 3: Check the strap pin outside the loop. 4: Fix the cleanup issue in f81601_pci_add_card(). 5: Remove unused "channels" in struct f81601_pci_card. drivers/net/can/sja1000/Kconfig | 8 ++ drivers/net/can/sja1000/Makefile | 1 + drivers/net/can/sja1000/f81601.c | 215 +++++++++++++++++++++++++++++++++++++++ 3 files changed, 224 insertions(+) create mode 100644 drivers/net/can/sja1000/f81601.c diff --git a/drivers/net/can/sja1000/Kconfig b/drivers/net/can/sja1000/Kconfig index f6dc89927ece..8588323c5138 100644 --- a/drivers/net/can/sja1000/Kconfig +++ b/drivers/net/can/sja1000/Kconfig @@ -101,4 +101,12 @@ config CAN_TSCAN1 IRQ numbers are read from jumpers JP4 and JP5, SJA1000 IO base addresses are chosen heuristically (first that works). +config CAN_F81601 + tristate "Fintek F81601 PCIE to 2 CAN Controller" + depends on PCI + help + This driver adds support for Fintek F81601 PCIE to 2 CAN Controller. + It had internal 24MHz clock source, but it can be changed by + manufacturer. We can use modinfo to get usage for parameters. + Visit http://www.fintek.com.tw to get more information. endif diff --git a/drivers/net/can/sja1000/Makefile b/drivers/net/can/sja1000/Makefile index 9253aaf9e739..6f6268543bd9 100644 --- a/drivers/net/can/sja1000/Makefile +++ b/drivers/net/can/sja1000/Makefile @@ -13,3 +13,4 @@ obj-$(CONFIG_CAN_PEAK_PCMCIA) += peak_pcmcia.o obj-$(CONFIG_CAN_PEAK_PCI) += peak_pci.o obj-$(CONFIG_CAN_PLX_PCI) += plx_pci.o obj-$(CONFIG_CAN_TSCAN1) += tscan1.o +obj-$(CONFIG_CAN_F81601) += f81601.o diff --git a/drivers/net/can/sja1000/f81601.c b/drivers/net/can/sja1000/f81601.c new file mode 100644 index 000000000000..3c378de8764d --- /dev/null +++ b/drivers/net/can/sja1000/f81601.c @@ -0,0 +1,215 @@ +// SPDX-License-Identifier: GPL-2.0 +/* Fintek F81601 PCIE to 2 CAN controller driver + * + * Copyright (C) 2019 Peter Hong + * Copyright (C) 2019 Linux Foundation + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "sja1000.h" + +#define F81601_PCI_MAX_CHAN 2 + +#define F81601_DECODE_REG 0x209 +#define F81601_IO_MODE BIT(7) +#define F81601_MEM_MODE BIT(6) +#define F81601_CFG_MODE BIT(5) +#define F81601_CAN2_INTERNAL_CLK BIT(3) +#define F81601_CAN1_INTERNAL_CLK BIT(2) +#define F81601_CAN2_EN BIT(1) +#define F81601_CAN1_EN BIT(0) + +#define F81601_TRAP_REG 0x20a +#define F81601_CAN2_HAS_EN BIT(4) + +struct f81601_pci_card { + void __iomem *addr; + spinlock_t lock; /* use this spin lock only for write access */ + struct pci_dev *dev; + struct net_device *net_dev[F81601_PCI_MAX_CHAN]; +}; + +static const struct pci_device_id f81601_pci_tbl[] = { + { PCI_DEVICE(0x1c29, 0x1703) }, + {}, +}; + +MODULE_DEVICE_TABLE(pci, f81601_pci_tbl); + +static bool internal_clk = 1; +module_param(internal_clk, bool, 0444); +MODULE_PARM_DESC(internal_clk, "Use internal clock, default 1 (24MHz)"); + +static unsigned int external_clk; +module_param(external_clk, uint, 0444); +MODULE_PARM_DESC(external_clk, "External Clock, must spec when internal_clk = 0"); + +static u8 f81601_pci_read_reg(const struct sja1000_priv *priv, int port) +{ + return readb(priv->reg_base + port); +} + +static void f81601_pci_write_reg(const struct sja1000_priv *priv, int port, + u8 val) +{ + struct f81601_pci_card *card = priv->priv; + unsigned long flags; + + spin_lock_irqsave(&card->lock, flags); + writeb(val, priv->reg_base + port); + readb(priv->reg_base); + spin_unlock_irqrestore(&card->lock, flags); +} + +static void f81601_pci_del_card(struct pci_dev *pdev) +{ + struct f81601_pci_card *card = pci_get_drvdata(pdev); + struct net_device *dev; + int i = 0; + + for (i = 0; i < ARRAY_SIZE(card->net_dev); i++) { + dev = card->net_dev[i]; + if (!dev) + continue; + + dev_info(&pdev->dev, "%s: Removing %s\n", __func__, dev->name); + + unregister_sja1000dev(dev); + free_sja1000dev(dev); + } + + pcim_iounmap(pdev, card->addr); +} + +/* Probe F81601 based device for the SJA1000 chips and register each + * available CAN channel to SJA1000 Socket-CAN subsystem. + */ +static int f81601_pci_add_card(struct pci_dev *pdev, + const struct pci_device_id *ent) +{ + struct sja1000_priv *priv; + struct net_device *dev; + struct f81601_pci_card *card; + int err, i, count; + u8 tmp; + + if (pcim_enable_device(pdev) < 0) { + dev_err(&pdev->dev, "Failed to enable PCI device\n"); + return -ENODEV; + } + + dev_info(&pdev->dev, "Detected card at slot #%i\n", + PCI_SLOT(pdev->devfn)); + + card = devm_kzalloc(&pdev->dev, sizeof(*card), GFP_KERNEL); + if (!card) + return -ENOMEM; + + card->dev = pdev; + spin_lock_init(&card->lock); + + pci_set_drvdata(pdev, card); + + tmp = F81601_IO_MODE | F81601_MEM_MODE | F81601_CFG_MODE | + F81601_CAN2_EN | F81601_CAN1_EN; + + if (internal_clk) { + tmp |= F81601_CAN2_INTERNAL_CLK | F81601_CAN1_INTERNAL_CLK; + + dev_info(&pdev->dev, + "F81601 running with internal clock: 24Mhz\n"); + } else { + dev_info(&pdev->dev, + "F81601 running with external clock: %dMhz\n", + external_clk / 1000000); + } + + pci_write_config_byte(pdev, F81601_DECODE_REG, tmp); + + card->addr = pcim_iomap(pdev, 0, pci_resource_len(pdev, 0)); + + if (!card->addr) { + err = -ENOMEM; + dev_err(&pdev->dev, "%s: Failed to remap BAR\n", __func__); + goto failure_cleanup; + } + + /* read CAN2_HW_EN strap pin to detect how many CANBUS do we have */ + count = ARRAY_SIZE(card->net_dev); + pci_read_config_byte(pdev, F81601_TRAP_REG, &tmp); + if (!(tmp & F81601_CAN2_HAS_EN)) + count = 1; + + /* Detect available channels */ + for (i = 0; i < count; i++) { + dev = alloc_sja1000dev(0); + if (!dev) { + err = -ENOMEM; + goto failure_cleanup; + } + + priv = netdev_priv(dev); + priv->priv = card; + priv->irq_flags = IRQF_SHARED; + priv->reg_base = card->addr + 0x80 * i; + priv->read_reg = f81601_pci_read_reg; + priv->write_reg = f81601_pci_write_reg; + + if (internal_clk) + priv->can.clock.freq = 24000000 / 2; + else + priv->can.clock.freq = external_clk / 2; + + priv->ocr = OCR_TX0_PUSHPULL | OCR_TX1_PUSHPULL; + priv->cdr = CDR_CBP; + + SET_NETDEV_DEV(dev, &pdev->dev); + dev->dev_id = i; + dev->irq = pdev->irq; + + /* Register SJA1000 device */ + err = register_sja1000dev(dev); + if (err) { + dev_err(&pdev->dev, + "%s: Registering device failed: %x\n", __func__, + err); + free_sja1000dev(dev); + goto failure_cleanup; + } + + card->net_dev[i] = dev; + dev_info(&pdev->dev, "Channel #%d, %s at 0x%p, irq %d\n", i, + dev->name, priv->reg_base, dev->irq); + } + + return 0; + +failure_cleanup: + dev_err(&pdev->dev, "%s: failed: %d. Cleaning Up.\n", __func__, err); + f81601_pci_del_card(pdev); + + return err; +} + +static struct pci_driver f81601_pci_driver = { + .name = "f81601", + .id_table = f81601_pci_tbl, + .probe = f81601_pci_add_card, + .remove = f81601_pci_del_card, +}; + +MODULE_DESCRIPTION("Fintek F81601 PCIE to 2 CANBUS adaptor driver"); +MODULE_AUTHOR("Peter Hong "); +MODULE_LICENSE("GPL v2"); + +module_pci_driver(f81601_pci_driver); -- 2.7.4