From patchwork Tue May 21 11:59:56 2019 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Masahisa Kojima X-Patchwork-Id: 164708 Delivered-To: patch@linaro.org Received: by 2002:a92:9e1a:0:0:0:0:0 with SMTP id q26csp1507681ili; Tue, 21 May 2019 05:00:30 -0700 (PDT) X-Google-Smtp-Source: APXvYqylV4+v3gUs/qumQYZZVP6DPptnwF/7Vzzdk+796lhrzltIFF1AHL20JF/GmiAU+XqrZSF3 X-Received: by 2002:aa7:8b12:: with SMTP id f18mr22539983pfd.178.1558440030129; Tue, 21 May 2019 05:00:30 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1558440030; cv=none; d=google.com; s=arc-20160816; b=UQuQJvF60fuKzWSQUkr17LSPUmLQn9LVgcEA4gbuCxeRj+JPjzCiXjdQqJe2bkQbEJ Ei7O/myKa19DlQTdKxwEnxZVfxdExYfErJAbc/UWf5UH2XefS/9V7HxCsEpVLHMVGlTy 35a1bFQvh+iRMhINVL+d50cnR5BoGue2ncHKkJaMx521P1JORSCnb+3gGyybzYVzM3vq i8CecytOiO1lS7F6E8ubWDLBYfGW6PsCA2BL4ueYYP30WaUQgqM1ozVeDrBEqk8MqANM B8D1Ulcbuszv5sn74i2Zc4yfeRY+wmoEHzAYDUtY1SygnZeor8wBsTBEBg2H3JKfFBYy p0CA== 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=WSzNCT45W84aER1GYHZSzkW24AjjUqqyhWIxjsh7obs=; b=cjpkM+J9xkg1XO49tENRJg7OiOmPj98n3BsWTxpK/Ecesoww806eVDGduM97WK06cD fZm7kKh00dR/KCWb/L+9zVy37lAM9eFI9CRzjSipNGwhio6tOjRvj7RcvvBDzCzRqcZW C1cjINMgV7viD3JS5/ymQG9NdHXPe1RbvhOEreOg1/1WwPlNOc4n2UuYFJUIbFCTc82N nxVirZUtc6Jg6guLdqmDPhjSfL7aL79ayOz66xfIzqJT+CZW/CkDeeo7wK94LmflJuSj tVU+SM+l5hGMe/acB+IX5+JcuO5kURpBdCXHj5IwLgCssybC7PephG8zzzdU49PuzNSy q4bg== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=IkS85lbY; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 j2si21179866pgh.192.2019.05.21.05.00.29; Tue, 21 May 2019 05:00:30 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of devicetree-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=IkS85lbY; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 S1726953AbfEUMA3 (ORCPT + 7 others); Tue, 21 May 2019 08:00:29 -0400 Received: from mail-pl1-f195.google.com ([209.85.214.195]:35202 "EHLO mail-pl1-f195.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726705AbfEUMA3 (ORCPT ); Tue, 21 May 2019 08:00:29 -0400 Received: by mail-pl1-f195.google.com with SMTP id p1so2972627plo.2 for ; Tue, 21 May 2019 05:00:29 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=WSzNCT45W84aER1GYHZSzkW24AjjUqqyhWIxjsh7obs=; b=IkS85lbYVaWWd/yJiDguqFGYHKQbp6L7nlZ4siblXOnlrm1tqw7U8NrNVBtOB8Rya+ fJb7P8uVWVVKbsHuM2oHvwBpK8ChMTPSps7mqxqXzb+i2b696+HZqfTXX4SD+z2n/uHZ AtJdAlDnWMpx9uJzj/hK2LoM0zFymkqcATOI26VpKD1wvSmY0Lz3kXAhyHgK+/+zZZ1Z +002Cn8a6Ifi65rKb7ZQ8lGpuvfBci/BAhy1qmuho8o+qiEoZmSUn4+Cnxo6CUnwsJmZ +f4erOHGDTiWuoCckQKs48CqcqgK1ZxpbhPnBUlAfRpn6BFJzxvCbKRgwskcKT6Zfmo2 05Pw== 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:in-reply-to :references; bh=WSzNCT45W84aER1GYHZSzkW24AjjUqqyhWIxjsh7obs=; b=RqEhuHhFJk4Z2VjiYah0/u9RlNLiqDRKIncI/7E+CdjQ87zVPwSH2lF9aPdDaAL6Bk c/Pq9nY4gKf/gBwtdR+gOi7SPA738zOUDSndQLN9edA2EuCWGKKufRwWx5ATLNC5VsFs AxTGxxu9QQbRLNlkPu9kMEkJ38y1rmAzf4N6+wYNd0Sg6ks2O+zkZtNDOKpm0aT7bV4a yWKrJvL5mBfNLpmq/xLSXnmOEt1m1bpnoosE2O7AwhngQ/OdTEgNq7tVZ3ra4vOVky9S E0yyKH+FHVL6gb9gArTMyxlzySX+C63sDMnLC3si8Z2aZMkTOI0Hpj0HPVSNQgstdre4 dv3Q== X-Gm-Message-State: APjAAAUYJYwipLOa9ha102EE3VHaReD/OkW7uBWrRzlgWpYXpu89sxNm Y6cx4oYuUunAzae79SM9qTLydQ== X-Received: by 2002:a17:902:848c:: with SMTP id c12mr32760824plo.17.1558440028783; Tue, 21 May 2019 05:00:28 -0700 (PDT) Received: from localhost ([121.95.100.191]) by smtp.gmail.com with ESMTPSA id 85sm15198121pfa.131.2019.05.21.05.00.27 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Tue, 21 May 2019 05:00:27 -0700 (PDT) From: Masahisa Kojima To: linux-spi@vger.kernel.org, devicetree@vger.kernel.org Cc: broonie@kernel.org, geert@linux-m68k.org, tpiepho@impinj.com, andy.shevchenko@gmail.com, robh+dt@kernel.org, mark.rutland@arm.com, ard.biesheuvel@linaro.org, jaswinder.singh@linaro.org, masami.hiramatsu@linaro.org, okamoto.satoru@socionext.com, osaki.yoshitoyo@socionext.com, Masahisa Kojima Subject: [PATCH v5 1/3] MAINTAINERS: Add entry for Synquacer SPI driver Date: Tue, 21 May 2019 20:59:56 +0900 Message-Id: <20190521115958.22504-2-masahisa.kojima@linaro.org> X-Mailer: git-send-email 2.14.2 In-Reply-To: <20190521115958.22504-1-masahisa.kojima@linaro.org> References: <20190521115958.22504-1-masahisa.kojima@linaro.org> Sender: devicetree-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org Add entry for the Synquacer spi driver and DT bindings. Signed-off-by: Masahisa Kojima Signed-off-by: Jassi Brar --- MAINTAINERS | 8 ++++++++ 1 file changed, 8 insertions(+) -- 2.14.2 diff --git a/MAINTAINERS b/MAINTAINERS index 0c55b0fedbe2..a4970349ef4c 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -14597,6 +14597,14 @@ S: Maintained F: drivers/net/ethernet/socionext/netsec.c F: Documentation/devicetree/bindings/net/socionext-netsec.txt +SOCIONEXT (SNI) Synquacer SPI DRIVER +M: Masahisa Kojima +M: Jassi Brar +L: linux-spi@vger.kernel.org +S: Maintained +F: drivers/spi/spi-synquacer.c +F: Documentation/devicetree/bindings/spi/spi-synquacer.txt + SOLIDRUN CLEARFOG SUPPORT M: Russell King S: Maintained From patchwork Tue May 21 11:59:57 2019 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Masahisa Kojima X-Patchwork-Id: 164709 Delivered-To: patch@linaro.org Received: by 2002:a92:9e1a:0:0:0:0:0 with SMTP id q26csp1507822ili; Tue, 21 May 2019 05:00:38 -0700 (PDT) X-Google-Smtp-Source: APXvYqwAB9tGUzdJT2SUCevr2BV7x7WUO7LpsDeZMzx9OqB8xdZUOeF6qJYyPSTf7jVfsf8F+1sH X-Received: by 2002:a62:86ce:: with SMTP id x197mr86873927pfd.218.1558440038558; Tue, 21 May 2019 05:00:38 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1558440038; cv=none; d=google.com; s=arc-20160816; b=cVJ+YpGNRCIBmkERlWdu/6SsMq7lVHqeO76sOVFflSLAft4x3xk9NnPhIUv+esGCGm bFYdTKSV5kGS6NUtZ8HZtKbXT/GC/b1TK/TYbXciOSpAZ4XQqik46TqWieKL5TrzrCm0 27a/Rls04AJfiy3FcN5/ohjUP6SUI1LZxFFhtM4oiLatvntcalcS/lJceik8MaYO59Qj 85xzGC6I5KSbQ1vf+BW9E3+Zjb9hYJZCN/FL+N/pkwMDFopb9I+PWk6EIYMA4L2qTKf/ 2GH/NQAcrVFnoaRY3PXIIofgyYukHWdKRqVLUCKXapT2zlcODvA7x9Uu3keTiHUXw3HM X4Ow== 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=J8He96LBvzgwWKZ93ZshlZgpeoHlqzqGddbd8HI8Wh0=; b=bPUQeqB/jy/CpZlM67sXdWWQlWFKMuArhhwwsCrNriwoYlVX8J1mkeQ28NxgrRG1Ha PgeyWnUaU9JB1S+Nzfz+I5YJcohrsgZOOyJ0oOtHauFudfqn0gGwdp+W95fjgraMFLJ8 Ud3cW51KphHHQbNTE1JKOKHwpsk9+LEgXshqMutDPr2qwADp/mYrUdulowynUAtarhvZ +uQxOmiIj9dEb9csMnq3X6FJ8LdFMnF4fnjPwf1zBcuYyZB5WNL2Eb4kJo4EgXRnTyr0 m6lkfkklgeNfSwH2Of7CiEUQmuPQx140XhU90fVIgwk1T8Tqi7ywNfTtdSpJFd62M1W4 L8Kw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=MgTzd8zf; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 j2si21179866pgh.192.2019.05.21.05.00.38; Tue, 21 May 2019 05:00:38 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of devicetree-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=MgTzd8zf; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 S1728045AbfEUMAi (ORCPT + 7 others); Tue, 21 May 2019 08:00:38 -0400 Received: from mail-pf1-f193.google.com ([209.85.210.193]:44391 "EHLO mail-pf1-f193.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726705AbfEUMAh (ORCPT ); Tue, 21 May 2019 08:00:37 -0400 Received: by mail-pf1-f193.google.com with SMTP id g9so8950860pfo.11 for ; Tue, 21 May 2019 05:00:37 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=J8He96LBvzgwWKZ93ZshlZgpeoHlqzqGddbd8HI8Wh0=; b=MgTzd8zfTg4rDQqa+BejqDdaIPAPvAbMt6W0peADav4XRHHw6xis2XW3YWPac1+25Y Uw9izIAmZGjeplWWV+N1cL38v5zayPYZwAWl/6QMELf1rW5qgcZG6rv/5i+nkTvZNEsh 1OcwvltOg9g1KLWexpXphqhoWE2B5ueGR39L4Gv8R73PqK1nmiwZsjr8lo/ZHwV/ni/a DLCN5avkG4y1OlvbL5GEyXH1faaB6PHGwbmnD8focTzezgnJIYLU0d1/jAJf1uDFlBAI INggD7MNlPrna9xjyyv0jauTtA/MdR5X4wPRcOPMFOVD20itb7WAST4UWnxLHyLGMCPo ym0Q== 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:in-reply-to :references; bh=J8He96LBvzgwWKZ93ZshlZgpeoHlqzqGddbd8HI8Wh0=; b=kQKGxHEoALQJ/d0ZLkcR9qrKrvh9P5eJX9w9tir9detufuzU9GhwoK2YjJ2hQvUBQR FC6LTg1OI5ceF0FJpk/zwkmm5WkY2dc1gpEuexCIjbWt5oktBkNAWHfAveyOoCTfah7j p4zijZixneupzSmUaVjEgAkCDgouRoy2jZTr3E875JPEiiCVYdCWUi/dybxClaO5ngPM 8+YMly012izI+WW31nRipKtBBvQ/aGAJmNlhFDTUgQqLC2G1GXzz55Vwck/6BQuimNOp RY8+6rAX5ZBvlhDxG/cAjijCMgBjYLTwVN3X03g6ej75YbK5Hz99F9uKzP94K08eHUy4 C5+w== X-Gm-Message-State: APjAAAUFY3dGuMkH+no8Q5EmOo9la38+CblZoJkCaRIGELMcBCmbyjMO quiRPSWqJRMJenOAl9U63xq9Fw== X-Received: by 2002:a62:fb18:: with SMTP id x24mr21966838pfm.76.1558440037087; Tue, 21 May 2019 05:00:37 -0700 (PDT) Received: from localhost ([121.95.100.191]) by smtp.gmail.com with ESMTPSA id r77sm28975782pgr.93.2019.05.21.05.00.35 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Tue, 21 May 2019 05:00:36 -0700 (PDT) From: Masahisa Kojima To: linux-spi@vger.kernel.org, devicetree@vger.kernel.org Cc: broonie@kernel.org, geert@linux-m68k.org, tpiepho@impinj.com, andy.shevchenko@gmail.com, robh+dt@kernel.org, mark.rutland@arm.com, ard.biesheuvel@linaro.org, jaswinder.singh@linaro.org, masami.hiramatsu@linaro.org, okamoto.satoru@socionext.com, osaki.yoshitoyo@socionext.com, Masahisa Kojima Subject: [PATCH v5 2/3] dt-bindings: spi: Add DT bindings for Synquacer Date: Tue, 21 May 2019 20:59:57 +0900 Message-Id: <20190521115958.22504-3-masahisa.kojima@linaro.org> X-Mailer: git-send-email 2.14.2 In-Reply-To: <20190521115958.22504-1-masahisa.kojima@linaro.org> References: <20190521115958.22504-1-masahisa.kojima@linaro.org> Sender: devicetree-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org This patch adds documentation for Device-Tree bindings for the Socionext Synquacer spi driver. Signed-off-by: Masahisa Kojima Signed-off-by: Jassi Brar Reviewed-by: Rob Herring --- .../devicetree/bindings/spi/spi-synquacer.txt | 27 ++++++++++++++++++++++ 1 file changed, 27 insertions(+) create mode 100644 Documentation/devicetree/bindings/spi/spi-synquacer.txt -- 2.14.2 diff --git a/Documentation/devicetree/bindings/spi/spi-synquacer.txt b/Documentation/devicetree/bindings/spi/spi-synquacer.txt new file mode 100644 index 000000000000..291dfa692d0a --- /dev/null +++ b/Documentation/devicetree/bindings/spi/spi-synquacer.txt @@ -0,0 +1,27 @@ +* Socionext Synquacer HS-SPI bindings + +Required Properties: +- compatible: should be "socionext,synquacer-spi" +- reg: physical base address of the controller and length of memory mapped + region. +- interrupts: should contain the "spi_rx", "spi_tx" and "spi_fault" interrupts. +- clocks: core clock iHCLK. Optional rate clock iPCLK (default is iHCLK) +- clock-names: Shall be "iHCLK" and "iPCLK" respectively + +Optional Properties: +- socionext,use-rtm: boolean, if required to use "retimed clock" for RX +- socionext,set-aces: boolean, if same active clock edges field to be set. + +Example: + + spi0: spi@ff110000 { + compatible = "socionext,synquacer-spi"; + reg = <0xff110000 0x1000>; + interrupts = , + , + ; + clocks = <&clk_hsspi>; + clock-names = "iHCLK"; + socionext,use-rtm; + socionext,set-aces; + }; From patchwork Tue May 21 11:59:58 2019 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Masahisa Kojima X-Patchwork-Id: 164710 Delivered-To: patch@linaro.org Received: by 2002:a92:9e1a:0:0:0:0:0 with SMTP id q26csp1508005ili; Tue, 21 May 2019 05:00:47 -0700 (PDT) X-Google-Smtp-Source: APXvYqzzAggDaJcNQLkeaxRZjUMqcTgJxzOyjHIJP/1S/oV/60okilmmTm7J5kMsNKlT3gyUbQBV X-Received: by 2002:a62:e303:: with SMTP id g3mr86098105pfh.220.1558440046937; Tue, 21 May 2019 05:00:46 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1558440046; cv=none; d=google.com; s=arc-20160816; b=gQhpmNPBMHrf4+NIZ3AgUBH3Y16ZQ6JRv1blxoM6qso/fPIHwC+d5vgI3X9wXlymqv wVC91rCtJh5F7+eguhQCpZ+WsYFYoMdzDv/mRzvPXMqAo6LGcKJsg+fXWXEY/MqXWb6S MQnKXfpf2ZBfBNQBRjddBIlRXv5C/y38Gn6BxbCmmdIMxNOu8KUlnSfbGaaM/1689GKS GRqxhpHbA2Xu+Kaywr6LeJyH+hUnsc9mlotLRso8i13mli5ONRqa/gUnTBItEbsAroVo bipRvn8LNMaynpj8lTdnk+OczLywalIOpj/2uWsskjgYUUC2w6pobniwsqESbwoOWVv6 3QJg== 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=dkncqCXO55SGa1MBMzGOr5xAs9kuJOojD6dsGVSaW2U=; b=LQe1BEaKWGb6Vnttk4QTO2/wIaxGXWXS6HDJxyS1j947Y1AxIBt5FzmE399xF6jfv1 +G/cdPtfCty3OcwoEa9nGSnubYmUQdZaSWWUe+SwbDLqKHa76p+TqxBze/bd9dHIuCTm K8c+5dl3FmZMs3jSfzcMDHHOcB89VpRlVp1PnH7gng8tW/cHc5gGzUHtuDq1TOBQBC1L 3sVDcx6uWwryehj6iU4ZcHNf3+PDD6KP39ZxiMnfeR3VcJKM2LSL90tGr5CJ7mEJDMNf qjv0E82qrc+SNVdxF0nTegMTqlmt+WmcP5u/2FEm/ymf+sw0WbdW0trVuJzE8rffYpCR FXPw== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=NTTCevJF; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 j2si21179866pgh.192.2019.05.21.05.00.46; Tue, 21 May 2019 05:00:46 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of devicetree-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=NTTCevJF; spf=pass (google.com: best guess record for domain of devicetree-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=devicetree-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 S1728057AbfEUMAq (ORCPT + 7 others); Tue, 21 May 2019 08:00:46 -0400 Received: from mail-pf1-f195.google.com ([209.85.210.195]:39224 "EHLO mail-pf1-f195.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728047AbfEUMAq (ORCPT ); Tue, 21 May 2019 08:00:46 -0400 Received: by mail-pf1-f195.google.com with SMTP id z26so8965447pfg.6 for ; Tue, 21 May 2019 05:00:45 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=from:to:cc:subject:date:message-id:in-reply-to:references; bh=dkncqCXO55SGa1MBMzGOr5xAs9kuJOojD6dsGVSaW2U=; b=NTTCevJFplPDaQN+2b3EnvsMaBuQrL9XyK6aO7vXBN7Kr9cck7t4jK9DgZuHJpLbBU DTpJvfTVaP6NQtUkm0Sp4usRQFQiCyWObf9vYn2SaRXhuBMDm2gYmBiQJieuxY2Xs32T MWD5J4phJ6y7Q5UuHboAV85dKBW9RpxnDtQ6qWfXnhqHeTc/IaZahZxy45qdHeSZesRr dTN/XdpsGEtMnS2m3MDVqC4Ly01rkh7DmnqXQprv9WxcHpSMuWXOnfO3yYy9D7km26Mx 3f5bC/8fO5csWZ/Sy2AsJ+wgJ3fW135fhia4qGcxhX4Rf5fR0623ZQwOQQkOx6SHLxdy Xj8Q== 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:in-reply-to :references; bh=dkncqCXO55SGa1MBMzGOr5xAs9kuJOojD6dsGVSaW2U=; b=NHjY29bJ9eZ3QdNutAkfWfF9UwTY+N5Vil+2Qdsc2W6xe6Pv6wYXU5myG1C+raUmDt gsqlAcox4RMjV5M7o6YKIYRzynJOWVJWcAVDq3h7P68Q5EPFJJY8eFUkM3a0SzIuJgxC /9s1o+2t76u5vh8gcbgMTpVAloJ4NW9UrIIJd/+oeEBXQAPz+96BEFw2ojC5JTfSwnKn MGbmoomLeFvYOpgCV/r1jBOtBM7uwDbgFDGaaV1yhoZoWn1ex3suDU4z+VlbHm/EKTrQ c2WFQX0EzK1UiZ/DNF6DRX2B9wAe1YE9JTcV+WsjmvnqNoMmtfxTb6M/96BecgSAtWj5 76Tw== X-Gm-Message-State: APjAAAWzXO6IhhLwXhpwd/NEl6n8sr79r0MjBFG/fifn3ordu5PpBu4b oTb8iBNqy2/IKtyDAHSkx6dR5Q== X-Received: by 2002:a63:5742:: with SMTP id h2mr80561532pgm.194.1558440044493; Tue, 21 May 2019 05:00:44 -0700 (PDT) Received: from localhost ([121.95.100.191]) by smtp.gmail.com with ESMTPSA id s77sm41968590pfa.63.2019.05.21.05.00.43 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Tue, 21 May 2019 05:00:43 -0700 (PDT) From: Masahisa Kojima To: linux-spi@vger.kernel.org, devicetree@vger.kernel.org Cc: broonie@kernel.org, geert@linux-m68k.org, tpiepho@impinj.com, andy.shevchenko@gmail.com, robh+dt@kernel.org, mark.rutland@arm.com, ard.biesheuvel@linaro.org, jaswinder.singh@linaro.org, masami.hiramatsu@linaro.org, okamoto.satoru@socionext.com, osaki.yoshitoyo@socionext.com, Masahisa Kojima Subject: [PATCH v5 3/3] spi: Add spi driver for Socionext Synquacer platform Date: Tue, 21 May 2019 20:59:58 +0900 Message-Id: <20190521115958.22504-4-masahisa.kojima@linaro.org> X-Mailer: git-send-email 2.14.2 In-Reply-To: <20190521115958.22504-1-masahisa.kojima@linaro.org> References: <20190521115958.22504-1-masahisa.kojima@linaro.org> Sender: devicetree-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: devicetree@vger.kernel.org This patch adds support for controller found on synquacer platforms. Signed-off-by: Masahisa Kojima Signed-off-by: Jassi Brar --- drivers/spi/Kconfig | 11 + drivers/spi/Makefile | 1 + drivers/spi/spi-synquacer.c | 731 ++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 743 insertions(+) create mode 100644 drivers/spi/spi-synquacer.c -- 2.14.2 diff --git a/drivers/spi/Kconfig b/drivers/spi/Kconfig index 0fba8f400c59..24a3eac72d12 100644 --- a/drivers/spi/Kconfig +++ b/drivers/spi/Kconfig @@ -732,6 +732,17 @@ config SPI_SUN6I help This enables using the SPI controller on the Allwinner A31 SoCs. +config SPI_SYNQUACER + tristate "Socionext's Synquacer HighSpeed SPI controller" + depends on ARCH_SYNQUACER || COMPILE_TEST + select SPI_BITBANG + help + SPI driver for Socionext's High speed SPI controller which provides + various operating modes for interfacing to serial peripheral devices + that use the de-facto standard SPI protocol. + + It also supports the new dual-bit and quad-bit SPI protocol. + config SPI_MXIC tristate "Macronix MX25F0A SPI controller" depends on SPI_MASTER diff --git a/drivers/spi/Makefile b/drivers/spi/Makefile index f2f78d03dc28..63dcab552bcb 100644 --- a/drivers/spi/Makefile +++ b/drivers/spi/Makefile @@ -106,6 +106,7 @@ obj-$(CONFIG_SPI_STM32_QSPI) += spi-stm32-qspi.o obj-$(CONFIG_SPI_ST_SSC4) += spi-st-ssc4.o obj-$(CONFIG_SPI_SUN4I) += spi-sun4i.o obj-$(CONFIG_SPI_SUN6I) += spi-sun6i.o +obj-$(CONFIG_SPI_SYNQUACER) += spi-synquacer.o obj-$(CONFIG_SPI_TEGRA114) += spi-tegra114.o obj-$(CONFIG_SPI_TEGRA20_SFLASH) += spi-tegra20-sflash.o obj-$(CONFIG_SPI_TEGRA20_SLINK) += spi-tegra20-slink.o diff --git a/drivers/spi/spi-synquacer.c b/drivers/spi/spi-synquacer.c new file mode 100644 index 000000000000..04ba76609a12 --- /dev/null +++ b/drivers/spi/spi-synquacer.c @@ -0,0 +1,731 @@ +// SPDX-License-Identifier: GPL-2.0 +// +// Synquacer HSSPI controller driver +// +// Copyright (c) 2015-2018 Socionext Inc. +// Copyright (c) 2018-2019 Linaro Ltd. +// + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +/* HSSPI register address definitions */ +#define SYNQUACER_HSSPI_REG_MCTRL 0x00 +#define SYNQUACER_HSSPI_REG_PCC0 0x04 +#define SYNQUACER_HSSPI_REG_PCC(n) (SYNQUACER_HSSPI_REG_PCC0 + (n) * 4) +#define SYNQUACER_HSSPI_REG_TXF 0x14 +#define SYNQUACER_HSSPI_REG_TXE 0x18 +#define SYNQUACER_HSSPI_REG_TXC 0x1C +#define SYNQUACER_HSSPI_REG_RXF 0x20 +#define SYNQUACER_HSSPI_REG_RXE 0x24 +#define SYNQUACER_HSSPI_REG_RXC 0x28 +#define SYNQUACER_HSSPI_REG_FAULTF 0x2C +#define SYNQUACER_HSSPI_REG_FAULTC 0x30 +#define SYNQUACER_HSSPI_REG_DMCFG 0x34 +#define SYNQUACER_HSSPI_REG_DMSTART 0x38 +#define SYNQUACER_HSSPI_REG_DMBCC 0x3C +#define SYNQUACER_HSSPI_REG_DMSTATUS 0x40 +#define SYNQUACER_HSSPI_REG_FIFOCFG 0x4C +#define SYNQUACER_HSSPI_REG_TX_FIFO 0x50 +#define SYNQUACER_HSSPI_REG_RX_FIFO 0x90 +#define SYNQUACER_HSSPI_REG_MID 0xFC + +/* HSSPI register bit definitions */ +#define SYNQUACER_HSSPI_MCTRL_MEN BIT(0) +#define SYNQUACER_HSSPI_MCTRL_COMMAND_SEQUENCE_EN BIT(1) +#define SYNQUACER_HSSPI_MCTRL_CDSS BIT(3) +#define SYNQUACER_HSSPI_MCTRL_MES BIT(4) +#define SYNQUACER_HSSPI_MCTRL_SYNCON BIT(5) + +#define SYNQUACER_HSSPI_PCC_CPHA BIT(0) +#define SYNQUACER_HSSPI_PCC_CPOL BIT(1) +#define SYNQUACER_HSSPI_PCC_ACES BIT(2) +#define SYNQUACER_HSSPI_PCC_RTM BIT(3) +#define SYNQUACER_HSSPI_PCC_SSPOL BIT(4) +#define SYNQUACER_HSSPI_PCC_SDIR BIT(7) +#define SYNQUACER_HSSPI_PCC_SENDIAN BIT(8) +#define SYNQUACER_HSSPI_PCC_SAFESYNC BIT(16) +#define SYNQUACER_HSSPI_PCC_SS2CD_SHIFT 5 +#define SYNQUACER_HSSPI_PCC_CDRS_MASK 0x7f +#define SYNQUACER_HSSPI_PCC_CDRS_SHIFT 9 + +#define SYNQUACER_HSSPI_TXF_FIFO_FULL BIT(0) +#define SYNQUACER_HSSPI_TXF_FIFO_EMPTY BIT(1) +#define SYNQUACER_HSSPI_TXF_SLAVE_RELEASED BIT(6) + +#define SYNQUACER_HSSPI_TXE_FIFO_FULL BIT(0) +#define SYNQUACER_HSSPI_TXE_FIFO_EMPTY BIT(1) +#define SYNQUACER_HSSPI_TXE_SLAVE_RELEASED BIT(6) + +#define SYNQUACER_HSSPI_RXF_FIFO_MORE_THAN_THRESHOLD BIT(5) +#define SYNQUACER_HSSPI_RXF_SLAVE_RELEASED BIT(6) + +#define SYNQUACER_HSSPI_RXE_FIFO_MORE_THAN_THRESHOLD BIT(5) +#define SYNQUACER_HSSPI_RXE_SLAVE_RELEASED BIT(6) + +#define SYNQUACER_HSSPI_DMCFG_SSDC BIT(1) +#define SYNQUACER_HSSPI_DMCFG_MSTARTEN BIT(2) + +#define SYNQUACER_HSSPI_DMSTART_START BIT(0) +#define SYNQUACER_HSSPI_DMSTOP_STOP BIT(8) +#define SYNQUACER_HSSPI_DMPSEL_CS_MASK 0x3 +#define SYNQUACER_HSSPI_DMPSEL_CS_SHIFT 16 +#define SYNQUACER_HSSPI_DMTRP_BUS_WIDTH_SHIFT 24 +#define SYNQUACER_HSSPI_DMTRP_DATA_MASK 0x3 +#define SYNQUACER_HSSPI_DMTRP_DATA_SHIFT 26 +#define SYNQUACER_HSSPI_DMTRP_DATA_TXRX 0 +#define SYNQUACER_HSSPI_DMTRP_DATA_RX 1 +#define SYNQUACER_HSSPI_DMTRP_DATA_TX 2 + +#define SYNQUACER_HSSPI_DMSTATUS_RX_DATA_MASK 0x1f +#define SYNQUACER_HSSPI_DMSTATUS_RX_DATA_SHIFT 8 +#define SYNQUACER_HSSPI_DMSTATUS_TX_DATA_MASK 0x1f +#define SYNQUACER_HSSPI_DMSTATUS_TX_DATA_SHIFT 16 + +#define SYNQUACER_HSSPI_FIFOCFG_RX_THRESHOLD_MASK 0xf +#define SYNQUACER_HSSPI_FIFOCFG_RX_THRESHOLD_SHIFT 0 +#define SYNQUACER_HSSPI_FIFOCFG_TX_THRESHOLD_MASK 0xf +#define SYNQUACER_HSSPI_FIFOCFG_TX_THRESHOLD_SHIFT 4 +#define SYNQUACER_HSSPI_FIFOCFG_FIFO_WIDTH_MASK 0x3 +#define SYNQUACER_HSSPI_FIFOCFG_FIFO_WIDTH_SHIFT 8 +#define SYNQUACER_HSSPI_FIFOCFG_RX_FLUSH BIT(11) +#define SYNQUACER_HSSPI_FIFOCFG_TX_FLUSH BIT(12) + +#define SYNQUACER_HSSPI_FIFO_DEPTH 16 +#define SYNQUACER_HSSPI_FIFO_TX_THRESHOLD 4 +#define SYNQUACER_HSSPI_FIFO_RX_THRESHOLD \ + (SYNQUACER_HSSPI_FIFO_DEPTH - SYNQUACER_HSSPI_FIFO_TX_THRESHOLD) + +#define SYNQUACER_HSSPI_TRANSFER_MODE_TX BIT(1) +#define SYNQUACER_HSSPI_TRANSFER_MODE_RX BIT(2) +#define SYNQUACER_HSSPI_TRANSFER_TMOUT_MSEC 2000 + +#define SYNQUACER_HSSPI_CLOCK_SRC_IHCLK 0 +#define SYNQUACER_HSSPI_CLOCK_SRC_IPCLK 1 + +#define SYNQUACER_HSSPI_NUM_CHIP_SELECT 4 + +struct synquacer_spi { + struct device *dev; + struct completion transfer_done; + unsigned int cs; + unsigned int bpw; + unsigned int mode; + unsigned int speed; + bool aces, rtm; + void *rx_buf; + const void *tx_buf; + struct clk *clk; + int clk_src_type; + void __iomem *regs; + unsigned int tx_words, rx_words; + unsigned int bus_width; + unsigned int transfer_mode; +}; + +static void read_fifo(struct synquacer_spi *sspi) +{ + u32 len = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTATUS); + + len = (len >> SYNQUACER_HSSPI_DMSTATUS_RX_DATA_SHIFT) & + SYNQUACER_HSSPI_DMSTATUS_RX_DATA_MASK; + len = min_t(unsigned int, len, sspi->rx_words); + + switch (sspi->bpw) { + case 8: + { + u8 *buf = sspi->rx_buf; + + readsb(sspi->regs + SYNQUACER_HSSPI_REG_RX_FIFO, buf, len); + sspi->rx_buf = buf + len; + break; + } + case 16: + { + u16 *buf = sspi->rx_buf; + + readsw(sspi->regs + SYNQUACER_HSSPI_REG_RX_FIFO, buf, len); + sspi->rx_buf = buf + len; + break; + } + default: + { + u32 *buf = sspi->rx_buf; + + readsl(sspi->regs + SYNQUACER_HSSPI_REG_RX_FIFO, buf, len); + sspi->rx_buf = buf + len; + break; + } + } + + sspi->rx_words -= len; +} + +static void write_fifo(struct synquacer_spi *sspi) +{ + u32 len = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTATUS); + + len = (len >> SYNQUACER_HSSPI_DMSTATUS_TX_DATA_SHIFT) & + SYNQUACER_HSSPI_DMSTATUS_TX_DATA_MASK; + len = min_t(unsigned int, SYNQUACER_HSSPI_FIFO_DEPTH - len, + sspi->tx_words); + + switch (sspi->bpw) { + case 8: + { + const u8 *buf = sspi->tx_buf; + + writesb(sspi->regs + SYNQUACER_HSSPI_REG_TX_FIFO, buf, len); + sspi->tx_buf = buf + len; + break; + } + case 16: + { + const u16 *buf = sspi->tx_buf; + + writesw(sspi->regs + SYNQUACER_HSSPI_REG_TX_FIFO, buf, len); + sspi->tx_buf = buf + len; + break; + } + default: + { + const u32 *buf = sspi->tx_buf; + + writesl(sspi->regs + SYNQUACER_HSSPI_REG_TX_FIFO, buf, len); + sspi->tx_buf = buf + len; + break; + } + } + sspi->tx_words -= len; +} + +static int synquacer_spi_config(struct spi_master *master, + struct spi_device *spi, + struct spi_transfer *xfer) +{ + struct synquacer_spi *sspi = spi_master_get_devdata(master); + unsigned int speed, mode, bpw, cs, bus_width, transfer_mode; + u32 rate, val, div; + + /* Full Duplex only on 1-bit wide bus */ + if (xfer->rx_buf && xfer->tx_buf && + (xfer->rx_nbits != 1 || xfer->tx_nbits != 1)) { + dev_err(sspi->dev, + "RX and TX bus widths must match for Full-Duplex!\n"); + return -EINVAL; + } + + if (xfer->tx_buf) { + bus_width = xfer->tx_nbits; + transfer_mode = SYNQUACER_HSSPI_TRANSFER_MODE_TX; + } else { + bus_width = xfer->rx_nbits; + transfer_mode = SYNQUACER_HSSPI_TRANSFER_MODE_RX; + } + + mode = spi->mode; + cs = spi->chip_select; + speed = xfer->speed_hz; + bpw = xfer->bits_per_word; + + /* return if nothing to change */ + if (speed == sspi->speed && + bus_width == sspi->bus_width && bpw == sspi->bpw && + mode == sspi->mode && cs == sspi->cs && + transfer_mode == sspi->transfer_mode) { + return 0; + } + + sspi->transfer_mode = transfer_mode; + rate = master->max_speed_hz; + + div = DIV_ROUND_UP(rate, speed); + if (div > 254) { + dev_err(sspi->dev, "Requested rate too low (%u)\n", + sspi->speed); + return -EINVAL; + } + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_PCC(cs)); + val &= ~SYNQUACER_HSSPI_PCC_SAFESYNC; + if (bpw == 8 && (mode & (SPI_TX_DUAL | SPI_RX_DUAL)) && div < 3) + val |= SYNQUACER_HSSPI_PCC_SAFESYNC; + if (bpw == 8 && (mode & (SPI_TX_QUAD | SPI_RX_QUAD)) && div < 6) + val |= SYNQUACER_HSSPI_PCC_SAFESYNC; + if (bpw == 16 && (mode & (SPI_TX_QUAD | SPI_RX_QUAD)) && div < 3) + val |= SYNQUACER_HSSPI_PCC_SAFESYNC; + + if (mode & SPI_CPHA) + val |= SYNQUACER_HSSPI_PCC_CPHA; + else + val &= ~SYNQUACER_HSSPI_PCC_CPHA; + + if (mode & SPI_CPOL) + val |= SYNQUACER_HSSPI_PCC_CPOL; + else + val &= ~SYNQUACER_HSSPI_PCC_CPOL; + + if (mode & SPI_CS_HIGH) + val |= SYNQUACER_HSSPI_PCC_SSPOL; + else + val &= ~SYNQUACER_HSSPI_PCC_SSPOL; + + if (mode & SPI_LSB_FIRST) + val |= SYNQUACER_HSSPI_PCC_SDIR; + else + val &= ~SYNQUACER_HSSPI_PCC_SDIR; + + if (sspi->aces) + val |= SYNQUACER_HSSPI_PCC_ACES; + else + val &= ~SYNQUACER_HSSPI_PCC_ACES; + + if (sspi->rtm) + val |= SYNQUACER_HSSPI_PCC_RTM; + else + val &= ~SYNQUACER_HSSPI_PCC_RTM; + + val |= (3 << SYNQUACER_HSSPI_PCC_SS2CD_SHIFT); + val |= SYNQUACER_HSSPI_PCC_SENDIAN; + + val &= ~(SYNQUACER_HSSPI_PCC_CDRS_MASK << + SYNQUACER_HSSPI_PCC_CDRS_SHIFT); + val |= ((div >> 1) << SYNQUACER_HSSPI_PCC_CDRS_SHIFT); + + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_PCC(cs)); + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + val &= ~(SYNQUACER_HSSPI_FIFOCFG_FIFO_WIDTH_MASK << + SYNQUACER_HSSPI_FIFOCFG_FIFO_WIDTH_SHIFT); + val |= ((bpw / 8 - 1) << SYNQUACER_HSSPI_FIFOCFG_FIFO_WIDTH_SHIFT); + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + val &= ~(SYNQUACER_HSSPI_DMTRP_DATA_MASK << + SYNQUACER_HSSPI_DMTRP_DATA_SHIFT); + + if (xfer->rx_buf) + val |= (SYNQUACER_HSSPI_DMTRP_DATA_RX << + SYNQUACER_HSSPI_DMTRP_DATA_SHIFT); + else + val |= (SYNQUACER_HSSPI_DMTRP_DATA_TX << + SYNQUACER_HSSPI_DMTRP_DATA_SHIFT); + + val &= ~(3 << SYNQUACER_HSSPI_DMTRP_BUS_WIDTH_SHIFT); + val |= ((bus_width >> 1) << SYNQUACER_HSSPI_DMTRP_BUS_WIDTH_SHIFT); + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + + sspi->bpw = bpw; + sspi->mode = mode; + sspi->speed = speed; + sspi->cs = spi->chip_select; + sspi->bus_width = bus_width; + + return 0; +} + +static int synquacer_spi_transfer_one(struct spi_master *master, + struct spi_device *spi, + struct spi_transfer *xfer) +{ + struct synquacer_spi *sspi = spi_master_get_devdata(master); + int ret; + int status = 0; + unsigned int words; + u8 bpw; + u32 val; + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + val |= SYNQUACER_HSSPI_FIFOCFG_RX_FLUSH; + val |= SYNQUACER_HSSPI_FIFOCFG_TX_FLUSH; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + + /* + * See if we can transfer 4-bytes as 1 word + * to maximize the FIFO buffer effficiency + */ + bpw = xfer->bits_per_word; + if (bpw == 8 && !(xfer->len % 4) && !(spi->mode & SPI_LSB_FIRST)) + xfer->bits_per_word = 32; + + ret = synquacer_spi_config(master, spi, xfer); + + /* restore */ + xfer->bits_per_word = bpw; + + if (ret) + return ret; + + reinit_completion(&sspi->transfer_done); + + sspi->tx_buf = xfer->tx_buf; + sspi->rx_buf = xfer->rx_buf; + + switch (sspi->bpw) { + case 8: + words = xfer->len; + break; + case 16: + words = xfer->len / 2; + break; + default: + words = xfer->len / 4; + break; + } + + if (xfer->tx_buf) + sspi->tx_words = words; + else + sspi->tx_words = 0; + + if (xfer->rx_buf) + sspi->rx_words = words; + else + sspi->rx_words = 0; + + if (xfer->tx_buf) + write_fifo(sspi); + + if (xfer->rx_buf) { + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + val &= ~(SYNQUACER_HSSPI_FIFOCFG_RX_THRESHOLD_MASK << + SYNQUACER_HSSPI_FIFOCFG_RX_THRESHOLD_SHIFT); + val |= ((sspi->rx_words > SYNQUACER_HSSPI_FIFO_DEPTH ? + SYNQUACER_HSSPI_FIFO_RX_THRESHOLD : sspi->rx_words) << + SYNQUACER_HSSPI_FIFOCFG_RX_THRESHOLD_SHIFT); + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_FIFOCFG); + } + + writel_relaxed(~0, sspi->regs + SYNQUACER_HSSPI_REG_TXC); + writel_relaxed(~0, sspi->regs + SYNQUACER_HSSPI_REG_RXC); + + /* Trigger */ + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + val |= SYNQUACER_HSSPI_DMSTART_START; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + + if (sspi->rx_words) { + val = SYNQUACER_HSSPI_RXE_FIFO_MORE_THAN_THRESHOLD | + SYNQUACER_HSSPI_RXE_SLAVE_RELEASED; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_RXE); + status = wait_for_completion_timeout(&sspi->transfer_done, + msecs_to_jiffies(SYNQUACER_HSSPI_TRANSFER_TMOUT_MSEC)); + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_RXE); + } + + if (xfer->tx_buf) { + val = SYNQUACER_HSSPI_TXE_FIFO_EMPTY; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_TXE); + status = wait_for_completion_timeout(&sspi->transfer_done, + msecs_to_jiffies(SYNQUACER_HSSPI_TRANSFER_TMOUT_MSEC)); + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_TXE); + } + + if (status < 0) { + dev_err(sspi->dev, "failed to transfer\n"); + return status; + } + + return 0; +} + +static void synquacer_spi_set_cs(struct spi_device *spi, bool enable) +{ + struct synquacer_spi *sspi = spi_master_get_devdata(spi->master); + u32 val; + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + val &= ~(SYNQUACER_HSSPI_DMPSEL_CS_MASK << + SYNQUACER_HSSPI_DMPSEL_CS_SHIFT); + val |= spi->chip_select << SYNQUACER_HSSPI_DMPSEL_CS_SHIFT; + + if (enable) { + val |= SYNQUACER_HSSPI_DMSTOP_STOP; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + + if (sspi->rx_buf) { + u32 buf[SYNQUACER_HSSPI_FIFO_DEPTH]; + + sspi->rx_buf = buf; + sspi->rx_words = SYNQUACER_HSSPI_FIFO_DEPTH; + read_fifo(sspi); + } + } else { + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + val &= ~SYNQUACER_HSSPI_DMSTOP_STOP; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMSTART); + } +} + +static int synquacer_spi_enable(struct spi_master *master) +{ + struct synquacer_spi *sspi = spi_master_get_devdata(master); + u32 val; + unsigned int retries = 0xfffff; + + /* Disable module */ + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_MCTRL); + while ((readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_MCTRL) & + SYNQUACER_HSSPI_MCTRL_MES) && --retries) + cpu_relax(); + if (!retries) + return -EBUSY; + + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_TXE); + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_RXE); + writel_relaxed(~0, sspi->regs + SYNQUACER_HSSPI_REG_TXC); + writel_relaxed(~0, sspi->regs + SYNQUACER_HSSPI_REG_RXC); + writel_relaxed(~0, sspi->regs + SYNQUACER_HSSPI_REG_FAULTC); + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_DMCFG); + val &= ~SYNQUACER_HSSPI_DMCFG_SSDC; + val &= ~SYNQUACER_HSSPI_DMCFG_MSTARTEN; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_DMCFG); + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_MCTRL); + if (sspi->clk_src_type == SYNQUACER_HSSPI_CLOCK_SRC_IPCLK) + val |= SYNQUACER_HSSPI_MCTRL_CDSS; + else + val &= ~SYNQUACER_HSSPI_MCTRL_CDSS; + + val &= ~SYNQUACER_HSSPI_MCTRL_COMMAND_SEQUENCE_EN; + val |= SYNQUACER_HSSPI_MCTRL_MEN; + val |= SYNQUACER_HSSPI_MCTRL_SYNCON; + writel_relaxed(val, sspi->regs + SYNQUACER_HSSPI_REG_MCTRL); + + return 0; +} + +static irqreturn_t sq_spi_rx_handler(int irq, void *priv) +{ + uint32_t val; + struct synquacer_spi *sspi = priv; + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_RXF); + if ((val & SYNQUACER_HSSPI_RXF_SLAVE_RELEASED) || + (val & SYNQUACER_HSSPI_RXF_FIFO_MORE_THAN_THRESHOLD)) + read_fifo(sspi); + + if (sspi->rx_words == 0) { + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_RXE); + complete(&sspi->transfer_done); + } + + return 0; +} + +static irqreturn_t sq_spi_tx_handler(int irq, void *priv) +{ + uint32_t val; + struct synquacer_spi *sspi = priv; + + val = readl_relaxed(sspi->regs + SYNQUACER_HSSPI_REG_TXF); + + if (val & SYNQUACER_HSSPI_TXF_FIFO_EMPTY) { + if (sspi->tx_words == 0) { + writel_relaxed(0, sspi->regs + SYNQUACER_HSSPI_REG_TXE); + complete(&sspi->transfer_done); + return 0; + } + write_fifo(sspi); + } + + return 0; +} + +static int synquacer_spi_probe(struct platform_device *pdev) +{ + struct device_node *np = pdev->dev.of_node; + struct spi_master *master; + struct synquacer_spi *sspi; + struct resource *res; + int ret; + int rx_irq, tx_irq; + + master = spi_alloc_master(&pdev->dev, sizeof(*sspi)); + if (!master) + return -ENOMEM; + + platform_set_drvdata(pdev, master); + + sspi = spi_master_get_devdata(master); + sspi->dev = &pdev->dev; + + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); + sspi->regs = devm_ioremap_resource(sspi->dev, res); + if (IS_ERR(sspi->regs)) { + ret = PTR_ERR(sspi->regs); + goto put_spi; + } + + if (of_property_match_string(np, "clock-names", "iHCLK") >= 0) { + sspi->clk_src_type = SYNQUACER_HSSPI_CLOCK_SRC_IHCLK; + sspi->clk = devm_clk_get(sspi->dev, "iHCLK"); + } else if (of_property_match_string(np, "clock-names", "iPCLK") >= 0) { + sspi->clk_src_type = SYNQUACER_HSSPI_CLOCK_SRC_IPCLK; + sspi->clk = devm_clk_get(sspi->dev, "iPCLK"); + } else { + dev_err(&pdev->dev, "specified wrong clock source\n"); + ret = -EINVAL; + goto put_spi; + } + if (IS_ERR(sspi->clk)) { + dev_err(&pdev->dev, "clock not found\n"); + ret = PTR_ERR(sspi->clk); + goto put_spi; + } + + sspi->aces = of_property_read_bool(np, "socionext,set-aces"); + sspi->rtm = of_property_read_bool(np, "socionext,use-rtm"); + + master->num_chipselect = SYNQUACER_HSSPI_NUM_CHIP_SELECT; + + init_completion(&sspi->transfer_done); + + rx_irq = platform_get_irq(pdev, 0); + if (rx_irq < 0) + dev_err(&pdev->dev, "get rx_irq failed\n"); + + tx_irq = platform_get_irq(pdev, 1); + if (tx_irq < 0) + dev_err(&pdev->dev, "get tx_irq failed\n"); + + ret = devm_request_irq(&pdev->dev, rx_irq, sq_spi_rx_handler, + 0, "synquacer-spi-rx", sspi); + ret = devm_request_irq(&pdev->dev, tx_irq, sq_spi_tx_handler, + 0, "synquacer-spi-tx", sspi); + + ret = clk_prepare_enable(sspi->clk); + if (ret) + goto put_spi; + + master->dev.of_node = np; + master->auto_runtime_pm = true; + master->bus_num = pdev->id; + + master->mode_bits = SPI_CPOL | SPI_CPHA | SPI_TX_DUAL | SPI_RX_DUAL | + SPI_TX_QUAD | SPI_RX_QUAD; + master->bits_per_word_mask = SPI_BPW_MASK(32) | SPI_BPW_MASK(24) | + SPI_BPW_MASK(16) | SPI_BPW_MASK(8); + + master->max_speed_hz = clk_get_rate(sspi->clk); + master->min_speed_hz = master->max_speed_hz / 254; + + master->set_cs = synquacer_spi_set_cs; + master->transfer_one = synquacer_spi_transfer_one; + + ret = synquacer_spi_enable(master); + if (ret) + goto fail_enable; + + pm_runtime_set_active(sspi->dev); + pm_runtime_enable(sspi->dev); + + ret = devm_spi_register_master(sspi->dev, master); + if (ret) + goto disable_pm; + + return 0; + +disable_pm: + pm_runtime_disable(sspi->dev); +fail_enable: + clk_disable_unprepare(sspi->clk); +put_spi: + spi_master_put(master); + + return ret; +} + +static int synquacer_spi_remove(struct platform_device *pdev) +{ + struct spi_master *master = platform_get_drvdata(pdev); + struct synquacer_spi *sspi = spi_master_get_devdata(master); + + pm_runtime_disable(sspi->dev); + clk_disable_unprepare(sspi->clk); + return 0; +} + +static int __maybe_unused synquacer_spi_suspend(struct device *dev) +{ + struct spi_master *master = dev_get_drvdata(dev); + struct synquacer_spi *sspi = spi_master_get_devdata(master); + int ret; + + ret = spi_master_suspend(master); + if (ret) + return ret; + + if (!pm_runtime_suspended(dev)) + clk_disable_unprepare(sspi->clk); + + return ret; +} + +static int __maybe_unused synquacer_spi_resume(struct device *dev) +{ + struct spi_master *master = dev_get_drvdata(dev); + struct synquacer_spi *sspi = spi_master_get_devdata(master); + int ret; + + if (!pm_runtime_suspended(dev)) { + /* Ensure reconfigure during next xfer */ + sspi->speed = 0; + + ret = clk_prepare_enable(sspi->clk); + if (ret < 0) { + dev_err(dev, "failed to enable clk (%d)\n", ret); + return ret; + } + + ret = synquacer_spi_enable(master); + if (ret) { + dev_err(dev, "failed to enable spi (%d)\n", ret); + return ret; + } + } + + ret = spi_master_resume(master); + if (ret < 0) + clk_disable_unprepare(sspi->clk); + + return ret; +} + +static SIMPLE_DEV_PM_OPS(synquacer_spi_pm_ops, synquacer_spi_suspend, + synquacer_spi_resume); + +static const struct of_device_id synquacer_spi_of_match[] = { + {.compatible = "socionext,synquacer-spi"}, + {} +}; +MODULE_DEVICE_TABLE(of, synquacer_spi_of_match); + +static struct platform_driver synquacer_spi_driver = { + .driver = { + .name = "synquacer-spi", + .pm = &synquacer_spi_pm_ops, + .of_match_table = synquacer_spi_of_match, + }, + .probe = synquacer_spi_probe, + .remove = synquacer_spi_remove, +}; +module_platform_driver(synquacer_spi_driver); + +MODULE_DESCRIPTION("Socionext Synquacer HS-SPI controller driver"); +MODULE_AUTHOR("Masahisa Kojima "); +MODULE_AUTHOR("Jassi Brar "); +MODULE_LICENSE("GPL v2");