Ashmem匿名共享內存驅動android
前面已經說過ashmem系統分爲三個層次,而這篇文章將要分析的就是cutils庫中的實現。函數
cutils庫就經過文件訪問操做open,ioctl來訪問驅動程序。另外cutils提供了五個C接口來訪問驅動程序,ashmem_create_region,ashmen_pin_region,ashmem_unpin_region,ashmem_set_prot_region,ashmem_get_size_region。spa
cutils庫層的實現位於/system/core/libcutils/ashmem-dev.c.net
這個方法用來向ashmem驅動程序請求爲應用程序建立一塊匿名共享內存,而且放回它的文件描述符。code
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; }
name和size分別表示匿名共享內存的名字和大小。blog
這個地方和2.3版本已經有些區別了,在2.3版本中咱們直接使用open函數打開驅動文件/dev/ashmem。以此和驅動程序創建關係。不過在8.0中,使用了__ashmem_open方法來打開驅動文件(其實是同樣的操做,只是多了同步鎖,還有重試和異常處理。)接口
在驅動程序講解中咱們知道,當咱們調用open函數,ashmem_open就會被調用,用來建立一個ashmem_area結構體表示一塊匿名共享內存。內存
而後就經過ioctl命令使用ASHMEM_SET_NAME爲其設置名稱,經過ASHMEM_SET_SIZE用來設置大小。get
該方法比較簡單同步
int ashmem_pin_region(int fd, size_t offset, size_t len) { struct ashmem_pin pin = { offset, len }; //2.3中不存在,可能會致使錯誤和被入侵風險,因此此處添加了對fd的類型驗證,確保是匿名共享內存的文件描述符 int ret = __ashmem_is_ashmem(fd, 1); if (ret < 0) { return ret; } return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_PIN, &pin)); }
向fd描述的文件發送io控制命令ASHMEM_PIN,用來鎖定一小塊內存區域。offset表示在匿名內存中的偏移位置,len表示長度。
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)); }
和 ashmem_pin_region 基本同樣,只是發送了ASHMEM_UNPIN命令用來解鎖一塊內存區域。
int ashmem_set_prot_region(int fd, int prot) { int ret = __ashmem_is_ashmem(fd, 1); if (ret < 0) { return ret; } return TEMP_FAILURE_RETRY(ioctl(fd, ASHMEM_SET_PROT_MASK, prot)); }
該方法僅用來設置匿名共享內存的訪問保護位。
用來獲取匿名共享內存大小。
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)); }