android匿名共享内存Ashmem(c库接口)

Ashmem

Android系统的匿名共享内存Ashmem驱动程序利用了Linux的共享内存子系统导出的接口来实现。

在Android系统中,匿名共享内存也是进程间通信方式的一种。

相比于malloc和anonymous/named mmap等传统的内存分配机制,Ashmem的优势是通过内核驱动提供了辅助内核的内存回收算法机制(pin/unpin)。

内存回收算法机制就是当你使用Ashmem分配了一块内存,但是其中某些部分却不会被使用时,那么就可以将这块内存unpin掉。

unpin后,内核可以将它对应的物理页面回收,以作他用。你也不用担心进程无法对unpin掉的内存进行再次访问,因为回收后的内存还可以再次被获得(通过缺页handler),因为unpin操作并不会改变已经 mmap的地址空间。

android匿名共享内存接口

源码是最好的老师,废话不多说,直接看代码。

源码路径:system/core/libcutils/ashmem-dev.c

android源码中,ashmem的实现:
打开共享内存:


/*
 * ashmem_create_region - creates a new ashmem region and returns the file
 * descriptor, or <0 on error
 *
 * `name' is an optional label to give the region (visible in /proc/pid/maps)
 * `size' is the size of the region, in page-aligned bytes
 */

int ashmem_create_region(const char *name, size_t size)
{
    int ret, save_errno;
    
    int fd = __ashmem_open();
    if (fd < 0) {
        return fd;
    }
    
    if (name) {
        char buf[ASHMEM_NAME_LEN] = {0};
        
        strlcpy(buf, name, sizeof(buf));
        ret = TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_NAME, buf));
        if (ret < 0) {
            goto error;
        }
    }
    
    ret = TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_SIZE, size));
    if (ret < 0) {
        goto error;
    }
    
    return fd;

error:
    save_errno = errno;
    close(fd);
    errno = save_errno;
    return ret;
}

在函数中调用驱动接口:

__ashmem_open

__ashmem_open函数的实现如下:

/* logistics of getting file descriptor for ashmem */
static int __ashmem_open_locked()
{
    int ret;
    struct stat st;

    int fd = TEMP_FAILURE_RETRY(open(ASHMEM_DEVICE, O_RDWR));
    if (fd < 0) {
        return fd;
    }

    ret = TEMP_FAILURE_RETRY(fstat(fd, &st));
    if (ret < 0) {
        int save_errno = errno;
        close(fd);
        errno = save_errno;
        return ret;
    }
    if (!S_ISCHR(st.st_mode) || !st.st_rdev) {
        close(fd);
        errno = ENOTTY;
        return -1;
    }

    __ashmem_rdev = st.st_rdev;
    return fd;
}

static int __ashmem_open()
{
    int fd;

    pthread_mutex_lock(&__ashmem_lock);
    fd = __ashmem_open_locked();
    pthread_mutex_unlock(&__ashmem_lock);

    return fd;
}

可见函数最后是通过open去操作ashmem驱动文件。
返回为一个文件描述符。

int ashmem_valid(int fd) 
{
    return __ashmem_is_ashmem(fd, 0) >= 0;
}

除此之外,源码中还提供了几个接口函数:

1. 锁定匿名共享内存块

int ashmem_pin_region(int fd, size_t offset, size_t len)
{
    struct ashmem_pin pin = { offset, len };

    int ret = __ashmem_is_ashmem(fd, 1);
    if (ret < 0) {
        return ret;
    }

    return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_PIN, &pin));
}

2. 解锁匿名共享内存块

int ashmem_unpin_region(int fd, size_t offset, size_t len)
{
    struct ashmem_pin pin = { offset, len };

    int ret = __ashmem_is_ashmem(fd, 1);
    if (ret < 0) {
        return ret;
    }

    return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_UNPIN, &pin));
}

3. 获取大小

int ashmem_get_size_region(int fd)
{
    int ret = __ashmem_is_ashmem(fd, 1);
    if (ret < 0) {
        return ret;
    }

    return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_GET_SIZE, NULL));
}


因为是文件描述符,所以关闭直接采用close.

close(fd)

使用例子

创建共享内存

fd = ashmem_create_region(NULL,length);
if(fd < 0)
    printf("Creating code cache, ashmem_create_region error.");          

将共享内存映射到用户空间

data = (char *)mmap(NULL, data.length, PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0); 
if(data != MAP_FAILED){
    printf("mmap sharemem success....");
    memcpy(data.data,gucDotBuffer,length);
}else{
    printf("mmap sharemem failed....'%s'",strerror(errno));
}

关闭映射并关闭共享内存文件

munmap(data,length);
close(fd);
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 14,469评论 2 59
  • 阅读之前,不妨先思考一个问题,在Android系统中,APP端View视图的数据是如何传递SurfaceFling...
    看书的小蜗牛阅读 63,881评论 31 159
  • 人心到底有多恶毒,细思极恐。 知乎上看到一句话:如果看书(《恶意》)的时候不知道后面会有多少页会更爽。 ...
    风继续吹SH阅读 3,392评论 0 0
  • 《红楼梦》这部巨著被后人称之为百科全书,它包括了诗词歌赋、琴棋书画、医疗烹饪、园林建筑、服饰收藏、社会风俗等等,当...
    jasmine南京阅读 4,748评论 0 1
  • 不知从何时起,内心变得异常柔弱、易感,仿佛被剥了壳的蜗牛,不堪一触...... 周末,姐妹们相约去赏花,本来是兴致...
    若素约时光阅读 3,882评论 0 1