net: dsa: microchip: Remove ksz_{read,write}24()
[linux-2.6-microblaze.git] / drivers / net / dsa / microchip / ksz9477_spi.c
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Microchip KSZ9477 series register access through SPI
4  *
5  * Copyright (C) 2017-2019 Microchip Technology Inc.
6  */
7
8 #include <asm/unaligned.h>
9
10 #include <linux/delay.h>
11 #include <linux/kernel.h>
12 #include <linux/module.h>
13 #include <linux/spi/spi.h>
14
15 #include "ksz_priv.h"
16 #include "ksz_spi.h"
17
18 /* SPI frame opcodes */
19 #define KS_SPIOP_RD                     3
20 #define KS_SPIOP_WR                     2
21
22 #define SPI_ADDR_SHIFT                  24
23 #define SPI_ADDR_MASK                   (BIT(SPI_ADDR_SHIFT) - 1)
24 #define SPI_TURNAROUND_SHIFT            5
25
26 /* Enough to read all switch port registers. */
27 #define SPI_TX_BUF_LEN                  0x100
28
29 static int ksz9477_spi_read_reg(struct spi_device *spi, u32 reg, u8 *val,
30                                 unsigned int len)
31 {
32         u32 txbuf;
33         int ret;
34
35         txbuf = reg & SPI_ADDR_MASK;
36         txbuf |= KS_SPIOP_RD << SPI_ADDR_SHIFT;
37         txbuf <<= SPI_TURNAROUND_SHIFT;
38         txbuf = cpu_to_be32(txbuf);
39
40         ret = spi_write_then_read(spi, &txbuf, 4, val, len);
41         return ret;
42 }
43
44 static int ksz9477_spi_write_reg(struct spi_device *spi, u32 reg, u8 *val,
45                                  unsigned int len)
46 {
47         u32 *txbuf = (u32 *)val;
48
49         *txbuf = reg & SPI_ADDR_MASK;
50         *txbuf |= (KS_SPIOP_WR << SPI_ADDR_SHIFT);
51         *txbuf <<= SPI_TURNAROUND_SHIFT;
52         *txbuf = cpu_to_be32(*txbuf);
53
54         return spi_write(spi, txbuf, 4 + len);
55 }
56
57 static int ksz_spi_read(struct ksz_device *dev, u32 reg, u8 *data,
58                         unsigned int len)
59 {
60         struct spi_device *spi = dev->priv;
61
62         return ksz9477_spi_read_reg(spi, reg, data, len);
63 }
64
65 static int ksz_spi_write(struct ksz_device *dev, u32 reg, void *data,
66                          unsigned int len)
67 {
68         struct spi_device *spi = dev->priv;
69
70         if (len > SPI_TX_BUF_LEN)
71                 len = SPI_TX_BUF_LEN;
72         memcpy(&dev->txbuf[4], data, len);
73         return ksz9477_spi_write_reg(spi, reg, dev->txbuf, len);
74 }
75
76 static const struct ksz_io_ops ksz9477_spi_ops = {
77         .read8 = ksz_spi_read8,
78         .read16 = ksz_spi_read16,
79         .read32 = ksz_spi_read32,
80         .write8 = ksz_spi_write8,
81         .write16 = ksz_spi_write16,
82         .write32 = ksz_spi_write32,
83         .get = ksz_spi_get,
84         .set = ksz_spi_set,
85 };
86
87 static int ksz9477_spi_probe(struct spi_device *spi)
88 {
89         struct ksz_device *dev;
90         int ret;
91
92         dev = ksz_switch_alloc(&spi->dev, &ksz9477_spi_ops, spi);
93         if (!dev)
94                 return -ENOMEM;
95
96         if (spi->dev.platform_data)
97                 dev->pdata = spi->dev.platform_data;
98
99         dev->txbuf = devm_kzalloc(dev->dev, 4 + SPI_TX_BUF_LEN, GFP_KERNEL);
100
101         ret = ksz9477_switch_register(dev);
102
103         /* Main DSA driver may not be started yet. */
104         if (ret)
105                 return ret;
106
107         spi_set_drvdata(spi, dev);
108
109         return 0;
110 }
111
112 static int ksz9477_spi_remove(struct spi_device *spi)
113 {
114         struct ksz_device *dev = spi_get_drvdata(spi);
115
116         if (dev)
117                 ksz_switch_remove(dev);
118
119         return 0;
120 }
121
122 static void ksz9477_spi_shutdown(struct spi_device *spi)
123 {
124         struct ksz_device *dev = spi_get_drvdata(spi);
125
126         if (dev && dev->dev_ops->shutdown)
127                 dev->dev_ops->shutdown(dev);
128 }
129
130 static const struct of_device_id ksz9477_dt_ids[] = {
131         { .compatible = "microchip,ksz9477" },
132         { .compatible = "microchip,ksz9897" },
133         { .compatible = "microchip,ksz9893" },
134         { .compatible = "microchip,ksz9563" },
135         {},
136 };
137 MODULE_DEVICE_TABLE(of, ksz9477_dt_ids);
138
139 static struct spi_driver ksz9477_spi_driver = {
140         .driver = {
141                 .name   = "ksz9477-switch",
142                 .owner  = THIS_MODULE,
143                 .of_match_table = of_match_ptr(ksz9477_dt_ids),
144         },
145         .probe  = ksz9477_spi_probe,
146         .remove = ksz9477_spi_remove,
147         .shutdown = ksz9477_spi_shutdown,
148 };
149
150 module_spi_driver(ksz9477_spi_driver);
151
152 MODULE_AUTHOR("Woojung Huh <Woojung.Huh@microchip.com>");
153 MODULE_DESCRIPTION("Microchip KSZ9477 Series Switch SPI access Driver");
154 MODULE_LICENSE("GPL");