1 // SPDX-License-Identifier: GPL-2.0-only
3 * Altera Passive Serial SPI Driver
5 * Copyright (c) 2017 United Western Technologies, Corporation
7 * Joshua Clayton <stillcompiling@gmail.com>
9 * Manage Altera FPGA firmware that is loaded over SPI using the passive
10 * serial configuration method.
11 * Firmware must be in binary "rbf" format.
12 * Works on Arria 10, Cyclone V and Stratix V. Should work on Cyclone series.
13 * May work on other Altera FPGAs.
16 #include <linux/bitrev.h>
17 #include <linux/delay.h>
18 #include <linux/fpga/fpga-mgr.h>
19 #include <linux/gpio/consumer.h>
20 #include <linux/module.h>
21 #include <linux/of_gpio.h>
22 #include <linux/of_device.h>
23 #include <linux/spi/spi.h>
24 #include <linux/sizes.h>
26 enum altera_ps_devtype
{
31 struct altera_ps_data
{
32 enum altera_ps_devtype devtype
;
33 int status_wait_min_us
;
34 int status_wait_max_us
;
39 struct altera_ps_conf
{
40 struct gpio_desc
*config
;
41 struct gpio_desc
*confd
;
42 struct gpio_desc
*status
;
43 struct spi_device
*spi
;
44 const struct altera_ps_data
*data
;
49 /* | Arria 10 | Cyclone5 | Stratix5 |
50 * t_CF2ST0 | [; 600] | [; 600] | [; 600] |ns
51 * t_CFG | [2;] | [2;] | [2;] |µs
52 * t_STATUS | [268; 3000] | [268; 1506] | [268; 1506] |µs
53 * t_CF2ST1 | [; 3000] | [; 1506] | [; 1506] |µs
54 * t_CF2CK | [3010;] | [1506;] | [1506;] |µs
55 * t_ST2CK | [10;] | [2;] | [2;] |µs
56 * t_CD2UM | [175; 830] | [175; 437] | [175; 437] |µs
58 static struct altera_ps_data c5_data
= {
59 /* these values for Cyclone5 are compatible with Stratix5 */
61 .status_wait_min_us
= 268,
62 .status_wait_max_us
= 1506,
67 static struct altera_ps_data a10_data
= {
69 .status_wait_min_us
= 268, /* min(t_STATUS) */
70 .status_wait_max_us
= 3000, /* max(t_CF2ST1) */
71 .t_cfg_us
= 2, /* max { min(t_CFG), max(tCF2ST0) } */
72 .t_st2ck_us
= 10, /* min(t_ST2CK) */
75 /* Array index is enum altera_ps_devtype */
76 static const struct altera_ps_data
*altera_ps_data_map
[] = {
81 static const struct of_device_id of_ef_match
[] = {
82 { .compatible
= "altr,fpga-passive-serial", .data
= &c5_data
},
83 { .compatible
= "altr,fpga-arria10-passive-serial", .data
= &a10_data
},
86 MODULE_DEVICE_TABLE(of
, of_ef_match
);
88 static enum fpga_mgr_states
altera_ps_state(struct fpga_manager
*mgr
)
90 struct altera_ps_conf
*conf
= mgr
->priv
;
92 if (gpiod_get_value_cansleep(conf
->status
))
93 return FPGA_MGR_STATE_RESET
;
95 return FPGA_MGR_STATE_UNKNOWN
;
98 static inline void altera_ps_delay(int delay_us
)
101 usleep_range(delay_us
, delay_us
+ 5);
106 static int altera_ps_write_init(struct fpga_manager
*mgr
,
107 struct fpga_image_info
*info
,
108 const char *buf
, size_t count
)
110 struct altera_ps_conf
*conf
= mgr
->priv
;
114 conf
->info_flags
= info
->flags
;
116 if (info
->flags
& FPGA_MGR_PARTIAL_RECONFIG
) {
117 dev_err(&mgr
->dev
, "Partial reconfiguration not supported.\n");
121 gpiod_set_value_cansleep(conf
->config
, 1);
123 /* wait min reset pulse time */
124 altera_ps_delay(conf
->data
->t_cfg_us
);
126 if (!gpiod_get_value_cansleep(conf
->status
)) {
127 dev_err(&mgr
->dev
, "Status pin failed to show a reset\n");
131 gpiod_set_value_cansleep(conf
->config
, 0);
133 min
= conf
->data
->status_wait_min_us
;
134 max
= conf
->data
->status_wait_max_us
;
139 /* wait for max { max(t_STATUS), max(t_CF2ST1) } */
140 for (i
= 0; i
< waits
; i
++) {
141 usleep_range(min
, min
+ 10);
142 if (!gpiod_get_value_cansleep(conf
->status
)) {
143 /* wait for min(t_ST2CK)*/
144 altera_ps_delay(conf
->data
->t_st2ck_us
);
149 dev_err(&mgr
->dev
, "Status pin not ready.\n");
153 static void rev_buf(char *buf
, size_t len
)
155 u32
*fw32
= (u32
*)buf
;
156 size_t extra_bytes
= (len
& 0x03);
157 const u32
*fw_end
= (u32
*)(buf
+ len
- extra_bytes
);
159 /* set buffer to lsb first */
160 while (fw32
< fw_end
) {
161 *fw32
= bitrev8x4(*fw32
);
166 buf
= (char *)fw_end
;
167 while (extra_bytes
) {
168 *buf
= bitrev8(*buf
);
175 static int altera_ps_write(struct fpga_manager
*mgr
, const char *buf
,
178 struct altera_ps_conf
*conf
= mgr
->priv
;
179 const char *fw_data
= buf
;
180 const char *fw_data_end
= fw_data
+ count
;
182 while (fw_data
< fw_data_end
) {
184 size_t stride
= min_t(size_t, fw_data_end
- fw_data
, SZ_4K
);
186 if (!(conf
->info_flags
& FPGA_MGR_BITSTREAM_LSB_FIRST
))
187 rev_buf((char *)fw_data
, stride
);
189 ret
= spi_write(conf
->spi
, fw_data
, stride
);
191 dev_err(&mgr
->dev
, "spi error in firmware write: %d\n",
201 static int altera_ps_write_complete(struct fpga_manager
*mgr
,
202 struct fpga_image_info
*info
)
204 struct altera_ps_conf
*conf
= mgr
->priv
;
205 static const char dummy
[] = {0};
208 if (gpiod_get_value_cansleep(conf
->status
)) {
209 dev_err(&mgr
->dev
, "Error during configuration.\n");
214 if (!gpiod_get_raw_value_cansleep(conf
->confd
)) {
215 dev_err(&mgr
->dev
, "CONF_DONE is inactive!\n");
221 * After CONF_DONE goes high, send two additional falling edges on DCLK
222 * to begin initialization and enter user mode
224 ret
= spi_write(conf
->spi
, dummy
, 1);
226 dev_err(&mgr
->dev
, "spi error during end sequence: %d\n", ret
);
233 static const struct fpga_manager_ops altera_ps_ops
= {
234 .state
= altera_ps_state
,
235 .write_init
= altera_ps_write_init
,
236 .write
= altera_ps_write
,
237 .write_complete
= altera_ps_write_complete
,
240 static const struct altera_ps_data
*id_to_data(const struct spi_device_id
*id
)
242 kernel_ulong_t devtype
= id
->driver_data
;
243 const struct altera_ps_data
*data
;
245 /* someone added a altera_ps_devtype without adding to the map array */
246 if (devtype
>= ARRAY_SIZE(altera_ps_data_map
))
249 data
= altera_ps_data_map
[devtype
];
250 if (!data
|| data
->devtype
!= devtype
)
256 static int altera_ps_probe(struct spi_device
*spi
)
258 struct altera_ps_conf
*conf
;
259 const struct of_device_id
*of_id
;
260 struct fpga_manager
*mgr
;
262 conf
= devm_kzalloc(&spi
->dev
, sizeof(*conf
), GFP_KERNEL
);
266 if (spi
->dev
.of_node
) {
267 of_id
= of_match_device(of_ef_match
, &spi
->dev
);
270 conf
->data
= of_id
->data
;
272 conf
->data
= id_to_data(spi_get_device_id(spi
));
278 conf
->config
= devm_gpiod_get(&spi
->dev
, "nconfig", GPIOD_OUT_LOW
);
279 if (IS_ERR(conf
->config
)) {
280 dev_err(&spi
->dev
, "Failed to get config gpio: %ld\n",
281 PTR_ERR(conf
->config
));
282 return PTR_ERR(conf
->config
);
285 conf
->status
= devm_gpiod_get(&spi
->dev
, "nstat", GPIOD_IN
);
286 if (IS_ERR(conf
->status
)) {
287 dev_err(&spi
->dev
, "Failed to get status gpio: %ld\n",
288 PTR_ERR(conf
->status
));
289 return PTR_ERR(conf
->status
);
292 conf
->confd
= devm_gpiod_get_optional(&spi
->dev
, "confd", GPIOD_IN
);
293 if (IS_ERR(conf
->confd
)) {
294 dev_err(&spi
->dev
, "Failed to get confd gpio: %ld\n",
295 PTR_ERR(conf
->confd
));
296 return PTR_ERR(conf
->confd
);
297 } else if (!conf
->confd
) {
298 dev_warn(&spi
->dev
, "Not using confd gpio");
301 /* Register manager with unique name */
302 snprintf(conf
->mgr_name
, sizeof(conf
->mgr_name
), "%s %s",
303 dev_driver_string(&spi
->dev
), dev_name(&spi
->dev
));
305 mgr
= devm_fpga_mgr_create(&spi
->dev
, conf
->mgr_name
,
306 &altera_ps_ops
, conf
);
310 spi_set_drvdata(spi
, mgr
);
312 return fpga_mgr_register(mgr
);
315 static int altera_ps_remove(struct spi_device
*spi
)
317 struct fpga_manager
*mgr
= spi_get_drvdata(spi
);
319 fpga_mgr_unregister(mgr
);
324 static const struct spi_device_id altera_ps_spi_ids
[] = {
325 { "cyclone-ps-spi", CYCLONE5
},
326 { "fpga-passive-serial", CYCLONE5
},
327 { "fpga-arria10-passive-serial", ARRIA10
},
330 MODULE_DEVICE_TABLE(spi
, altera_ps_spi_ids
);
332 static struct spi_driver altera_ps_driver
= {
334 .name
= "altera-ps-spi",
335 .owner
= THIS_MODULE
,
336 .of_match_table
= of_match_ptr(of_ef_match
),
338 .id_table
= altera_ps_spi_ids
,
339 .probe
= altera_ps_probe
,
340 .remove
= altera_ps_remove
,
343 module_spi_driver(altera_ps_driver
)
345 MODULE_LICENSE("GPL v2");
346 MODULE_AUTHOR("Joshua Clayton <stillcompiling@gmail.com>");
347 MODULE_DESCRIPTION("Module to load Altera FPGA firmware over SPI");