一、概述
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文件
spi@b00 {
status = "okay";
pinctrl-names = "default";
pinctrl-0 = <&spi_pins>, <&spi_cs1_pins>;
cs-gpios = <0>, <0>, <&gpio1 5 1>;
m25p80@0 {
#address-cells = <1>;
#size-cells = <1>;
compatible = "w25q256";
reg = <0 0>;
linux,modalias = "m25p80", "w25q256";
spi-max-frequency = <40000000>;
m25p,chunked-io = <31>;
... ...
};
spidev@1 {
#address-cells = <1>;
#size-cells = <1>;
compatible = "spidev";
reg = <1>;
spi-max-frequency = <40000000>;
};
spidev@2 {
#address-cells = <1>;
#size-cells = <1>;
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函數中:
- 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函數中:
- 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設備衝突。