MT7688 dts 和pinmux 关系

本文详细解析了MT7628平台的GPIO Pinmux配置,包括PWM、UART、I2C等外设的功能选择及引脚分配。深入介绍了不同功能对应的引脚模式设置。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

文件路径
1./build_dir/target-mipsel_24kec+dsp_uClibc-0.9.33.2/linux-ramips_mt7688/linux-3.18.29/arch/mips/include/asm/mach-ralink

pinmux.h

#ifndef _RT288X_PINMUX_H__
#define _RT288X_PINMUX_H__

//name=节点名字,value=节点值 pin_first=开始pin,pin_count=有几个pin
#define FUNC(name, value, pin_first, pin_count) { name, value, pin_first, pin_count }

#define GRP(_name, _func, _mask, _shift) \
	{ .name = _name, .mask = _mask, .shift = _shift, \
	  .func = _func, .gpio = _mask, \
	  .func_count = ARRAY_SIZE(_func) }

#define GRP_G(_name, _func, _mask, _gpio, _shift) \
	{ .name = _name, .mask = _mask, .shift = _shift, \
	  .func = _func, .gpio = _gpio, \
	  .func_count = ARRAY_SIZE(_func) }

struct rt2880_pmx_group;

struct rt2880_pmx_func {
	const char *name;
	const char value;

	int pin_first;
	int pin_count;
	int *pins;

	int *groups;
	int group_count;

	int enabled;
};

struct rt2880_pmx_group {
	const char *name;
	int enabled;

	const u32 shift;
	const char mask;
	const char gpio;

	struct rt2880_pmx_func *func;
	int func_count;
};

2.build_dir/target-mipsel_24kec+dsp_uClibc-0.9.33.2/linux-ramips_mt7688/linux-3.18.29/arch/mips/ralink
mt7620.c

static struct rt2880_pmx_func pwm1_grp_mt7628[] = {
	FUNC("sdxc d6", 3, 19, 1),
	FUNC("utif", 2, 19, 1),
	FUNC("gpio", 1, 19, 1),
	FUNC("pwm1", 0, 19, 1),
};

static struct rt2880_pmx_func pwm0_grp_mt7628[] = {
	FUNC("sdxc d7", 3, 18, 1),
	FUNC("utif", 2, 18, 1),
	FUNC("gpio", 1, 18, 1),
	FUNC("pwm0", 0, 18, 1),
};

static struct rt2880_pmx_func uart2_grp_mt7628[] = {
	FUNC("sdxc d5 d4", 3, 20, 2),
	FUNC("pwm", 2, 20, 2),
	FUNC("gpio", 1, 20, 2),
	FUNC("uart2", 0, 20, 2),
};

static struct rt2880_pmx_func uart1_grp_mt7628[] = {
	FUNC("sw_r", 3, 45, 2),
	FUNC("pwm", 2, 45, 2),
	FUNC("gpio", 1, 45, 2),
	FUNC("uart1", 0, 45, 2),
};

static struct rt2880_pmx_func i2c_grp_mt7628[] = {
	FUNC("-", 3, 4, 2),
	FUNC("debug", 2, 4, 2),
	FUNC("gpio", 1, 4, 2),
	FUNC("i2c", 0, 4, 2),
};

static struct rt2880_pmx_func refclk_grp_mt7628[] = { FUNC("reclk", 0, 37, 1) };
static struct rt2880_pmx_func perst_grp_mt7628[] = { FUNC("perst", 0, 36, 1) };
static struct rt2880_pmx_func wdt_grp_mt7628[] = { FUNC("wdt", 0, 38, 1) };
static struct rt2880_pmx_func spi_grp_mt7628[] = { FUNC("spi", 0, 7, 4) };

static struct rt2880_pmx_func sd_mode_grp_mt7628[] = {
	FUNC("jtag", 3, 22, 8),
	FUNC("utif", 2, 22, 8),
	FUNC("gpio", 1, 22, 8),
	FUNC("sdxc", 0, 22, 8),
};

static struct rt2880_pmx_func uart0_grp_mt7628[] = {
	FUNC("-", 3, 12, 2),
	FUNC("-", 2, 12, 2),
	FUNC("gpio", 1, 12, 2),
	FUNC("uart0", 0, 12, 2),
};

static struct rt2880_pmx_func i2s_grp_mt7628[] = {
	FUNC("antenna", 3, 0, 4),
	FUNC("pcm", 2, 0, 4),
	FUNC("gpio", 1, 0, 4),
	FUNC("i2s", 0, 0, 4),
};

static struct rt2880_pmx_func spi_cs1_grp_mt7628[] = {
	FUNC("-", 3, 6, 1),
	FUNC("refclk", 2, 6, 1),
	FUNC("gpio", 1, 6, 1),
	FUNC("spi cs1", 0, 6, 1),
};

static struct rt2880_pmx_func spis_grp_mt7628[] = {
	FUNC("pwm", 3, 14, 4),
	FUNC("util", 2, 14, 4),
	FUNC("gpio", 1, 14, 4),
	FUNC("spis", 0, 14, 4),
};

static struct rt2880_pmx_func gpio_grp_mt7628[] = {
	FUNC("pcie", 3, 11, 1),
	FUNC("refclk", 2, 11, 1),
	FUNC("gpio", 1, 11, 1),
	FUNC("gpio", 0, 11, 1),
};

struct rt2880_pmx_func wled_an_grp_mt7628[] = {
        FUNC("rsvd", 3, 44, 1),
        FUNC("rsvd", 2, 44, 1),
        FUNC("gpio", 1, 44, 1),
        FUNC("wled_an", 0,44, 1),
};


static struct rt2880_pmx_func ephy_p1_grp_mt7628[] = {
        FUNC("rsvd", 3, 42, 1),
        FUNC("rsvd", 2, 42, 1),
        FUNC("gpio", 1, 42, 1),
        FUNC("ephy", 0, 42, 1),
};
static struct rt2880_pmx_func ephy_p2_grp_mt7628[] = {
        FUNC("rsvd", 3, 41, 1),
        FUNC("rsvd", 2, 41, 1),
        FUNC("gpio", 1, 41, 1),
        FUNC("ephy", 0, 41, 1),
};
static struct rt2880_pmx_func ephy_p3_grp_mt7628[] = {
        FUNC("rsvd", 3, 40, 1),
        FUNC("rsvd", 2, 40, 1),
        FUNC("gpio", 1, 40, 1),
        FUNC("ephy", 0, 40, 1),
};
static struct rt2880_pmx_func ephy_p4_grp_mt7628[] = {
        FUNC("rsvd", 3, 39, 1),
        FUNC("rsvd", 2, 39, 1),
        FUNC("gpio", 1, 39, 1),
        FUNC("ephy", 0, 39, 1),
};

#define MT7628_GPIO_MODE_MASK		0x3
#define MT7628_GPIO_MODE_EPHY_P4        42
#define MT7628_GPIO_MODE_EPHY_P3        40
#define MT7628_GPIO_MODE_EPHY_P2        38
#define MT7628_GPIO_MODE_EPHY_P1        36
#define MT7628_GPIO_MODE_WLED_AN	32
#define MT7628_GPIO_MODE_PWM1		30
#define MT7628_GPIO_MODE_PWM0		28
#define MT7628_GPIO_MODE_UART2		26
#define MT7628_GPIO_MODE_UART1		24
#define MT7628_GPIO_MODE_I2C		20
#define MT7628_GPIO_MODE_REFCLK		18
#define MT7628_GPIO_MODE_PERST		16
#define MT7628_GPIO_MODE_WDT		14
#define MT7628_GPIO_MODE_SPI		12
#define MT7628_GPIO_MODE_SDMODE		10
#define MT7628_GPIO_MODE_UART0		8
#define MT7628_GPIO_MODE_I2S		6
#define MT7628_GPIO_MODE_CS1		4
#define MT7628_GPIO_MODE_SPIS		2
#define MT7628_GPIO_MODE_GPIO		0

static struct rt2880_pmx_group mt7628an_pinmux_data[] = {
	GRP_G("pwm1", pwm1_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_PWM1),
	GRP_G("pwm0", pwm0_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_PWM0),
	GRP_G("uart2", uart2_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_UART2),
	GRP_G("uart1", uart1_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_UART1),
	GRP_G("i2c", i2c_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_I2C),
	GRP("refclk", refclk_grp_mt7628, 1, MT7628_GPIO_MODE_REFCLK),
	GRP("perst", perst_grp_mt7628, 1, MT7628_GPIO_MODE_PERST),
	GRP("wdt", wdt_grp_mt7628, 1, MT7628_GPIO_MODE_WDT),
	GRP("spi", spi_grp_mt7628, 1, MT7628_GPIO_MODE_SPI),
	GRP_G("sdmode", sd_mode_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_SDMODE),
	GRP_G("uart0", uart0_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_UART0),
	GRP_G("i2s", i2s_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_I2S),
	GRP_G("spi cs1", spi_cs1_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_CS1),
	GRP_G("spis", spis_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_SPIS),
	GRP_G("gpio", gpio_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_GPIO),
	GRP_G("wled_an", wled_an_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_WLED_AN),
        GRP_G("ephy_p1", ephy_p1_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_EPHY_P1),
        GRP_G("ephy_p2", ephy_p2_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_EPHY_P2),
        GRP_G("ephy_p3", ephy_p3_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_EPHY_P3),
        GRP_G("ephy_p4", ephy_p4_grp_mt7628, MT7628_GPIO_MODE_MASK, 1, MT7628_GPIO_MODE_EPHY_P4),

	{ 0 }
};

 

设备树(device tree)机制是Linux内核从linux-3.x版本开始引进的一种机制,目的是解决内核源码的arch/arm目录下代码混乱的问题:随着ARM生态的快速发展,在内核源码的arch/arm目录下,存放着几十种arm芯片几百个开发板相关的源文件,很多开发板处理器的中断、寄存器等相关硬件资源都在这个目录下以.c或.h的文件格式定义。而对于内核来说,与这些硬件耦合,会导致内核代码混乱不堪,每个开发板上运行的内核镜像都必须单独编译配置,无法通用。什么时候Linux内核能像Windows镜像那样,无论你的电脑什么配置,一个Windows安装包,都可以直接下载安装运行呢?设备树机制,实现了Linux内核硬件平台的解耦:每个硬件平台的硬件资源使用一个设备树文件(xxx.dts)来描述,而不是在arch/arm下以.c 或 .h 文件来定义。Linux内核是一个通用的内核,在启动过程中,在通过解析设备树中的硬件资源来初始化某个具体的平台。 引入设备树后,很多内核驱动开发的工作也发生了变化:以往驱动工程师关注的头文件宏定义、寄存器定义,现在这些基本上不用关注,关注的重点则转向了如何根据硬件平台去配置修改设备树文件。很多驱动的编程接口也发生了变化,开始慢慢使用device tree提供的编程接口去开发驱动。本期课程主要面向嵌入式开发人员,分享Linux下驱动开发所需要的设备树知识必备技能
<think>好的,用户的问题是关于在MT7981芯片上通过DTS配置将四个GPIO引脚设置为SPI功能。我需要先理解用户的需求,然后结合提供的引用资料来解答。 首先,用户提到了MT7981芯片,这应该是联发科的一款处理器,常用于路由器等嵌入式设备。DTS(设备树源文件)用于描述硬件配置,特别是在Linux内核中。用户想配置四个GPIO作为SPI使用,这通常涉及到引脚复用(pinctrl)SPI控制器的设置。 根据引用内容,引用[3]提到了MT7688/MT7628的GPIO复用配置SPI的DTS配置,这可能与MT7981类似,但需要确认MT7981的具体情况。另外,引用[1]中有一个DTS示例,里面提到了leds的配置,虽然不直接相关,但展示了DTS的结构。引用[2]提到了GPIO-LEDS的配置,可能涉及到GPIO的编号方式,比如使用&gpio1这样的节点。 首先,SPI通常需要四个引脚:CS(片选)、CLK(时钟)、MOSI(主出从入)、MISO(主入从出)。所以用户需要配置这四个GPIO为SPI功能。 步骤可能包括: 1. 确定要使用的GPIO引脚编号,并检查它们是否支持SPI复用。 2. 在DTS中配置pinctrl,将这四个GPIO复用为SPI功能。 3. 配置SPI控制器节点,引用这些pinctrl设置。 4. 可能需要处理片选(CS)引脚,特别是如果有多个CS的话。 引用[3]中提到,SPI_CS1的配置需要修改pinctrl,并注意pinctrl-0中的设置。例如,使用系统SPI控制器时,需要配置CS1为SPI_CS1功能,并调整pinctrl设置。这可能涉及到在pinctrl节点中定义SPI相关的引脚组。 另外,GPIO的编号方式可能像引用[2]中提到的,使用类似&pio 35 0的格式,其中35是GPIO号,0可能表示激活电平。但在SPI配置中,可能需要使用不同的属性,比如functiongroups来设置复用功能。 需要注意的是,不同芯片的GPIO复用方式可能不同,所以必须参考MT7981的具体文档。但用户提供的引用中没有MT7981的具体SPI配置,只有MT7688/MT7628的例子,因此可能需要类比处理。 可能的DTS配置步骤: - 在pinctrl节点中定义SPI的引脚组,设置复用功能为SPI。 - 在SPI控制器节点中引用这些pinctrl设置。 - 配置CS引脚,可能需要单独的GPIO控制,特别是如果使用软件CS。 例如,在MT7688中(引用[3]),SPI的配置包括pinctrl-0pinctrl-names,以及设置CS引脚。对于MT7981,可能需要类似的配置,但具体GPIO编号复用功能需要调整。 总结,用户需要: 1. 确定四个GPIO对应的SPI功能引脚。 2. 在DTS的pinctrl部分配置这些引脚为SPI功能。 3. 在SPI控制器节点中启用并配置这些引脚。 4. 处理片选信号,可能需要额外的GPIO配置。 可能遇到的问题包括GPIO编号是否正确,复用功能名称是否正确,以及是否与其他功能冲突。需要确保没有其他节点使用了相同的GPIO,并且pinctrl配置正确。 最后,生成相关问题,可能涉及SPI的其他配置、调试方法或性能优化。</think>### MT7981芯片DTS配置SPI功能分步说明 #### 1. 确定GPIO复用关系 首先需查阅MT7981芯片手册确认四个GPIO对应的SPI功能引脚: - **SPI_CLK**:时钟信号线 - **SPI_MOSI**:主设备输出从设备输入 - **SPI_MISO**:主设备输入从设备输出 - **SPI_CS**:片选信号线(可配置多个) 示例假设使用GPIO35作为SPI_CLK,GPIO36作为MOSI,GPIO37作为MISO,GPIO38作为CS0(具体编号需根据实际芯片手册调整)[^3]。 #### 2. 配置pinctrl复用 在DTS文件中添加引脚控制组: ```dts &pinctrl { spi_pins: spi { pins-spi { pinmux = <GPIO35_FUNC_SPI_CLK>, // 复用功能值需查手册 <GPIO36_FUNC_SPI_MOSI>, <GPIO37_FUNC_SPI_MISO>, <GPIO38_FUNC_SPI_CS0>; bias-disable; // 关闭上下拉 drive-strength = <8>;// 驱动强度设置 }; }; }; ``` #### 3. 配置SPI控制器节点 启用SPI控制器并关联引脚: ```dts &spi0 { status = "okay"; // 启用SPI控制器 pinctrl-names = "default"; // 引脚组名称 pinctrl-0 = <&spi_pins>; // 绑定定义的引脚组 // 配置SPI参数 num-cs = <1>; // 片选信号数量 cs-gpios = <&pio 38 0>; // CS0使用GPIO38 // 挂接SPI设备示例(如Flash) flash@0 { compatible = "jedec,spi-nor"; reg = <0>; spi-max-frequency = <50000000>; }; }; ``` #### 4. 关键配置说明 - **GPIO复用函数**:`GPIOXX_FUNC_XXX`需替换为芯片手册中定义的宏,例如MT7628芯片中`GPIO38`作为SPI_CS1时对应`SPI_CS1`功能[^3] - **片选信号处理**:若使用硬件CS信号需配置`num-cs`参数,若需软件控制CS可增加`cs-gpios`定义 - **驱动强度**:根据实际信号质量调整`drive-strength`参数(单位mA) #### 5. 完整配置示例 ```dts / { compatible = "mediatek,mt7981"; pinctrl { spi_pins: spi { pins-spi { pinmux = <GPIO35_FUNC_SPI_CLK>, <GPIO36_FUNC_SPI_MOSI>, <GPIO37_FUNC_SPI_MISO>, <GPIO38_FUNC_SPI_CS0>; bias-disable; drive-strength = <8>; }; }; }; }; &spi0 { status = "okay"; pinctrl-names = "default"; pinctrl-0 = <&spi_pins>; num-cs = <1>; cs-gpios = <&pio 38 0>; flash@0 { compatible = "jedec,spi-nor"; reg = <0>; spi-max-frequency = <50000000>; }; }; ``` #### 6. 编译验证 执行`make menuconfig`确认SPI驱动已启用,编译固件后通过`dmesg | grep spi`查看设备注册状态,使用逻辑分析仪或示波器验证信号波形。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值