单片机内存管理剖析

一、概述

在单片机系统中,内存资源通常是有限的,因此高效的内存管理至关重要。合理地分配和使用内存可以提高系统的性能和稳定性,避免内存泄漏和碎片化问题。单片机的内存主要包括程序存储器(如 Flash)和数据存储器(如 RAM),其中数据存储器又可进一步分为静态数据区、栈区和堆区。动态内存分配主要发生在堆区,而 sbrkmallocfree 这三个函数 在堆内存管理中起着关键作用。

二、sbrk:底层的内存边界调整

2.1 原理

sbrk 是一个底层的系统调用(在某些单片机库中也有对应的实现),其核心功能是调整进程数据段的结束地址,也就是 break 指针。通过改变 break 指针的位置,可以实现堆内存的扩展和收缩。当传入一个正的增量值时,break 指针向后移动,堆内存得到扩展;当传入一个负的增量值时,break 指针向前移动,堆内存被收缩。

2.2 源码示例与解释

#include <stdint.h>
#include <errno.h>

// 假设这是链接脚本定义的堆起始和结束地址
extern char _end[];
extern char _heap_end[];
// 当前堆指针
static char *curbrk = _end;
// sbrk 函数实现
void *_sbrk(int incr) {
    char *old_brk = curbrk;
    char *new_brk = curbrk + incr;

    // 边界检查
    if (new_brk < _end || new_brk > _heap_end) {
        errno = ENOMEM;  // 设置错误号表示内存不足
        return (void *)-1;
    }

    curbrk = new_brk;
    return (void *)old_brk;
}
  • 全局变量
    • _end:由链接脚本确定,代表堆的起始地址。
    • _heap_end:同样由链接脚本确定,代表堆的最大可用地址。
    • curbrk:静态变量,记录当前堆的结束地址,初始化为 _end
  • 函数逻辑
    1. 保存当前的 curbrkold_brk 中,这将作为函数的返回值。
    2. 根据传入的 incr 计算新的堆结束地址 new_brk
    3. 进行边界检查,确保 new_brk 在合法范围内(不小于 _end 且不大于 _heap_end)。如果超出范围,设置 errnoENOMEM 并返回 (void *)-1 表示内存分配失败。
    4. 如果边界检查通过,更新 curbrknew_brk,并返回 old_brk,它指向新分配内存的起始位置。

2.3 使用场景和注意事项

  • 使用场景sbrk 通常作为底层的内存分配原语,为更高级的内存分配函数(如 malloc)提供支持。在一些简单的单片机应用中,如果只需要简单的内存扩展和收缩操作,也可以直接使用 sbrk
  • 注意事项
    • 由于 sbrk 直接操作堆的边界,使用不当可能会导致内存越界访问,破坏其他重要的数据。
    • sbrk 分配的内存是连续的,频繁的扩展和收缩操作可能会导致内存碎片化,降低内存的利用率。

三、malloc:用户级的动态内存分配

3.1 原理

malloc 是 C 标准库中提供的用于动态内存分配的函数,它建立在 sbrk 的基础之上。malloc 函数的主要任务是根据用户请求的内存大小,在堆中找到合适的空闲内存块并返回其起始地址。为了管理堆中的空闲内存,malloc 通常会维护一个空闲块链表,使用不同的分配策略(如首次适配、最佳适配等)来查找合适的空闲块。

3.2 源码示例与解释

#include <stdio.h>
#include <stdint.h>
#include <errno.h>
// 内存块结构体
typedef struct mem_block {
    size_t size;
    int is_free;
    struct mem_block *next;
} MemBlock;

// 空闲链表头指针
static MemBlock *free_list = NULL;

// 分配内存
void *malloc(size_t size) {
    MemBlock *current = free_list;
    MemBlock *prev = NULL;

    // 查找合适的空闲块
    while (current != NULL) {
        if (current->is_free && current->size >= size) {
            current->is_free = 0;
            // 如果空闲块比需求大,分割空闲块
            if (current->size > size + sizeof(MemBlock)) {
                MemBlock *new_free_block = (MemBlock *)((char *)current + sizeof(MemBlock) + size);
                new_free_block->size = current->size - size - sizeof(MemBlock);
                new_free_block->is_free = 1;
                new_free_block->next = current->next;
                current->size = size;
                current->next = new_free_block;
            }
            return (void *)(current + 1);
        }
        prev = current;
        current = current->next;
    }

    // 没有合适的空闲块,调用 sbrk 扩展堆
    size_t total_size = size + sizeof(MemBlock);
    MemBlock *new_block = (MemBlock *)_sbrk(total_size);
    if (new_block == (MemBlock *)-1) {
        return NULL;
    }
    new_block->size = size;
    new_block->is_free = 0;
    new_block->next = NULL;

    if (prev != NULL) {
        prev->next = new_block;
    } else {
        free_list = new_block;
    }

    return (void *)(new_block + 1);
}
  • 数据结构

    • MemBlock
      

      结构体:用于表示堆中的内存块,包含三个成员:

      • size:记录内存块的大小。
      • is_free:标记该内存块是否空闲。
      • next:指向下一个内存块的指针,用于构建空闲块链表。
    • free_list:指向空闲块链表的头指针,初始化为 NULL

  • 函数逻辑

    1. 查找空闲块:遍历空闲块链表 free_list,使用首次适配策略查找第一个大小足够的空闲块。
    2. 分割空闲块:如果找到的空闲块比请求的大小大,将其分割为两部分:一部分用于满足当前请求,另一部分作为新的空闲块插入到链表中。
    3. 扩展堆:如果在空闲块链表中没有找到合适的空闲块,调用 sbrk 函数扩展堆空间,分配一块新的内存,并将其初始化为一个新的内存块。
    4. 返回内存地址:返回分配的内存块的起始地址(跳过 MemBlock 结构体部分)。

在这里插入图片描述

四、free:动态内存的释放

4.1 原理

free 函数用于释放 malloccallocrealloc 分配的内存块。当调用 free 时,它会将指定的内存块标记为空闲,并尝试合并相邻的空闲块,以减少内存碎片化。

4.2 源码示例与解释

// 释放内存
void free(void *ptr) {
    if (ptr == NULL) return;

    // 获取内存块头部
    MemBlock *block = (MemBlock *)ptr - 1;
    block->is_free = 1;

    // 合并相邻的空闲块
    MemBlock *current = free_list;
    MemBlock *prev = NULL;

    // 找到合适的插入位置
    while (current != NULL && current < block) {
        prev = current;
        current = current->next;
    }

    // 合并前一个空闲块
    if (prev != NULL && prev->is_free) {
        prev->size += block->size + sizeof(MemBlock);
        prev->next = block->next;
        block = prev;
    }

    // 合并后一个空闲块
    if (current != NULL && current->is_free) {
        block->size += current->size + sizeof(MemBlock);
        block->next = current->next;
    }

    // 如果没有前一个块,更新空闲链表头
    if (prev == NULL) {
        free_list = block;
    } else {
        prev->next = block;
    }
    block->next = current;
}
  • 函数逻辑
    1. 空指针检查:如果传入的指针 ptrNULL,直接返回,不进行任何操作。
    2. 标记为空闲:通过指针计算得到内存块的头部信息(MemBlock 结构体),将其 is_free 标记设置为 1,表示该内存块已空闲。
    3. 合并相邻空闲块
      • 遍历空闲块链表,找到合适的位置插入该空闲块。
      • 检查前一个和后一个内存块是否空闲,如果是,则将它们合并成一个更大的空闲块。
    4. 更新空闲链表:根据合并结果更新空闲块链表的指针,确保链表的正确性。

4.3 使用场景和注意事项

  • 使用场景:在不再需要使用动态分配的内存时,必须调用 free 函数释放内存,以避免内存泄漏。
  • 注意事项
    • 只能释放由 malloccallocrealloc 分配的内存,释放其他内存可能会导致未定义行为。
    • 不要多次释放同一块内存,这会导致双重释放错误,可能会破坏内存管理数据结构。

在这里插入图片描述

sbrkmallocfree 是单片机内存管理中重要的工具,它们相互协作,实现了堆内存的动态分配和释放。sbrk 作为底层的系统调用,提供了基本的内存扩展和收缩功能;malloc 基于 sbrk 实现了用户级的动态内存分配接口,方便程序员在运行时分配所需的内存;free 则负责释放不再使用的内存,避免内存泄漏和碎片化。在实际应用中,需要合理使用这些函数,注意内存的分配和释放规则,以确保系统的稳定性和性能。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络 (61)移动IP

前言 移动IP&#xff08;Mobile IP&#xff09;是由Internet工程任务小组&#xff08;Internet Engineering Task Force&#xff0c;IETF&#xff09;提出的一个协议&#xff0c;旨在解决移动设备在不同网络间切换时的通信问题&#xff0c;确保移动设备可以在离开原有网络或子网…

线性回归、协同过滤、基于内容过滤、主成分分析(PCA)

线性回归 使用item特征用户打分标签线性回归训练&#xff0c;最小化成本函数&#xff0c;得到每个用户的参数 协同过滤 协同过滤基于一个核心假设&#xff1a;相似的用户会有相似的兴趣&#xff0c;因此可以通过分析相似用户历史行为&#xff0c;来预测当前用户可能感兴趣的i…

引领产品创新: 2025 年 PM 效能倍增法则

本文讲述 PM 如何利用 AI 做到效率倍增&#xff0c;非常有借鉴意义&#xff0c;故而翻译于此。 原文链接&#xff1a;https://www.news.aakashg.com/p/the-ai-pms-playbook 在产品圈有一个广为流传的说法&#xff1a; “每个产品经理都应该成为 AI 产品经理。” 这个观点有一…

vscode无法格式化go代码的问题

CTRLshiftp 点击Go:Install/Update Tools 点击全选&#xff0c;OK&#xff01;

【外文原版书阅读】《机器学习前置知识》1.线性代数的重要性,初识向量以及向量加法

目录 ​编辑 ​编辑 1.Chapter 2 Why Linear Algebra? 2.Chapter 3 What Is a Vector? 个人主页&#xff1a;Icomi 大家好&#xff0c;我是Icomi&#xff0c;本专栏是我阅读外文原版书《Before Machine Learning》对于文章中我认为能够增进线性代数与机器学习之间的理解的…

对神经网络基础的理解

目录 一、《python神经网络编程》 二、一些粗浅的认识 1&#xff09; 神经网络也是一种拟合 2&#xff09;神经网络不是真的大脑 3&#xff09;网络构建需要反复迭代 三、数字图像识别的实现思路 1&#xff09;建立一个神经网络类 2&#xff09;权重更新的具体实现 3&am…

SOME/IP--协议英文原文讲解1

前言 SOME/IP协议越来越多的用于汽车电子行业中&#xff0c;关于协议详细完全的中文资料却没有&#xff0c;所以我将结合工作经验并对照英文原版协议做一系列的文章。基本分三大块&#xff1a; 1. SOME/IP协议讲解 2. SOME/IP-SD协议讲解 3. python/C举例调试讲解 一、SOM…

移动光猫怎么自己改桥接模式?

环境&#xff1a; 型号H3-8s 问题描述&#xff1a; 家里宽带用的是H3-8s 光猫&#xff0c;想改桥接模式。 解决方案&#xff1a; 1.默认管理员账号和密码&#xff1a; 账号&#xff1a;CMCCAdmin 密码&#xff1a;aDm8H%MdAWEB页面我试了登陆不了&#xff0c;显示错误 …

2D 超声心动图视频到 3D 心脏形状重建的临床应用| 文献速递-医学影像人工智能进展

Title 题目 2D echocardiography video to 3D heart shape reconstruction for clinicalapplication 2D 超声心动图视频到 3D 心脏形状重建的临床应用 01 文献速递介绍 超声心动图是心血管医学中一种至关重要且广泛应用的影像学技术&#xff0c;利用超声波技术捕捉心脏及其…

web端ActiveMq测试工具

如何用vue3创建简单的web端ActiveMq测试工具&#xff1f; 1、复用vue3模板框架 创建main.js,引入APP文件&#xff0c;createApp创建文件&#xff0c;并加载element插件&#xff0c;然后挂载dom节点 2、配置vue.config.js脚本配置 mport { defineConfig } from "vite&qu…

STM32 GPIO配置 点亮LED灯

本次是基于STM32F407ZET6做一个GPIO配置&#xff0c;实现点灯实验。 新建文件 LED.c、LED.h文件&#xff0c;将其封装到Driver文件中。 双击Driver文件将LED.c添加进来 编写头文件&#xff0c;这里注意需要将Driver头文件声明一下。 在LED.c、main.c里面引入头文件LED.h LED初…

DroneXtract:一款针对无人机的网络安全数字取证工具

关于DroneXtract DroneXtract是一款使用 Golang 开发的适用于DJI无人机的综合数字取证套件&#xff0c;该工具可用于分析无人机传感器值和遥测数据、可视化无人机飞行地图、审计威胁活动以及提取多种文件格式中的相关数据。 功能介绍 DroneXtract 具有四个用于无人机取证和审…

用Python和PyQt5打造一个股票涨幅统计工具

在当今的金融市场中&#xff0c;股票数据的实时获取和分析是投资者和金融从业者的核心需求之一。无论是个人投资者还是专业机构&#xff0c;都需要一个高效的工具来帮助他们快速获取股票数据并进行分析。本文将带你一步步用Python和PyQt5打造一个股票涨幅统计工具&#xff0c;不…

大模型正确调用方式

1、ollama 安装 curl -fsSL https://ollama.com/install.sh | sh 如果是AutoDl服务器&#xff0c;可以开启学术加速。 source /etc/network_turbo 本次使用腾讯云Cloud Studio&#xff0c;所以已经安装好了 Ollama 2、启动 ollama run 模型的名字 ollama serve # 开启服务 olla…

计算机网络 (62)移动通信的展望

一、技术发展趋势 6G技术的崛起 内生智能&#xff1a;6G将强调自适应网络架构&#xff0c;通过AI驱动的智能算法提升通信能力。例如&#xff0c;基于生成式AI的6G内生智能架构将成为重要研究方向&#xff0c;实现低延迟、高效率的智能通信。信息编码与调制技术&#xff1a;新型…

KVM/ARM——基于ARM虚拟化扩展的VMM

1. 前言 ARM架构为了支持虚拟化做了些扩展&#xff0c;称为虚拟化扩展(Virtualization Extensions)。原先为VT-x创建的KVM(Linux-based Kernel Virtual Machine)适配了ARM体系结构&#xff0c;引入了KVM/ARM (the Linux ARM hypervisor)。KVM/ARM没有在hypervisor中引入复杂的…

python:taichi 模拟一维波场

在 Taichi 中模拟一维波场&#xff0c;通常是利用 Taichi 编程语言的特性来对一维空间中的波动现象进行数值模拟&#xff0c;以下是相关介绍&#xff1a; 原理基础 波动方程&#xff1a;一维波动方程的一般形式为 &#xff0c;其中 u(x,t) 表示在位置x 和时间t 处的波的状态&…

LeetCode - Google 大模型校招10题 第1天 Attention 汇总 (3题)

欢迎关注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://spike.blog.csdn.net/article/details/145368666 GroupQueryAttention(分组查询注意力机制) 和 KVCache(键值缓存) 是大语言模型中的常见架构&#xff0c;GroupQueryAttention 是注意力…

视觉语言模型 (VLMs):跨模态智能的探索

文章目录 一. VLMs 的重要性与挑战&#xff1a;连接视觉与语言的桥梁 &#x1f309;二. VLMs 的核心训练范式&#xff1a;四种主流策略 &#x1f5fa;️1. 对比训练 (Contrastive Training)&#xff1a;拉近正例&#xff0c;推远负例 ⚖️2. 掩码方法 (Masking)&#xff1a;重构…

java8-日期时间Api

目录 LocalDate更新时间LocalTimeLocalDateTimeInstantPeriod Duration格式化、解析日期-时间对象时区 java.util.Date java.util.Calendar 不支持时区 线程不安全 月份从0起线程不安全&#xff0c;只有包裹在ThreadLocal中才安全 java.text.DateFormat java.text.SimpleDateFo…