ncnn fastMalloc函数 内存对齐式分配 代码浅析

1 起因

工作需要,简单熟悉下ncnn(腾讯的一个神经网络前向计算框架)相关的源码。看到有关内存对齐的一段代码,忍不住分析一番。

涉及到C语言的指针、内存,总会让人有些头大,但我们又不得不承认,它也确是非常有趣。

2 源码 & Demo & 问题

nccc git地址
https://github.com/Tencent/ncnn
简单Demo
allocator.h 中的 alignPtrfastMalloc 两个函数提取出来,做一个简单的调用Demo,具体如下。

Demo & 问题

如果对上述问题感兴趣,你可以继续往下看了:)

3 分析

Step n 和 第2部分代码中的是一一对应的哦!

3.1 Step1 : 分配内存

unsigned char* udata = (unsigned char*)malloc(size + sizeof(void*) + MALLOC_ALIGN);
内存分配
  • 问题1:为什么要多分配 sizeof(void *) 大小的内存?(图中8B部分)
    因为设计者期望保存对其前的内存指针。
  • 问题2:为什么要多分配 MALLOC_ALIGN 大小的内存?(图中16B部分)
    因为对齐单位是16(MALLOC_ALIGN),而对其后,数据指针不一定指在分配的内存的起始位置,为了保证对其后仍然有100B目标内存可用,要+16。

3.2 Step2 : 对其方法调用分析

unsigned char** adata = alignPtr((unsigned char**)udata + 1, MALLOC_ALIGN);
关于udata+1
  • 问题3:为什么要将udata转换成(unsigned char **)?
    1)现在udata是char *类型,假设udata指向的内存是0x0000000102803030,那么udata+1 = 0x0000000102803031, 而(unsigned char**)udata + 1 = 0x0000000102803038,我们期望在内存地址的开始初留出一个存放指针的空间(8B),所以选择后者。
    2)我们要使用留出的8B的空间存储对齐前的内存地址,而对齐前的指针的类型(在这个函数体中)是unsigned char *,那么将内存看做一个unsigned char * 的数组的指针,再以此存储我们的对齐前内存地址(unsigned char *类型),多么的优雅,多么的符合概念啊!

Step3 : 实现偏移

template<typename _Tp> static inline _Tp* alignPtr(_Tp* ptr, int n=(int)sizeof(_Tp)) {
    return (_Tp*)(((size_t)ptr + n-1) & -n);
}
实现地址偏移
  • 问题4:_Tp是什么类型?
    unsigned char *
  • 问题5:& -n 如何理解?
    1)目的:n = 16 (B),这边的目的是让内存指向的地址为16的整数倍
    2)目的达成分析: 16的二进制值是 0001 0000,-16的二进制值是16二进制值的各位取反(...0001 1111),再加1(...1111 0000)。如果一个地址是64位的,这么操作后就是0xffff ffff ffff fff0,与目标数字按位与后,自然就是一个能被16整除的数字了。
    3)举一反三: 不仅16,2^n的数字都可以这么操作哦
  • 问题6:什么是&,什么是按位与
    ……这个……不解释……

Step4 : 设置未对齐前的内存地址

adata[-1] = udata;
设置对齐前内存地址
  • 问题7:为什么要设置?
    宝贝儿,因为内存释放的时候要用~ 代码如下,不解释咯!
static inline void fastFree(void* ptr) {
    if (ptr) {
        unsigned char* udata = ((unsigned char**)ptr)[-1];
        free(udata);
    }
}

Step5 : 返回用户关注的内存地址

return adata;
image.png

用户拿到的内存指针指向的是数据的开始位置哦!

  • 问题8:为啥要内存对齐,而且为了对齐还浪费了部分内存?
    简单来说,为了提高运算速度,更深层的原因比较复杂,不赘述咯。

4 草草结束

完~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容

  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 10,505评论 6 13
  • ———————————————回答好下面的足够了---------------------------------...
    恒爱DE问候阅读 1,673评论 0 4
  • 史上最全的iOS面试题及答案 iOS面试小贴士———————————————回答好下面的足够了----------...
    Style_伟阅读 2,318评论 0 35
  • 常常在半睡半醒的时候起来,不知道为什么,静静坐着,像痴呆一样。老半天想不起什么,仿佛眼前只是一片荒芜的苍白。 又慢...
    禅笑大泉阅读 742评论 0 0
  • 曾经见过一句话,原文不记得了,大意就是:男孩子要学会赞美姑娘漂亮,如果她不漂亮,就要赞美她有气质,如果她既不漂亮也...
    曲晓岩阅读 393评论 0 2