关于malloc源码中的bin_at宏定义的个人见解
0x01:简介
在堆中的内存申请和释放中,为了减少使用系统调用函数对内存操作,malloc_state(分配区)结构中使用了fastbinsY数组和bins数组。当chunk被free后,bins链会将这些free chunk组织起来。当下次malloc时,会先对bins链中的free chunk进行遍历,有适合的则使用,无合适的再进行下一步操作。
在申请和释放操作时,一般是有一个arena(分配区,其为malloc_state结构类型)。通过宏定义bin_at(m,i)即可获得对相应bins链进行操作,m为分配区,i为索引,当i=1是为unsorted bin,i=2~63是small bin,i=64~126为large bin。
bin_at可以这样使用:
p = bin_at(m,1) //unsorted bin
FD = p -> fd //FD指向该链的第一个free chunk
BK = p -> bk //BK指向该链的最后一个free chunk
0x02:个人理解
该宏定义为这样子,主要是为了操作方便(可直接fd、bk指向)和在节省空间之间取平衡。既操作方便又不浪费空间。
bin_at(m,i)宏定义原型:
#define bin_at(m, i) \
(mbinptr) (((char *) &((m)->bins[((i) - 1) * 2])) \
- offsetof (struct malloc_chunk, fd))
解释:
1、&((m)->bins[((i)-1)*2]),该式子根据bin的索引i,i为1时,即最开始的bin(unsorted bin)。
获得bins[0]的地址,乘于2主要是因为fd和bk是一对存储的。
2、offsetof(struct malloc_chunk,fd):得到fd成员在malloc_chunk结构中的偏移量。在64位系统系统下为16。
把第一步得到地址(令其为pt)转化为char*型,这样子减偏移值: pt - offsetof = pt - offsetof*(sizeof(char))。
3、重点:
在第二步下,第一步得到的地址指向往后推移了两个单元,比如图中的bin_at(m,1)得到的是bins[0]前两个单元所在的地址。
然后经过最后的(mbinptr)转化为malloc_chunk*类型,这样就可以有->fd、->bk操作。
相当于是糊弄了操作系统,让操作系统误以为是个chunk结构。

语言表达不够,画图来凑~~
0x03:测试
1 测试代码:
2 #include<stdio.h>
3 #include<malloc.h>
4
5 int main()
6 {
7 void* p1 = malloc(0x100);
8 void* f1 = malloc(0x10); //防止合并
9 void* p2 = malloc(0x100);
10 void* f2 = malloc(0x10);
11 void* p3 = malloc(0x100);
12 void* f3 = malloc(0x10);
13
14 sleep(0); //方便下断点
15
16 free(p1);
17 free(p2);
18 free(p3);
19
20 sleep(0); //下图断在了这里
21 return 0;
22 }

可以看出unsorted bin的第一个单元是指向bin链的第一个free chunk,另外一个单元指向bin链的最后一个free chunk。图中我已经标出了指向top chunk的那个单元,也有:bin_at(m,1)指向该单元 ,或者更本质一点:bin_at(m,1) = 0x7ffffdd1b78。
0x04:思考
在刚开始看这个宏定义时,一脸懵逼,觉得这个宏定义是有问题吧,减去16不是超出了范围了么?后来继续分析源码时发现了->fd和->bk的操作,陷入了沉思。巧的是,群里有师傅刚好提到这个问题,所以和该师傅交流了一下,最后得到了一个比较合理的解释。
但是令我疑惑的是,为什么在后面会留两个空白单元嘞?留在前面不是更恰当么?bin_at(m,1)指向了bins数组外面的单元(存top chunk地址的单元),这样会不会存有遗患呢?有想法的欢迎与我交流^_^,QQ:1623093551。
tolele 2022-03-16
关于malloc源码中的bin_at宏定义的个人见解的更多相关文章
- Kafka源码中的Producer Record定义
1.ProducerRecord 含义: 发送给Kafka Broker的key/value 值对 2.内部数据结构: -- Topic (名字) -- PartitionID ( 可选) -- Ke ...
- Eclipse与Android源码中ProGuard工具的使用
由于工作需要,这两天和同事在研究android下面的ProGuard工具的使用,通过查看android官网对该工具的介绍以及网络上其它相关资料,再加上自己的亲手实践,算是有了一个基本了解.下面将自己的 ...
- php源码分析之PHPAPI宏的作用
在PHP源码中,我们经常会看到很多函数前面有个PHPAPI,但这是什么呢? 于是我在php源码/main/php.h中找到了它的定义 #ifdef PHP_WIN32 # include " ...
- Eclipse与Android源码中ProGuard工具的使用(代码混淆)
由于工作需要,这两天和同事在研究android下面的ProGuard工具的使用,通过查看android官网对该工具的介绍以及网络上其它相关资料,再加上自己的亲手实践,算是有了一个基本了解.下面将自己的 ...
- 从express源码中探析其路由机制
引言 在web开发中,一个简化的处理流程就是:客户端发起请求,然后服务端进行处理,最后返回相关数据.不管对于哪种语言哪种框架,除去细节的处理,简化后的模型都是一样的.客户端要发起请求,首先需要一个标识 ...
- Android 网络框架之Retrofit2使用详解及从源码中解析原理
就目前来说Retrofit2使用的已相当的广泛,那么我们先来了解下两个问题: 1 . 什么是Retrofit? Retrofit是针对于Android/Java的.基于okHttp的.一种轻量级且安全 ...
- String源码中的"avoid getfield opcode"
引言: 之前一篇文章梳理了String的不变性原则,还提到了一段源码中注释"avoid getfield opcode",当时通过查阅资料发现,这是为了防止 getfield(获取 ...
- android源码中修改wifi热点默认始终开启
在项目\frameworks\base\wifi\java\android\net\wifi\WifiStateMachine.java里面,有如下的代码,是设置wifi热点保持状态的:如下: pri ...
- rxjava源码中的线程知识
rxjava源码中的线程知识 rx的最精简的总结就是:异步 这里说一下以下的五个类 1.Future2.ConcurrentLinkedQueue3.volatile关键字4.AtomicRefere ...
随机推荐
- Static块和类加载顺序
原创:转载需注明原创地址 https://www.cnblogs.com/fanerwei222/p/11451040.html 版本:Java8 直接上代码: public class Stati ...
- 使用Docker安装ElasticSearch和可视化界面Kibana【图文教学】
一.前言 Elasticsearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful web接口.Elasticsearch是用Java语言开发的,并 ...
- 4G无线全网通太阳能水文设备电源监测系统BMS110
钡铼技术BMS110模块可实现4路电池电压.2路模拟量.2路数字量和1路温度测量,支持Modbus RTU over TCP和MQTT通讯协议,DC9-36V电源供电.BMS110可应用于各种有使用蓄 ...
- 274-基于XC7V690T的3U VPX信号处理板
一.板卡概述 本板卡系我司自主研发的基于3U VPX导冷架构的信号处理板,适用于高速图像处理,雷达信号处理等.芯片采用工业级设计.该处理板包含1片Xilinx公司的Virtex7系列FPGA-XC7V ...
- Solution -「ARC 101D」「AT4353」Robots and Exits
\(\mathcal{Description}\) Link. 有 \(n\) 个小球,坐标为 \(x_{1..n}\):还有 \(m\) 个洞,坐标为 \(y_{1..m}\),保证上述坐标 ...
- git忽略文件提交 .gitignore
Git 忽略规则 详细的忽略规则可以参考官方英文文档 Git 忽略规则优先级 在 .gitingore 文件中,每一行指定一个忽略规则,Git 检查忽略规则的时候有多个来源,它的优先级如下(由高到低) ...
- 深入MySQL(四):MySQL的SQL查询语句性能优化概述
关于SQL查询语句的优化,有一些一般的优化步骤,本节就介绍一下通用的优化步骤. 一条查询语句是如何执行的 首先,我们如果要明白一条查询语句所运行的过程,这样我们才能针对过程去进行优化. 参考我之前画的 ...
- 多表查询思路、navicat可视化软件、python操作MySQL、SQL注入问题以及其他补充知识
昨日内容回顾 外键字段 # 就是用来建立表与表之间的关系的字段 表关系判断 # 一对一 # 一对多 # 多对多 """通过换位思考判断""" ...
- logstash获取nginx日志的配置
nginx部分配置直接用json,省去很多麻烦 log_format json '{"@timestamp":"$time_iso8601",' '" ...
- rinetd基于内网TCP端口转发
在Linux系统中大多数情况选择用iptables来实现端口转发,iptables虽然强大,但配置不便,而且新手容易出错.在此分享另一个TCP/UDP端口转发工具rinetd,rinetd体积小巧,配 ...