先看关于寄存器描述。
写着IOPIN为只读寄存器。
但其实可以向IOPIN里写数据,如http://home.eeworld.com.cn/my/space-uid-440873-blogid-77805.html 所示。
可以使用IOPIN使GPIO管脚上输出瞬时的0和1
先写IOSET再写IOCLR寄存器可使管脚先输出1再输出0。有的系统允许两个有效输出之间存在这段
延时时间。但某些应用要求一个GPIO口的一组管脚同时输出一个二进制数(0和1混合)。这可通过写端
口的IOPIN寄存器来实现。
下面的代码所实现的功能是:P0.[31:16]和P0.[7:0]输出保持不变的同时将P0.[15:8]设置成0xA5,不管
P0.[15:8]之前是何值:
IO0PIN=(IO0PIN && # 0xFFFF00FF)‖# 0x0000A500
但有陷阱,如http://blog.pfan.cn/niao0311/38996.html 所示。
下面是解释:http://bbs.zlgmcu.com/dv_rss.asp?s=xhtml&boardid=29&id=18359&page=3
“先写IOSET再写IOCLR寄存器可使管脚先输出1再输出0。有的系统允许两个有效输出之间存在这段延时时间。但某此应用要求一个GPIO口的一组管脚同时输出一个二进制(0和1混合)。这可通过写端口的IOPIN寄存器来实现(但是,Philips不推荐这种用法,非必要时不要使用)。”
以上是《深入浅出ARM7-LPC213x_214x(上册)》P138中的一段文字。
问题是:既然使用IOPIN能够避免使用IOSET/IOCLR 输出1/0出现不必要的中间态,为什么Philips不推荐这种用法?非必要时不要使用,那在何时才用IOPIN的方法。
回答:非必须时不要使用,是因为IOPIN是没有位屏蔽操作的,即改变这个寄存器32个引脚都可能会发生改变,所以必须要用与或的操作实现引脚状态的改变,但与或操作即读改写有一问题,就是读改写被打断时(可能被中断或任务打断)时,如打断后IOPIN的值改了,那再写回去的值就不正确了,所以应用时要十分注意。