使用 ATTiny 和 Atmel Studio 处理 32 位数字
32 bit number handling with ATTiny and Atmel Studio
我想知道 ATTiny,特别是 ATTiny24 如何在内存中存储 32 位无符号整数。我正在尝试获取 32 位值并将其写入 EEPROM 中的 32 位位置。我曾尝试使用一个简单的掩码,但每次我尝试时,我都会正确地获得低两个字节 (lsb),而高两个字节全为零。例如,当我尝试写入:0x12345678 时,输出为:0x00005678。 Atmel Studio 中是否有我需要设置的设置,或者我需要使用屏蔽以外的其他方法。
最终我希望能够读取一个 32 位计数器值并将其写入 EEPROM 中的特定位置。我正在修改现有电路,因此无法使用串行输出进行调试。
代码片段:
主线:
unsigned long test_val = 305419896; //0x12345678
EEprom_Long_Write(0x25,test_val);
函数:
EEprom_Long_Write:
void EEprom_Long_Write(unsigned char eeadr, unsigned long EE_Data)
{
unsigned char temp=0;
unsigned char count= eeadr + 3;
unsigned long mask=0;
unsigned char position=24;
while (eeadr <= count)
{
mask = ((1<<8)-1) << position;
temp = (EE_Data & mask) >> position;
EEPROM_write(eeadr, temp);
position = position-8;
eeadr++;
}
}
EEPROM_write:
void EEPROM_write(unsigned char ucAddress, unsigned char ucData)
{
while(EECR & (1<<EEPE)); //Wait for completion of previous write
EECR =(0<<EEPM1) | (0>>EEPM0); // Set Programming mode
EEARL = ucAddress; // Setup address and data registers
EEDR = ucData; // Load Data Register
EECR |= (1<<EEMPE); // Write logical one to EEMPE
EECR |= (1<<EEPE); // Start eeprom write be setting EEPE
}
你掉进了integer promotion and coercion的坑里。 ((1<<8)-1)
被处理为 int
,而不是 long
(更好:uint32_t
)。在 AVR 上,int
具有标准允许的最小大小:16 位。
反正就是太复杂了,可以用下面的:
uint8_t shift = 32U; // enough is enough
do {
shift -= 8U;
EEPROM_write(eeadr++, (uint8_t)(EE_Data >> shift));
} while ( shift ) ;
这为您节省了一个额外的班次以及显式屏蔽和一些寄存器。
请注意我使用的 stdint.h
类型(当然,您必须包括 header)。您应该相应地更正所有声明。转换为 uint8_t
意味着屏蔽。
改变这个:
mask = ((1<<8)-1) << position;
temp = (EE_Data & mask) >> position;
对此:
temp = (EE_Data >> position) & 0xFF;
我想知道 ATTiny,特别是 ATTiny24 如何在内存中存储 32 位无符号整数。我正在尝试获取 32 位值并将其写入 EEPROM 中的 32 位位置。我曾尝试使用一个简单的掩码,但每次我尝试时,我都会正确地获得低两个字节 (lsb),而高两个字节全为零。例如,当我尝试写入:0x12345678 时,输出为:0x00005678。 Atmel Studio 中是否有我需要设置的设置,或者我需要使用屏蔽以外的其他方法。
最终我希望能够读取一个 32 位计数器值并将其写入 EEPROM 中的特定位置。我正在修改现有电路,因此无法使用串行输出进行调试。
代码片段:
主线:
unsigned long test_val = 305419896; //0x12345678
EEprom_Long_Write(0x25,test_val);
函数:
EEprom_Long_Write:
void EEprom_Long_Write(unsigned char eeadr, unsigned long EE_Data)
{
unsigned char temp=0;
unsigned char count= eeadr + 3;
unsigned long mask=0;
unsigned char position=24;
while (eeadr <= count)
{
mask = ((1<<8)-1) << position;
temp = (EE_Data & mask) >> position;
EEPROM_write(eeadr, temp);
position = position-8;
eeadr++;
}
}
EEPROM_write:
void EEPROM_write(unsigned char ucAddress, unsigned char ucData)
{
while(EECR & (1<<EEPE)); //Wait for completion of previous write
EECR =(0<<EEPM1) | (0>>EEPM0); // Set Programming mode
EEARL = ucAddress; // Setup address and data registers
EEDR = ucData; // Load Data Register
EECR |= (1<<EEMPE); // Write logical one to EEMPE
EECR |= (1<<EEPE); // Start eeprom write be setting EEPE
}
你掉进了integer promotion and coercion的坑里。 ((1<<8)-1)
被处理为 int
,而不是 long
(更好:uint32_t
)。在 AVR 上,int
具有标准允许的最小大小:16 位。
反正就是太复杂了,可以用下面的:
uint8_t shift = 32U; // enough is enough
do {
shift -= 8U;
EEPROM_write(eeadr++, (uint8_t)(EE_Data >> shift));
} while ( shift ) ;
这为您节省了一个额外的班次以及显式屏蔽和一些寄存器。
请注意我使用的 stdint.h
类型(当然,您必须包括 header)。您应该相应地更正所有声明。转换为 uint8_t
意味着屏蔽。
改变这个:
mask = ((1<<8)-1) << position;
temp = (EE_Data & mask) >> position;
对此:
temp = (EE_Data >> position) & 0xFF;