C语言中的虚拟地址

news2025/3/1 0:43:17

虚拟地址

虚拟地址空间

  • 对于操作系统而言,每个进程所得到的虚拟地址都在一个独立的固定的范围内,不会超过这个范围,我们把这个范围称为虚拟地址空间。
  • 所谓的虚拟地址空间本质就是一个地址范围,表示程序的寻址能力。
  • 对于32位系统而言,虚拟地址空间从0x00000000~0xFFFFFFFF,也就是4G
    • 0 ~ 3G-1是归用户所使用,称为用户地址空间
    • 3G ~ 4G-1是归内核使用,称为内核地址空间
  • 对于64位系统而言,因为应用程序没有那么大的内存需求,所以不支持完全的64位虚拟地址
    • 0x0000 0000 0000 0000 ~ 0x0000 FFFF FFFF FFFF是用户地址空间
    • 0xFFFF 0000 0000 0000 ~ 0xFFFF FFFF FFFF FFFF是内核地址空间
    • 内核地址空间和用户地址空间直接是不规范地址空间,不允许使用,强制使用会出现段错误
    • 用户地址空间的代码不能直接访问内核空间的代码和数据,但可以通过系统调用进入内核态,间接与系统内核交互

以下是一个32位系统存储数据的示例图

在这里插入图片描述

从上图中可以看到每个进程都有自己独立的虚拟地址池,它们之间相互隔离,不会相互干扰或冲突,当给内核操作的虚拟地址没有在映射关系表中找到对应关系时,就会出现段错误

虚拟地址空间布局

  • 程序中不同性质的数据,加载到内存中,其虚拟地址会被映射到虚拟地址空间中不同区域

在这里插入图片描述

代码如下:

// 虚拟地址空间布局
#include <stdio.h>
#include <stdlib.h>

const int const_global = 1;  // 常全局变量
int init_global = 2;  // 初始化全局变量
int uninit_global;  // 未初始化全局变量

int main(int argc, char *argv[], char *envp[])
{
    static const int const_static = 3;  // 常静态变量
    static int init_static = 4;  // 初始化静态变量
    static int uninit_static; // 未初始化静态变量
    const int const_local = 5;  // 常局部变量
    int local;  // 局部变量
    char *string = "hello";  // 字面值常量
    int *heap = malloc(sizeof(int)); // 堆变量

    printf("----------参数和环境----------\n");
    printf("         命令行参数: %p\n", argv);
    printf("           环境变量: %p\n", envp);
    printf("------------栈区-------------\n");
    printf("         常局部变量: %p\n", &const_local);
    printf("           局部变量: %p\n", &local);
    printf("------------堆区-------------\n");
    printf("             堆变量: %p\n", heap);
    printf("------------BSS区-------------\n");
    printf("    未初始化全局变量: %p\n", &uninit_global);
    printf("    未初始化静态变量: %p\n", &uninit_static);
    printf("------------数据区-------------\n");
    printf("      初始化全局变量: %p\n", &init_global);
    printf("      初始化静态变量: %p\n", &init_static);
    printf("------------代码区-------------\n");
    printf("          常全局变量: %p\n", &const_global);
    printf("          常静态变量: %p\n", &const_static);
    printf("          字面值常量: %p\n", string);
    printf("               函数: %p\n", main);
    printf("-------------------------------\n");

    return 0;
}

内存映射的建立与解除

  • 没有与物理地址建立映射关系的虚拟地址,无法直接使用,如果强行使用被报段错误,我们可以通过系统调用mmap函数手动建立虚拟地址和物理地址之间的映射关系。
  • 补充:基本上所有的头文件都是放在这个路径下的:/usr/include
  • 下面那个函数需要引入mman.h这个头文件:#include <sys/mman.h>
  • mmap函数
    • void *mmap(void *start, size_t length, int prot, int flags, int fd, off_t offset);
      • 功能:建立虚拟内存到物理内存或磁盘文件的映射
      • 参数
        • start:映射区虚拟内存的起始地址,NULL系统自动选定后返回。
        • length:映射区字节数,自动按页圆整,4k为一页。
        • prot:映射区操作权限,可取以下值,多选使用|隔开
          • PROT_READ:映射区可读
          • PROT_WRITE:映射区可写
          • PROT_EXEC:映射区可执行
          • PROT_NONE:映射区不可访问
        • flags:映射标志,可取以下值,多选使用|隔开
          • MAP_ANONYMOUS:匿名映射,将虚拟内存映射到物理内存中而非文件,使用这个就可以忽略df和offset参数,将这两个参数设为0即可
          • MAP_PRIVATE:对映射区的写操作只反映到缓冲区并不会真正写入文件,与MAP_SHARED二选一,不能同时存在
          • MAP_SHARED:对映射区的写操作直接反映到文件中
          • MAP_DENYWRITE:拒绝其他对文件的写操作
          • MAP_FIXED:若在start上无法创建映射,则失败(无此标志系统会自动调整)
        • fd:文件描述符,flags使用MAP_ANONYMOUS时,这里填0即可
        • offset:文件偏移量,自动按页(4K)对齐,flags使用MAP_ANONYMOUS时,这里填0即可
      • 返回值:成功返回映射区虚拟地址的起始地址,失败返回MAP_FAILED(-1)
  • munmap函数
    • int munmap(void *start, size_t length);
      • 功能:解除虚拟内存到物理内存或磁盘文件的映射
      • 参数:
        • start:映射区虚拟内存的起始地址
        • length:映射区字节数,自动按页圆整,4k为一页
      • 返回值:成功返回0,失败返回-1
      • munmap允许对映射区的一部分映射,但必须按页处理

使用示例:

#include <stdio.h>
#include <string.h>
#include <sys/mman.h>

int main(void)
{
    // 1. 建立映射: 返回值是void*类型,这里使用的是char*,内部会自动强转的
    char *start = mmap(NULL, 8192, PROT_READ | PROT_WRITE, MAP_ANONYMOUS | MAP_PRIVATE, 0, 0);
    if(start == MAP_FAILED)
    {
        perror("mmap");
        return -1;
    }

    // 2. 写入数据
    strcpy(start, "哈哈");
    printf("输出: %s\n", start);

    // 3. 解除一部分映射
    if(munmap(start, 4096) == -1)
    {
        perror("munmap");
        return -1;
    }

    // strcpy(start, "呵呵");  // 这里测试使用了已经解除的虚拟地址,出现了 段错误 (核心已转储)

    // 4. 在剩下的那一部分中写入数据
    start += 4096;  // 首先先将start往上移
    strcpy(start, "嘿嘿");
    printf("输出: %s\n", start);

    // 5. 解除
    if(munmap(start, 4096) == -1)
    {
        perror("munmap");
        return -1;
    }
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1030384.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Python 在 JMeter 中如何使用?

要在JMeter中使用Python&#xff0c;需要使用JSR223 Sampler元素来执行Python脚本。使用JSR223 Sampler执行Python脚本时&#xff0c;需要确保已在JMeter中配置了Python解释器&#xff0c;并设置了正确的环境路径。 1、确保JMeter已安装Python解释器&#xff0c;并将解释器的路…

时序预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元时间序列预测

时序预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元时间序列预测 目录 时序预测 | MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向门控循环单元时间序列预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 MATLAB实现POA-CNN-BiGRU鹈鹕算法优化卷积双向…

Jenkins学习笔记6

开发者开发代码一般会使用IDE集成开发工具&#xff08;比如pycharm这种),那么使用pycharm开发的代码能否直接利用自动发布系统发布到业务服务器上呢&#xff1f; 答案是肯定的。 然后进行下测试&#xff1a; 那说明SSH免密是成功的。 将Pycharm修改为原来的界面&#xff0c;然…

抖音短视频矩阵系统搭建

企业在进行短视频矩阵运营时&#xff0c;搭建一个矩阵号是非常必要的。矩阵号可以绑定多个不同平台的账号&#xff0c;批量制作和定时发布短视频&#xff0c;提高企业的曝光量和粉丝互动。但是&#xff0c;如何搭建一个有效的短视频矩阵号呢&#xff1f;以下是几个关键步骤。 一…

STM32 NVIC中断优先级管理通过结构图快速理解

STM32 NVIC中断优先级管理通过结构图快速理解 &#x1f4d1;抢占优先级和响应优先级基本常识 &#x1f33f;抢占优先级的级别高于响应优先级。&#x1f33f;抢占优先级数值编号越小&#xff0c;所代表的优先级就越高&#xff1b;同理&#xff0c;响应优先级也是如此。&#x1…

为什么要选择Spring cloud Sentinel

为什么要选择Spring cloud Sentinel &#x1f34e;对比Hystrix&#x1f342;雪崩问题及解决方案&#x1f342;雪崩问题&#x1f342;.超时处理&#x1f342;仓壁模式&#x1f342;断路器&#x1f342;限流&#x1f342;总结 &#x1f34e;对比Hystrix 在SpringCloud当中支持多…

使用 FHE 实现加密大语言模型

近来&#xff0c;大语言模型 (LLM) 已被证明是提高编程、内容生成、文本分析、网络搜索及远程学习等诸多领域生产力的可靠工具。 大语言模型对用户隐私的影响 尽管 LLM 很有吸引力&#xff0c;但如何保护好 输入给这些模型的用户查询中的隐私 这一问题仍然存在。一方面&#xf…

《从菜鸟到大师之路 Redis 篇》

《从菜鸟到大师之路 Redis 篇》 &#xff08;一&#xff09;&#xff1a;Redis 基础理论与安装配置 Nosql 数据库介绍 是一种 非关系型 数据库服务&#xff0c;它能 解决常规数据库的并发能力 &#xff0c;比如 传统的数据库的IO与性能的瓶颈 &#xff0c;同样它是关系型数据…

Android 11.0 禁止二次展开QuickQSPanel设置下拉QSPanel高度

1.前言 在11.0的系统定制化需求中,在进行systemui的ui定制开发中,有些产品中有需求对原生systemui下拉状态栏中的二次展开QSPanel修改成 一次展开禁止二次展开,所以就需要修改QuickQSpanel的高度,然后在QuickQsPanel做定制,然后禁止二次展开就可以了 如图: 2.禁止二次展开…

32.3D文本旋转动画效果

特效 源码 index.html <!DOCTYPE html> <html> <head> <title>CSS 3D Text Rotation</title> <link rel="stylesheet" type="text/css" href="style.css"> </head> <body><div class=&quo…

C++实现观察者模式(包含源码)

文章目录 观察者模式一、基本概念二、实现方式三、角色四、过程五、结构图六、构建思路七、完整代码 观察者模式 一、基本概念 观察者模式&#xff08;又被称为模型&#xff08;Model&#xff09;-视图&#xff08;View&#xff09;模式&#xff09;是软件设计模式的一种。在…

5G通信与蜂窝模组之间的关系

5G通信是第五代移动通信技术的简称&#xff0c;它代表了一种新一代的无线通信技术标准。5G通信的主要目标是提供更高的数据传输速度、更低的延迟、更大的网络容量以及更可靠的连接&#xff0c;以支持各种新兴应用和服务&#xff0c;包括高清视频流、虚拟现实、物联网&#xff0…

【软考中级】网络工程师:7.下一代互联网

IPv4问题与改进 IPv4存在以下著名的问题&#xff1a; 网络地址短缺&#xff08;32位&#xff09;以二进制数串表示&#xff0c;v4仅有43亿个地址&#xff0c;而IPv6有128位&#xff0c;且以十六进制数串表示。&#xff08;现在还能用v4得益于NAT地址转换&#xff09;地址分配…

pwn学习(3)BUUCTF-rip

下载文件&#xff0c;查看文件信息 IDA64打开&#xff0c;发现危险函数gets(),可以判断存在栈溢出漏洞 接着查看fun()函数&#xff0c;发现是system函数&#xff0c;system是C语言下的一个可以执行shell命令的函数 接下来思路就清晰了&#xff0c;需要用gets函数获取一个长字符…

电力安全智慧云平台:引领更安全的用电新时

电力能源是人类社会不可或缺的重要资源&#xff0c;其安全稳定供应关系到各行各业的正常运转和千家万户的生活质量。然而&#xff0c;随着电力使用的普及&#xff0c;电力安全问题也日益凸显&#xff0c;一旦发生电力事故&#xff0c;不仅会造成巨大的经济损失&#xff0c;还会…

python随手小练

题目&#xff1a; 使用python做一个简单的英雄联盟商城登录界面 具体操作&#xff1a; print("英雄联盟商城登录界面") print("~ * "*15 "~") #找其规律 a "1、用户登录" b "2、新用户注册" c "3、退出系统&quo…

rv1126-rv1109-test

测试指令 播放音频:aplay aigei.wav 测试时间: 查看系统时间:date 设置时间:date -s "2023-09-21 16:00:00" 设置芯片时间:hwclock -w 查看芯片时间:hwclock 测试背光: echo 0 > sys/class/backlight/backlight/brightness echo 50 > sys/class/backlig…

[python 刷题] 167 Two Sum II - Input Array Is Sorted 15 3Sum

[python 刷题] 167 Two Sum II - Input Array Is Sorted & 15 3Sum 虽然 3 sum 出来的比较早&#xff0c;不过按照解法来说&#xff0c;2 sum II 算是 3 sum 的前置解法 167 Two Sum II - Input Array Is Sorted 题目&#xff1a; Given a 1-indexed array of integers …

图像处理软件Photoshop 2024 mac新增功能

Photoshop 2024 mac是一款图像处理软件的最新版本。ps2024提供了丰富的功能和工具&#xff0c;使用户能够对照片、插图、图形等进行精确的编辑和设计。 Photoshop 2024 mac软件特点 快速性能&#xff1a;Photoshop 2024 提供了更快的渲染速度和更高效的处理能力&#xff0c;让用…

中秋国庆抢票用便签软件记录抢票细则

今年的中秋国庆又连为一体&#xff0c;大家都在庆幸着有一个八天的小长假&#xff0c;一年难得有一次这样的机会&#xff0c;趁着这个小长假必然是要好好出去玩一玩的&#xff1b;如果涉及到长途旅游&#xff0c;少不得提前在12306上进行抢票&#xff0c;怎么才能快速抢到票呢&…