其他
漫谈C变量——优化天敌“volatile”
【说在前面的话】
【正文】
static volatile uint32_t s_wVPort = 0;
void set_vport_u8(uint8_t chValue, uint8_t chOffset)
{
uint32_t wMask = 0xFF <<chOffset; //!<获取正确的掩码
s_wVPort &= ~wMask; //!<步骤1:将掩码对应的位置清零
s_wVPort |= ((uint32_t)chValue<<chOffset); //!<步骤2:设置新值到虚拟端口
}
void set_vport_u8(uint8_t chValue, uint8_t chOffset)
{
uint32_t wMask = 0xFF <<chOffset; //!<获取正确的掩码
//! s_wVPort&= ~wMask; 的等效展开
uint32_t wTemp1 = s_wVPort; //!<步骤1.1 读取s_wVPort
wTemp1 &= ~wMask; //!<步骤1.2 改写wTemp1
s_wVPort = wTemp1; //!<步骤1.3 将wTemp1写回s_wVPort
//! s_wVPort |= ((uint32_t)chValue<<chOffset);的等效展开
uint32_t wTemp1 = s_wVPort; //!<步骤2.1 读取s_wVPort
wTemp1 |= ((uint32_t)chValue<<chOffset); //!<步骤2.2 改写wTemp1
s_wVPort = wTemp1; //!<步骤2.3 将wTemp1写回s_wVPort
}
void set_vport_u8(uint8_t chValue, uint8_t chOffset)
{
uint32_t wMask = 0xFF <<chOffset; //!<获取正确的掩码
//! 将s_wVPort读取到通用寄存器中(wTemp1编译器会用通用寄存器来保存)
uint32_t wTemp1 = s_wVPort; //!<步骤1.1 读取s_wVPort
//! 对保存在通用寄存器中的值进行统一修改
wTemp1 &= ~wMask; //!<步骤1.2 改写wTemp1
wTemp1 |= ((uint32_t)chValue<<chOffset); //!<步骤2.2 改写wTemp1
//! 将修改后的值写回s_wVPort
s_wVPort = wTemp1; //!<步骤2.3 将wTemp1写回s_wVPort
}
//!已知某32位外设寄存器的地址为 XXXXX_IO_REG_BASE_ADDRESS,则对应的寄存器可以定义为
#defineXXXXX_IO_REG ( *((volatile uint32_t*)XXXX_IO_REG_BASE_ADDRESS) )
考虑到这种情况,应用中很多针对外设寄存器的连续操作都可以通过“手工窥孔优化”来大幅度提高效率。如果可能(在保证程序逻辑正确的情况下),应该尽可能减少volatile的使用;或者是限制其使用的范围;万不得已的情况下,则应该对volatile参与的运算热点进行“手工窥孔优化”。