日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > Android >内容正文

Android

Android 图形系统之gralloc

發布時間:2024/4/11 Android 38 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Android 图形系统之gralloc 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

Gralloc1::Loader 與 gralloc 模塊加載

Gralloc1::Loader 用于加載 HAL gralloc 模塊。其類定義(位于 frameworks/native/include/ui/Gralloc1.h)如下:

class Loader { public:Loader();~Loader();std::unique_ptr<Device> getDevice();private:static std::unique_ptr<Gralloc1On0Adapter> mAdapter;std::unique_ptr<Device> mDevice; };

這個類只包含構造函數,析夠函數和一個 Get 函數用于獲得 Gralloc1::Device。該類全部方法實現(位于frameworks/native/libs/ui/Gralloc1.cpp)如下:

Loader::Loader(): mDevice(nullptr) {hw_module_t const* module;int err = hw_get_module(GRALLOC_HARDWARE_MODULE_ID, &module);uint8_t majorVersion = (module->module_api_version >> 8) & 0xFF;uint8_t minorVersion = module->module_api_version & 0xFF;gralloc1_device_t* device = nullptr;if (majorVersion == 1) {gralloc1_open(module, &device);} else {if (!mAdapter) {mAdapter = std::make_unique<Gralloc1On0Adapter>(module);}device = mAdapter->getDevice();}mDevice = std::make_unique<Gralloc1::Device>(device); }Loader::~Loader() {}std::unique_ptr<Device> Loader::getDevice() {return std::move(mDevice); }

Gralloc1::Loader 在其構造函數中通過 hw_get_module() 函數完成 HAL gralloc 模塊的加載,隨后創建一個 Gralloc1::Device 。當 gralloc API 版本為 1 時,通過 frameworks/native/include/ui/Gralloc1.h 中定義的 gralloc1_open() 函數創建:

static inline int gralloc1_open(const struct hw_module_t* module,gralloc1_device_t** device) {return module->methods->open(module, GRALLOC_HARDWARE_MODULE_ID,(struct hw_device_t**) device); }

當 gralloc API 版本不為 1 時,則創建 Gralloc1On0Adapter 作為 Gralloc1::Device,相關函數(位于frameworks/native/include/ui/Gralloc1On0Adapter.h)如下:

class Gralloc1On0Adapter : public gralloc1_device_t { public:Gralloc1On0Adapter(const hw_module_t* module);~Gralloc1On0Adapter();gralloc1_device_t* getDevice() {return static_cast<gralloc1_device_t*>(this);}

Gralloc1::Loader 的構造函數為 hw_get_module() 傳入模塊 ID 和 hw_module_t 指針 module 加載 gralloc 模塊,其中 module 用于接收加載的結果。gralloc 的模塊 ID 定義(位于 hardware/libhardware/include/hardware/gralloc1.h)如下:

#define GRALLOC_HARDWARE_MODULE_ID "gralloc"

hw_get_module() 函數用于加載 HAL gralloc 模塊,其定義(位于 hardware/libhardware/hardware.c)如下:

static const char *variant_keys[] = {"ro.hardware", /* This goes first so that it can pick up a differentfile on the emulator. */"ro.product.board","ro.board.platform","ro.arch" };static const int HAL_VARIANT_KEYS_COUNT =(sizeof(variant_keys)/sizeof(variant_keys[0])); . . . . . . int hw_get_module_by_class(const char *class_id, const char *inst,const struct hw_module_t **module) {int i = 0;char prop[PATH_MAX] = {0};char path[PATH_MAX] = {0};char name[PATH_MAX] = {0};char prop_name[PATH_MAX] = {0};if (inst)snprintf(name, PATH_MAX, "%s.%s", class_id, inst);elsestrlcpy(name, class_id, PATH_MAX);/** Here we rely on the fact that calling dlopen multiple times on* the same .so will simply increment a refcount (and not load* a new copy of the library).* We also assume that dlopen() is thread-safe.*//* First try a property specific to the class and possibly instance */snprintf(prop_name, sizeof(prop_name), "ro.hardware.%s", name);if (property_get(prop_name, prop, NULL) > 0) {if (hw_module_exists(path, sizeof(path), name, prop) == 0) {goto found;}}/* Loop through the configuration variants looking for a module */for (i=0 ; i<HAL_VARIANT_KEYS_COUNT; i++) {if (property_get(variant_keys[i], prop, NULL) == 0) {continue;}if (hw_module_exists(path, sizeof(path), name, prop) == 0) {goto found;}}/* Nothing found, try the default */if (hw_module_exists(path, sizeof(path), name, "default") == 0) {goto found;} return -ENOENT;found:/* load the module, if this fails, we're doomed, and we should not try* to load a different variant. */ return load(class_id, path, module); }int hw_get_module(const char *id, const struct hw_module_t **module) { return hw_get_module_by_class(id, NULL, module); }

hw_get_module() 調用 hw_get_module_by_class() 實現其功能。hw_get_module_by_class() 通過兩個步驟完成 gralloc 模塊的加載:
1. 找到 gralloc 模塊文件的路徑。
2. 加載 gralloc 模塊文件。

hw_get_module_by_class() 首先通過模塊 ID 和模塊實例名稱得到模塊名稱,對于 gralloc 而言,模塊 ID為 gralloc,模塊實例名稱為空,然后獲得模塊子名稱。模塊子名稱來自于系統屬性。hw_get_module_by_class() 依次嘗試從如下幾個系統屬性中獲得模塊子名稱:

ro.hardware.gralloc ro.hardware ro.product.board ro.board.platform ro.arch

對于 Pixel 設備而言,系統屬性 ro.hardware.gralloc 和 ro.arch 未定義,其它幾個系統屬性值如下:

[ro.hardware]: [sailfish] [ro.product.board]: [sailfish] [ro.board.platform]: [msm8996]

無法由從系統屬性獲得的模塊子名稱找到所對應的模塊文件時,則以 default 作為模塊子名稱繼續查找。

有了模塊名稱和模塊子名稱,hw_get_module_by_class() 則依次在 /odm -> /vendor -> /system 等目錄下查找相應的模塊文件:

#if defined(__LP64__) #define HAL_LIBRARY_PATH1 "/system/lib64/hw" #define HAL_LIBRARY_PATH2 "/vendor/lib64/hw" #define HAL_LIBRARY_PATH3 "/odm/lib64/hw" #else #define HAL_LIBRARY_PATH1 "/system/lib/hw" #define HAL_LIBRARY_PATH2 "/vendor/lib/hw" #define HAL_LIBRARY_PATH3 "/odm/lib/hw" #endif . . . . . . static int hw_module_exists(char *path, size_t path_len, const char *name,const char *subname) {snprintf(path, path_len, "%s/%s.%s.so",HAL_LIBRARY_PATH3, name, subname);if (access(path, R_OK) == 0)return 0;snprintf(path, path_len, "%s/%s.%s.so",HAL_LIBRARY_PATH2, name, subname);if (access(path, R_OK) == 0)return 0;snprintf(path, path_len, "%s/%s.%s.so",HAL_LIBRARY_PATH1, name, subname);if (access(path, R_OK) == 0)return 0;return -ENOENT; }

對于 Pixel 設備,可以在 /system/lib64/hw/ 目錄下找到如下的 gralloc 模塊文件:

1|sailfish:/ # ls /vendor/lib64/hw/ | grep gralloc gralloc.default.so gralloc.msm8996.so

最終將采用 /vendor/lib64/hw/gralloc.msm8996.so 作為 gralloc 模塊文件。關于這一點,可以從 surfacefinger 進程的內存映射鏡像中看到:

7105377000-710537d000 r-xp 00000000 fd:00 1575 /system/lib64/hw/gralloc.msm8996.so 710537d000-710537e000 r--p 00005000 fd:00 1575 /system/lib64/hw/gralloc.msm8996.so 710537e000-710537f000 rw-p 00006000 fd:00 1575 /system/lib64/hw/gralloc.msm8996.so

hw_get_module_by_class() 通過 load() 函數來加載模塊文件:

static int load(const char *id,const char *path,const struct hw_module_t **pHmi) {int status = -EINVAL;void *handle = NULL;struct hw_module_t *hmi = NULL;/** load the symbols resolving undefined symbols before* dlopen returns. Since RTLD_GLOBAL is not or'd in with* RTLD_NOW the external symbols will not be global*/handle = dlopen(path, RTLD_NOW);if (handle == NULL) {char const *err_str = dlerror();ALOGE("load: module=%s\n%s", path, err_str?err_str:"unknown");status = -EINVAL;goto done;}/* Get the address of the struct hal_module_info. */const char *sym = HAL_MODULE_INFO_SYM_AS_STR;hmi = (struct hw_module_t *)dlsym(handle, sym);if (hmi == NULL) {ALOGE("load: couldn't find symbol %s", sym);status = -EINVAL;goto done;}/* Check that the id matches */if (strcmp(id, hmi->id) != 0) {ALOGE("load: id=%s != hmi->id=%s", id, hmi->id);status = -EINVAL;goto done;}hmi->dso = handle;/* success */status = 0;done:if (status != 0) {hmi = NULL;if (handle != NULL) {dlclose(handle);handle = NULL;}} else {ALOGV("loaded HAL id=%s path=%s hmi=%p handle=%p",id, path, *pHmi, handle);}*pHmi = hmi;return status; }

加載過程主要是加載動態鏈接庫文件,找到其中名為 HAL_MODULE_INFO_SYM_AS_STR,類型為 struct hw_module_t 的變量,并返回給調用者。

HAL 層要求 HAL 模塊中都需要定義一個名為 HAL_MODULE_INFO_SYM_AS_STR 的 struct hw_module_t 變量。HAL_MODULE_INFO_SYM_AS_STR 定義(位于 hardware/libhardware/include/hardware/hardware.h)如下:

/*** Name of the hal_module_info*/ #define HAL_MODULE_INFO_SYM HMI/*** Name of the hal_module_info as a string*/ #define HAL_MODULE_INFO_SYM_AS_STR "HMI"

前面看到的 gralloc 實現 gralloc.msm8996.so 位于( gralloc 接口實現) hardware/qcom/display/msm8996/libgralloc,或位于 hardware/qcom/display/msm8996/libgralloc1( gralloc1 接口實現)。

gralloc.default.so 實現位于 hardware/libhardware/modules/gralloc,其 struct hw_module_t 定義(位于 hardware/libhardware/modules/gralloc/gralloc.cpp)如下:

struct private_module_t HAL_MODULE_INFO_SYM = {.base = {.common = {.tag = HARDWARE_MODULE_TAG,.version_major = 1,.version_minor = 0,.id = GRALLOC_HARDWARE_MODULE_ID,.name = "Graphics Memory Allocator Module",.author = "The Android Open Source Project",.methods = &gralloc_module_methods},.registerBuffer = gralloc_register_buffer,.unregisterBuffer = gralloc_unregister_buffer,.lock = gralloc_lock,.unlock = gralloc_unlock,},.framebuffer = 0,.flags = 0,.numBuffers = 0,.bufferMask = 0,.lock = PTHREAD_MUTEX_INITIALIZER,.currentBuffer = 0, };

不同類型的 HAL 模塊,不同類型 HAL 模塊的具體實現通常會定義自己的 struct hw_module_t 結構,如 gralloc.default.so 的 struct private_module_t,其定義(位于 hardware/libhardware/modules/gralloc/gralloc_priv.h)如下:

struct private_module_t {gralloc_module_t base;private_handle_t* framebuffer;uint32_t flags;uint32_t numBuffers;uint32_t bufferMask;pthread_mutex_t lock;buffer_handle_t currentBuffer;int pmem_master;void* pmem_master_base;struct fb_var_screeninfo info;struct fb_fix_screeninfo finfo;float xdpi;float ydpi;float fps; };

struct private_module_t 的成員 base 類型為 gralloc_module_t,該類型定義如下:

typedef struct gralloc_module_t {struct hw_module_t common;int (*registerBuffer)(struct gralloc_module_t const* module,buffer_handle_t handle);int (*unregisterBuffer)(struct gralloc_module_t const* module,buffer_handle_t handle);int (*lock)(struct gralloc_module_t const* module,buffer_handle_t handle, int usage,int l, int t, int w, int h,void** vaddr);int (*unlock)(struct gralloc_module_t const* module,buffer_handle_t handle);int (*perform)(struct gralloc_module_t const* module,int operation, ... );int (*lock_ycbcr)(struct gralloc_module_t const* module,buffer_handle_t handle, int usage,int l, int t, int w, int h,struct android_ycbcr *ycbcr);int (*lockAsync)(struct gralloc_module_t const* module,buffer_handle_t handle, int usage,int l, int t, int w, int h,void** vaddr, int fenceFd);int (*unlockAsync)(struct gralloc_module_t const* module,buffer_handle_t handle, int* fenceFd);int (*lockAsync_ycbcr)(struct gralloc_module_t const* module,buffer_handle_t handle, int usage,int l, int t, int w, int h,struct android_ycbcr *ycbcr, int fenceFd);/* reserved for future use */void* reserved_proc[3]; } gralloc_module_t;

gralloc_module_t 類型的第一個成員 common 類型為 struct hw_module_t。這實際是 C 語言對于繼承的實現 —— 父結構體總是作為子結構體的第一個成員存在。對于 HAL gralloc 模塊,其 struct hw_module_t 將總是 struct gralloc_module_t 結構的子結構,各個 gralloc 模塊實現,通常又都會定義自己的私有 struct gralloc_module_t 結構。

按照接口約定,實現了 gralloc1 接口的 HAL gralloc 模塊,其 module->methods->open() 函數應該通過參數 device 返回一個類型為 gralloc1_device_t 的結構,一個 struct hw_device_t 結構的子結構;而實現了 gralloc 接口的 HAL gralloc 模塊,則應該通過相同的方法,返回 struct hw_device_t 結構的子結構 struct alloc_device_t。

gralloc1_device_t 定義(位于 hardware/libhardware/include/hardware/gralloc1.h)如下:

typedef struct gralloc1_device {/* Must be the first member of this struct, since a pointer to this struct* will be generated by casting from a hw_device_t* */struct hw_device_t common;void (*getCapabilities)(struct gralloc1_device* device, uint32_t* outCount,int32_t* /*gralloc1_capability_t*/ outCapabilities);gralloc1_function_pointer_t (*getFunction)(struct gralloc1_device* device,int32_t /*gralloc1_function_descriptor_t*/ descriptor); } gralloc1_device_t;

gralloc1_device_t 新定義了兩個函數指針成員,它們也是 gralloc1 的實現模塊必須要提供的函數。其中 getCapabilities 用于獲得設備支持的 capabilities 的列表,getFunction 則用于獲得實現不同功能的函數的指針。

struct alloc_device_t 接口定義如下:

typedef struct alloc_device_t {struct hw_device_t common;int (*alloc)(struct alloc_device_t* dev,int w, int h, int format, int usage,buffer_handle_t* handle, int* stride);int (*free)(struct alloc_device_t* dev,buffer_handle_t handle);void (*dump)(struct alloc_device_t *dev, char *buff, int buff_len);void* reserved_proc[7]; } alloc_device_t;

struct alloc_device_t 新定義了用于分配和釋放圖形緩沖區的接口。

Gralloc1::Loader 的構造函數中,當 majorVersion 不為 1 時,將創建 Gralloc1On0Adapter 把 gralloc 接口下的 struct alloc_device_t 轉為 gralloc1 接口下的 gralloc1_device_t。

前面看到的 gralloc.default.so,它實現了 gralloc 接口,而不是 gralloc1。

Gralloc1::Device

Gralloc1::Device 是對 gralloc1_device_t 的一個包裝。Gralloc1::Device 定義(位于 frameworks/native/include/ui/Gralloc1.h)如下:

class Device {friend class Gralloc1::Descriptor;public:Device(gralloc1_device_t* device);bool hasCapability(gralloc1_capability_t capability) const;std::string dump();std::shared_ptr<Descriptor> createDescriptor();gralloc1_error_t getStride(buffer_handle_t buffer, uint32_t* outStride);gralloc1_error_t allocate(const std::vector<std::shared_ptr<const Descriptor>>& descriptors,std::vector<buffer_handle_t>* outBuffers);gralloc1_error_t allocate(const std::shared_ptr<const Descriptor>& descriptor,gralloc1_backing_store_t id, buffer_handle_t* outBuffer);gralloc1_error_t retain(buffer_handle_t buffer);gralloc1_error_t retain(const GraphicBuffer* buffer);gralloc1_error_t release(buffer_handle_t buffer);gralloc1_error_t getNumFlexPlanes(buffer_handle_t buffer,uint32_t* outNumPlanes);gralloc1_error_t lock(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion, void** outData,const sp<Fence>& acquireFence);gralloc1_error_t lockFlex(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion,struct android_flex_layout* outData, const sp<Fence>& acquireFence);gralloc1_error_t lockYCbCr(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion, struct android_ycbcr* outData,const sp<Fence>& acquireFence);gralloc1_error_t unlock(buffer_handle_t buffer, sp<Fence>* outFence);private:std::unordered_set<gralloc1_capability_t> loadCapabilities();bool loadFunctions();template <typename LockType, typename OutType>gralloc1_error_t lockHelper(LockType pfn, buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion, OutType* outData,const sp<Fence>& acquireFence) {int32_t intError = pfn(mDevice, buffer,static_cast<uint64_t>(producerUsage),static_cast<uint64_t>(consumerUsage), accessRegion, outData,acquireFence->dup());return static_cast<gralloc1_error_t>(intError);}gralloc1_device_t* const mDevice;const std::unordered_set<gralloc1_capability_t> mCapabilities;template <typename PFN, gralloc1_function_descriptor_t descriptor>struct FunctionLoader {FunctionLoader() : pfn(nullptr) {}bool load(gralloc1_device_t* device, bool errorIfNull) {gralloc1_function_pointer_t rawPointer =device->getFunction(device, descriptor);pfn = reinterpret_cast<PFN>(rawPointer);if (errorIfNull && !rawPointer) {ALOG(LOG_ERROR, GRALLOC1_LOG_TAG,"Failed to load function pointer %d", descriptor);}return rawPointer != nullptr;}template <typename ...Args>typename std::result_of<PFN(Args...)>::type operator()(Args... args) {return pfn(args...);}PFN pfn;};// Function pointersstruct Functions {FunctionLoader<GRALLOC1_PFN_DUMP, GRALLOC1_FUNCTION_DUMP> dump;FunctionLoader<GRALLOC1_PFN_CREATE_DESCRIPTOR,GRALLOC1_FUNCTION_CREATE_DESCRIPTOR> createDescriptor;FunctionLoader<GRALLOC1_PFN_DESTROY_DESCRIPTOR,GRALLOC1_FUNCTION_DESTROY_DESCRIPTOR> destroyDescriptor;FunctionLoader<GRALLOC1_PFN_SET_CONSUMER_USAGE,GRALLOC1_FUNCTION_SET_CONSUMER_USAGE> setConsumerUsage;FunctionLoader<GRALLOC1_PFN_SET_DIMENSIONS,GRALLOC1_FUNCTION_SET_DIMENSIONS> setDimensions;FunctionLoader<GRALLOC1_PFN_SET_FORMAT,GRALLOC1_FUNCTION_SET_FORMAT> setFormat;FunctionLoader<GRALLOC1_PFN_SET_PRODUCER_USAGE,GRALLOC1_FUNCTION_SET_PRODUCER_USAGE> setProducerUsage;FunctionLoader<GRALLOC1_PFN_GET_BACKING_STORE,GRALLOC1_FUNCTION_GET_BACKING_STORE> getBackingStore;FunctionLoader<GRALLOC1_PFN_GET_CONSUMER_USAGE,GRALLOC1_FUNCTION_GET_CONSUMER_USAGE> getConsumerUsage;FunctionLoader<GRALLOC1_PFN_GET_DIMENSIONS,GRALLOC1_FUNCTION_GET_DIMENSIONS> getDimensions;FunctionLoader<GRALLOC1_PFN_GET_FORMAT,GRALLOC1_FUNCTION_GET_FORMAT> getFormat;FunctionLoader<GRALLOC1_PFN_GET_PRODUCER_USAGE,GRALLOC1_FUNCTION_GET_PRODUCER_USAGE> getProducerUsage;FunctionLoader<GRALLOC1_PFN_GET_STRIDE,GRALLOC1_FUNCTION_GET_STRIDE> getStride;FunctionLoader<GRALLOC1_PFN_ALLOCATE,GRALLOC1_FUNCTION_ALLOCATE> allocate;FunctionLoader<GRALLOC1_PFN_RETAIN,GRALLOC1_FUNCTION_RETAIN> retain;FunctionLoader<GRALLOC1_PFN_RELEASE,GRALLOC1_FUNCTION_RELEASE> release;FunctionLoader<GRALLOC1_PFN_GET_NUM_FLEX_PLANES,GRALLOC1_FUNCTION_GET_NUM_FLEX_PLANES> getNumFlexPlanes;FunctionLoader<GRALLOC1_PFN_LOCK,GRALLOC1_FUNCTION_LOCK> lock;FunctionLoader<GRALLOC1_PFN_LOCK_FLEX,GRALLOC1_FUNCTION_LOCK_FLEX> lockFlex;FunctionLoader<GRALLOC1_PFN_LOCK_YCBCR,GRALLOC1_FUNCTION_LOCK_YCBCR> lockYCbCr;FunctionLoader<GRALLOC1_PFN_UNLOCK,GRALLOC1_FUNCTION_UNLOCK> unlock;// Adapter-only functionsFunctionLoader<GRALLOC1_PFN_RETAIN_GRAPHIC_BUFFER,GRALLOC1_FUNCTION_RETAIN_GRAPHIC_BUFFER> retainGraphicBuffer;FunctionLoader<GRALLOC1_PFN_ALLOCATE_WITH_ID,GRALLOC1_FUNCTION_ALLOCATE_WITH_ID> allocateWithId;} mFunctions;}; // class android::Gralloc1::Device

Gralloc1::Device 定義了接口以提供圖形內存的分配/釋放、lock/unlock 操作,所有的這些操作都依賴于從 gralloc1_device_t 獲得的函數指針實現。在
Gralloc1::Device 構造期間,會加載相關的函數指針,以及設備的 capabilities:

Device::Device(gralloc1_device_t* device): mDevice(device),mCapabilities(loadCapabilities()),mFunctions() {if (!loadFunctions()) {ALOGE("Failed to load a required function, aborting");abort();} } . . . . . . std::unordered_set<gralloc1_capability_t> Device::loadCapabilities() {std::vector<int32_t> intCapabilities;uint32_t numCapabilities = 0;mDevice->getCapabilities(mDevice, &numCapabilities, nullptr);intCapabilities.resize(numCapabilities);mDevice->getCapabilities(mDevice, &numCapabilities, intCapabilities.data());std::unordered_set<gralloc1_capability_t> capabilities;for (const auto intCapability : intCapabilities) {capabilities.emplace(static_cast<gralloc1_capability_t>(intCapability));}return capabilities; }bool Device::loadFunctions() {// Functions which must always be presentif (!mFunctions.dump.load(mDevice, true)) {return false;}if (!mFunctions.createDescriptor.load(mDevice, true)) {return false;}if (!mFunctions.destroyDescriptor.load(mDevice, true)) {return false;}if (!mFunctions.setConsumerUsage.load(mDevice, true)) {return false;}if (!mFunctions.setDimensions.load(mDevice, true)) {return false;}if (!mFunctions.setFormat.load(mDevice, true)) {return false;}if (!mFunctions.setProducerUsage.load(mDevice, true)) {return false;}if (!mFunctions.getBackingStore.load(mDevice, true)) {return false;}if (!mFunctions.getConsumerUsage.load(mDevice, true)) {return false;}if (!mFunctions.getDimensions.load(mDevice, true)) {return false;}if (!mFunctions.getFormat.load(mDevice, true)) {return false;}if (!mFunctions.getProducerUsage.load(mDevice, true)) {return false;}if (!mFunctions.getStride.load(mDevice, true)) {return false;}if (!mFunctions.retain.load(mDevice, true)) {return false;}if (!mFunctions.release.load(mDevice, true)) {return false;}if (!mFunctions.getNumFlexPlanes.load(mDevice, true)) {return false;}if (!mFunctions.lock.load(mDevice, true)) {return false;}if (!mFunctions.lockFlex.load(mDevice, true)) {return false;}if (!mFunctions.unlock.load(mDevice, true)) {return false;}if (hasCapability(GRALLOC1_CAPABILITY_ON_ADAPTER)) {// These should always be present on the adapterif (!mFunctions.retainGraphicBuffer.load(mDevice, true)) {return false;}if (!mFunctions.lockYCbCr.load(mDevice, true)) {return false;}// allocateWithId may not be present if we're only able to map in this// processmFunctions.allocateWithId.load(mDevice, false);} else {// allocate may not be present if we're only able to map in this processmFunctions.allocate.load(mDevice, false);}return true; }

Gralloc1::Device 中定義的 FunctionLoader 模板,即可以加載函數指針,同時它實現了 operator(),也是一個函數對象:

template <typename PFN, gralloc1_function_descriptor_t descriptor>struct FunctionLoader {FunctionLoader() : pfn(nullptr) {}bool load(gralloc1_device_t* device, bool errorIfNull) {gralloc1_function_pointer_t rawPointer =device->getFunction(device, descriptor);pfn = reinterpret_cast<PFN>(rawPointer);if (errorIfNull && !rawPointer) {ALOG(LOG_ERROR, GRALLOC1_LOG_TAG,"Failed to load function pointer %d", descriptor);}return rawPointer != nullptr;}template <typename ...Args>typename std::result_of<PFN(Args...)>::type operator()(Args... args) {return pfn(args...);}PFN pfn;};

FunctionLoader 模板藉由 gralloc1_device_t 的 getFunction() 接口獲得函數指針。Gralloc1::Device 的函數函數類型定義像下面這樣:

typedef void (*GRALLOC1_PFN_DUMP)(gralloc1_device_t* device, uint32_t* outSize,char* outBuffer); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_CREATE_DESCRIPTOR)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t* outDescriptor); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_DESTROY_DESCRIPTOR)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t descriptor); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_SET_CONSUMER_USAGE)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t descriptor,uint64_t /*gralloc1_consumer_usage_t*/ usage); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_SET_DIMENSIONS)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t descriptor,uint32_t width, uint32_t height); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_SET_FORMAT)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t descriptor,int32_t /*android_pixel_format_t*/ format); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_SET_PRODUCER_USAGE)(gralloc1_device_t* device, gralloc1_buffer_descriptor_t descriptor,uint64_t /*gralloc1_producer_usage_t*/ usage); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_BACKING_STORE)(gralloc1_device_t* device, buffer_handle_t buffer,gralloc1_backing_store_t* outStore); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_CONSUMER_USAGE)(gralloc1_device_t* device, buffer_handle_t buffer,uint64_t* /*gralloc1_consumer_usage_t*/ outUsage); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_DIMENSIONS)(gralloc1_device_t* device, buffer_handle_t buffer, uint32_t* outWidth,uint32_t* outHeight); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_FORMAT)(gralloc1_device_t* device, buffer_handle_t descriptor,int32_t* outFormat); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_PRODUCER_USAGE)(gralloc1_device_t* device, buffer_handle_t buffer,uint64_t* /*gralloc1_producer_usage_t*/ outUsage); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_GET_STRIDE)(gralloc1_device_t* device, buffer_handle_t buffer, uint32_t* outStride); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_ALLOCATE)(gralloc1_device_t* device, uint32_t numDescriptors,const gralloc1_buffer_descriptor_t* descriptors,buffer_handle_t* outBuffers); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_RETAIN)(gralloc1_device_t* device, buffer_handle_t buffer); typedef int32_t /*gralloc1_error_t*/ (*GRALLOC1_PFN_RELEASE)(gralloc1_device_t* device, buffer_handle_t buffer);

而所謂的函數 descriptor 則是用于標識函數的整數:

typedef enum {GRALLOC1_FUNCTION_INVALID = 0,GRALLOC1_FUNCTION_DUMP = 1,GRALLOC1_FUNCTION_CREATE_DESCRIPTOR = 2,GRALLOC1_FUNCTION_DESTROY_DESCRIPTOR = 3,GRALLOC1_FUNCTION_SET_CONSUMER_USAGE = 4,GRALLOC1_FUNCTION_SET_DIMENSIONS = 5,GRALLOC1_FUNCTION_SET_FORMAT = 6,GRALLOC1_FUNCTION_SET_PRODUCER_USAGE = 7,GRALLOC1_FUNCTION_GET_BACKING_STORE = 8,GRALLOC1_FUNCTION_GET_CONSUMER_USAGE = 9,GRALLOC1_FUNCTION_GET_DIMENSIONS = 10,GRALLOC1_FUNCTION_GET_FORMAT = 11,GRALLOC1_FUNCTION_GET_PRODUCER_USAGE = 12,GRALLOC1_FUNCTION_GET_STRIDE = 13,GRALLOC1_FUNCTION_ALLOCATE = 14,GRALLOC1_FUNCTION_RETAIN = 15,GRALLOC1_FUNCTION_RELEASE = 16,GRALLOC1_FUNCTION_GET_NUM_FLEX_PLANES = 17,GRALLOC1_FUNCTION_LOCK = 18,GRALLOC1_FUNCTION_LOCK_FLEX = 19,GRALLOC1_FUNCTION_UNLOCK = 20,GRALLOC1_LAST_FUNCTION = 20, } gralloc1_function_descriptor_t;

Gralloc1::Device 的所有功能實現,都依賴于從 gralloc1_device_t 的函數指針:

std::string Device::dump() {uint32_t length = 0;mFunctions.dump(mDevice, &length, nullptr);std::vector<char> output;output.resize(length);mFunctions.dump(mDevice, &length, output.data());return std::string(output.cbegin(), output.cend()); }std::shared_ptr<Descriptor> Device::createDescriptor() {gralloc1_buffer_descriptor_t descriptorId;int32_t intError = mFunctions.createDescriptor(mDevice, &descriptorId);auto error = static_cast<gralloc1_error_t>(intError);if (error != GRALLOC1_ERROR_NONE) {return nullptr;}auto descriptor = std::make_shared<Descriptor>(*this, descriptorId);return descriptor; }gralloc1_error_t Device::getStride(buffer_handle_t buffer, uint32_t* outStride) {int32_t intError = mFunctions.getStride(mDevice, buffer, outStride);return static_cast<gralloc1_error_t>(intError); }static inline bool allocationSucceded(gralloc1_error_t error) {return error == GRALLOC1_ERROR_NONE || error == GRALLOC1_ERROR_NOT_SHARED; }gralloc1_error_t Device::allocate(const std::vector<std::shared_ptr<const Descriptor>>& descriptors,std::vector<buffer_handle_t>* outBuffers) {if (mFunctions.allocate.pfn == nullptr) {// Allocation is not supported on this devicereturn GRALLOC1_ERROR_UNSUPPORTED;}std::vector<gralloc1_buffer_descriptor_t> deviceIds;for (const auto& descriptor : descriptors) {deviceIds.emplace_back(descriptor->getDeviceId());}std::vector<buffer_handle_t> buffers(descriptors.size());int32_t intError = mFunctions.allocate(mDevice,static_cast<uint32_t>(descriptors.size()), deviceIds.data(),buffers.data());auto error = static_cast<gralloc1_error_t>(intError);if (allocationSucceded(error)) {*outBuffers = std::move(buffers);}return error; }gralloc1_error_t Device::allocate(const std::shared_ptr<const Descriptor>& descriptor,gralloc1_backing_store_t id, buffer_handle_t* outBuffer) {gralloc1_error_t error = GRALLOC1_ERROR_NONE;if (hasCapability(GRALLOC1_CAPABILITY_ON_ADAPTER)) {buffer_handle_t buffer = nullptr;int32_t intError = mFunctions.allocateWithId(mDevice,descriptor->getDeviceId(), id, &buffer);error = static_cast<gralloc1_error_t>(intError);if (allocationSucceded(error)) {*outBuffer = buffer;}} else {std::vector<std::shared_ptr<const Descriptor>> descriptors;descriptors.emplace_back(descriptor);std::vector<buffer_handle_t> buffers;error = allocate(descriptors, &buffers);if (allocationSucceded(error)) {*outBuffer = buffers[0];}}return error; }gralloc1_error_t Device::retain(buffer_handle_t buffer) {int32_t intError = mFunctions.retain(mDevice, buffer);return static_cast<gralloc1_error_t>(intError); }gralloc1_error_t Device::retain(const GraphicBuffer* buffer) {if (hasCapability(GRALLOC1_CAPABILITY_ON_ADAPTER)) {return mFunctions.retainGraphicBuffer(mDevice, buffer);} else {return retain(buffer->getNativeBuffer()->handle);} }gralloc1_error_t Device::release(buffer_handle_t buffer) {int32_t intError = mFunctions.release(mDevice, buffer);return static_cast<gralloc1_error_t>(intError); }gralloc1_error_t Device::getNumFlexPlanes(buffer_handle_t buffer,uint32_t* outNumPlanes) {uint32_t numPlanes = 0;int32_t intError = mFunctions.getNumFlexPlanes(mDevice, buffer, &numPlanes);auto error = static_cast<gralloc1_error_t>(intError);if (error == GRALLOC1_ERROR_NONE) {*outNumPlanes = numPlanes;}return error; }gralloc1_error_t Device::lock(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion, void** outData,const sp<Fence>& acquireFence) {ALOGV("Calling lock(%p)", buffer);return lockHelper(mFunctions.lock.pfn, buffer, producerUsage,consumerUsage, accessRegion, outData, acquireFence); }gralloc1_error_t Device::lockFlex(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion,struct android_flex_layout* outData,const sp<Fence>& acquireFence) {ALOGV("Calling lockFlex(%p)", buffer);return lockHelper(mFunctions.lockFlex.pfn, buffer, producerUsage,consumerUsage, accessRegion, outData, acquireFence); }gralloc1_error_t Device::lockYCbCr(buffer_handle_t buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t* accessRegion,struct android_ycbcr* outData,const sp<Fence>& acquireFence) {ALOGV("Calling lockYCbCr(%p)", buffer);return lockHelper(mFunctions.lockYCbCr.pfn, buffer, producerUsage,consumerUsage, accessRegion, outData, acquireFence); }gralloc1_error_t Device::unlock(buffer_handle_t buffer, sp<Fence>* outFence) {int32_t fenceFd = -1;int32_t intError = mFunctions.unlock(mDevice, buffer, &fenceFd);auto error = static_cast<gralloc1_error_t>(intError);if (error == GRALLOC1_ERROR_NONE) {*outFence = new Fence(fenceFd);}return error; }

我們可以看一下 Gralloc1On0Adapter,來簡單了解 HAL gralloc 模塊,gralloc1_device_t 可能的實現和功能。Gralloc1On0Adapter 繼承自 gralloc1_device_t:

class Gralloc1On0Adapter : public gralloc1_device_t { public:Gralloc1On0Adapter(const hw_module_t* module);~Gralloc1On0Adapter();gralloc1_device_t* getDevice() {return static_cast<gralloc1_device_t*>(this);}

Gralloc1On0Adapter 需要提供 gralloc1_device_t 所必需的 getCapabilities 和 getFunction 這兩個回調:

namespace android {Gralloc1On0Adapter::Gralloc1On0Adapter(const hw_module_t* module): mModule(reinterpret_cast<const gralloc_module_t*>(module)),mMinorVersion(mModule->common.module_api_version & 0xFF),mDevice(nullptr) {ALOGV("Constructing");getCapabilities = getCapabilitiesHook;getFunction = getFunctionHook;int error = ::gralloc_open(&(mModule->common), &mDevice);if (error) {ALOGE("Failed to open gralloc0 module: %d", error);}ALOGV("Opened gralloc0 device %p", mDevice); }

這兩個回調的實現在定義類的頭文件里:

private:static inline Gralloc1On0Adapter* getAdapter(gralloc1_device_t* device) {return static_cast<Gralloc1On0Adapter*>(device);}// getCapabilitiesvoid doGetCapabilities(uint32_t* outCount,int32_t* /*gralloc1_capability_t*/ outCapabilities);static void getCapabilitiesHook(gralloc1_device_t* device,uint32_t* outCount,int32_t* /*gralloc1_capability_t*/ outCapabilities) {getAdapter(device)->doGetCapabilities(outCount, outCapabilities);};// getFunctiongralloc1_function_pointer_t doGetFunction(int32_t /*gralloc1_function_descriptor_t*/ descriptor);static gralloc1_function_pointer_t getFunctionHook(gralloc1_device_t* device,int32_t /*gralloc1_function_descriptor_t*/ descriptor) {return getAdapter(device)->doGetFunction(descriptor);}

兩個回調最終通過 doGetCapabilities() 和 doGetFunction() 兩個成員函數完成其功能,這兩個函數實現如下:

void Gralloc1On0Adapter::doGetCapabilities(uint32_t* outCount,int32_t* outCapabilities) {if (outCapabilities == nullptr) {*outCount = 1;return;}if (*outCount >= 1) {*outCapabilities = GRALLOC1_CAPABILITY_ON_ADAPTER;*outCount = 1;} }gralloc1_function_pointer_t Gralloc1On0Adapter::doGetFunction(int32_t intDescriptor) {constexpr auto lastDescriptor =static_cast<int32_t>(GRALLOC1_LAST_ADAPTER_FUNCTION);if (intDescriptor < 0 || intDescriptor > lastDescriptor) {ALOGE("Invalid function descriptor");return nullptr;}auto descriptor =static_cast<gralloc1_function_descriptor_t>(intDescriptor);switch (descriptor) {case GRALLOC1_FUNCTION_DUMP:return asFP<GRALLOC1_PFN_DUMP>(dumpHook);case GRALLOC1_FUNCTION_CREATE_DESCRIPTOR:return asFP<GRALLOC1_PFN_CREATE_DESCRIPTOR>(createDescriptorHook);case GRALLOC1_FUNCTION_DESTROY_DESCRIPTOR:return asFP<GRALLOC1_PFN_DESTROY_DESCRIPTOR>(destroyDescriptorHook);case GRALLOC1_FUNCTION_SET_CONSUMER_USAGE:return asFP<GRALLOC1_PFN_SET_CONSUMER_USAGE>(setConsumerUsageHook);case GRALLOC1_FUNCTION_SET_DIMENSIONS:return asFP<GRALLOC1_PFN_SET_DIMENSIONS>(setDimensionsHook);case GRALLOC1_FUNCTION_SET_FORMAT:return asFP<GRALLOC1_PFN_SET_FORMAT>(setFormatHook);case GRALLOC1_FUNCTION_SET_PRODUCER_USAGE:return asFP<GRALLOC1_PFN_SET_PRODUCER_USAGE>(setProducerUsageHook);case GRALLOC1_FUNCTION_GET_BACKING_STORE:return asFP<GRALLOC1_PFN_GET_BACKING_STORE>(bufferHook<decltype(&Buffer::getBackingStore),&Buffer::getBackingStore, gralloc1_backing_store_t*>);case GRALLOC1_FUNCTION_GET_CONSUMER_USAGE:return asFP<GRALLOC1_PFN_GET_CONSUMER_USAGE>(getConsumerUsageHook);case GRALLOC1_FUNCTION_GET_DIMENSIONS:return asFP<GRALLOC1_PFN_GET_DIMENSIONS>(bufferHook<decltype(&Buffer::getDimensions),&Buffer::getDimensions, uint32_t*, uint32_t*>);case GRALLOC1_FUNCTION_GET_FORMAT:return asFP<GRALLOC1_PFN_GET_FORMAT>(bufferHook<decltype(&Buffer::getFormat),&Buffer::getFormat, int32_t*>);case GRALLOC1_FUNCTION_GET_PRODUCER_USAGE:return asFP<GRALLOC1_PFN_GET_PRODUCER_USAGE>(getProducerUsageHook);case GRALLOC1_FUNCTION_GET_STRIDE:return asFP<GRALLOC1_PFN_GET_STRIDE>(bufferHook<decltype(&Buffer::getStride),&Buffer::getStride, uint32_t*>);case GRALLOC1_FUNCTION_ALLOCATE:// Not provided, since we'll use ALLOCATE_WITH_IDreturn nullptr;case GRALLOC1_FUNCTION_ALLOCATE_WITH_ID:if (mDevice != nullptr) {return asFP<GRALLOC1_PFN_ALLOCATE_WITH_ID>(allocateWithIdHook);} else {return nullptr;}case GRALLOC1_FUNCTION_RETAIN:return asFP<GRALLOC1_PFN_RETAIN>(managementHook<&Gralloc1On0Adapter::retain>);case GRALLOC1_FUNCTION_RELEASE:return asFP<GRALLOC1_PFN_RELEASE>(managementHook<&Gralloc1On0Adapter::release>);case GRALLOC1_FUNCTION_RETAIN_GRAPHIC_BUFFER:return asFP<GRALLOC1_PFN_RETAIN_GRAPHIC_BUFFER>(retainGraphicBufferHook);case GRALLOC1_FUNCTION_GET_NUM_FLEX_PLANES:return asFP<GRALLOC1_PFN_GET_NUM_FLEX_PLANES>(bufferHook<decltype(&Buffer::getNumFlexPlanes),&Buffer::getNumFlexPlanes, uint32_t*>);case GRALLOC1_FUNCTION_LOCK:return asFP<GRALLOC1_PFN_LOCK>(lockHook<void*, &Gralloc1On0Adapter::lock>);case GRALLOC1_FUNCTION_LOCK_FLEX:return asFP<GRALLOC1_PFN_LOCK_FLEX>(lockHook<struct android_flex_layout,&Gralloc1On0Adapter::lockFlex>);case GRALLOC1_FUNCTION_LOCK_YCBCR:return asFP<GRALLOC1_PFN_LOCK_YCBCR>(lockHook<struct android_ycbcr,&Gralloc1On0Adapter::lockYCbCr>);case GRALLOC1_FUNCTION_UNLOCK:return asFP<GRALLOC1_PFN_UNLOCK>(unlockHook);case GRALLOC1_FUNCTION_INVALID:ALOGE("Invalid function descriptor");return nullptr;}ALOGE("Unknown function descriptor: %d", intDescriptor);return nullptr; }

最終 Gralloc1On0Adapter 的這些功能函數有將依賴于 alloc_device_t 和 gralloc_module_t 提供的那些回調來完成其功能,如 alloc() 函數依賴于 alloc_device_t:

gralloc1_error_t Gralloc1On0Adapter::allocate(const std::shared_ptr<Descriptor>& descriptor,gralloc1_backing_store_t store,buffer_handle_t* outBufferHandle) {ALOGV("allocate(%" PRIu64 ", %#" PRIx64 ")", descriptor->id, store);// If this function is being called, it's because we handed out its function// pointer, which only occurs when mDevice has been loaded successfully and// we are permitted to allocateint usage = static_cast<int>(descriptor->producerUsage) |static_cast<int>(descriptor->consumerUsage);buffer_handle_t handle = nullptr;int stride = 0;ALOGV("Calling alloc(%p, %u, %u, %i, %u)", mDevice, descriptor->width,descriptor->height, descriptor->format, usage);auto error = mDevice->alloc(mDevice,static_cast<int>(descriptor->width),static_cast<int>(descriptor->height), descriptor->format,usage, &handle, &stride);if (error != 0) {ALOGE("gralloc0 allocation failed: %d (%s)", error,strerror(-error));return GRALLOC1_ERROR_NO_RESOURCES;}*outBufferHandle = handle;auto buffer = std::make_shared<Buffer>(handle, store, *descriptor, stride,true);std::lock_guard<std::mutex> lock(mBufferMutex);mBuffers.emplace(handle, std::move(buffer));return GRALLOC1_ERROR_NONE; }

lock/unlock 操作則都依賴于 gralloc_module_t,以 lock 為例:

gralloc1_error_t Gralloc1On0Adapter::lock(const std::shared_ptr<Buffer>& buffer,gralloc1_producer_usage_t producerUsage,gralloc1_consumer_usage_t consumerUsage,const gralloc1_rect_t& accessRegion, void** outData,const sp<Fence>& acquireFence) {if (mMinorVersion >= 3) {int result = mModule->lockAsync(mModule, buffer->getHandle(),static_cast<int32_t>(producerUsage | consumerUsage),accessRegion.left, accessRegion.top, accessRegion.width,accessRegion.height, outData, acquireFence->dup());if (result != 0) {return GRALLOC1_ERROR_UNSUPPORTED;}} else {acquireFence->waitForever("Gralloc1On0Adapter::lock");int result = mModule->lock(mModule, buffer->getHandle(),static_cast<int32_t>(producerUsage | consumerUsage),accessRegion.left, accessRegion.top, accessRegion.width,accessRegion.height, outData);ALOGV("gralloc0 lock returned %d", result);if (result != 0) {return GRALLOC1_ERROR_UNSUPPORTED;}}return GRALLOC1_ERROR_NONE; }

關于 HAL gralloc 模塊更具體的實現,這里不再深追。

總結一下 Android 圖形系統中,對于 gralloc 接口的情況,與圖形緩沖區分配、映射有關的組件之間的結構,如下圖所示:

如果是 gralloc1 接口的情況,則稍微有一點不同:

此外,在前面 BufferQueueCore 中創建 IGraphicBufferAlloc 時,我們看到這個對象總是會通過 surfacefinger 創建,然后通過 Binder IPC 傳遞給創建 BufferQueue 的進程一個句柄。后續在創建 BufferQueue 的進程中的生產者要分配 GraphicBuffer,則這一分配過程實際也將發生在 surfaceflinger 進程中,創建的 GraphicBuffer經過序列化之后,傳回給創建 BufferQueue 的進程。在 GraphicBuffer 對象真正創建的時候,也會直接為其分配圖形緩沖區。也就是說,在 Android 系統中,真正會分配圖形緩沖區的進程只有 surfaceflinger,盡管可能會創建 BufferQueue 的進程有多個。

GraphicBufferAllocator 和 GraphicBufferMapper 在對象創建的時候,都會通過 Gralloc1::Loader 加載 HAL gralloc。只有需要分配圖形緩沖區的進程才需要創建 GraphicBufferAllocator,只有需要訪問圖形緩沖區的進程,才需要創建 GraphicBufferMapper 對象。從 Android 的日志分析,可以看到,只有 surfaceflinger 進程中,同時發生了為創建 GraphicBufferAllocator 和 GraphicBufferMapper 而加載 HAL gralloc。而為創建 GraphicBufferMapper 而加載 HAL gralloc 則發生在 zygote、bootanimation 等多個進程。可見能夠訪問圖形緩沖區的進程包括 Android Java 應用等多個進程。

Done.

Android OpenGL 圖形系統分析系列文章

在 Android 中使用 OpenGL
Android 圖形驅動初始化
EGL Context 創建
Android 圖形系統之圖形緩沖區分配
Android 圖形系統之gralloc

總結

以上是生活随笔為你收集整理的Android 图形系统之gralloc的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

日韩一二区在线观看 | 亚州av网站大全 | 成人在线视频你懂的 | 97在线视频免费看 | 久久精品这里都是精品 | 手机成人在线电影 | 一级一片免费看 | japanesexxxxfreehd乱熟| 五月婷婷欧美视频 | 日韩二区精品 | 国产在线一卡 | 国产日韩欧美在线一区 | 五月婷婷色 | 亚洲午夜小视频 | 欧美激情视频一二区 | 国产亚洲精品久久久久5区 成人h电影在线观看 | 国产精品日韩久久久久 | 国产999精品久久久久久 | 中文字幕第一页在线 | 亚洲美女精品 | 中文字幕久久精品一区 | 香蕉视频在线观看免费 | 免费看一级黄色 | 欧美精品亚州精品 | 一本一道久久a久久精品 | 亚洲午夜精品一区 | 日韩av片在线 | 狠狠色丁香婷婷综合久小说久 | 欧美成年黄网站色视频 | 日本久久片 | 四川妇女搡bbbb搡bbbb搡 | 久久久五月婷婷 | 亚洲国产资源 | 国产午夜小视频 | 欧美日韩性视频 | 国产精品人人做人人爽人人添 | 天天操夜夜曰 | 天天操天天操天天爽 | 激情五月av | 成年人在线电影 | 日韩美女一级片 | 久久国产日韩 | 亚洲精选在线 | 超碰久热 | 久久精品1区 | 午夜精品久久久久久久爽 | 在线观看岛国 | 日韩精品一区二区三区视频播放 | 久久国产精品99久久久久久老狼 | 久久久久久久久亚洲精品 | 亚洲影院天堂 | 中日韩免费视频 | 91九色丨porny丨丰满6 | 精品欧美一区二区精品久久 | 黄色大片网 | 五月香视频在线观看 | 欧美极品久久 | 久久精品久久久久久久 | 91九色蝌蚪视频 | 免费一级黄色 | 色婷婷综合激情 | 欧美日韩精品在线一区二区 | 中文字幕亚洲欧美日韩 | 国内精品久久天天躁人人爽 | 国产精品久久二区 | av一级片在线观看 | 精品一区91 | 国产手机在线播放 | 日韩日韩日韩日韩 | 日本黄色黄网站 | 日本黄色a级大片 | 日韩大片在线免费观看 | 国产精品wwwwww | 欧美成人久久 | 国产中文字幕一区二区 | 免费在线观看黄色网 | 久久国产电影院 | 日韩在线欧美在线 | 视频一区二区免费 | 日精品 | 国产高清日韩欧美 | 成人在线视频你懂的 | 欧美成人按摩 | 久久久久黄 | 国产理伦在线 | av黄网站 | 最新的av网站 | 久久av网址 | 91麻豆精品国产91久久久久 | 欧美精品xxx | 日韩网站在线 | 欧美老少交 | 一级片免费观看 | 国产群p视频 | 99精品国产兔费观看久久99 | 人成免费网站 | 国产99免费视频 | 在线观看日韩专区 | 久久精品视频国产 | 久久99精品久久久久久三级 | 在线亚洲小视频 | 久久成人资源 | 国产视频亚洲精品 | 成人在线黄色电影 | 亚洲另类视频在线 | 丁香色婷婷 | 成人黄色电影免费观看 | 久久,天天综合 | 国产精品久久久久久超碰 | 国产一区视频免费在线观看 | 欧美淫aaa免费观看 日韩激情免费视频 | 免费高清国产 | 免费黄色小网站 | 国产拍揄自揄精品视频麻豆 | 夜夜婷婷 | 在线国产视频 | 日本精品免费看 | av888av.com| 六月婷操| 国产综合婷婷 | 色偷偷88888欧美精品久久 | 97视频精品 | 成人午夜电影免费在线观看 | 狠狠的操你| 亚洲成年人在线播放 | 久草在线观看 | 日韩中文字幕在线 | 日本久久视频 | 黄网在线免费观看 | 天天干天天操天天射 | 欧美疯狂性受xxxxx另类 | 色久综合 | 欧美日韩亚洲国产一区 | 亚洲免费在线 | 国产 在线观看 | www黄com| 国产午夜三级一区二区三桃花影视 | 色悠悠久久综合 | 黄色影院在线免费观看 | 欧美色图30p | 免费婷婷 | 久久久精品 | 久久官网 | 又黄又爽的视频在线观看网站 | 国产免费观看高清完整版 | 国产特级毛片 | 国产精品毛片久久蜜 | 99精品国产亚洲 | 天天干天天做天天操 | 三级av网站 | 国产小视频精品 | 亚洲精品一区二区在线观看 | 另类老妇性bbwbbw高清 | 国产91小视频 | 日韩一区二区三区在线观看 | 四虎欧美 | 亚洲h在线播放在线观看h | 91资源在线视频 | 欧美激情在线看 | 四虎5151久久欧美毛片 | 成人一区影院 | 91视频在线国产 | 亚洲一区二区视频在线 | 色999五月色| 亚洲另类视频在线 | 国产色女人 | 亚洲麻豆精品 | 国产在线不卡一区 | 婷婷国产视频 | av成人免费在线观看 | av免费高清观看 | 精品美女久久久久久免费 | 国产成人一区二区三区在线观看 | 国产精品www | 久久久久久久久久久精 | 亚洲综合色丁香婷婷六月图片 | 久久免费毛片 | 在线免费观看视频a | 国产精品21区| 国产视频精品免费 | 五月婷婷激情五月 | 国产精品免费人成网站 | 成人在线视频在线观看 | 91久久丝袜国产露脸动漫 | 国产在线久久久 | 国产精品影音先锋 | 99色免费视频 | 在线观看视频中文字幕 | 二区三区av | 在线视频免费观看 | 亚洲精品视频免费看 | 婷婷综合伊人 | 亚洲精品中文在线资源 | 人人干天天射 | 久久韩国免费视频 | 亚洲精品婷婷 | 亚洲免费专区 | 色婷婷av在线 | 国产成人亚洲精品自产在线 | 日本黄色免费网站 | 亚洲欧美视频在线播放 | 中文字幕电影一区 | 91激情视频在线播放 | 色操插 | 在线国产专区 | 国产精品免费一区二区三区 | 亚洲最大av网站 | 91欧美视频网站 | 天天干天天操天天 | 久 久久影院| 久久久免费看视频 | 亚洲成av人影院 | 999成人 | 国产精品福利无圣光在线一区 | 蜜臀av性久久久久蜜臀aⅴ涩爱 | 久久夜av | 欧美日韩国产精品一区 | 精品国产一区二区三区久久影院 | 欧美动漫一区二区三区 | 国产精品自在线拍国产 | 免费久久网 | 蜜臀av免费一区二区三区 | 天天插日日操 | 久久国产成人午夜av影院宅 | 国内精品亚洲 | 国内精品二区 | 成人电影毛片 | 在线国产高清 | 欧美一级视频在线观看 | 激情av综合 | 久久人人爽人人爽人人片av免费 | 日韩免费不卡av | 欧美色图p | 97超碰在线播放 | 亚洲日本欧美 | 欧美成人性战久久 | av成人免费在线看 | 国产中文字幕网 | av官网| 在线看国产视频 | 四虎最新入口 | 日韩精品中文字幕av | 久久成人精品电影 | 美女网站一区 | 91亚洲精| 国产一二三四在线观看视频 | 亚洲黄色av一区 | 国产亚洲激情视频在线 | 久久久久国产视频 | 丰满少妇一级片 | 一区二区三区国产精品 | 国产福利在线不卡 | 91人人揉日日捏人人看 | 国产不卡视频在线播放 | 园产精品久久久久久久7电影 | 日日色综合 | 欧美日韩在线第一页 | 91色影院 | 免费看的黄网站 | 国产手机在线视频 | 中文字幕高清免费日韩视频在线 | 成人黄色影片在线 | wwwav视频| 欧美福利视频 | 国产亚州精品视频 | 成人免费在线播放视频 | 久久综合九色综合97_ 久久久 | 亚州国产精品久久久 | 在线观看国产v片 | 国内揄拍国产精品 | 色wwwww| 五月综合色婷婷 | 亚洲精品在线免费观看视频 | 色综合亚洲精品激情狠狠 | 午夜免费福利视频 | 国产精品剧情 | 欧美日韩亚洲精品在线 | 日韩中文字幕免费视频 | 欧美在线观看视频一区二区三区 | 成人午夜剧场在线观看 | 五月天婷婷免费视频 | 国产精品视频久久 | 美女天天操 | a级国产乱理论片在线观看 特级毛片在线观看 | www.天天成人国产电影 | 人人澡超碰碰 | 婷婷在线视频 | 在线亚洲播放 | 97超碰国产精品女人人人爽 | 狠狠精品| 狠狠久久综合 | 夜夜操天天干, | av成人在线观看 | 中文免费在线观看 | 国产高清久久久久 | 精品久久久久久国产偷窥 | 成人a免费视频 | 色多视频在线观看 | 亚洲国产中文在线观看 | 91在线看视频免费 | 精品亚洲免费视频 | 91精品国产成人 | 国产中文字幕视频在线 | av片在线看 | 激情欧美丁香 | 狠狠色丁香婷婷综合久小说久 | 摸bbb搡bbb搡bbbb | 欧美 国产 视频 | 久草在线视频免赞 | 香蕉视频在线网站 | 激情五月色播五月 | 五月天欧美精品 | 国产精品一区二区av日韩在线 | 黄污网站在线 | 色综合天天综合 | 免费情缘| 人人爽影院 | 亚洲精品综合久久 | 欧美夫妻性生活电影 | 国产精品国产三级国产不产一地 | 韩日av一区二区 | 国产一区久久久 | 久久歪歪 | 国产又粗又猛又黄视频 | 国产精品网站 | 亚州激情视频 | 成人av中文字幕在线观看 | 国产精品6999成人免费视频 | 91丨九色丨首页 | 免费看一级片 | 激情丁香综合五月 | 成人av片在线观看 | 欧美精品久久99 | 天天干天天操天天搞 | 国产精品亚洲片在线播放 | 国产看片免费 | 亚洲狠狠| 91av在线视频播放 | 国产91精品看黄网站 | 色姑娘综合天天 | 一区二区亚洲精品 | 探花视频在线版播放免费观看 | 97色se| 欧美激情视频一区二区三区 | 日韩免费在线一区 | 欧美人操人| 视频国产在线 | www.夜夜骑.com | 亚洲精品国偷自产在线99热 | 中文字幕在线观看免费高清完整版 | 中文字幕日韩精品有码视频 | 91色视频 | 黄在线免费观看 | 中文字幕久久精品亚洲乱码 | 国产精品2020 | 午夜日b视频 | 一区二区久久 | 国产视频欧美视频 | 国产黑丝一区二区 | 欧美精品首页 | 在线国产不卡 | 国产精品精品久久久 | 久久天天躁狠狠躁夜夜不卡公司 | 亚洲涩涩涩 | 免费高清在线观看成人 | 久久久久久久久久久网 | 国产精品九九九九九 | 高清免费在线视频 | 精品国内自产拍在线观看视频 | 成人av电影在线观看 | av天天澡天天爽天天av | 手机av电影在线观看 | 色香蕉在线 | 中文字幕一区二区三区乱码不卡 | 超碰在线97免费 | 久久网址 | 久久av在线 | 999抗病毒口服液 | 国产精品久久久久久妇 | 国产18精品乱码免费看 | 国产视频99 | 草在线| 一本—道久久a久久精品蜜桃 | 精品产品国产在线不卡 | 四虎永久精品在线 | 99久久久国产精品免费99 | 色88久久| 国产精品国产自产拍高清av | 成av在线 | 国产xxxx性hd极品 | 成年人免费电影在线观看 | 国产一线二线三线性视频 | 成年人免费看片网站 | 日日操夜 | www.少妇| 97在线看片| 夜夜爱av| 久草在线资源网 | 97视频免费观看 | 女人18精品一区二区三区 | 久久久久久久久久久久影院 | 2021国产精品视频 | 日韩在线视频观看 | 久久成人资源 | 国产精品麻豆一区二区三区 | 国产又粗又猛又黄又爽的视频 | 久久精品三 | 色开心| 欧美一区二区三区在线播放 | 国产精品一区二区免费在线观看 | 婷婷丁香激情五月 | 一区二区三区免费播放 | 亚洲日本va中文字幕 | 国产精品系列在线观看 | 激情欧美日韩一区二区 | 久久99国产精品视频 | 欧美一级免费在线 | 黄色av电影免费观看 | 天天色草| 久久久久久高清 | 日韩成人中文字幕 | 天天操天天干天天爽 | 国产精品麻豆一区二区三区 | 亚洲精品视频在线观看免费 | 久久精品国产久精国产 | 国产精品久久嫩一区二区免费 | 在线观看视频三级 | 成人蜜桃网 | 国产福利在线 | 中文字幕4 | 中文字幕丰满人伦在线 | 天堂av免费在线 | 亚洲一级片免费观看 | 亚洲人成免费网站 | 亚洲视频1区2区 | 久久久久人人 | 国产一级精品视频 | 中文字幕日韩有码 | 国产美女久久 | 在线观看一级片 | 成人黄色小视频 | 在线不卡的av | zzijzzij亚洲成熟少妇 | 免费观看黄色12片一级视频 | 日韩精品中文字幕av | 亚洲 欧美日韩 国产 中文 | 99中文字幕视频 | 亚洲精品乱码久久久久久写真 | 西西人体www444 | 2021国产精品视频 | 免费色av| 久久夜色精品国产亚洲aⅴ 91chinesexxx | 亚洲综合激情小说 | 五月综合激情婷婷 | 久久久久国产精品一区二区 | 欧美另类成人 | 久草在线观 | 久久精品久久精品久久39 | 欧美 高跟鞋交 xxxxhd | 99精品视频观看 | 久久夜夜夜 | 狠狠色丁香婷婷综合最新地址 | 中文字幕欧美激情 | 亚洲乱码中文字幕综合 | 91爱在线| 亚洲乱码中文字幕综合 | 99久久精品无码一区二区毛片 | 欧美日韩在线视频一区二区 | 久久久久久美女 | 久久综合九色综合久久久精品综合 | 91精品视频免费看 | 亚洲激情五月 | 亚洲九九爱 | 青草视频免费观看 | 精品亚洲视频在线 | 在线观看黄a | 久久久久国产精品免费网站 | 国产精品刺激对白麻豆99 | 91精品国产入口 | 精品国产一区二区三区蜜臀 | 99精品国产在热久久 | 成人在线一区二区 | 久久天天躁 | www.久久婷婷| 69绿帽绿奴3pvideos | 国产精品欧美久久久久三级 | 久久99国产精品久久99 | 国产免费久久精品 | 免费观看特级毛片 | 91丨九色丨蝌蚪丨老版 | 天天·日日日干 | 奇米影视8888在线观看大全免费 | 久久视频这里有久久精品视频11 | 久久婷婷色综合 | 国产精品粉嫩 | 18av在线视频 | 黄色aaaaa| 91麻豆精品国产91久久久久 | 我要看黄色一级片 | 婷婷在线网| 欧美色噜噜噜 | 国产一区二区精 | 免费在线观看国产黄 | 久久伊人操 | 午夜精品福利一区二区 | 在线观看av中文字幕 | 美女网站色 | 五月天伊人 | 亚洲国产精品久久久久久 | 夜夜骑日日 | av不卡免费在线观看 | 国产又粗又猛又色又黄网站 | 日韩系列| 国产视频中文字幕 | 婷婷丁香六月天 | 成人日批视频 | 国产精品视频免费 | 国产精品入口久久 | 国产三级香港三韩国三级 | va视频在线| 国产成人精品aaa | 91精品久久久久久久91蜜桃 | 中文字幕在线观看你懂的 | 欧美性成人 | 婷婷亚洲综合五月天小说 | 免费日韩高清 | 欧美激情精品久久久久久变态 | 中文字幕第 | 国产美腿白丝袜足在线av | 日韩最新在线 | 日韩在线第一 | 国产96视频| 亚洲成人av在线播放 | 公开超碰在线 | 久久不卡日韩美女 | 久久精品国产精品 | 国产精品一区二区果冻传媒 | 天天插日日插 | 激情影音先锋 | 狠狠干狠狠久久 | 欧美亚洲一级片 | 麻豆久久精品 | 2021国产视频 | 国产精品爽爽爽 | 日韩高清av在线 | 五月婷婷导航 | 日韩精品一区在线播放 | 丁香六月婷 | 精品一二三区视频 | 久久久午夜视频 | 午夜精品福利在线 | 日韩久久视频 | 日日操狠狠干 | 激情综合网婷婷 | 97热在线观看 | 国产一区二区播放 | 天天射天天爽 | 亚洲五月 | 午夜视频黄| 国产91电影在线观看 | 99视频精品全部免费 在线 | 亚洲综合激情网 | 五月婷婷激情 | 99 色| 麻豆影视在线播放 | 久久精品电影院 | 亚洲一区二区视频在线播放 | 中文字幕成人网 | 免费看一及片 | 粉嫩一二三区 | 国产xx视频| 少妇资源站 | 欧美日韩在线观看视频 | 日韩欧美一区二区三区免费观看 | 成人h视频在线播放 | 欧美国产精品一区二区 | 国产麻豆电影在线观看 | 免费观看一级视频 | 国产精品一区二区精品视频免费看 | 日日夜夜精品视频天天综合网 | 欧美久久久久久久久久 | 婷婷六月天综合 | 国产拍在线 | 五月天婷亚洲天综合网精品偷 | 久久不卡电影 | 国产免费不卡 | 色视频网站在线 | 日韩精品免费在线 | 九色免费视频 | 91精品国产高清自在线观看 | 人人看97| 欧美日韩免费视频 | 免费不卡中文字幕视频 | 日韩免费一区二区 | 国产一级a毛片视频爆浆 | 色就色,综合激情 | 色婷婷久久久综合中文字幕 | 国产一卡二卡四卡国 | 日韩精品免费一区二区 | 天天操天天射天天 | 中文字幕在线影院 | 国产探花在线看 | 国产美女视频 | 婷婷丁香狠狠爱 | 国产精品麻豆欧美日韩ww | 狠狠躁日日躁狂躁夜夜躁av | 97视频免费观看2区 亚洲视屏 | 激情在线网| 成人黄色电影在线播放 | 天天操天天操天天操天天操天天操 | 在线观看亚洲专区 | 久久人人爽人人爽人人片av免费 | 五月婷婷综合激情网 | 91福利在线导航 | 中文字幕久久精品亚洲乱码 | 五月天久久婷 | 亚洲成年人免费网站 | 日韩av中文在线 | 日韩在线影视 | 激情伊人| 日韩视频中文字幕在线观看 | 久久任你操 | 久草免费在线 | 麻豆传媒视频观看 | 午夜丰满寂寞少妇精品 | 亚洲人片在线观看 | 深爱五月激情网 | 亚洲热久久 | 日韩欧美在线第一页 | 国产在线精品播放 | 中文字幕婷婷 | 久久久久久久久久影院 | 国产aa免费视频 | 四川bbb搡bbb爽爽视频 | 国产精品你懂的在线观看 | 日韩欧美在线国产 | 国产精品美女免费看 | 午夜视频免费播放 | 欧美一级日韩三级 | 免费日韩一区 | 久久国产精品色av免费看 | 丁香六月激情婷婷 | 中国精品少妇 | 国产精品久久久久久久久久久不卡 | 贫乳av女优大全 | 天天干天天插 | 欧美九九九 | 精品久久久久一区二区国产 | 午夜av在线 | 最新日韩中文字幕 | 玖玖视频国产 | av成人免费在线看 | 欧美日韩精品在线播放 | 久久久久北条麻妃免费看 | 视频福利在线观看 | 欧美色图一区 | 五月天久久 | 99亚洲视频| 国产在线高清视频 | 超碰97在线资源站 | av电影免费 | 欧美激情精品久久久久久免费印度 | 国产日韩欧美在线影视 | 免费看黄色小说的网站 | 久久久久久久久免费视频 | 玖玖爱免费视频 | 色中色亚洲| 人人涩| 激情深爱五月 | 最近最新中文字幕视频 | www.夜夜草 | 久久精品视频在线观看免费 | 91人人干 | 成人a免费| 日韩在线字幕 | 在线黄色国产电影 | 久久不卡视频 | 日韩免费三区 | 欧美一区二区三区免费观看 | 一区二区三区韩国免费中文网站 | 中国一区二区视频 | 成人综合日日夜夜 | 久久久久久久亚洲精品 | 在线蜜桃视频 | 怡红院av久久久久久久 | 欧美一区二区精美视频 | 色婷婷激情电影 | 91精品伦理 | 豆豆色资源网xfplay | 成人91免费视频 | 亚洲激情中文 | 免费在线观看污 | 欧美日韩3p | 一区二区精品在线 | 丁香六月天 | 久草在线观看资源 | 国产亚洲精品久久久久秋 | 国产老太婆免费交性大片 | 综合黄色网| 免费国产一区二区 | 精品久久久国产 | 私人av| 一本一道久久a久久精品蜜桃 | 国产无遮挡猛进猛出免费软件 | 黄网在线免费观看 | 欧美激情视频免费看 | 午夜电影一区 | 亚洲精品乱码久久久久久按摩 | 午夜天天操 | 色综合亚洲精品激情狠狠 | 香蕉在线影院 | 麻豆一区二区 | av电影在线观看 | 亚洲成av人片在线观看香蕉 | 99亚洲国产精品 | 精品国产成人 | 免费av一级电影 | 麻豆国产网站 | 福利一区在线视频 | 国产精品一区在线播放 | 在线观看午夜av | 成人激情开心网 | 人人爽人人爽人人片av | 欧美色综合久久 | 91精品国产欧美一区二区成人 | 91精品视频在线 | 激情图片qvod | 波多野结衣精品在线 | 久久99视频免费 | 免费久久视频 | 中文字幕在线播放一区 | 韩日精品在线观看 | 国产一区欧美日韩 | 成人av在线看 | 国产精品久久久一区二区三区网站 | 久久视频在线免费观看 | 在线性视频日韩欧美 | 夜夜爱av| 草久在线 | 最近更新中文字幕 | 免费在线电影网址大全 | 国产夫妻av在线 | 久久草草热国产精品直播 | 亚洲精品国产精品国自产 | a黄色影院 | 91精品视频在线免费观看 | 久在线观看视频 | 亚洲高清久久久 | 亚洲三级视频 | 激情五月在线观看 | a在线观看免费视频 | 成人av片在线观看 | 狠狠狠干 | 99r在线视频| 亚洲免费av片| 久久人人插 | 欧美精品久久久久性色 | .国产精品成人自产拍在线观看6 | 日韩精品久久久久 | 国产精品美女www爽爽爽视频 | 精品在线不卡 | 国产精品丝袜 | 国产成人福利片 | 国产99久久精品 | 亚洲视频一区二区三区在线观看 | 欧美婷婷色 | 天堂视频中文在线 | www.久久久 | 91麻豆精品国产91久久久久久久久 | 国产一区av在线 | 色先锋资源网 | 亚洲精品乱码久久久一二三 | 综合天天 | 亚洲成av人片一区二区梦乃 | 狠狠的干狠狠的操 | 91福利在线导航 | 欧美一区二区在线刺激视频 | 亚洲免费av观看 | 五月婷网站 | 中文字幕av在线免费 | 中文字幕久久精品亚洲乱码 | www.国产在线 | 中文字幕av影院 | 久久精品欧美一区二区三区麻豆 | 综合激情av | 成年人免费在线看 | 国产 日韩 在线 亚洲 字幕 中文 | 国产综合91| 日日干天天 | 亚洲精品日韩在线观看 | 中文字幕久久久精品 | 亚洲视频 在线观看 | 亚洲精品在线免费播放 | 欧美在线一 | 国产在线一区二区 | 国产午夜三级一区二区三桃花影视 | 国产在线a视频 | 69久久久 | 日韩欧美在线观看一区二区三区 | 欧美日韩亚洲国产一区 | 在线免费黄| 久久成年人视频 | 久久dvd| 国产一级视屏 | 日韩免费视频播放 | 草久在线观看视频 | av免费观看网址 | 天天操天天能 | 久久艹精品 | 日日摸日日添日日躁av | 日韩成片 | 国产免费观看av | 国产99在线免费 | 亚洲精品午夜久久久久久久久久久 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 中文av一区二区 | 免费av大片| 综合铜03| 国产精品久久久久亚洲影视 | 在线 视频 一区二区 | 亚洲精品玖玖玖av在线看 | 日韩精品专区在线影院重磅 | 日韩欧美在线观看一区二区三区 | 欧美日韩国产高清视频 | 九九久久国产 | 国产在线观看av | 在线观看成人一级片 | 久久亚洲私人国产精品 | 国产a国产 | 国产精品久久久久久久久久久久冷 | 久久呀| 欧美精品免费一区二区 | 久草精品在线 | 成人精品影视 | 亚洲h在线播放在线观看h | 国产精品久免费的黄网站 | 久久亚洲福利视频 | zzijzzij日本成熟少妇 | 日韩精品观看 | av黄色大片 | 日本激情中文字幕 | 在线精品一区二区 | 激情在线网址 | 99理论片 | 99国产免费网址 | 精品国产一区在线观看 | 99热这里只有精品在线观看 | 99在线高清视频在线播放 | 欧美日韩精品久久久 | 欧美激情视频三区 | 国产这里只有精品 | 亚洲高清视频一区二区三区 | 久久99精品国产99久久 | 337p日本大胆噜噜噜噜 | 免费视频 你懂的 | 成人在线观看免费 | 久久国产精品久久精品 | 91亚洲精品乱码久久久久久蜜桃 | 天天插天天操天天干 | 日韩一区二区在线免费观看 | 黄色小网站在线 | 亚洲一区二区天堂 | 亚洲成人中文在线 | 99精品视频免费看 | 伊人电影天堂 | 999久久久精品视频 日韩高清www | 婷婷综合影院 | 欧美福利在线播放 | 久久人人插 | 国产剧情一区二区 | 在线观看中文 | 国产亚洲精品美女久久 | 日本69hd| 在线韩国电影免费观影完整版 | 免费观看www视频 | 丰满少妇一级 | 色99导航| 日韩中文字幕国产精品 | 在线观看av网站 | 欧美成人h版在线观看 | 中文字幕观看av | 国产精品久久久久久久久蜜臀 | 一区二区男女 | 欧美日韩三级 | 婷婷久久网 | 日韩欧美在线一区 | 国产四虎在线 | 在线中文字幕一区二区 | 日本三级在线观看中文字 | 911久久香蕉国产线看观看 | 日韩av一区二区在线影视 | 久久久久久久久久久久国产精品 | 国产高清不卡一区二区三区 | 欧美一二三视频 | 亚洲免费在线 | 四虎国产视频 | 日本精品免费看 | 婷婷亚洲综合 | 精品毛片久久久久久 | 97偷拍视频 | 亚洲一级黄色av | 久久免费黄色 | 国产精品久久久久久久久久久久久 | 日韩欧美视频免费在线观看 | 国产91影院| 精品欧美日韩 | 玖玖视频国产 | 91精品久久久久久综合五月天 | 久久久久国产成人精品亚洲午夜 | 久久久久夜色 | 欧美男女爱爱视频 | 欧美日韩久久 | 国产在线精品视频 | 亚洲九九爱 | 国产一级久久 | 国产69久久 | 欧美性另类| 日韩在线免费小视频 | 91成人精品一区在线播放69 | 精品国产乱码 | 四虎国产免费 | 天天爱天天射天天干天天 | 精品无人国产偷自产在线 | 女人高潮特级毛片 | 96久久欧美麻豆网站 | 天天操天天操天天干 | 国产xxxx| 久久久精品 一区二区三区 国产99视频在线观看 | 深夜免费小视频 | 日韩精品首页 | 国产黄在线免费观看 | 69久久夜色精品国产69 | 91在线免费看片 | 黄色大片入口 | 日韩电影在线视频 | 亚洲综合导航 | 中文字幕日韩国产 | 精品久久一二三区 | 免费看片亚洲 | a√国产免费a | 日韩精品久久一区二区三区 | 超碰在线人人 | 欧美一区二区三区特黄 | 免费在线色 | 黄色成人影院 | 欧美性网站 | 蜜臀aⅴ国产精品久久久国产 | 久久久久久久久久影视 | 久久成熟 | 日韩免费大片 | 爱干视频 | 日韩精品播放 | 亚洲 欧美变态 另类 综合 | 亚洲一区在线看 | 黄影院| 久久久影视 | 久久久久久久久久免费 | 蜜臀av夜夜澡人人爽人人 | 黄色电影在线免费观看 | 91麻豆精品国产91久久久使用方法 | 91在线免费公开视频 | 999久久 | 成人午夜黄色 | 精品自拍sae8—视频 | 激情视频综合网 | www.com操| 奇米先锋 | 日本丶国产丶欧美色综合 | 国产黄色片在线 | 国产精品一区二区三区电影 | 91成人看片| 涩涩资源网| 在线а√天堂中文官网 | 中文字幕 国产视频 | 国产黄在线看 | 婷婷五天天在线视频 | 久久久国产在线视频 | 欧美日韩国产精品爽爽 | 亚洲精品ww| 黄色av一级 | 日韩欧美一区二区三区免费观看 | 麻豆成人小视频 | 91色九色 | 97超碰总站| 99久久夜色精品国产亚洲96 | 麻花豆传媒一二三产区 | 福利久久 | 婷婷成人综合 | 香蕉在线观看视频 | 香蕉久久久久久av成人 | 国产aa免费视频 | 欧洲高潮三级做爰 | 国产裸体视频bbbbb | 啪啪午夜免费 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 超碰人人乐 | 国产高清网站 | 亚洲伊人第一页 |