美文网首页Redis源码学习笔记
Redis源码学习之内存管理

Redis源码学习之内存管理

作者: lixin_karl | 来源:发表于2019-04-04 18:16 被阅读0次

    zmalloc.h/zmalloc.c

    //内存管理函数 malloc 封装的版本 考虑到字节对齐64位的话就是8字节对齐
    
    static size_t used_memory = 0;
    //申请内存
    void *zmalloc(size_t size){
        void *ptr = malloc(size+PREFIX_SIZE);//申请内存size+一个长度 需要的内存
    
        if (!ptr) zmalloc_oom_handler(size);//申请失败处理
    #ifdef HAVE_MALLOC_SIZE
        update_zmalloc_stat_alloc(zmalloc_size(ptr));
        return ptr;
    #else
        *((size_t*)ptr) = size;
        update_zmalloc_stat_alloc(size+PREFIX_SIZE);//更新used_memory
        return (char*)ptr+PREFIX_SIZE;//返回需要的申请出来的内存的首地址
    #endif
    }
    
    //对于64位机器有全局变量used_memory增加((__n%8?==0?0:8-__n%8) +__n) 即字节对齐所做的操作
    #define update_zmalloc_stat_alloc(__n) do { \
        size_t _n = (__n); \
        if (_n&(sizeof(long)-1)) _n += sizeof(long)-(_n&(sizeof(long)-1)); \
        atomicIncr(used_memory,__n); \
    } while(0)
    
    void *zcalloc(size_t size);//同上
    void *zrealloc(void *ptr, size_t size);//重新申请内存,与申请操作相反
    void zfree(void *ptr);//释放内存
    char *zstrdup(const char *s);//复制C字符串
    size_t zmalloc_used_memory(void);//获取used_memory的值  线程安全的
    void zmalloc_set_oom_handler(void (*oom_handler)(size_t));//内存申请失败的处理函数
    
    //以下都是smaps中的相关数据
    size_t zmalloc_get_rss(void);//-Resident Set Size 实际使用物理内存
    int zmalloc_get_allocator_info(size_t *allocated, size_t *active, size_t *resident);//得到已经申请了内存数,活跃的,长期存在的内存数
    size_t zmalloc_get_private_dirty(long pid);//smap中得到Private_Dirty大小
    size_t zmalloc_get_smap_bytes_by_field(char *field, long pid);//得到特定key对应的值
    size_t zmalloc_get_memory_size(void);//RAM大小
    void zlibc_free(void *ptr);//释放内存
    
    
    size_t zmalloc_size(void *ptr);//返回后实际申请的内存长度
    size_t zmalloc_usable(void *ptr);//返回除去长度内存后实际数据使用的长度
    

    smaps介绍

    smaps在/proc/pid(进程id)/ 文件夹里

    VmFlags: mr mw me sd 
    
    7fa786ee2000-7fa786ee3000 r--p 00061000 fc:00 262679                     /lib/x86_64-linux-gnu/libisc-export.so.160.0.0
    上一行个字段意义为:
    1. 地址空间的开始地址 - 结束地址  
    2. 属性。前三个是rwx(读、写、可执行),如果不具有则为“-”。最后一个是p/s(私有/共享) 
    3. 偏移量。如果这段内存是从文件里映射过来的,则偏移量为这4.
    4. 段内容在文件中的偏移量。如果不是从文件里面映射过来的则为0. 
    5. 如果是从文件映射过来的则分号两边分别表示最大,最小文件字符
    6. 如果是从文件映射过来的则表示文件编号
    7. 如果是从文件映射过来的则表示文件名
    Size:                  4 kB
    Rss:                   4 kB
    Rss-Resident Set Size 实际使用物理内存(包含共享库占用的
    内存) Rss=Shared_Clean+Shared_Dirty+Private_Clean+Private_Dirty 
    Pss:                   4 kB
    Pss 实际使用的物理内存(按比例包含共享库占用的内存)。比如四个进程
    共享同一个占内存1000MB的共享库,每个进程算进250MB在Pss。
    Shared_Clean:          0 kB//共享的干净数据
    Shared_Dirty:          0 kB//共享的脏数据
    Private_Clean:         0 kB//私有干净数据
    Private_Dirty:         4 kB//私有脏数据
    Referenced:            4 kB//
    Anonymous:             4 kB
    AnonHugePages:         0 kB
    Shared_Hugetlb:        0 kB
    Private_Hugetlb:       0 kB
    Swap:                  0 kB
    SwapPss:               0 kB
    KernelPageSize:        4 kB
    MMUPageSize:           4 kB
    Locked:                0 kB
    

    参考

    相关文章

      网友评论

        本文标题:Redis源码学习之内存管理

        本文链接:https://www.haomeiwen.com/subject/padqiqtx.html