我正在使用MSP430F2619和Code Composer Studio 6.1为串行通信网关编写代码。
网关有两种不同的操作模式:
Master 这是默认模式。它将数据轮询为RS485总线上的总线主控器,然后将其发送到具有不同协议的另一个板。
从属,在其中响应串口上的PC应用程序。
程序将处于主模式或从模式,因为PC应用程序和其他从属板之间共享相同的RS-485总线。
我试图弄清楚两个缓冲区是否可以使用相同的物理内存,而不是为每种模式使用2个不同的内存缓冲区。
我尝试搜索Code Composer Studio的内存叠加功能但没有成功。
基本上,我有这个:
echo $words[$value];
我希望这两个缓冲区共享相同的内存,以便使用2KB内存而不是4KB。
所以我需要一种让char Slaverxbuffer[2048];
char Masterrxbuffer[2048];
和Slaverxbuffer
使用相同内存区域的方法,例如:
Masterrxbuffer
答案 0 :(得分:10)
您可以只使用一个缓冲区并在两种操作模式之间共享吗?这可能是最简单的解决方案。
否则,在C和C ++中,您可以通过使两个缓冲区为union类型的字段来实现此目的。联合中的每个成员在内存中具有相同的基址,并且联合的总大小是最大成员的大小。
代码看起来像这样:
union OverlaidBuffer
{
char Slaverxbuffer[2048];
char Masterrxbuffer[2048];
};
union OverLaidBuffer overlay;
/* Code to use the slave buffer */
for(unsigned int index = 0; index < 2048; ++index)
{
overlay.Slaverxbuffer[index] = some_slave_value();
}
/* Code to use the master buffer */
for(unsigned int index = 0; index < 2048; ++index)
{
overlay.Masterrxbuffer[index] = some_master_value();
}
请注意,如果您使用联合,则应始终从最近写入的成员中读取。写入一个成员,然后从另一个成员读取undefined behavior,可能会导致错误的代码。例如,代码:
overlay.Slaverxbuffer[i] = 1;
overlay.Masterrxbuffer[i] = 2;
x = overlay.Slaverxbuffer[i];
可能会将x
设置为1,因为它“显然”是overlay.Slaverxbuffer[i]
中存储的最后一个值,即使该内存地址现在包含值2.
实现此目的的另一种方法是使用编译器和链接器指令将两个缓冲区放在内存中的特定位置。大多数嵌入式工具链允许您通过源代码中的#pragma
指令执行此操作。指令的确切格式取决于特定的工具链;下面是几个例子。
TI Code Composer Studio(参考MSP430 Optimizing C/C++ Compiler User's Guide)
#pragma DATA_SECTION(Slaverxbuffer, "some_section")
char Slaverxbuffer[2048];
IAR Embedded Workbench(参考IAR C/C++ Development Guide)
#pragma section = "some_section"
char Slaverxbuffer[2048];
然后在链接器文件中,添加指令以将符号放在特定位置:
TI Code Composer Studio(参考MSP430 Assembly Language Tools User's Guide)
SECTIONS
{
.some_section : load = 0x12345678
}
IAR Embedded Workbench(参考IAR C/C++ Development Guide)
place at address mem:0x12345678 { readwrite section some_section };
答案 1 :(得分:4)
正如你所说,如果你不想要一个联合,你可以只有一个缓冲区和一个单独的变量来记录你所处的模式吗?
char rxBuffer[2048];
bool rxMode; // false = master, true = slave
这样,您可以将缓冲区放在固定的内存位置(由链接器定义)并链接到缓冲区