当前位置: 首页 > news >正文

内核编程十:进程的虚拟地址空间

请思考当以下这个经典的hello world代码,调入内存并且在cpu上运行变成一个进程的时候,代码中的各个数据在内存中是如何进行分布的

#include <stdio.h>

int main() {
	printf("Hello, World!\n");
    return 0;
}

进程内存布局

进程在内存中的数据分布通常分为多个段(Segment),每个段负责存储不同类型的数据。进程使用的是虚拟空间的地址,这些段在虚拟地址空间中是连续的,但实际物理内存可能分散或不连续。以下是典型的进程内存布局:

1. 代码段(Text Segment)

  • 内容:存储程序的机器指令(即编译后的代码)。

  • 特性:只读,不可修改,多个进程可共享同一代码段。

2. 数据段(Data Segment)

  • 内容:存储全局变量和静态变量。

  • 分类

    • 已初始化数据段:存储显式初始化的全局和静态变量。

    • 未初始化数据段(BSS段):存储未初始化的全局和静态变量,程序加载时初始化为零。

3. 堆(Heap)

  • 内容:动态分配的内存(如mallocnew等)。

  • 特性:手动管理,大小不固定,向高地址增长。

4. 栈(Stack)

  • 内容:存储局部变量、函数参数、返回地址等。

  • 特性:自动管理,大小有限,向低地址增长。

5. 共享库(Shared Libraries)

  • 内容:存储动态链接库的代码和数据。

  • 特性:多个进程可共享,减少内存占用。

6. 环境变量和命令行参数

  • 内容:存储环境变量和命令行参数。

  • 位置:通常位于栈的上方。

7. 内存映射区(Memory Mapped Region)

  • 内容:用于内存映射文件或匿名映射。

  • 特性:允许文件直接映射到内存,便于高效访问。

典型内存布局(从低到高地址)

+---------------------------+ 高地址
| 环境变量和命令行参数     	|
+---------------------------+
| 栈(Stack)                | <--- 向低地址增长
|			|				|
|			|				|
|			v				|
+---------------------------+
| 内存映射区                  |
+---------------------------+
| 共享库                     |
+---------------------------+
|			^				|
|			|				|
|			|				|
| 堆(Heap)                 | <--- 向高地址增长
+---------------------------+
| 未初始化数据段(BSS)     	|
+---------------------------+
| 已初始化数据段(Data)    	|
+---------------------------+
| 代码段(Text)            	|
+---------------------------+ 低地址

Hello world在内存中的分布

#include <stdio.h>

int main() {
	int i = 42;
    printf("Hello, World!\n");
	printf("int i = %d\n", i);
    return 0;
}
+---------------------------+ 高地址
| 环境变量和命令行参数      	|
+---------------------------+
| 栈(Stack)              	| <--- 向低地址增长
|   - main函数的栈帧         	|
|   - 局部变量 int i = 42   	|
|   - printf的参数           |
|     - "Hello, World!\n"   |
|     - "int i = %d\n", i   |
+---------------------------+
| 共享库(Shared Libraries) 	|
|   - printf的实现代码       	|
+---------------------------+
| 堆(Heap)                	| <--- 向高地址增长
|   (该程序未使用堆)       	|
+---------------------------+
| 未初始化数据段(BSS)     	|
|   (该程序未使用BSS)      	|
+---------------------------+
| 已初始化数据段(Data)    	|
|   (该程序未使用数据段)   	|
+---------------------------+
| 代码段(Text)            	|
|   - main函数的机器指令     	|
|   - printf的调用指令       	|	
+---------------------------+ 低地址

用户态栈:每个进程在用户空间拥有独立的栈(位于用户虚拟地址空间),用于存储函数调用、局部变量等。

内核态栈:每个进程在内核空间还有一个独立的内核栈(通常较小,如8KB或16KB),专门用于内核态执行时的调用链(如系统调用、中断处理)。

内核线程(Kernel Thread):仅有内核栈,无用户栈(因为不运行用户态代码)。

内核栈大小固定,设计紧凑以保证性能和安全性

http://www.dtcms.com/a/87764.html

相关文章:

  • Docker Hub Mirror 终极解决方案——0成本,超高速!
  • RAG优化:python从零实现时间管理大师Self-RAG
  • 红宝书第十二讲:详解JavaScript中的工厂模式与原型模式等各种设计模式
  • 第八章:防火墙
  • [实操]Mysql8 InnoDB引擎主从配置三节点操作流程
  • 论文阅读笔记:Denoising Diffusion Implicit Models
  • 消息队列ActiveMQ、RabbitMQ、RocketMQ、Kafka对比分析和选型
  • 【杂谈】Godot4.4 对象池(附脚本)
  • 盖泽 寻边器 帮助类
  • 开发中后端返回下划线数据,要不要统一转驼峰?
  • 【HTML5游戏开发教程】零基础入门合成大西瓜游戏实战 | JS物理引擎+Canvas动画+完整源码详解
  • C# BULK INSERT导入大数据文件数据到SqlServer
  • centos7.9镜像源及Python引入ssl问题处理
  • OLED中英文混合显示
  • 如何设计一个处理物联网设备数据流的后端系统。
  • SpringMVC 配置详解
  • 《深度剖析:DevEco Studio 如何实现人工智能模型的高效可视化开发》
  • 交换机(access端口)
  • Vue中的状态管理器Vuex被Pinia所替代-上手使用指南
  • 数据预处理习题
  • EtherCAT转CANopen配置CANopen侧的PDO映射
  • JavaScript性能优化实战手册:从V8引擎到React的毫秒级性能革命
  • 大数据平台各组件功能与协同作用全解析
  • Python Excel表格数据对比工具
  • Spring MVC配置详解:从历史到实战
  • 多路径PKL文件读取与合并
  • 云服务器怎么设置端口禁用呢?
  • Python 迭代器与生成器:深入理解与实践
  • 资源分配图(RAG)检测死锁算法实现
  • 【数据库】sql错题详解