ADS的默认连接顺序是怎样的呢?例如下边从2440init.s中摘出的编译器符号又该怎样理解呢?

BaseOfROM    DCD    |Image##RO##Base|
TopOfROM     DCD    |Image##RO##Limit|
BaseOfBSS    DCD    |Image##RW##Base|
BaseOfZero   DCD    |Image##ZI##Base|
EndOfBSS     DCD    |Image##ZI##Limit|
 
 
说明:由于“$ $”(实际上中间没有空格,但是不知为什么只要这两个字符并排着放在一起,提交后显示的时候总是出现问题)排版的问题,所以本文中所有用到的“$ $”都用##代替。

我根据TQ2440_Test.mcp做了一个测试。我知道下边的图片对于使用ADS的朋友很熟悉,但是我觉得还是有必要先把这个工程的编译配置选项贴出来。

为了说明这些问题,我不得不把ADS生成的List.txt内容列出来,尽管它很长,不过只去注意那些对问题有意义的地方。

================================================================================

Memory Map of the image

Image Entry point : 0x30000000

Load Region LR_1 (Base: 0x30000000, Size: 0x00086c14, Max: 0xffffffff, ABSOLUTE)

Execution Region ER_RO (Base: 0x30000000,Size: 0x0000b400, Max: 0xffffffff, ABSOLUTE)

Base Addr    Size         Type   Attr  Idx  E Section Name        Object

0x30000000   0x00000544   Code   RO     1   * Init                2440init.o
    0x30000544   0x0000026c   Code   RO    10     .text               nand.o
    0x300007b0   0x00000aa0   Code   RO    59     .text               2440lib.o
    0x30001250   0x00000550   Code   RO   145     .text               Main.o
    0x300017a0   0x00000288   Code   RO   210     .text               mmu.o
    0x30001a28   0x0000046c   Code   RO   226     .text               dma.o
    0x30001e94   0x0000019c   Code   RO   264     .text               Adc.o
    0x30002030   0x00000e7c   Code   RO   279     .text               camif.o
    0x30002eac   0x000007f0   Code   RO   411     .text               IIC.o
    0x3000369c   0x0000026c   Code   RO   445     .text               keyscan.o
    0x30003908   0x00000120   Code   RO   501     .text               RTC.o
    0x30003a28   0x000003e8   Code   RO   513     .text               Test_OV9650.o
    0x30003e10   0x0000026c   Code   RO   606     .text               Touchpanel.o
    0x3000407c   0x000005b4   Code   RO   620     .text               UDA1341.o
    0x30004630   0x00000cdc   Code   RO   643     .text               AudioDrv.o
    0x3000530c   0x00000af0   Code   RO   731     .text               LCD_TFT.o
    0x30005dfc   0x00001018   Code   RO   766     .text               SD_MMC.o
    0x30006e14   0x00000034   Code   RO   831     .text               atoi.o(c_a__un.l)
    0x30006e48   0x00000014   Code   RO   833     .text               rt_ctype_table.o(c_a__un.l)
    0x30006e5c   0x000000d4   Code   RO   835     .text               rt_sdiv.o(c_a__un.l)
    0x30006f30   0x000000c0   Code   RO   837     .text               rt_udiv.o(c_a__un.l)
    0x30006ff0   0x00000068   Code   RO   839     .text               strlen.o(c_a__un.l)
    0x30007058   0x00000058   Code   RO   841     .text               vsprintf.o(c_a__un.l)
    0x300070b0   0x00000d38   Code   RO   860     .text               __vfpntf.o(c_a__un.l)
    0x30007de8   0x0000001c   Code   RO   862     .text               _sputc.o(c_a__un.l)
    0x30007e04   0x0000003c   Code   RO   864     .text               lc_ctype_c.o(c_a__un.l)
    0x30007e40   0x0000000c   Code   RO   867     .text               libspace.o(c_a__un.l)
    0x30007e4c   0x0000000c   Code   RO   870     .text               rt_div0.o(c_a__un.l)
    0x30007e58   0x0000000c   Code   RO   872     .text               rt_errno_addr.o(c_a__un.l)
    0x30007e64   0x00000010   Code   RO   874     .text               rt_fp_status_addr.o(c_a__un.l)
    0x30007e74   0x000000c0   Code   RO   876     .text               strtol.o(c_a__un.l)
    0x30007f34   0x00000924   Code   RO   894     .text               _fp_disp.o(c_a__un.l)
    0x30008858   0x00000060   Code   RO   896     .text               _fptrap.o(c_a__un.l)
    0x300088b8   0x00000108   Code   RO   898     .text               _strtoul.o(c_a__un.l)
    0x300089c0   0x00000098   Code   RO   900     .text               lludiv10.o(c_a__un.l)
    0x30008a58   0x00000018   Code   RO   902     .text               rt_raise.o(c_a__un.l)
    0x30008a70   0x0000002c   Code   RO   904     .text               rtudiv10.o(c_a__un.l)
    0x30008a9c   0x000000a0   Code   RO   906     .text               strcmp.o(c_a__un.l)
    0x30008b3c   0x00000064   Code   RO   910     .text               __raise.o(c_a__un.l)
    0x30008ba0   0x00000020   Code   RO   912     .text               _chval.o(c_a__un.l)
    0x30008bc0   0x0000015c   Code   RO   914     .text               bigflt0.o(c_a__un.l)
    0x30008d1c   0x0000003c   Code   RO   919     .text               lc_numeric_c.o(c_a__un.l)
    0x30008d58   0x00000040   Code   RO   922     .text               rtsdiv10.o(c_a__un.l)
    0x30008d98   0x00000018   Code   RO   924     .text               sys_exit.o(c_a__un.l)
    0x30008db0   0x00000048   Code   RO   928     .text               classify.o(m_a_pu.l)
    0x30008df8   0x00000054   Code   RO   930     .text               dtoi.o(m_a_pu.l)
    0x30008e4c   0x00000160   Code   RO   932     .text               defsig.o(c_a__un.l)
    0x30008fac   0x00000004   Code   RO   936     .text               use_semi.o(c_a__un.l)
    0x30008fb0   0x00000018   Code   RO   938     .text               sys_wrch.o(c_a__un.l)
    0x30008fc8   0x000001b4   Code   RO     6     C$$code             2440slib.o
    0x3000917c   0x00000898   Code   RO   917     CL$$btodstuff       btod.o(c_a__un.l)
    0x30009a14   0x00000014   Code   RO   446     i.ClearPending      keyscan.o
    0x30009a28   0x00000048   Code   RO   516     i.OV9650_sccb_end   Test_OV9650.o
    0x30009a70   0x00000054   Code   RO   515     i.OV9650_sccb_start  Test_OV9650.o
    0x30009ac4   0x00000050   Code   RO   517     i.OV9650_sccb_write_bit  Test_OV9650.o
    0x30009b14   0x00000034   Code   RO   518     i.OV9650_sccb_writechar  Test_OV9650.o
    0x30009b48   0x00000038   Code   RO   644     i.SetPlayDma        AudioDrv.o
    0x30009b80   0x00000038   Code   RO   645     i.SetRecDma         AudioDrv.o
    0x30009bb8   0x00000054   Code   RO   514     i.set_gpio_ctrl     Test_OV9650.o
    0x30009c0c   0x0000049c   Code   RO   845     x$fpl$dadd          daddsub.o(f_a_p.l)
    0x3000a0a8   0x00000040   Code   RO   880     x$fpl$dcheck        dcheck.o(f_a_p.l)
    0x3000a0e8   0x00000018   Code   RO   882     x$fpl$dcheck1       dcheck1.o(f_a_p.l)
    0x3000a100   0x000005b0   Code   RO   847     x$fpl$ddiv          ddiv.o(f_a_p.l)
    0x3000a6b0   0x000000a4   Code   RO   849     x$fpl$dfix          dfix.o(f_a_p.l)
    0x3000a754   0x00000064   Code   RO   852     x$fpl$dflt          dflt.o(f_a_p.l)
    0x3000a7b8   0x000002b4   Code   RO   854     x$fpl$dmul          dmul_mull.o(f_a_p.l)
    0x3000aa6c   0x00000230   Code   RO   884     x$fpl$dunder        dunder.o(f_a_p.l)
    0x3000ac9c   0x00000164   Code   RO   886     x$fpl$exception     except.o(f_a_p.l)
    0x3000ae00   0x0000003c   Code   RO   926     x$fpl$ieeestatus    istatus.o(f_a_p.l)
    0x3000ae3c   0x00000004   Code   RO   878     x$fpl$printf1       printf1.o(f_a_p.l)
    0x3000ae40   0x00000098   Code   RO   908     x$fpl$retnan        retnan.o(f_a_p.l)
    0x3000aed8   0x000000a4   Code   RO   892     x$fpl$trapveneer    trapv.o(f_a_p.l)
    0x3000af7c   0x000000d3   Data   RO   147     .constdata          Main.o
    0x3000b04f   0x00000001   PAD    (说明:为了使代码4字节对齐,起到占位作用,占了一个字节)
    0x3000b050   0x00000058   Data   RO   622     .constdata          UDA1341.o
    0x3000b0a8   0x00000110   Data   RO   865     .constdata          lc_ctype_c.o(c_a__un.l)
    0x3000b1b8   0x00000094   Data   RO   915     .constdata          bigflt0.o(c_a__un.l)
    0x3000b24c   0x0000001c   Data   RO   920     .constdata          lc_numeric_c.o(c_a__un.l)
    0x3000b268   0x0000012b   Data   RO   933     .constdata          defsig.o(c_a__un.l)
    0x3000b393   0x00000001   PAD    (说明:为了使代码4字节对齐,起到占位作用,占了一个字节)
    0x3000b394   0x00000040   Data   RO   521     .constdata_check_OV9650  Test_OV9650.o
    0x3000b3d4   0x0000002c   Data   RO   522     .constdata_s3c2440_camif_init  Test_OV9650.o

Execution Region ER_RW (Base: 0x3000b400, Size: 0x0007b814, Max: 0xffffffff, ABSOLUTE)

Base Addr    Size         Type   Attr  Idx  E Section Name        Object

0x3000b400   0x00000004   Data   RW    60     .data               2440lib.o
    0x3000b404   0x00000068   Data   RW   146     .data               Main.o
    0x3000b46c   0x00000004   Data   RW   265     .data               Adc.o
    0x3000b470   0x000003ec   Data   RW   519     .data               Test_OV9650.o
    0x3000b85c   0x0003b760   Data   RW   621     .data               UDA1341.o
    0x30046fbc   0x00000058   Data   RW   646     .data               AudioDrv.o
    0x30047014   0x0003fc00   Data   RW   760     .data               TQ_LOGO.o

Execution Region ER_ZI (Base: 0x30086c14, Size: 0x000415b8, Max: 0xffffffff, ABSOLUTE)

Base Addr    Size         Type   Attr  Idx  E Section Name        Object

0x30086c14   0x00000004   Zero   RW    61     .bss                2440lib.o
    0x30086c18   0x00000020   Zero   RW   148     .bss                Main.o
    0x30086c38   0x00000004   Zero   RW   153     .bss                Main.o
    0x30086c3c   0x00000004   Zero   RW   155     .bss                Main.o
    0x30086c40   0x00000020   Zero   RW   227     .bss                dma.o
    0x30086c60   0x00000004   Zero   RW   266     .bss                Adc.o
    0x30086c64   0x00000024   Zero   RW   280     .bss                camif.o
    0x30086c88   0x00000230   Zero   RW   412     .bss                IIC.o
    0x30086eb8   0x00001400   Zero   RW   520     .bss                Test_OV9650.o
    0x300882b8   0x0000000c   Zero   RW   607     .bss                Touchpanel.o
    0x300882c4   0x00000004   Zero   RW   623     .bss                UDA1341.o
    0x300882c8   0x00000064   Zero   RW   647     .bss                AudioDrv.o
    0x3008832c   0x0003fc00   Zero   RW   732     .bss                LCD_TFT.o
    0x300c7f2c   0x00000240   Zero   RW   767     .bss                SD_MMC.o
    0x300c816c   0x00000060   Zero   RW   868     .bss                libspace.o(c_a__un.l)
================================================================================

Image component sizes
      Code    RO Data    RW Data    ZI Data      Debug

29120        407     505876     267608     114304   Object Totals
     15804        747          0         96       5096   Library Totals

================================================================================
      Code    RO Data    RW Data    ZI Data      Debug

44924       1154     505876     267704     119400   Grand Totals

================================================================================

Total RO  Size(Code + RO Data)                46078 (  45.00kB)          (0x0000B3FE)
    Total RW  Size(RW Data + ZI Data)            773580 ( 755.45kB)       (0x000BCDCC)     
    Total ROM Size(Code + RO Data + RW Data)     551954 ( 539.02kB)  (0x00086C12)

================================================================================

分析:

(1)从Image Entry point : 0x30000000这句话中,我们可以看出程序的入口地址是0x30000000。

(2)上边橙色的部分是"RO Data"段,List.txt并没有把这一部分单列出来,而是与Code段放在一起。

(3)从Load Region LR_1 (Base: 0x30000000, Size: 0x00086c14, Max: 0xffffffff, ABSOLUTE)这句话中,我们可以知道最终下载程序(TQ2440_Test.bin)的大小是0x00086C14。如果想知道为什么是这个数,可以通过Jlink软件把TQ2440_Test.bin打开,然后看代码量就可以知道。

从Total ROM Size(Code + RO Data + RW Data)     551954 ( 539.02kB)  (0x00086C12)这句,我们可以看出下载程序的代码量应该是0x00086C12,这与0x00086C14是矛盾的。其实不然,这是因为在"RO Data"段为了4字节对齐,用了两个单字节占位引起的多余空间。

(4)如果最终下载程序(TQ2440_Test.bin)的大小是0x00086C14,而并没有把bss段或者叫ZI段(0x000415b8)这部分算进去。也就是说最终的下载程序是不包括bss段的,bss段不会被下载到Flash或者RAM中,这也是为什么启动代码(比如说boot.s)要进行bss段初始化的原因。

(5)ADS默认的链接顺序就是代码段(.text)、只读数据段(.rodata)、(已初始化)数据段(.data)、未初始化数据段(.bss)。假设projiect里有三个文件,head.s、nand.c、maic.c,那么默认为的就先依次放这三个文件的代码段(head.s(.text)、nand.c(.text)、maic.c(.text)),再依次放只读数据段(head.s(.rodata)、nand.c(.rodata)、maic.c(.rodata)),依次再放这三个文件的数据段(head.s(.data)、nand.c(.data)、maic.c(.data)),最后再放(head.s(.bss)、nand.c(.bss)、maic.c(.bss))。

(6)为了理解编译器符号,我从TQ2440_Test.dis摘出的对应的反汇编代码如下。

300003b4 <BaseOfROM>:
300003b4:         andcc      r0, r0, r0 300003b8 <TopOfROM>:
300003b8:   3000b400      andcc      fp, r0, r0, lsl # 300003bc <BaseOfBSS>:
300003bc:   3000b400      andcc      fp, r0, r0, lsl # 300003c0 <BaseOfZero>:
300003c0:   30086c14       andcc      r6, r8, r4, lsl ip 300003c4 <EndOfBSS>:
300003c4:   300c81cc       andcc      r8, ip, ip, asr #

注意把这些值与List.txt中深蓝色字体对比,你就会发现其中的奥秘。30000000、3000b400、30086c14都可以在深蓝色字体中找到,需要指出的是300c81cc这个数字。这个数字是这样理解的, 从“Execution Region ER_ZI (Base: 0x30086c14, Size: 0x000415b8, Max: 0xffffffff, ABSOLUTE)”这句话中我们可以看出,bss段(ZI段)的基址是0x30086c14,大小是 0x000415b8,把这两个数字加起来就可以得到bss段的终址是300c81cc。

由此,我们得出对ADS编译器生成的各段的认识如下所示。

BaseOfROM      DCD       |Image##RO##Base|      代码段的首地址

TopOfROM       DCD       |Image##RO##Limit|      代码段的结束地址

BaseOfBSS        DCD       |Image##RW##Base|     数据段的起始地址

BaseOfZero       DCD       |Image##ZI##Base|      未初始化数据段的首地址

EndOfBSS         DCD       |Image##ZI##Limit|       未初始化数据段的结束地址

附注:|Image##RW##Limit|  =  |Image##ZI##Limit|

怎样调用ADS编译器产生的参数?

从汇编代码“BaseOfROM      DCD       |Image##RO##Base|”中,我们就可以看出汇编程序是如何调用编译器产生的参数。但是,C语言又该怎样调用这些参数呢?我摘出了一段程序可以看出。

extern char Image##RW##Limit[];()
void *mallocPt=Image##RW##Limit;()

怎样理解呢?我是这样认为的。第(1)句在地址Image##RW##Limit处定义了一个数组,显然数组的首地址就是“Image##RW##Limit”。而第二句,就去调用这个首地址。

什么可以作为只读数据段(.rodata)?

下边的代码是从main.c中摘出来的。

void Temp_function() { Uart_Printf("\nPlease input 1-11 to select test!!!\n"); 
struct {
       void (*fun)(void);
       char *tip;
}CmdTip[] = {
                            { Temp_function, "Please input 1-11 to select test" } ,
                            { BUZZER_PWM_Test, "Test PWM" } ,
                            { RTC_Display, "RTC time display" } ,
                            { Test_Adc, "Test ADC" } ,
                            { KeyScan_Test, "Test interrupt and key scan" } ,
                            { Test_Touchpanel, "Test Touchpanel" } ,
                            { Lcd_TFT_Test, "Test TFT LCD" } ,
                            { Test_Iic, "Test IIC EEPROM" } ,
                            { PlayMusicTest, "UDA1341 play music" } ,
                            { RecordTest, "UDA1341 record voice" } ,
                            { Test_SDI, "Test SD Card" } ,
                            { Camera_Test, "Test CMOS Camera"},
                            { , }                                      
                     };
void Main(void)
{
       Beep(, );       
       Uart_SendByte('\n');
       Uart_Printf("<***************************************>\n");
       Uart_Printf("               TQ2440 Test Program\n");
       Uart_Printf("                www.embedsky.net\n");
//     Uart_Printf("      Build time is: %s  %s\n", __DATE__ , __TIME__  );
       Uart_Printf("<***************************************>\n");
       while()
       {
              U8 idx;
              Uart_Printf("\nPlease select function : \n");  
              for(i=; CmdTip[i].fun!=; i++)
              {      Uart_Printf("%d : %s\n", i, CmdTip[i].tip);
                     idx = Uart_GetIntNum_GJ() ; 
                     if(idx<i)
                     {
                           (*CmdTip[idx].fun)();
                           Delay();
                           Uart_Init( , );
                     }
              }     
       }     
}

通过反汇编我们可以看出,main()函数中的字符串不会被当做只读数据段处理,而被视作程序代码(.text)的一部分。只有以变量形式定义的字符串才会被视作只读数据段,例如Main()函数之前内容中的字符串。

ADS的默认连接分析及编译器产生符号解惑的更多相关文章

  1. Laravel 使用Voyager导致多个数据库连接总是返回默认连接?

    问题与分析 最近的项目碰到一个奇怪的问题,在Laravel(5.3)中想建立多个数据库连接连到MySQL的不同数据库(另一个连接名为conn2),执行如下语句得到却发现得到的仍然是默认连接: $con ...

  2. 修改 EF的默认连接工厂为 Sql Server 而不是LocalDb

      当你用EF6创建一个新项目,不知你是否注意到默认的连接字符串使用了LocalDb而不是SQLServer.但你如果想把默认连接改用SQLSErver而不是LocalDb.这个其实很简单:只需修改下 ...

  3. QSqlDatabase::addDatabase第一次运行的时候,生成SQLite文件的同时会产生一个默认连接

    QSqlDatabase::addDatabase第一次运行的时候,生成SQLite文件的同时会产生一个默认连接: QSqlDatabase database = QSqlDatabase::addD ...

  4. Spring Boot之默认连接池配置策略

    注意:如果我们使用spring-boot-starter-jdbc 或 spring-boot-starter-data-jpa “starters”坐标,Spring Boot将自动配置Hikari ...

  5. SpringBoot:关于默认连接池Hikari的源码剖析

    1.起因 因为这两天在给公司的一个项目升级SpringBoot版本,遇到了一些坑,升级项目版本:SpringBoot1.5.x到SpringBoot2.0.x 今天早上双库操作遇到一个问题:jdbcU ...

  6. JVM学习三:JVM之类加载器之连接分析

    学习完类加载之加载篇后,让我们继续来看加载之连接,连接分为三个步骤:验证.准备和解析三步,我们将一一分析之. 连接就是将已经读入到内存的类的二进制数据合并到虚拟机的运行时环境中去. 类加载完毕之后进入 ...

  7. 完整全面的Java资源库(包括构建、操作、代码分析、编译器、数据库、社区等等)

    构建 这里搜集了用来构建应用程序的工具. Apache Maven:Maven使用声明进行构建并进行依赖管理,偏向于使用约定而不是配置进行构建.Maven优于Apache Ant.后者采用了一种过程化 ...

  8. 【转】HTTP中的长连接和短连接分析

    1. HTTP协议与TCP/IP协议的关系 HTTP的长连接和短连接本质上是TCP长连接和短连接.HTTP属于应用层协议,在传输层使用TCP协议,在网络层使用IP协议.IP协议主要解决网络路由和寻址问 ...

  9. centos 7.0 修改ssh默认连接22端口 和 添加防火墙firewalld 通过端口

    首先 先做的就是 修改ssh的默认端口22 需要修改文件 /etc/ssh/sshd_config 使用命令 vi /etc/ssh/sshd_config [root@localhost ~]# v ...

随机推荐

  1. Delphi调用C++写的dll示例

    最近做一个读市民卡的项目,读卡器公司提供的读市民卡dll是用C++写的. 下面记录一些自己的心得,供需要的朋友参考. 声明dll函数要加上stdcall关键字,否则可能会报地址非法的错误. 代码: u ...

  2. 在Linux环境中使用Ext3文件系统

      Linux缺省情况下使用的文件系统为Ext2,ext2文件系统的确高效稳定.但是,随着Linux系统在关键业务中的应用,Linux文件系统的弱点也渐渐显露出来了:其中系统缺省使用的ext2文件系统 ...

  3. 使用DrawerLayout实现QQ5.0侧拉菜单效果

    在上一篇文章中,我们介绍了怎么使用DrawerLayout来实现一个简单的侧拉菜单(使用DrawerLayout实现侧拉菜单),也就是我们常说的抽屉效果,GitHub上类似效果的实现方式非常多,实现出 ...

  4. Java基础知识强化之集合框架笔记73:如何选择使用哪种集合

    1. 到底使用那种集合.    看需求 是否是键值对象形式: 是:Map 键是否需要排序: 是:TreeMap 否:HashMap 不知道,就使用HashMap. 否:Collection 元素是否唯 ...

  5. Java基础知识强化之集合框架笔记66:Map集合面试题之HashMap和Hashtable区别(重要)

    1. HashMap和Hashtable区别 ? • Hashtable:线程安全,效率低.不允许null键和null值 • HashMap:线程不安全,效率高.允许null键和null值 packa ...

  6. opai_suki

  7. 监控Linux内存使用情况

    cat mem.sh#!/bin/bashIP=`ifconfig | grep 'inet ' | grep -v '127.0.0.1' | awk -F ' ' '{print $2}'| aw ...

  8. 利用php给图片添加文字水印--面向对象与面向过程俩种方法的实现

    1: 面向过程的编写方法 //指定图片路径 $src = '001.png'; //获取图片信息 $info = getimagesize($src); //获取图片扩展名 $type = image ...

  9. python中关于正则表达式二

    2.2 反向引用 \1, \2... 表达式在匹配时,表达式引擎会将小括号 "( )" 包含的表达式所匹配到的字符串记录下来.在获取匹配结果的时候,小括号包含的表达式所匹配到的字符 ...

  10. 学习笔记6_Java_day11_JSP_基础和入门(1、2)

    主要内容:1. JSP基础2. Cookie3. HttpSession ================================ JSP基础 1. jsp的作用: * Servlet: &g ...