1.未初始化的全局变量(.bss段)

bss段用来存放 没有被初始化已经被初始化为0 的全局变量。如下例代码:

#include<stdio.h>

int bss_array[1024*1024];
int main(int argc, char *argv[])
{
return 0;
}

编译并查看:

$ gcc -g mainbss.c -o mainbss
$ ls -l mainbss
-rwxrwxr-x. 1 hy hy 8330 Apr 22 19:33 mainbss
$ objdump -h mainbss |grep bss
mainbss: file format elf32-i386
24 .bss 00400020 0804a020 0804a020 00001018 2**5
$ size mainbss
text data bss dec hex filename
1055 272 4194336 4195663 40054f mainbss

全局变量bss_array的大小为4MB = 1024*1024*sizeof(int) Byte = 4194304 Byte。 通过size 查看可知数据被存在了 bss 段

  而 可执行文件mainbss 有8KB左右,命令 ls -l mainbss 查看得知。可知,bss类型的全局变量只占用 运行时的内存空间,而不占用可执行文件自身的文件空间

2017-11-24

今天再读此书时,联想到 OR1200 编译过程中的 RAM 和 链接脚本,产生一个问题。在 OR 仿真验证中,我的RAM只有8KB的空间,若我有一个定义上诉数组,可执行文件本身不大,但是运行时需要占用4M的空间。那程序岂不是会崩溃? 带着这个问题。在OR的验证程序中做了测试。

个人结论: 若在运行时这个空间不足的问题,编译器会帮忙检查的。这些是数据,若是在运行时的 堆和栈 不足,这点编译器没法检查。

#include"uart.h"
volatile unsigned long timestamp = ;
#define DRAM_MEM_SIZE (0x200)
#define DRAM_ADDR_START (0x1FFF-DRAM_MEM_SIZE+1)
#define DRAM_ADDR_END (0x1FFF) #define Struct_Section __attribute__((unused, section(".stdata_mem_type"), \
aligned())) typedef unsigned int u32;
typedef signed int s32; char txbuf[]={0x01,0x03,0x07,0x0f,0x10,0x30,0x70};
char testbss;
char testdata=;
typedef struct {
char ch[];
unsigned int addrinit;
unsigned int len;
}STDATA_MEM_TYPE;
STDATA_MEM_TYPE arry_dynamic Struct_Section;
int t_array[*]={}; int main(void)
{
//省略
}

编译报错如下, .bss 段不够用:

openrisc@openrisc-VirtualBox:~/mc-or-lngit/or_bootsim_shpy/hyor_ramboot$ make
or32-elf-gcc -c reset.S -o reset.o
or32-elf-gcc -c main.c -o main.o
or32-elf-gcc -c uart.c -o uart.o
or32-elf-ld reset.o main.o uart.o -T ram.ld -o hyor_ramboot.or32
or32-elf-ld: hyor_ramboot.or32 section `.bss' will not fit in region `ram'
or32-elf-ld: region `ram' overflowed by 4192276 bytes
make: *** [hyor_ramboot.or32] Error

2. 已被初始化为非零的全局变量(.data段)

  data段用来存放已经被初始化为非零的全局变量。如下代码,只将矩阵的第一个元素初试化为1:

#include<stdio.h>

int data_array[1024*1024]={1};
int main(int argc, char *argv[])
{
return 0;
}

编译查看

[hy@localhost memcfg]$ gcc -g maindata.c -o maindata
[hy@localhost memcfg]$ ls -l maindata
-rwxrwxr-x. 1 hy hy 4202682 Apr 22 19:48 maindata
[hy@localhost memcfg]$ objdump -h maindata |grep \\.data
23 .data 00400020 0804a020 0804a020 00001020 2**5
[hy@localhost memcfg]$ size maindata
text data bss dec hex filename
1055 4194604 4 4195663 40054f maindata

而 可执行文件maindata 有4MB左右。通过size 查看可知数据被存在了 data 段

可知,data类型的全局变量既占用运行时的内存空间,也占用可执行文件自身的文件空间

3.常量数据(.rodata段)

1)rodata用来存放常量数据。 ro: read only

2)字符串会被编译器自动放在rodata中,加 const 关键字的常量数据会被放在 rodata 中

3)在有的嵌入式系统中, rodata放在 ROM(或 NOR Flash)里,运行时直接读取,不须加载到RAM内存中。

所以,在嵌入式开发中,常将已知的常量系数,表格数据等造表加以 const 关键字。存在ROM中,避免占用RAM空间。

4.代码(.text段)

  text段存放代码(如:函数)和部分整数常量(应该指的是一些立即数),这个段是可执行的。

5.栈(stack)

  1)stack 存放函数的局部变量和函数参数

  2)被调用函数的参数和返回值 被存储到当前程序的栈区,之后被调用函数再为自身的自动变量和临时变量在栈区上分配空间

  3)函数返回时,栈区的数据会被释放掉,先入后出(FILO)的顺序。

6.堆(heap)

  heap用来动态分配内存,由程序自身决定开辟和释放。

6.1 malloc/free

#include<stdio.h>
#include<stdlib.h> int main(int argc, char *argv[])
{
int *p = (int *)malloc(10*1);
// p= (int *)malloc(10*1);
if(p==NULL) {
printf("malloc p err\n");
return -1;
}
free(p);
printf("p = %4x\n",p);
p = NULL;
printf("p = %4x\n",p);
return 0;
}

程序运行结果:

[hy@localhost memcfg]$ gcc maindata.c
[hy@localhost memcfg]$ ./a.out
p = 9cf4008
p = 0

1)开辟了空间,就要适时的释放。释放时,指针应指向开辟时的内存空间,所以在使用指针时,要注意不要修改了其地址,或者将开辟时的起始地址保存起来。

2)对指针free后,其地址不一定就为NULL。如代码中的 p,在 free(p)后,printf("p=%4x",p)后并非为0。所以建议在free(p)后,立即加一句p=NULL。

3)检查p的地址 if(p!=NULL){ ... }

6.2 calloc/realloc

1)calloc()函数

void *calloc(size_t nmemb, size_t size);

  参数nmemb 表示要分配元素的个数,size表示每个元素的大小,分频的内存空间大小是 nmemb*size; 返回值是 void* 类型的指针,指向分配好的内存首地址。

用法一:分配1024*sizeof(int)字节大小的内存,并清空为0

int *p = (int *)calloc(1024,sizeof(int));

用法二:与 alloc等价的 malloc 用法

int *p = (int *)malloc(1024*sizeof(int));
memset(p,0,1024*sizeof(int));

差异:用法一calloc,会根据分配的的类型来初始化为0,如:分配int型,则初始化为(int)0; 若为指针类型,则初始化为空指针;若为浮点,则初始化为浮点型。

    用法二memset,不能保证初试化为空指针值和浮点型。(与NULL常量和浮点型的定义有关)

2)realloc()函数

  realloc()用来重新分配正在使用的一块内存大小。

定义:

void *realloc(void *ptr, size_t size);

用法示例:

int *p = (int *)malloc(1024);    //
p = (int *)realloc(512); // 重新分配为 512字节大小内存,缩小数据丢失
p = (int *)realloc(2048); // 重新分配为2048字节大小内存

注意:经过realloc()调整后的内存空间起始地址有可能与原来的不同。

摘录书籍

《ARM嵌入式Linux系统开发详解》 弓雷 等编著

《高质量嵌入式Linux C编程》 梁庚 等编著

笔记:程序内存管理 .bss .data .rodata .text stack heap的更多相关文章

  1. XV6学习笔记(2) :内存管理

    XV6学习笔记(2) :内存管理 在学习笔记1中,完成了对于pc启动和加载的过程.目前已经可以开始在c语言代码中运行了,而当前已经开启了分页模式,不过是两个4mb的大的内存页,而没有开启小的内存页.接 ...

  2. C++复习12.程序内存管理

    程序内存管理 20131006 一个程序在运行期间的内存是如何的对编写程序至关重要,之前整理的C++内存管理的知识和Java程序内存管理的知识.今天我们系统的整理一下程序的内存. 1.一个程序的内存有 ...

  3. linux kernel学习笔记-5内存管理_转

    void * kmalloc(size_t size, gfp_t gfp_mask); kmalloc()第一个参数是要分配的块的大小,第一个参数为分配标志,用于控制kmalloc()的行为. km ...

  4. C程序内存管理

    C程序的内存管理 熟悉Java语言的肯定知道,Java中内存管理是由虚拟机帮助我们完毕的,在C/C++中可不是这样,程序猿须要自己去分配和回收内存空间.本文记录了C程序可运行文件的存储结构.在内存中的 ...

  5. Magenta源代码笔记(3) —— 内存管理【转】

    转自:http://blog.csdn.net/boymax2/article/details/52550197 版权声明:本文为博主原创文章,未经博主允许不得转载. Magenta内核支持虚拟地址的 ...

  6. Java复习2.程序内存管理

    前言: 国庆节的第三天,大家都回家了,一个人在宿舍好无聊.不过这年头与其说是出去玩不如是说出去挤,所以在学校里还是清闲的好.找工作不用担心了,到时候看着你们慢慢忙:插个话题,大学都没有恋爱过,总之各种 ...

  7. bss、data、text、heap(堆)与stack(栈)

    bss段: bss段(bss segment)通常是指用来存放程序中未初始化的全局变量和静态变量(static)的一块内存区域. bss是英文Block Started by Symbol的简称. b ...

  8. JVM读书笔记之内存管理

    对于从事C.C++程序开发人员来说,在内存管理领域,他们既是拥有最高权力的“皇帝”又是从事最基础工作的“劳动人民”--既拥有每一个对象的“所有权”,又负责每一个对象生命开始到终结的维护责任. 对于Ja ...

  9. Linux System Programming 学习笔记(九) 内存管理

    1. 进程地址空间 Linux中,进程并不是直接操作物理内存地址,而是每个进程关联一个虚拟地址空间 内存页是memory management unit (MMU) 可以管理的最小地址单元 机器的体系 ...

随机推荐

  1. uboot的配置流程分析

    简单介绍一下uboot的基本配置流程.和绝大多数源码编译安装一样,uboot在执行make之前需要执行make XXXconfig来配置相关信息,而且uboot本身是针对多种平台的bootloader ...

  2. tomcat 应用部署方式(转)

    tomcat部署web应用的三种方式 1.直接放到Webapps目录下    Tomcat的Webapps目录是Tomcat默认的应用目录,当服务器启动时,会加载所有这个目录下的应用.也可以将JSP程 ...

  3. 对比Haproxy和Nginx负载均衡效果

    为了对比Hproxy和Nginx负载均衡的效果,分别在测试机上(以下实验都是在单机上测试的,即负载机器和后端机器都在一台机器上)做了这两个负载均衡环境,并各自抓包分析.下面说下这两种负载均衡环境下抓包 ...

  4. 解决ASP.NET上传文件大小限制

    第一种方法,主要适用于IIS6.0版本 一.修改配置Web.Config文件中的httpRuntime节点对于asp.net,默认只允许上传4M文件,增加如下配置,一般可以自定义最大文件大小.一.修改 ...

  5. CPU中断和轮询

    轮询顾名思义,CPU不停的询问,CPU很忙. 中断和轮询不同,是硬件上的,就像一个开关被按下了,没有什么东西在不停的询问开关有没有被按下,开关只是被按下了,就通电了,通电了就会发声一些事情. 可以看看 ...

  6. 分享“12306 P2P平台”创业Idea

      结合云平台抢票,社区,P2P等性质,实施供求抢票平台,能有效提高和整合抢票市场,抢票优势以杜绝黄牛,给散户提供更有利的抢票途径.本在11月计划实施,后去搞比特币,故分享摘要集思广益.     背景 ...

  7. jquery点滴

    1.toggle 2.next prev after before 3.on 4.当我们使用jquery的attr('checked',true)或者attr('checked','checked') ...

  8. laravel实现数据库多库配置,读写分离配置或者多读写分离配置

    'connections' => array( //默认mysql配置,访问test库 'mysql' => array( 'driver' => 'mysql', 'host' = ...

  9. 记录rewrite url我之前不知道的地方

    大部分url重写的需求是伪静态,当然有很多第三方开源组件,但是这种需求的核心方法其实就是context.rewritePath() 要是系统像ARR那样,用重写做代理和反向代理,一般的重写就不行了,c ...

  10. sql查询慢优化

    SELECT g.goods_id, g.type_id, g.user_id, g.productname, g.img, g.intro, g.attr, u.companyname, u.enl ...