- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在为AVR微控制器开发软件。坦白说,现在我只有LED和按钮可以调试。问题是,如果我将字符串文字传递给以下函数:
void test_char(const char *str) {
if (str[0] == -1)
LED_PORT ^= 1 << 7; /* Test */
}
main()
中的某处
test_char("AAAAA");
str[0]
等于
'A'
。为什么会这样呢?
button
的结构,如下所示:
typedef struct {
int8_t seq[BTN_SEQ_COUNT]; /* The sequence of button */
int8_t seq_count; /* The number of buttons registered */
int8_t detected; /* The detected button */
uint8_t released; /* Whether the button is released
after a hold */
} button;
button btn = {
.seq = {-1, -1, -1},
.detected = -1,
.seq_count = 0,
.released = 0
};
btn.seq_count
却以
-1
开头。
seq_count
设置为
-1
,也不能解释为什么字符串常量中的字符等于
-1
。
void LED_on() {
PORTA = 0x00;
}
void LED_off() {
PORTA = 0xFF;
}
void port_init() {
PORTA = 0xFF;
DDRA |= 0xFF;
}
void test_char(const char* str) {
if (str[0] == -1) {
LED_on();
}
}
void main() {
port_init();
test_char("AAAAA");
while(1) {
}
}
void test_char(const char* str) {
switch(pgm_read_byte(str++)) {
case '\0': return;
case 'A': LED_on(); break;
case 'B': LED_off(); break;
}
}
void main() {
const char* test = "ABABA";
port_init();
test_char(test);
while(1) {
}
}
avr-gcc -v
Using built-in specs.
COLLECT_GCC=avr-gcc
COLLECT_LTO_WRAPPER=/home/carl/Softwares/AVR/libexec/gcc/avr/4.6.4/lto-wrapper
Target: avr
Configured with: ../configure --prefix=/home/carl/Softwares/AVR --target=avr --enable-languages=c,c++ --disable-nls --disable-libssp --with-dwarf2
Thread model: single
gcc version 4.6.4 (GCC)
最佳答案
从头开始重写,以期消除一些混乱。
首先,一些重要背景:
AVR微控制器具有用于RAM和ROM / Flash(“程序存储器”)的单独地址空间。
GCC生成的代码假定所有数据始终位于RAM中。 (较早的版本曾经具有特殊类型,例如prog_char
,它们引用了ROM地址空间中的数据,但是较新的GCC版本不并且也不支持此类数据类型。)
当链接到avr-libc时,链接器添加代码(__do_copy_data
),以将所有初始化的数据从程序存储器复制到RAM。如果您同时安装了avr-gcc和avr-libc软件包,并且使用avr-gcc -Wall -O2 -fomit-frame-pointer -mmcu=AVRTYPE source.c -o binary.elf
之类的东西将源文件编译为程序二进制文件,请使用avr-objcopy
将elf文件转换为固件实用程序支持的格式,您正在针对avr-libc进行链接。
如果使用avr-gcc
仅生成目标文件source.o
,并且使用某些其他实用程序来链接程序并将其上载到微控制器,则可能不会发生从程序存储器到RAM的复制。这取决于您使用的链接器和库。
由于大多数AVR只有几十到几百个字节的可用RAM,因此用完RAM非常非常容易。我不确定avr-gcc和avr-libc是否可以可靠地检测到何时您拥有的初始化数据多于可用的RAM。如果您指定任何包含字符串的数组,则很可能您已经超出了RAM,从而导致出现各种有趣的错误。avr/pgmspace.h
头文件是avr-libc的一部分,并定义了一个宏PROGMEM
,该宏可用于指定仅由采用程序存储器地址(指针)的函数(例如
pgm_read_byte()
。链接器不会将此类变量复制到RAM中,但是编译器也不会告诉您是否使用了错误的变量。
如果同时使用avr-gcc和avr-libc,则建议对所有只读数据使用以下方法:
#include <avr/pgmspace.h>
/*
* Define LED_init(), LED_on(), and LED_off() functions.
*/
void blinky(const char *str)
{
while (1) {
switch (pgm_read_byte(str++)) {
case '\0': return;
case 'A': LED_on(); break;
case 'B': LED_off(); break;
}
/* Add a sleep or delay here,
* or you won't be able to see the LED flicker. */
}
}
static const char example1[] PROGMEM = "AB";
const char example2[] PROGMEM = "AAAA";
int main(void)
{
static const char example3[] PROGMEM = "ABABB";
LED_init();
while (1) {
blinky(example1);
blinky(example2);
blinky(example3);
}
}
strcmp_P()
属性只能与变量一起使用;如果引用类型,则不执行任何操作。因此,您需要使用上述形式之一将字符串指定为字符数组。 (
PROGMEM
仅在此编译单元中可见,
example1
也可以从其他编译单元中引用,并且
example2
仅在其定义的函数中可见。此处,visible是指您可以引用的位置变量;与内容无关。)
example3
属性实际上并不更改GCC生成的代码。它所做的只是将内容放在
PROGMEM
部分,如果没有它,它们将放在
.progmem.data
中。所有的魔力实际上都在链接以及链接的库代码中。
.rodata
属性,因为它们确定了内容将以哪个部分结尾。可变(非const)数据应以
const
结尾部分,而不变(const)数据则以
.data
部分结尾。请记住从变量本身开始,从右到左阅读说明符,并以'*'分隔:最左边的是内容,而最右边的是变量。换一种说法,
const char *s = p;
.rodata
,以便可以更改变量的值,但是变量所指向的内容是不可变的(不可更改/常量);而
char *const s = p;
s
,以便您不能修改变量本身,但是可以使内容-
s
指向的内容是可变的,可修改的。此外,
const char *s = "literal";
s
指向文字字符串(并且您可以修改
s
,例如,使其指向其他文字字符串),但是您不能修改内容;和
char s[] = "string";
s
定义为一个字符数组(长度为6;字符串长度+ 1(表示字符串结尾的char)),并且碰巧已初始化为
s
。
{ 's', 't', 'r', 'i', 'n', 'g', '\0' }
节的内容复制到RAM,并且仅将
.rodata
保留在程序存储器中。)
.progmem.data
查看ELF二进制文件是否包含任何库代码。我认为,您应该期望在该列表中至少看到
avr-objdump -d binary.elf | grep -e '^[0-9a-f]* <_'
,
<__do_clear_bss>:
和
<_exit>:
。
<__stop_program>:
和何时基于类型的
ld
),则可能是不可能的即使所有工具之间都能很好地通信,也可以在该库中使用avr-gcc。
lpm
部分)保存在程序存储器中,而我自己必须在需要时将任何不可变数据显式复制到RAM。这样,我可以在独立模式下使用简单的特定于微控制器的链接描述文件和GCC(完全不使用C库),并获得对微控制器的完全控制。另一方面,您会丢失avr-libc和其他C库提供的所有漂亮的预定义宏和功能。
关于c - C字符串文字作为参数等于在AVR-GCC中为-1?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25826813/
简而言之:我想从可变参数模板参数中提取各种选项,但不仅通过标签而且通过那些参数的索引,这些参数是未知的 标签。我喜欢 boost 中的方法(例如 heap 或 lockfree 策略),但想让它与 S
我可以对单元格中的 excel IF 语句提供一些帮助吗? 它在做什么? 对“BaselineAmount”进行了哪些评估? =IF(BaselineAmount, (Variance/Baselin
我正在使用以下方法: public async Task Save(Foo foo,out int param) { ....... MySqlParameter prmparamID
我正在使用 CodeGear RAD Studio IDE。 为了使用命令行参数测试我的应用程序,我多次使用了“运行 -> 参数”菜单中的“参数”字段。 但是每次我给它提供一个新值时,它都无法从“下拉
我已经为信用卡类编写了一些代码,粘贴在下面。我有一个接受上述变量的构造函数,并且正在研究一些方法将这些变量格式化为字符串,以便最终输出将类似于 号码:1234 5678 9012 3456 截止日期:
MySql IN 参数 - 在存储过程中使用时,VarChar IN 参数 val 是否需要单引号? 我已经像平常一样创建了经典 ASP 代码,但我没有更新该列。 我需要引用 VarChar 参数吗?
给出了下面的开始,但似乎不知道如何完成它。本质上,如果我调用 myTest([one, Two, Three], 2); 它应该返回元素 third。必须使用for循环来找到我的解决方案。 funct
将 1113355579999 作为参数传递时,该值在函数内部变为 959050335。 调用(main.c): printf("%d\n", FindCommonDigit(111335557999
这个问题在这里已经有了答案: Is Java "pass-by-reference" or "pass-by-value"? (92 个回答) 关闭9年前。 public class StackOve
我真的很困惑,当像 1 == scanf("%lg", &entry) 交换为 scanf("%lg", &entry) == 1 没有区别。我的实验书上说的是前者,而我觉得后者是可以理解的。 1 =
我正在尝试使用调用 SetupDiGetDeviceRegistryProperty 的函数使用德尔福 7。该调用来自示例函数 SetupEnumAvailableComPorts .它看起来像这样:
我需要在现有项目上实现一些事件的显示。我无法更改数据库结构。 在我的 Controller 中,我(从 ajax 请求)传递了一个时间戳,并且我需要显示之前的 8 个事件。因此,如果时间戳是(转换后)
rails 新手。按照多态关联的教程,我遇到了这个以在create 和destroy 中设置@client。 @client = Client.find(params[:client_id] || p
通过将 VM 参数设置为 -Xmx1024m,我能够通过 Eclipse 运行 Java 程序-Xms256M。现在我想通过 Windows 中的 .bat 文件运行相同的 Java 程序 (jar)
我有一个 Delphi DLL,它在被 Delphi 应用程序调用时工作并导出声明为的方法: Procedure ProduceOutput(request,inputs:widestring; va
浏览完文档和示例后,我还没有弄清楚 schema.yaml 文件中的参数到底用在哪里。 在此处使用 AWS 代码示例:https://github.com/aws-samples/aws-proton
程序参数: procedure get_user_profile ( i_attuid in ras_user.attuid%type, i_data_group in data_g
我有一个字符串作为参数传递给我的存储过程。 dim AgentString as String = " 'test1', 'test2', 'test3' " 我想在 IN 中使用该参数声明。 AND
这个问题已经有答案了: When should I use "this" in a class? (17 个回答) 已关闭 6 年前。 我运行了一些java代码,我看到了一些我不太明白的东西。为什么下
我输入 scroll(0,10,200,10);但是当它运行时,它会传递字符串“xxpos”或“yypos”,我确实在没有撇号的情况下尝试过,但它就是行不通。 scroll = function(xp
我是一名优秀的程序员,十分优秀!