下面是我用于解析令牌的一段代码。
在底部附近有一条标有>>>>
的行不再需要,但如果我将其注释掉,cmd_parse_value_lookup()
函数将失败。如果我保留它,代码运行正常。谁能告诉我为什么,并解释发生了什么?
void cmd_parse(void)
{
cmd_parse_value=0;
int cmd_parse_counter = 1;
char *cmd_parse_pointer;
cmd_parse_pointer = strtok(cmd_buffer_in, " ");
if (cmd_parse_pointer!=NULL)
{
cmd_parse_value_lookup(cmd_parse_pointer);
}
while (cmd_parse_pointer != NULL)
{
cmd_parse_counter++;
cmd_parse_pointer = strtok(NULL, " ");
if (cmd_parse_pointer!=NULL)
{
>>>>cmd_buffer_sprintf_return = sprintf(cmd_buffer_sprintf,"%i: %s\r\n", cmd_parse_counter, cmd_parse_pointer); //WHY DO I NEED THIS LINE
cmd_parse_value_lookup(cmd_parse_pointer);
}
}
}
void cmd_parse_value_lookup(char *cmd_command)
{
if (strcmp(cmd_command,"show")==0)
{
cmd_parse_value |= 1;
}
else if (strcmp(cmd_command,"get")==0)
{
cmd_parse_value |= 1;
}
else if (strcmp(cmd_command,"set")==0)
{
cmd_parse_value |= 2;
}
else if (strcmp(cmd_command,"system")==0)
{
cmd_parse_value |= 4;
}
}
编辑:这是完整的代码:
/** C O M M A N D ************************************************************/
#include "generic.h"
#include <stdio.h>
#include <string.h>
void btm_out_character (char character);
void btm_out_string(char *string);
void cmd_parse(void);
char cmd_buffer_in[81]="\0 ";
int cmd_buffer_in_position=0;
unsigned long long cmd_parse_value=0;
char cmd_buffer_sprintf[81];
int cmd_buffer_sprintf_return;
void cmd_parse_value_lookup(char *cmd_command);
void cmd_buffer_in_add(int character);
void cmd_init(void)
{
}
void cmd_cls(void)
{
if (dbg_mode==1){btm_out_string("\033[2J\033[1;33;40m\033[H\r\n");}
if (dbg_mode==1){btm_out_string("================================================================================");}
if (dbg_mode==1){btm_out_string("\033[24;0H================================================================================");}
if (dbg_mode==1){btm_out_string("\033[1;36;40m\033[8;66H DEBUG MODE ");}
if (dbg_mode==1){btm_out_string("\033[24;7H ***** ******** ********** Engineering Limited, All Rights Reserved ");}
if (dbg_mode==1){btm_out_string("\033[10;0H");}
}
void cmd_prompt(void)
{
cmd_buffer_in_position = 0; // Clear buffer position
cmd_buffer_in[0]=0; // Clear buffer
if (dbg_mode==1){btm_out_string("\r\n\033[1;37;40m***:> ");}
}
void cmd_buffer_in_add(int character)
{
switch (character)
{
case 8:
if (cmd_buffer_in_position>0)
{
if (dbg_mode==1){btm_out_string("\b \b");}
cmd_buffer_in[(int)cmd_buffer_in_position-1] = (char)character;
cmd_buffer_in[(int)cmd_buffer_in_position] = 0;
cmd_buffer_in_position--;
}
else
{
if (dbg_mode==1){btm_out_character(7);} //BELL alert (too long a string)
cmd_buffer_in[0] = 0;
cmd_buffer_in_position = 0;
}
break;
case 13:
cmd_parse();
cmd_buffer_sprintf_return = sprintf(cmd_buffer_sprintf,"\r\nparse value=%llu\r\n", cmd_parse_value);
if (dbg_mode==1){btm_out_string(cmd_buffer_sprintf);}
cmd_prompt();
break;
default:
if (cmd_buffer_in_position<73)
{
if (dbg_mode==1){btm_out_character((char)character);} //Echo character to host
cmd_buffer_in[(int)cmd_buffer_in_position] = (char)character;
cmd_buffer_in[(int)cmd_buffer_in_position+1] = 0;
cmd_buffer_in_position++;
}
else
{
if (dbg_mode==1){btm_out_character(7);} //BELL alert (too long a string)
}
}//switch
}//cmd_buffer_in_add
void cmd_parse(void)
{
cmd_parse_value=0;
/* this was code for testing */
//cmd_buffer_sprintf_return = sprintf(cmd_buffer_sprintf,"\r\nClone %s",cmd_buffer_in);
//cmd_buffer_sprintf_return = sprintf(cmd_buffer_sprintf,"Returnval='%i''%i''%i'",cmd_buffer_in[(int)0],cmd_buffer_in[(int)1],cmd_buffer_in[(int)2]);
//if (dbg_mode==1){btm_out_string(cmd_buffer_sprintf);}
int cmd_parse_counter = 1;
char *cmd_parse_pointer;
cmd_parse_pointer = strtok(cmd_buffer_in, " ");
if (cmd_parse_pointer!=NULL)
{
cmd_parse_value_lookup(cmd_parse_pointer);
}
while (cmd_parse_pointer != NULL)
{
cmd_parse_counter++;
cmd_parse_pointer = strtok(NULL, " ");
if (cmd_parse_pointer!=NULL)
{
cmd_buffer_sprintf_return = sprintf(cmd_buffer_sprintf,"%i: %s\r\n", cmd_parse_counter, cmd_parse_pointer); //WHY DO I NEED THIS LINE
//if (dbg_mode==1){btm_out_string(cmd_buffer_sprintf);}
cmd_parse_value_lookup(cmd_parse_pointer);
}
}
}
void cmd_parse_value_lookup(char *cmd_command)
{
if (strcmp(cmd_command,"show")==0)
{
cmd_parse_value |= 1;
}
else if (strcmp(cmd_command,"get")==0)
{
cmd_parse_value |= 1;
}
else if (strcmp(cmd_command,"set")==0)
{
cmd_parse_value |= 2;
}
else if (strcmp(cmd_command,"system")==0)
{
cmd_parse_value |= 4;
}
else if (strcmp(cmd_command,"sys")==0)
{
cmd_parse_value |= 4;
}
else if (strcmp(cmd_command,"adc")==0)
{
cmd_parse_value |= 8;
}
else if (strcmp(cmd_command,"a2d")==0)
{
cmd_parse_value |= 8;
}
else if (strcmp(cmd_command,"channel1")==0)//
{
cmd_parse_value |= 16;
}
else if (strcmp(cmd_command,"ch1")==0)
{
cmd_parse_value |= 16;
}
}
答案 0 :(得分:0)
由于您在其他任何地方都没有使用cmd_buffer_sprintf
,因此该行代码似乎毫无用处。情况可能是cmd_buffer_sprintf
是您在其他地方使用的全局变量,但在您提供的代码中,该行不会改变任何内容。
答案 1 :(得分:0)
如果你拿出那条线,cmd_buffer_sprintf
可能没有被初始化。
尝试在*cmd_buffer_sprintf = '\0'
中的某处添加main()
,看看是否可以解决问题。
答案 2 :(得分:0)
非常感谢你的downvote,我相信无论谁给了我,我都会立即猜到有一个编译器错误导致了这个问题,即使他们只是在学习C。
无论如何,问题的答案是编译器对unsigned long longs的粗略(最多)支持。由于某种原因,该行让它工作。我重新编写了避免使用unsigned long longs的部分,现在一切都已经排序了。感谢您的支持,试图解决这个问题。
答案 3 :(得分:0)
我们可以看到,一旦您评论了该行,您尝试评论的行上使用的var cmd_parse_counter就会变得无用。
然后,编译器的逻辑行为是从汇编代码中删除它。
这可能会修复代码中其他位置发生的缓冲区溢出或堆栈错误。
要测试此理论,请将cmd_parse_counter声明为volatile int cmd_parse_counter并注释掉该行并检查代码是否运行。如果它确实运行,那么你可以进一步找到问题。这可能与你发现的很长时间有关。
要检查的另一件事是你用空格填充缓冲区,然后使用空格标记进行拆分。每个分裂你做一个strcmp。如果你的缓冲区没有以0结尾,那么你肯定会因使用没有0字符的strcmp而崩溃。您可以尝试使用for循环使用全0来初始化缓冲区。