一、概述
MT7688硬件自带两个SPI的CS,其中一个用于flash,可用就只有一个。在实际应用中可能会连接多个SPI设备,显然一个CS是不够用的。内核中虽然实现了SPI bitbang的master使用gpio模拟spi,但性能无法与cpu的spi控制器相比。本文目的在于使用GPIO扩展spi的CS,其他spi控制线共用。网上也有很多博文实现GPIO扩展SPI CS的,但实际操作中总会有些问题,本文方法亲测是可以的。
二、软件环境
- linux发行版:ubuntu 14.04LTS
- Openwrt版本:15.05.1
- 硬件:MT7688A
三、配置过程
3.1 target/linux/ramips/dts/
根据实际使用的目标设备,修改该目录下对应的dts文件1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35 spi@b00 {
status = "okay";
pinctrl-names = "default";
pinctrl-0 = <&spi_pins>, <&spi_cs1_pins>;
cs-gpios = <0>, <0>, <&gpio1 5 1>;
m25p80@0 {
compatible = "w25q256";
reg = <0 0>;
linux,modalias = "m25p80", "w25q256";
spi-max-frequency = <40000000>;
m25p,chunked-io = <31>;
... ...
};
spidev@1 {
compatible = "spidev";
reg = <1>;
spi-max-frequency = <40000000>;
};
spidev@2 {
compatible = "spidev";
reg = <2>;
spi-max-frequency = <40000000>;
};
};
cs-gpios的两个0表示CS0和CS1,<&gpio1 5 1>表示使用GPIO37作为CS2。
spidev@1和spidev@2表示CS1和CS2的spi设备,系统启动后会在/dev下出现spidev32766.1和spidev32766.2设备符,应用程序可直接使用
注意:对于作为SPI CS的片选,如果是复用功能,需要配置成GPIO模式。
3.2 SPI驱动的修改
drivers/spi/spi.c的spi_add_device函数中:1
2
3
4
5
6
7
8
9
10
11- if (master->cs_gpios)
+ if (master->cs_gpios) {
spi->cs_gpio = master->cs_gpios[spi->chip_select];
+ if (spi->cs_gpio >= 0) {
+ status = gpio_request(spi->cs_gpio, dev_name(&spi->dev));
+ dev_info(dev, "cs_gpio = %d, status = %d\n", spi->cs_gpio, status);
+ if (status)
+ goto done;
+ gpio_direction_output(spi->cs_gpio, !(spi->mode & SPI_CS_HIGH));
+ }
+ }
drivers/spi/spi-mt7621.c的mt7621_spi_set_cs函数中:1
2
3
4
5
6
7
8
9
10
11
12- mt7621_spi_reset(rs, cs);
- if (enable)
- polar = BIT(cs);
- mt7621_spi_write(rs, MT7621_SPI_POLAR, polar);
+ mt7621_spi_reset(rs, cs);
+ if (spi->cs_gpio >= 0) {
+ gpio_set_value_cansleep(spi->cs_gpio, (spi->mode & SPI_CS_HIGH) ? enable : !enable);
+ } else {
+ if (enable)
+ polar = BIT(cs);
+ mt7621_spi_write(rs, MT7621_SPI_POLAR, polar);
+ }
四、注意事项
因使用GPIO扩展SPI 的CS,注意上电过程该管脚的默认电平,以免与flash等其他spi设备冲突。