dts文件中的spi節點
&ecspi2{ /* spi控制器節點 */
fsl,spi-num-chipselects= < 1 >;
cs-gpios = <&gpio5 13 0 > ;/* 片選的io口 */
pinctrl-names = "default";
pinctrl-0 = <&pinctrl_ecspi2 >;
status = "okay"; /* status屬性值為"okay" 表示該節點使能*/
spidev@0x00{
compatible = "spidev","rohm,dh2228fv";/* 此屬性值用於與spi設備驅動匹配 */
reg = <0>; /*spi設備是沒有設備地址的, 這里是指使用spi控制器的cs-gpios里的第幾個片選io */
spi-max-frequency = <10000000>; /* 指定spi設備的最大工作時鍾 */
/*以下為自定義屬性 用於指定工作時序方式及其它功能設置等*/
...
buswidth = <8>; /* 傳輸以8位為單位 */
mode = <0>; /* 使用第幾種工作時序(CPOL, CPHA) */
/*但在現用的內核源碼里發現, spi設備的工作時序並不是用mode屬性值來指定的*/
/* 如CPOL需要設1, 則只需在spi設備節點里加上"spi-cpol"屬性即可; CPOL設0,則不寫"spi-cpol"屬性即可 */
/* CPHA設1時, 則在設備節點里加上"spi-cpha"屬性即可 */
};
};
dtsi文件中的設備樹節點
ecspi2: ecspi@30830000 {
compatible = "fsl,imx6ul-ecspi", "fsl,imx51-ecspi";
reg = <0x0 0x30830000 0x0 0x10000>;
interrupts = <GIC_SPI 32 IRQ_TYPE_LEVEL_HIGH>;
clocks = <&clk IMX8MQ_CLK_ECSPI2_ROOT>,
<&clk IMX8MQ_CLK_ECSPI2_ROOT>;
clock-names = "ipg", "per";
interrupt-parent = <&gpc>;
status = "disabled";
};
一般帶spi名稱的節點表示spi控制器, 它會先被轉換為platform_device, 在內核中有對應的platform_driver;(根據compatible屬性來匹配)一般為廠商所配套的platform_driver文件(freescale的處理文件為spi-imx.c),platform_driver的probe函數會調用spi_register_master(), 下面是spi節點在內核中的轉化過程:
spi_imx_probe //drivers/spi/spi-imx.c
spi_bitbang_start
spi_register_master
of_register_spi_devices
for_each_available_child_of_node(ctlr->dev.of_node, nc) {
spi = of_register_spi_device(ctlr, nc); // 讀取設備樹中的spi子節點的屬性
spi = spi_alloc_device(ctlr);
rc = spi_add_device(spi); //添加spi設備
}
下面來重點看一下of_register_spi_device()函數,該函數的主要作用是讀取spi節點內的各種值
static struct spi_device *
of_register_spi_device(struct spi_master *master, struct device_node *nc)
{
struct spi_device *spi;
int rc;
u32 value;
/* Alloc an spi_device 分配一個spi設備*/
spi = spi_alloc_device(master);
if (!spi) {
dev_err(&master->dev, "spi_device alloc error for %s\n",
nc->full_name);
rc = -ENOMEM;
goto err_out;
}
/* Select device driver 獲取 compatibel 屬性 用於匹配spi driver*/
rc = of_modalias_node(nc, spi->modalias,
sizeof(spi->modalias));
if (rc < 0) {
dev_err(&master->dev, "cannot find modalias for %s\n",
nc->full_name);
goto err_out;
}
/* Device address 獲取 reg 屬性作為片選編號*/
rc = of_property_read_u32(nc, "reg", &value);
if (rc) {
dev_err(&master->dev, "%s has no valid 'reg' property (%d)\n",
nc->full_name, rc);
goto err_out;
}
spi->chip_select = value;
/* Mode (clock phase/polarity/etc.) spi mode選擇*/
if (of_find_property(nc, "spi-cpha", NULL))
spi->mode |= SPI_CPHA;
if (of_find_property(nc, "spi-cpol", NULL))
spi->mode |= SPI_CPOL;
if (of_find_property(nc, "spi-cs-high", NULL))
spi->mode |= SPI_CS_HIGH;
if (of_find_property(nc, "spi-3wire", NULL))
spi->mode |= SPI_3WIRE;
if (of_find_property(nc, "spi-lsb-first", NULL))
spi->mode |= SPI_LSB_FIRST;
/* Device DUAL/QUAD mode */
if (!of_property_read_u32(nc, "spi-tx-bus-width", &value)) {
switch (value) {
case 1:
break;
case 2:
spi->mode |= SPI_TX_DUAL;
break;
case 4:
spi->mode |= SPI_TX_QUAD;
break;
default:
dev_warn(&master->dev,
"spi-tx-bus-width %d not supported\n",
value);
break;
}
}
if (!of_property_read_u32(nc, "spi-rx-bus-width", &value)) {
switch (value) {
case 1:
break;
case 2:
spi->mode |= SPI_RX_DUAL;
break;
case 4:
spi->mode |= SPI_RX_QUAD;
break;
default:
dev_warn(&master->dev,
"spi-rx-bus-width %d not supported\n",
value);
break;
}
}
/* Device speed spi速度設置*/
rc = of_property_read_u32(nc, "spi-max-frequency", &value);
if (rc) {
dev_err(&master->dev, "%s has no valid 'spi-max-frequency' property (%d)\n",
nc->full_name, rc);
goto err_out;
}
spi->max_speed_hz = value;
/* Store a pointer to the node in the device structure */
of_node_get(nc); //保存設備樹節點
spi->dev.of_node = nc;
/* Register the new device 注冊新的spi設備*/
rc = spi_add_device(spi);
if (rc) {
dev_err(&master->dev, "spi_device register error %s\n",
nc->full_name);
goto err_of_node_put;
}
return spi;
err_of_node_put:
of_node_put(nc);
err_out:
spi_dev_put(spi);
return ERR_PTR(rc);
}
生成了spi設備之后,會使用spi_match_device()匹配對應的spi driver。匹配成功之后生成對應的spi設備節點。
補充幾點最近的發現:(針對於freescale im8mq 內核版本:linux-4.9.88)
1、spi節點下不能定義兩個使用同一個片選信號的節點,如果有相同的節點會出現其中一個結點無效的情況。(未具體分析源碼)
2、在spi節點下可聲明除spi相關節點之外的節點,gpio的初始化需要放入spi根節點的pinctrl中。並且需要注意使用的io管腳不能在其他pinctrl節點中出現,不然會造成節點初始化失敗的情況。
pinctrl_ecspi2: ecspi2grp {
fsl,pins = <
MX8MQ_IOMUXC_ECSPI2_SS0_GPIO5_IO13 0x16
MX8MQ_IOMUXC_ECSPI2_MOSI_ECSPI2_MOSI 0x16
MX8MQ_IOMUXC_ECSPI2_MISO_ECSPI2_MISO 0x16
MX8MQ_IOMUXC_ECSPI2_SCLK_ECSPI2_SCLK 0x1816
MX8MQ_IOMUXC_NAND_RE_B_GPIO3_IO15 0x16
>;
};
pinctrl_uart3: uart3grp {
fsl,pins = <
MX8MQ_IOMUXC_UART3_TXD_UART3_DCE_TX 0x49 /* UART3_TXD */
MX8MQ_IOMUXC_UART3_RXD_UART3_DCE_RX 0x49 /* UART3_RXD */
MX8MQ_IOMUXC_ECSPI1_MISO_UART3_DCE_CTS_B 0x49 /* ECSPI1_MISO UART3_CTS */
MX8MQ_IOMUXC_ECSPI1_SS0_UART3_DCE_RTS_B 0x49 /* ECSPI1_SS0 UART3_RTS */
//MX8MQ_IOMUXC_NAND_RE_B_GPIO3_IO15 0x56 /*需要屏蔽此處相同的管腳*/
MX8MQ_IOMUXC_GPIO1_IO00_ANAMIX_REF_CLK_32K 0x14 /* REF_CLK_32K */
>;
};
3、在spi設備匹配時會自動處理節點中的中斷號,不需要自己處理了(下列函數位於driver/spi/spi.c)
static int spi_drv_probe(struct device *dev)
{
const struct spi_driver *sdrv = to_spi_driver(dev->driver);
struct spi_device *spi = to_spi_device(dev);
int ret;
ret = of_clk_set_defaults(dev->of_node, false);
if (ret)
return ret;
if (dev->of_node) {
spi->irq = of_irq_get(dev->of_node, 0); //獲取中斷號
if (spi->irq == -EPROBE_DEFER)
return -EPROBE_DEFER;
if (spi->irq < 0)
spi->irq = 0;
}
ret = dev_pm_domain_attach(dev, true);
if (ret != -EPROBE_DEFER) {
ret = sdrv->probe(spi);
if (ret)
dev_pm_domain_detach(dev, true);
}
return ret;
}
————————————————
版權聲明:本文為CSDN博主「弋陽yoga」的原創文章,遵循CC 4.0 BY-SA版權協議,轉載請附上原文出處鏈接及本聲明。
原文鏈接:https://blog.csdn.net/qq_17270067/article/details/106993990