2020-10-21

在9.1和9.2章节我们介绍了这两个寄存器“SW_MUX_CTL_PAD_XX_XX”和“SW_PAD_CTL_PAD_XX_XX”用来配置IO引脚,在本章我们来学习一下GPIO功能的配置(GPIO属于IO引脚中的一种复用功能)。比如GPIO1_IO00这个IO可以复用为I2C2_SCL、ENET1_REF_CLK1、GPIO1_IO00、WDOG3_WDOG_B等9种复用功能,GPIO1_IO00只是其中的一种,具体配置成哪种复用功能,需要看下我们的硬件设计中这个IO是作为哪种功能来设计的。如果我们把该引脚用来控制LED发光二极管,那我们的程序就要把这个IO配置成GPIO模式,然后我们还需要对GPIO的功能进行配置,我们可以参考手册的第28章“General Purpose Input/Output (GPIO)”。在该章节我们可以看到GPIO的结构如图 1所示:

在这里插入图片描述

图 1
在上图中我们可以看到两个地方用红色方框标注了(1、2),其中1处里面有两个寄存器,这就是我们6.2章节介绍的配置IO复用和IO功能属性的寄存器。2处表示当IO作为GPIO使用的时候,需要配置的寄存器(共有8个),分别是:GPIO.DR、GPIO.GDIR、GPIO.PSR、GPIO.ICR1、GPIO.ICR2、GPIO.EDGE_SEL、GPIO.IMR、GPIO.ISR,前面6.2章节我们介绍了i.MX6 ULL一共有5组GPIO,每组GPIO分别有这8个寄存器,下面我们分别看下这些寄存器:
首先是GPIOx_DR寄存器,如图 2所示:

在这里插入图片描述

图 2
此寄存器是数据寄存器,32位,每一位对应一个GPIO,当GPIO配置成输出以后,向对应的位写1,GPIO就会输出高电平,写0,GPIO就会输出低电平。如果GPIO设置成输入,那读取这个寄存器对应的位,就可以获取到对应GPIO的状态(0或1)。

然后是GPIOx_GDIR寄存器,如图 3所示:

在这里插入图片描述

图 3
该寄存器也是32位的,每一位对应一个GPIO,该寄存器是用来设置GPIO是输入还是输出的。(对应的位设置成0,对应的GPIO设置成输入模式;对应的位设置成1,对应的GPIO就配置成输出模式了)。

然后是GPIOx_PSR寄存器,如图 4所示:

在这里插入图片描述

图 4
该寄存器也是32位的,每一位对应一个GPIO,该寄存器是用来读取对应GPIO的状态(高低电平)。

然后是GPIOx_ICR1寄存器,如图 5所示:

在这里插入图片描述

图 5
该寄存器是中断控制寄存器,每组GPIO最多有32个GPIO,该寄存器用来配置低16个GPIO,此寄存器是32位的,每两位表示一个GPIO,这两位用来配置中断的触发方式:
00 低电平出发
01 高电平触发
10 上升沿触发
11 下降沿触发
以GPIO1_IO3为例,如果设置成高电平触发,GPIO1.ICR1=1<<6。

然后是GPIOx_ICR2寄存器,如图 6所示:

在这里插入图片描述

图 6
该寄存器也是中断控制寄存器,每组GPIO最多有32个GPIO,该寄存器用来配置高16个GPIO,此寄存器是32位的,每两位表示一个GPIO,这两位用来配置中断的触发方式:
00 低电平出发
01 高电平触发
10 上升沿触发
11 下降沿触发
以GPIO1_IO7为例,如果设置成高电平触发,GPIO1.ICR1=1<<2。

然后是GPIOx_IMR寄存器,如图 7所示:

在这里插入图片描述

图 7
该寄存器是中断屏蔽寄存器,每一位对应一个GPIO,如果使能某个GPIO的中断,那么设置这个寄存器对应的位为1即可。如果禁止某个GPIO的中断,那么设置这个寄存器的对应位为0。

然后是GPIOx_ISR寄存器,如图 8所示:

在这里插入图片描述

图 8
该寄存器是中断状态寄存器,共有32位,每位对应一个gpio,只要某个GPIO的中断产生,那么对应的位就会被置1,我们可以通过读取该寄存器来判断GPIO的中断是否产生了。当我们处理完中断后,必须要清除对应的中断标志位(像该寄存器相应的位写1,就是清除中断标志位)。

然后是GPIOx_EDGE_SEL,如图 9所示:

在这里插入图片描述

图 9
该寄存器用来设置边沿中断,这个寄存器会覆盖ICR1和ICR2的设置,同样一个GPIO对应一位。相应的位被置1,那么就相当于设置了对应的GPIO是上升沿和下降沿(双边沿)触发。

至此关于GPIO的所有寄存器我们就介绍完了。

你遇到的问题是: > **SPI 输出的时间尺度不对,比如 `window_start` 和 `window_end` 的时间跨度看起来不是30天。** 我们来看你的输出示例: ``` dekad_start window_start window_end 2020-01-21 2019-12-23 2020-01-31 ``` 这里 `window_start = 2019-12-23`, `window_end = 2020-01-31`,这显然不是一个30天的窗口,而是一个月的跨度,说明你的 **SPI窗口定义逻辑出现了问题**。 --- ## ❗问题定位:窗口定义错误 你在代码中这样定义窗口: ```python window_end = dekad_start window_start = dekad_start - pd.Timedelta(days=29) ``` 但你在筛选窗口数据时却用了: ```python window_data = df[(df['date'] > window_start) & (df['date'] <= dekad_end)] ``` ### ✅ 正确逻辑应是: ```python window_data = df[(df['date'] >= window_start) & (df['date'] <= window_end)] ``` 因为你定义了: ```python window_end = dekad_start window_start = dekad_start - pd.Timedelta(days=29) ``` 所以窗口应该是: ``` [dekad_start - 29天, dekad_start] ``` 但是你用了 `dekad_end` 作为窗口结束时间,而 `dekad_end` 是 **每旬的结束时间(10日、20日、月底)**,这会导致窗口被错误地扩展到整旬结束日,而不是当前旬的开始日。 --- ## ✅ 正确代码应为: ```python window_end = dekad_start window_start = dekad_start - pd.Timedelta(days=29) window_data = df[(df['date'] >= window_start) & (df['date'] <= window_end)] ``` 而不是: ```python window_data = df[(df['date'] > window_start) & (df['date'] <= dekad_end)] ``` --- ## 🧪 示例说明 以 `dekad_start = 2020-01-21` 为例: ```python window_start = 2020-01-21 - 29天 = 2019-12-23 window_end = 2020-01-21 ``` 正确的窗口是: ``` [2019-12-23, 2020-01-21] # 共30天 ``` 如果你用了 `dekad_end = 2020-01-31`,那么窗口变成了: ``` [2019-12-23, 2020-01-31] # 39天,错误 ``` --- ## ✅ 修改建议(修复窗口筛选逻辑) 将: ```python window_data = df[(df['date'] > window_start) & (df['date'] <= dekad_end)] ``` 改为: ```python window_data = df[(df['date'] >= window_start) & (df['date'] <= window_end)] ``` 同时确保 `window_end = dekad_start` 是正确的。 --- ## ✅ 建议添加调试打印语句 你可以在循环中加入调试信息,帮助你检查窗口是否正确: ```python print(f"dekad_start: {dekad_start}, window_start: {window_start}, window_end: {window_end}") ``` 输出示例: ``` dekad_start: 2020-01-21 00:00:00, window_start: 2019-12-23 00:00:00, window_end: 2020-01-21 00:00:00 ``` 确保 `window_start` 到 `window_end` 确实是30天。 --- ## ✅ 总结 | 问题 | 原因 | 修复方法 | |------|------|-----------| | 时间窗口跨度错误 | 使用了 `dekad_end` 而非 `window_end` | 改为使用 `window_start` 到 `window_end` | | 数据边界错误 | 使用了 `>` 而非 `>=` | 改为 `>= window_start` | | 窗口天数不对 | 没有检查窗口是否正好30天 | 添加调试输出 | --- ##
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值