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

歡迎訪問 生活随笔!

生活随笔

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

Android

深入浅出Android BufferQueue

發布時間:2024/2/28 Android 44 豆豆
生活随笔 收集整理的這篇文章主要介紹了 深入浅出Android BufferQueue 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.




1. 背景

對業務開發來說,無法接觸到BufferQueue,甚至不知道BufferQueue是什么東西。對系統來說,BufferQueue是很重要的傳遞數據的組件,Android顯示系統依賴于BufferQueue,只要顯示內容到“屏幕”(此處指抽象的屏幕,有時候還可以包含編碼器),就一定需要用到BufferQueue,可以說在顯示/播放器相關的領悟中,BufferQueue無處不在。即使直接調用Opengl?ES來繪制,底層依然需要BufferQueue才能顯示到屏幕上。


弄明白BufferQueue,不僅可以增強對Android系統的了解,還可以弄明白/排查相關的問題,如為什么Mediacodec調用dequeueBuffer老是返回-1?為什么普通View的draw方法直接繪制內容即可,SurfaceView在draw完畢后還需要unlockCanvasAndPost?


注:本文分析的代碼來自于Android6.0.1。


2. BufferQueue內部運作方式

BufferQueue是Android顯示系統的核心,它的設計哲學是生產者-消費者模型,只要往BufferQueue中填充數據,則認為是生產者,只要從BufferQueue中獲取數據,則認為是消費者。有時候同一個類,在不同的場景下既可能是生產者也有可能是消費者。如SurfaceFlinger,在合成并顯示UI內容時,UI元素作為生產者生產內容,SurfaceFlinger作為消費者消費這些內容。而在截屏時,SurfaceFlinger又作為生產者將當前合成顯示的UI內容填充到另一個BufferQueue,截屏應用此時作為消費者從BufferQueue中獲取數據并生產截圖。


以下是Android官網對其的介紹:



以下是常見的BufferQueue使用步驟:

  • 初始化一個BufferQueue

  • 圖形數據的生產者通過BufferQueue申請一塊GraphicBuffer,對應圖中的dequeueBuffer方法

  • 申請到GraphicBuffer后,獲取GraphicBuffer,通過函數requestBuffer獲取

  • 獲取到GraphicBuffer后,通過各種形式往GraphicBuffer中填充圖形數據后,然后將GraphicBuffer入隊到BufferQueue中,對應上圖中的queueBuffer方法

  • 在新的GraphicBuffer入隊BufferQueue時,BufferQueue會通過回調通知圖形數據的消費者,有新的圖形數據被生產出來了

  • 然后消費者從BufferQueue中出隊一個GraphicBuffer,對應圖中的acquireBuffer方法

  • 待消費者消費完圖形數據后,將空的GraphicBuffer還給BufferQueue以便重復利用,此時對應上圖中的releaseBuffer方法

  • 此時BufferQueue再通過回調通知圖形數據的生產者有空的GraphicBuffer了,圖形數據的生產者又可以從BufferQueue中獲取一個空的GraphicBuffer來填充數據

  • 一直循環2-8步驟,這樣就有條不紊的完成了圖形數據的生產-消費

  • 當然圖形數據的生產者可以不用等待BufferQueue的回調再生產數據,而是一直生產數據然后入隊到BufferQueue,直到BufferQueue滿為止。圖形數據的消費者也可以不用等BufferQueue的回調通知,每次都從BufferQueue中嘗試獲取數據,獲取失敗則嘗試,只是這樣效率比較低,需要不斷的輪訓BufferQueue(因為BufferQueue有同步阻塞和非同步阻塞兩種機種,在非同步阻塞機制下獲取數據失敗不會阻塞該線程直到有數據才喚醒該線程,而是直接返回-1)。


    同時使用BufferQueue的生產者和消費者往往處在不同的進程,BufferQueue內部使用共享內存和Binder在不同的進程傳遞數據,減少數據拷貝提高效率。


    和BufferQueue有關的幾個類分別是:

  • BufferBufferCore:BufferQueue的實際實現

  • BufferSlot:用來存儲GraphicBuffer

  • BufferState:表示GraphicBuffer的狀態

  • IGraphicBufferProducer:BufferQueue的生產者接口,實現類是BufferQueueProducer

  • IGraphicBufferConsumer:BufferQueue的消費者接口,實現類是BufferQueueConsumer

  • GraphicBuffer:表示一個Buffer,可以填充圖像數據

  • ANativeWindow_Buffer:GraphicBuffer的父類

  • ConsumerBase:實現了ConsumerListener接口,在數據入隊列時會被調用到,用來通知消費者


  • BufferQueue中用BufferSlot來存儲GraphicBuffer,使用數組來存儲一系列BufferSlot,數組默認大小為64。

    GraphicBuffer用BufferState來表示其狀態,有以下狀態:

  • FREE:表示該Buffer沒有被生產者-消費者所使用,該Buffer的所有權屬于BufferQueue

  • DEQUEUED:表示該Buffer被生產者獲取了,該Buffer的所有權屬于生產者

  • QUEUED:表示該Buffer被生產者填充了數據,并且入隊到BufferQueue了,該Buffer的所有權屬于BufferQueue

  • ACQUIRED:表示該Buffer被消費者獲取了,該Buffer的所有權屬于消費者


  • 為什么需要這些狀態呢??假設不需要這些狀態,實現一個簡單的BufferQueue,假設是如下實現:

    BufferQueue{ vector<GraphicBuffer> slots; void push(GraphicBuffer slot){ slots.push(slot); }
    GraphicBuffer pull(){ return slots.pull(); }}


    生產者生產完數據后,通過調用BufferQueue的push函數將數據插入到vector中。消費者調用BufferQueue的pull函數出隊一個Buffer數據。


    上述實現的問題在于,生產者每次都需要自行創建GraphicBuffer,而消費者每次消費完數據后的GraphicBuffer就被釋放了,GraphicBuffer沒有得到循環利用。而在Android中,由于BufferQueue的生產者-消費者往往處于不同的進程,GraphicBuffer內部是需要通過共享內存來連接生成者-消費者進程的,每次創建GraphicBuffer,即意味著需要創建共享內存,效率較低。


    而BufferQueue中用BufferState來表示GraphicBuffer的狀態則解決了這個問題。每個GraphicBuffer都有當前的狀態,通過維護GraphicBuffer的狀態,完成GraphicBuffer的復用。


    由于BufferQueue內部實現是BufferQueueCore,下文均用BufferQueueCore代替BufferQueue。先介紹下BufferQueueCore內部相應的數據結構,再介紹BufferQueue的狀態扭轉過程和生產-消費過程。

    以下是Buffer的入隊/出隊操作和BufferState的狀態扭轉的過程,這里只介紹非同步阻塞模式。


    2.1 BufferQueueCore內部數據結構

    核心數據結構如下:

    BufferQueueDefs::SlotsType?mSlots:用數組存放的Slot,數組默認大小為BufferQueueDefs::NUM_BUFFER_SLOTS,具體是64,代表所有的Slotstd::set<int>?mFreeSlots:當前所有的狀態為FREE的Slot,這些Slot沒有關聯上具體的GraphicBuffer,后續用的時候還需要關聯上GraphicBufferstd::list<int>?mFreeBuffers:當前所有的狀態為FREE的Slot,這些Slot已經關聯上具體的GraphicBuffer,可以直接使用Fifo?mQueue:一個先進先出隊列,保存了生產者生產的數據


    在BufferQueueCore初始化時,由于此時隊列中沒有入隊任何數據,按照上面的介紹,此時mFreeSlots應該包含所有的Slot,元素大小和mSlots一致,初始化代碼如下:

    for (int slot = 0;?slot?<?BufferQueueDefs::NUM_BUFFER_SLOTS;?++slot)?{ mFreeSlots.insert(slot); }


    2.2 生產者dequeueBuffer

    當生產者可以生產圖形數據時,首先向BufferQueue中申請一塊GraphicBuffer。調用函數是BufferQueueProducer.dequeueBuffer,如果當前BufferQueue中有可用的GraphicBuffer,則返回其對用的索引,如果不存在,則返回-1,代碼在BufferQueueProducer,流程如下:

    status_t?BufferQueueProducer::dequeueBuffer(int *outSlot,????????sp<android::Fence>?*outFence,?bool async, uint32_t width, uint32_t height, PixelFormat format, uint32_t usage) {
    //1. 尋找可用的Slot,可用指Buffer狀態為FREE status_t status = waitForFreeSlotThenRelock("dequeueBuffer", async, &found, &returnFlags); if?(status?!=?NO_ERROR)?{ return status; } //2.找到可用的Slot,將Buffer狀態設置為DEQUEUED,由于步驟1找到的Slot狀態為FREE,因此這一步完成了FREE到DEQUEUED的狀態切換 *outSlot = found; ATRACE_BUFFER_INDEX(found); attachedByConsumer = mSlots[found].mAttachedByConsumer;????????????mSlots[found].mBufferState?=?BufferSlot::DEQUEUED; //3. 找到的Slot如果需要申請GraphicBuffer,則申請GraphicBuffer,這里采用了懶加載機制,如果內存沒有申請,申請內存放在生產者來處理 if (returnFlags & BUFFER_NEEDS_REALLOCATION) { status_t error; sp<GraphicBuffer> graphicBuffer(mCore->mAllocator->createGraphicBuffer(width, height, format, usage, &error)); graphicBuffer->setGenerationNumber(mCore->mGenerationNumber); mSlots[*outSlot].mGraphicBuffer = graphicBuffer; }}


    關鍵在于尋找可用Slot,waitForFreeSlotThenRelock的流程如下:

    status_t?BufferQueueProducer::waitForFreeSlotThenRelock(const char* caller, bool async, int* found, status_t* returnFlags) const {
    //1. mQueue 是否太多 bool tooManyBuffers = mCore->mQueue.size()> static_cast<size_t>(maxBufferCount); if (tooManyBuffers) {
    } else { // 2. 先查找mFreeBuffers中是否有可用的,由2.1介紹可知,mFreeBuffers中的元素關聯了GraphicBuffer,直接可用 if?(!mCore->mFreeBuffers.empty())?{ auto slot = mCore->mFreeBuffers.begin(); *found = *slot; mCore->mFreeBuffers.erase(slot); } else if?(mCore->mAllowAllocation?&&?!mCore->mFreeSlots.empty())?{ //?3.?再查找mFreeSlots中是否有可用的,由2.1可知,初始化時會填充滿這個列表,因此第一次調用一定不會為空。同時用這個列表中的元素需要關聯上GraphicBuffer才可以直接使用,關聯的過程由外層函數來實現 auto slot = mCore->mFreeSlots.begin(); // Only return free slots up to the max buffer count if (*slot < maxBufferCount) { *found = *slot; mCore->mFreeSlots.erase(slot); } } }
    ?????????tryAgain?=?(*found?==?BufferQueueCore::INVALID_BUFFER_SLOT)?|| tooManyBuffers; //4. 如果找不到可用的Slot或者Buffer太多(同步阻塞模式下),則可能需要等 if (tryAgain) { if (mCore->mDequeueBufferCannotBlock && (acquiredCount <= mCore->mMaxAcquiredBufferCount)) { return WOULD_BLOCK; } mCore->mDequeueCondition.wait(mCore->mMutex); }}


    waitForFreeSlotThenRelock函數會嘗試尋找一個可用的Slot,可用的Slot狀態一定是FREE(因為是從兩個FREE狀態的列表中獲取的),然后dequeueBuffer將狀態改變為DEQUEUED,即完成了狀態的扭轉。


    waitForFreeSlotThenRelock返回可用的Slot分為兩種:

  • 從mFreeBuffers中獲取到的,mFreeBuffers中的元素關聯了GraphicBuffer,直接可用

  • 從mFreeSlots中獲取到的,沒有關聯上GraphicBuffer,因此需要申請GraphicBuffer并和Slot關聯上,通過createGraphicBuffer申請一個GraphicBuffer,然后賦值給Slot的mGraphicBuffer完成關聯


  • 小結dequeueBuffer:嘗試找到一個Slot,并完成Slot與GraphicBuffer的關聯(如果需要),然后將Slot的狀態由FREE扭轉成DEQUEUED。返回Slot在BufferQueueCore中mSlots對應的索引。


    2.3 生產者requestBuffer

    dequeueBuffer函數獲取到了可用Slot的索引后,通過requestBuffer獲取到對應的GraphicBuffer。流程如下:

    status_t?BufferQueueProducer::requestBuffer(int?slot,?sp<GraphicBuffer>*?buf)?{
    // 1. 判斷slot參數是否合法 if (slot < 0?||?slot?>=?BufferQueueDefs::NUM_BUFFER_SLOTS)?{ BQ_LOGE("requestBuffer:?slot?index?%d?out?of?range?[0,?%d)",????????????????slot,?BufferQueueDefs::NUM_BUFFER_SLOTS); return BAD_VALUE; } else if?(mSlots[slot].mBufferState?!=?BufferSlot::DEQUEUED)?{ BQ_LOGE("requestBuffer:?slot?%d?is?not?owned?by?the?producer?" "(state = %d)", slot, mSlots[slot].mBufferState); return BAD_VALUE; }
    //2. 將mRequestBufferCalled置為true mSlots[slot].mRequestBufferCalled = true; *buf = mSlots[slot].mGraphicBuffer; return NO_ERROR;}


    這一步不是必須的。業務層可以直接通過Slot的索引獲取到對應的GraphicBuffer。


    2.4 生產者queueBuffer

    上文dequeueBuffer獲取到一個Slot后,就可以在Slot對應的GraphicBuffer上完成圖像數據的生產了,可以是View的主線程Draw過程,也可以是SurfaceView的子線程繪制過程,甚至可以是MediaCodec的解碼過程。


    填充完圖像數據后,需要將Slot入隊BufferQueueCore(數據寫完了,可以傳給生產者-消費者隊列,讓消費者來消費了),入隊調用queueBuffer函數。queueBuffer的流程如下:

    status_t?BufferQueueProducer::queueBuffer(int?slot, const QueueBufferInput &input, QueueBufferOutput *output) {
    // 1. 先判斷傳入的Slot是否合法 if (slot < 0 || slot >= maxBufferCount) { BQ_LOGE("queueBuffer:?slot?index?%d?out?of?range?[0,?%d)", slot, maxBufferCount); return BAD_VALUE; }
    //2. 將Buffer狀態扭轉成QUEUED,此步完成了Buffer的狀態由DEQUEUED到QUEUED的過程 mSlots[slot].mFence = fence;????????mSlots[slot].mBufferState?=?BufferSlot::QUEUED; ++mCore->mFrameCounter; mSlots[slot].mFrameNumber = mCore->mFrameCounter;
    //3. 入隊mQueue if (mCore->mQueue.empty()) { mCore->mQueue.push_back(item); frameAvailableListener = mCore->mConsumerListener; }
    // 4. 回調frameAvailableListener,告知消費者有數據入隊了 if?(frameAvailableListener?!=?NULL) { frameAvailableListener->onFrameAvailable(item); } else if?(frameReplacedListener?!=?NULL) { frameReplacedListener->onFrameReplaced(item); }}

    從上面的注釋可以看到,queueBuffer的主要步驟如下:

  • 將Buffer狀態扭轉成QUEUED,此步完成了Buffer的狀態由DEQUEUED到QUEUED的過程

  • 將Buffer入隊到BufferQueueCore的mQueue隊列中

  • 回調frameAvailableListener,告知消費者有數據入隊,可以來消費數據了,frameAvailableListener是消費者注冊的回調


  • 小結queueBuffer:將Slot的狀態扭轉成QUEUED,并添加到mQueue中,最后通知消費者有數據入隊。


    2.5 消費者acquireBuffer

    在消費者接收到onFrameAvailable回調時或者消費者主動想要消費數據,調用acquireBuffer嘗試向BufferQueueCore獲取一個數據以供消費。消費者的代碼在BufferQueueConsumer中,acquireBuffer流程如下:

    status_t?BufferQueueConsumer::acquireBuffer(BufferItem*?outBuffer, nsecs_t expectedPresent, uint64_t maxFrameNumber) {
    //1. 如果隊列為空,則直接返回 if (mCore->mQueue.empty()) { return NO_BUFFER_AVAILABLE; }
    //2. 取出mQueue隊列的第一個元素,并從隊列中移除????????BufferQueueCore::Fifo::iterator?front(mCore->mQueue.begin()); int slot = front->mSlot; *outBuffer = *front; mCore->mQueue.erase(front);
    //3. 處理expectedPresent的情況,這種情況可能會連續丟幾個Slot的“顯示”時間小于expectedPresent的情況,這種情況下這些Slot已經是“過時”的,直接走下文的releaseBuffer消費流程,代碼比較長,忽略了
    //4. 更新Slot的狀態為ACQUIRED if (mCore->stillTracking(front)) { mSlots[slot].mAcquireCalled = true; mSlots[slot].mNeedsCleanupOnRelease = false;????????????mSlots[slot].mBufferState?=?BufferSlot::ACQUIRED;????????????mSlots[slot].mFence?=?Fence::NO_FENCE; }
    //5. 如果步驟3有直接releaseBuffer的過程,則回調生產者,有數據被消費了 if?(listener?!=?NULL) { for (int i = 0; i < numDroppedBuffers; ++i) { listener->onBufferReleased(); } }
    }


    從上面的注釋可以看到,acquireBuffer的主要步驟如下:

  • 從mQueue隊列中取出并移除一個元素

  • 改變Slot對應的狀態為ACQUIRED

  • 如果有丟幀邏輯,回調告知生產者有數據被消費,生產者可以準備生產數據了


  • 小結acquireBuffer:將Slot的狀態扭轉成ACQUIRED,并從mQueue中移除,最后通知生產者有數據出隊。


    2.6 消費者releaseBuffer

    消費者獲取到Slot后開始消費數據(典型的消費如SurfaceFlinger的UI合成),消費完畢后,需要告知BufferQueueCore這個Slot被消費者消費完畢了,可以給生產者重新生產數據,releaseBuffer流程如下:

    status_t?BufferQueueConsumer::releaseBuffer(int?slot,?uint64_t?frameNumber, const sp<Fence>& releaseFence, EGLDisplay eglDisplay,EGLSyncKHR eglFence) {
    //1. 檢查Slot是否合法 if (slot < 0?||?slot?>=?BufferQueueDefs::NUM_BUFFER_SLOTS?||????????? return BAD_VALUE; }
    //2.?容錯處理:如果要處理的Slot存在于mQueue中,那么說明這個Slot的來源不合法,并不是從2.5的acquireBuffer獲取的Slot,拒絕處理????????BufferQueueCore::Fifo::iterator?current(mCore->mQueue.begin()); while?(current?!=?mCore->mQueue.end())?{ if (current->mSlot == slot) { return BAD_VALUE; } ++current; }
    // 3. 將Slot的狀態扭轉為FREE,之前是ACQUIRED,并將該Slot添加到BufferQueueCore的mFreeBuffers列表中(mFreeBuffers的定義參考2.1的介紹) if?(mSlots[slot].mBufferState?==?BufferSlot::ACQUIRED)?{ mSlots[slot].mEglDisplay = eglDisplay; mSlots[slot].mEglFence = eglFence; mSlots[slot].mFence = releaseFence;????????????????mSlots[slot].mBufferState?=?BufferSlot::FREE; mCore->mFreeBuffers.push_back(slot); listener = mCore->mConnectedProducerListener; BQ_LOGV("releaseBuffer:?releasing?slot?%d", slot); }
    // 4. 回調生產者,有數據被消費了 if?(listener?!=?NULL) { listener->onBufferReleased(); }}


    從上面的注釋可以看到,releaseBuffer的主要步驟如下:

  • 將Slot的狀態扭轉為FREE

  • 將被消費的Slot添加到mFreeBuffers供后續的生產者dequeueBuffer使用

  • 回調告知生產者有數據被消費,生產者可以準備生產數據了


  • 小結releaseBuffer:將Slot的狀態扭轉成FREE,并添加到BufferQueueCore?mFreeBuffers隊列中,最后通知生產者有數據出隊。


    總結下狀態變化的過程:


    上面主要介紹了BufferQueue的設計思想和內部實現。


    下面將繼續介紹BufferQueue,著重介紹Android中對于BufferQueue的常用封裝,以及SurfaceView中使用BufferQueue的具體實現。


    3.BufferQueue常用封裝類

    在實際應用中,除了直接使用BuferQueue外,更多的是使用Surface/SurfaceTexture,其對BufferQueue做了包裝,方便業務更方便的使用BufferQueue。Surface作為BufferQueue的生產者,SurfaceTexture作為BufferQueue的消費者。


    3.1 Surface

    Surface的構造函數如下:

    Surface::Surface( const sp<IGraphicBufferProducer>& bufferProducer, bool controlledByApp)????:?mGraphicBufferProducer(bufferProducer), mGenerationNumber(0)


    構造函數需要傳入一個生產者的引用,和BufferQueue的交互均有這個生產者的引用來完成。dequeueBuffer的流程如下:

    int?Surface::dequeueBuffer(android_native_buffer_t**?buffer,?int* fenceFd) {
    // 1. 調用mGraphicBufferProducer的dequeueBuffer方法,嘗試獲取一個Slot索引 int buf = -1; sp<Fence> fence; status_t result = mGraphicBufferProducer->dequeueBuffer(&buf, &fence, swapIntervalZero, reqWidth, reqHeight, reqFormat, reqUsage);
    if (result < 0) { ALOGV("dequeueBuffer:?IGraphicBufferProducer::dequeueBuffer(%d,?%d,?%d,?%d,?%d)" "failed:?%d", swapIntervalZero, reqWidth, reqHeight, reqFormat, reqUsage, result); return result; } // 2. 調用mGraphicBufferProducer的requestBuffer方法,嘗試獲取Slot sp<GraphicBuffer>& gbuf(mSlots[buf].buffer); if?((result?&?IGraphicBufferProducer::BUFFER_NEEDS_REALLOCATION)?||?gbuf?==?0) { result = mGraphicBufferProducer->requestBuffer(buf, &gbuf); if?(result?!=?NO_ERROR)?{ ALOGE("dequeueBuffer:?IGraphicBufferProducer::requestBuffer?failed:?%d", result); mGraphicBufferProducer->cancelBuffer(buf, fence); return result; } }
    // 3. 返回GraphicBuffer *buffer = gbuf.get();}


    queueBuffer也是如下,流程如下:

    int?Surface::queueBuffer(android_native_buffer_t* buffer, int fenceFd) {
    ????IGraphicBufferProducer::QueueBufferOutput?output;????IGraphicBufferProducer::QueueBufferInput input(timestamp, isAutoTimestamp, mDataSpace, crop, mScalingMode, mTransform ^ mStickyTransform, mSwapIntervalZero, fence, mStickyTransform); // 1. 直接調用mGraphicBufferProducer的queueBuffer方法即可 status_t err = mGraphicBufferProducer->queueBuffer(i, input, &output); if?(err?!=?OK)??{ ALOGE("queueBuffer:?error?queuing?buffer?to?SurfaceTexture,?%d", err); }}


    Surface還提供了lock函數,用來支持雙緩沖,內部也是調用dequeueBuffer方法獲取最新的Buffer:

    status_t?Surface::lock( ANativeWindow_Buffer* outBuffer, ARect* inOutDirtyBounds){
    ANativeWindowBuffer* out; int fenceFd = -1; //1. 獲取實際Buffer status_t err = dequeueBuffer(&out, &fenceFd);
    //2. 處理雙緩沖 if (canCopyBack) { // copy the area that is invalid and not repainted this round const Region copyback(mDirtyRegion.subtract(newDirtyRegion)); if?(!copyback.isEmpty()) copyBlt(backBuffer, frontBuffer, copyback); }}


    Surface也提供了unlockAndPost方法,將數據給到BufferQueue:

    status_t?Surface::unlockAndPost(){ if (mLockedBuffer == 0) { ALOGE("Surface::unlockAndPost?failed,?no?locked?buffer"); return INVALID_OPERATION; }
    int fd = -1; status_t err = mLockedBuffer->unlockAsync(&fd); ALOGE_IF(err, "failed unlocking buffer (%p)", mLockedBuffer->handle);
    //1. 將生產好的數據給到BufferQueue err = queueBuffer(mLockedBuffer.get(), fd); ALOGE_IF(err, "queueBuffer (handle=%p) failed (%s)", mLockedBuffer->handle, strerror(-err));
    mPostedBuffer = mLockedBuffer; mLockedBuffer = 0; return err;}


    3.2 SurfaceTexture

    SurfaceTexture作為BufferQueue的消費者,其初始化代碼如下:

    static void SurfaceTexture_init(JNIEnv* env, jobject thiz, jboolean isDetached, jint texName, jboolean singleBufferMode, jobject weakThiz){
    sp<IGraphicBufferProducer> producer; sp<IGraphicBufferConsumer> consumer; //1. 創建一個BufferQueue????BufferQueue::createBufferQueue(&producer,?&consumer);
    if (singleBufferMode) { consumer->disableAsyncBuffer(); consumer->setDefaultMaxBufferCount(1); }
    //2. 創建一個消費者實例surfaceTexture sp<GLConsumer> surfaceTexture; if (isDetached) { surfaceTexture = new GLConsumer(consumer, GL_TEXTURE_EXTERNAL_OES, true, true); } else { surfaceTexture = new GLConsumer(consumer, texName, GL_TEXTURE_EXTERNAL_OES, true, true); }
    //3. 將消費者實例和該BufferQueue對應的生產者保存到java層,這樣Surface構造時,就可以獲取到該BufferQueue對應的生產者了 SurfaceTexture_setSurfaceTexture(env, thiz, surfaceTexture); SurfaceTexture_setProducer(env, thiz, producer);
    }


    消費的方法是updateTexImage,流程如下:

    static void SurfaceTexture_updateTexImage(JNIEnv* env, jobject thiz){ // 1. 先獲取到初始化時構造的消費者 sp<GLConsumer> surfaceTexture(SurfaceTexture_getSurfaceTexture(env, thiz)); // 2. 調用消費者的updateTexImage方法 status_t err = surfaceTexture->updateTexImage方法(); if (err == INVALID_OPERATION) { jniThrowException(env, IllegalStateException, "Unable to update texture contents (see " "logcat for details)"); } else if (err < 0) { jniThrowRuntimeException(env, "Error during updateTexImage (see logcat for details)"); }}


    GLConsumer的updateTextImage實現如下:

    status_t?GLConsumer::updateTexImage()?{ BufferItem item; //1. 調用自身的acquireBufferLocked方法 err = acquireBufferLocked(&item, 0);:updateTexImage()?{
    // Release the previous buffer. err = updateAndReleaseLocked(item); if?(err?!=?NO_ERROR)?{ glBindTexture(mTexTarget, mTexName); return err; }
    }


    acquireBufferLocked方法,最終走到了ConsumerBase的acquireBufferLocked方法。

    status_t?ConsumerBase::acquireBufferLocked(BufferItem?*item, nsecs_t presentWhen, uint64_t maxFrameNumber) { //1. 最終還是走到了消費者的acquireBuffer方法,消費者對應上面的BufferQueueConsumer status_t err = mConsumer->acquireBuffer(item, presentWhen, maxFrameNumber); if?(err?!=?NO_ERROR)?{ return err; }
    return OK;}


    同理,消費者消費數據的方法是releaseTexImage,最終也會走到BufferQueueConsumer的releaseBufferLocked方法,這里不再描述了。


    4.BufferQueue的實例

    上述介紹了BufferQueue的內部實現,以及常用的封裝類。接下來將介紹一個具體的實例。


    Android中,SurfaceView作為系統提供的組件,因為可以在子線程中繪制提高性能,SurfaceView擁有自身的Surface,不需要和Activity的Surface共享,在SurfaceFlinger中,Activity的Surface和SurfaceView的Surface是平級且互相獨立的,可以獨立的進行合成。那我們來看一下SurfaceView是怎么使用BufferQueue的。


    4.1 數據的生產過程

    SurfaceView的Surface創建過程,這里不關注,有興趣的可以參考?android?SurfaceView繪制實現原理解析?這篇文章,我們主要關注其中與BufferQueue相關的繪制和顯示步驟。

    使用SuerfaceView繪制偽碼如下:?

    Canvas canvas = null; try { canvas = holder.lockCanvas(null); //實際的draw }catch (Exception e) { // TODO: handle exception e.printStackTrace(); }finally { if(canvas?!=?null) { holder.unlockCanvasAndPost(canvas); }

    需要調用lockCanvas和unlockCanvasAndPost方法,這兩個方法的作用是什么呢?


    先看下lockCanvas,調用流程是:

  • SurfaceHolder.lockCanvas

  • SurfaceHolder.internalLockCanvas

  • Surface.lockCanvas?

  • Surface.nativeLockCanvas?

  • nativeLockCanvas實現如下:

    static jlong nativeLockCanvas(JNIEnv* env, jclass clazz, jlong nativeObject, jobject canvasObj, jobject dirtyRectObj) { sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject));
    ANativeWindow_Buffer outBuffer; //1.?通過Surface::lock方法,獲取一個合適的Buffer status_t err = surface->lock(&outBuffer, dirtyRectPtr);
    //2. 構造一個Bitmap,地址指向步驟1獲取的Buffer的地址,這樣在這個Bitmap上繪制的內容,直接繪制到了GraphicBuffer,如果GraphicBuffer的內存是SurfaceFlinger通過共享內存申請的,那么SurfaceFlinger就能直接看到繪制的圖形數據????SkImageInfo?info?=?SkImageInfo::Make(outBuffer.width,?outBuffer.height, convertPixelFormat(outBuffer.format), kPremul_SkAlphaType); SkBitmap bitmap; ssize_t bpr = outBuffer.stride * bytesPerPixel(outBuffer.format); bitmap.setInfo(info, bpr); if (outBuffer.width > 0 && outBuffer.height > 0) { bitmap.setPixels(outBuffer.bits); } else { // be safe with an empty bitmap. bitmap.setPixels(NULL); }
    // 3. 將創建的Bitmap設置給Canvas,作為畫布????Canvas*?nativeCanvas?=?GraphicsJNI::getNativeCanvas(env,?canvasObj); nativeCanvas->setBitmap(bitmap);
    }


    從這里可以看到,nativeLockCanvas的步驟主要如下:

  • 通過調用Surface::lock方法(內部也是調用dequeueBuffer和requestBuffer方法),獲取到一個GraphicBuffer

  • 將步驟1獲取的GraphicBuffer構造成一個Bitmap,設置給Canvas

  • 應用通過這個Canvas就可以繪制圖形了



  • 在繪制圖形完成后,調用unlockCanvasAndPost方法,調用流程是:

  • SurfaceHolder.unlockCanvasAndPost

  • Surface.unlockCanvasAndPost

  • Surface.nativeUnlockCanvasAndPost


  • nativeUnlockCanvasAndPost?的實現如下:

    static void nativeUnlockCanvasAndPost(JNIEnv* env, jclass clazz, jlong nativeObject, jobject canvasObj) { sp<Surface> surface(reinterpret_cast<Surface *>(nativeObject)); if?(!isSurfaceValid(surface))?{ return; }
    // detach the canvas from the surface????Canvas*?nativeCanvas?=?GraphicsJNI::getNativeCanvas(env,?canvasObj); nativeCanvas->setBitmap(SkBitmap());
    // 直接調用Surface的unlockAndPost方法,有上文可知unlockAndPost內部最終也會調用到qeueBuffer方法 status_t err = surface->unlockAndPost方法,有上文可知unlockAndPost內部最終也會調用到qeueBuffer方法(); if (err < 0) { doThrowIAE(env); }}


    從注釋可以看到,這個方法,最終會調用到Surface的unlockAndPost方法方法,而該方法內部最終也會調用到BufferQueueProducer的queueBuffer方法。即完成了數據的生產和入隊。


    4.2 數據的消費過程

    SurfaceView繪制的數據,傳遞過BufferQueue后,最終由SurfaceFlinger進行合成消費。SurfaceFlinger的消費由SurfaceFlingerConsumer實現,流程如下:

    status_t?SurfaceFlingerConsumer::updateTexImage(BufferRejecter*?rejecter, const DispSync& dispSync, uint64_t maxFrameNumber){ BufferItem item; // 1. 調用acquireBufferLocked獲取一個Slot err = acquireBufferLocked(&item, computeExpectedPresent(dispSync), maxFrameNumber); if?(err?!=?NO_ERROR)?{ return err; }

    //2. 消費完畢,釋放Slot err = updateAndReleaseLocked(item); if?(err?!=?NO_ERROR)?{ return err; }}


    acquireBufferLocked的實現如下:

    status_t?SurfaceFlingerConsumer::acquireBufferLocked(BufferItem*?item, nsecs_t presentWhen, uint64_t maxFrameNumber) { //1.?調用?GLConsumer::acquireBufferLocked,最終會調用到BufferQueueConsumer的acquireBuffer方法 status_t?result?=?GLConsumer::acquireBufferLocked(item,?presentWhen, maxFrameNumber); if (result == NO_ERROR) { mTransformToDisplayInverse = item->mTransformToDisplayInverse; mSurfaceDamage = item->mSurfaceDamage; } return result;}


    而updateAndReleaseLocked方法的流程如下:

    status_t?GLConsumer::updateAndReleaseLocked(const BufferItem& item){ // Do whatever sync ops we need to do before releasing the old slot. err = syncForReleaseLocked(mEglDisplay); if?(err?!=?NO_ERROR)?{ //1. releaseBufferLocked釋放Slot,最終會調用到BufferQueueConsumer的releaseBuffer方法 releaseBufferLocked(buf, mSlots[buf].mGraphicBuffer, mEglDisplay, EGL_NO_SYNC_KHR); return err; }}


    5. 總結

    本文對BufferQueue的內部實現做了介紹,結合入隊/出對說明了BufferQueue內部Slot的狀態扭轉過程,并介紹了常用的BufferQueue封裝類,最后介紹了一個基于BufferQueue的例子。




    總結

    以上是生活随笔為你收集整理的深入浅出Android BufferQueue的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    久久www免费视频 | 91亚色视频| 久久久久免费精品视频 | 国产精品手机播放 | 日韩a免费 | 99久免费精品视频在线观看 | 亚洲综合色站 | 久久99精品久久久久久久久久久久 | 精品久久久久久久 | a午夜在线 | 精品久久网| 色黄视频免费观看 | 精品国内自产拍在线观看视频 | 国产这里只有精品 | 欧美在线久久 | 亚洲欧美综合精品久久成人 | 欧美久久久久久久久久久久 | 国产精品福利av | 久久刺激视频 | 久久久久久蜜桃一区二区 | 久久99深爱久久99精品 | 99精品国产一区二区 | 成人欧美一区二区三区在线观看 | 亚洲人成在线观看 | 国产精品www | 国内免费久久久久久久久久久 | 亚洲国产mv | 久久综合网色—综合色88 | 97超碰免费在线 | 91视视频在线直接观看在线看网页在线看 | 久久综合九色综合欧美就去吻 | 欧美激情另类 | 九九免费精品视频在线观看 | 免费精品视频 | 亚洲热久久 | 国产一区二区在线免费播放 | 香蕉精品在线观看 | 欧美日本不卡视频 | 婷婷av网站| 中文字幕在线日 | 国产精品九九九 | 亚洲成av人片在线观看无 | 91九色视频在线观看 | 日韩精品欧美专区 | 欧美另类成人 | 777奇米四色 | av观看免费在线 | 精品国产乱码久久久久久三级人 | 日韩在线观看精品 | 亚洲va欧美va人人爽 | 欧美久久久久久久久久久 | 不卡国产在线 | 久久老司机精品视频 | 伊人影院在线观看 | 欧美日韩精品免费观看 | 人人爽久久久噜噜噜电影 | 亚洲精品tv久久久久久久久久 | 国产日韩三级 | 国产一级一级国产 | 91精品第一页 | 亚洲视频在线播放 | 国产在线一卡 | 日韩av资源站 | 色综合天天综合网国产成人网 | 99久在线精品99re8热视频 | 日本中文在线播放 | 一区二区三区av在线 | 九九爱免费视频 | 99久久夜色精品国产亚洲96 | 中文字幕国产视频 | 91激情视频在线 | 夜夜操夜夜干 | 中文视频在线 | 国产精品久久久久久久久久直播 | 亚洲精品国产成人av在线 | 69国产精品成人在线播放 | 97精品国产97久久久久久粉红 | 日韩 精品 一区 国产 麻豆 | 最新av免费在线观看 | 免费观看v片在线观看 | 国产流白浆高潮在线观看 | 在线免费观看涩涩 | 国产精品久久久久久一二三四五 | 看片一区二区三区 | 欧美日韩国产区 | 国产精品一区二区中文字幕 | www狠狠| 久久精品视频在线观看免费 | 91免费观看国产 | 久久综合偷偷噜噜噜色 | 中文字幕123区| 五月婷婷一区二区三区 | 成人少妇影院yyyy | 特级aaa毛片| 综合国产视频 | 国产91综合一区在线观看 | 亚洲精品乱码久久久久 | 久久看看 | 国产免费久久精品 | 国产精品高 | 免费观看成人网 | 婷婷六月中文字幕 | 国产日韩欧美在线免费观看 | 成人在线一区二区 | 在线免费高清视频 | 天天操操 | 日韩一二区在线观看 | 亚洲精品在线免费播放 | 日韩午夜在线播放 | 超碰免费在线公开 | 超碰在线观看97 | 久久社区视频 | 欧美国产在线看 | 337p西西人体大胆瓣开下部 | 国产一区二区影院 | 国产精品久久久久免费 | 激情网综合 | 色国产在线 | 免费精品在线观看 | 精品xxx| 激情丁香 | 免费特级黄色片 | 日韩一级电影在线 | 黄色国产高清 | 亚洲经典视频在线观看 | 一区二区中文字幕在线播放 | 日韩精品1区2区 | 亚洲欧洲精品一区 | 99精品国产一区二区三区不卡 | 国产96在线观看 | 久草在线在线精品观看 | 免费午夜网站 | 久久99久久99精品免观看粉嫩 | 日日夜夜精品视频天天综合网 | 啪啪免费视频网站 | 日韩美女黄色片 | 国产精品乱码一区二区视频 | 免费麻豆视频 | 久久久久久免费视频 | 免费看片亚洲 | 成人免费在线视频 | 少妇搡bbb| 国产91免费观看 | 91精品色 | 国产一级片观看 | 日韩精品视频在线免费观看 | 91精品网站在线观看 | 日韩高清不卡一区二区三区 | 久久精品视频99 | 国产精品视频app | 亚洲小视频在线 | 亚洲精品无 | 国产麻豆剧果冻传媒视频播放量 | 久草资源在线 | 日韩免费视频一区二区 | 日韩欧美一区二区三区视频 | 97色国产 | 狠狠网亚洲精品 | 一区二区电影在线观看 | 欧美另类xxxxx| 黄色成人在线 | 福利电影一区二区 | 久久久久综合精品福利啪啪 | 亚洲欧洲久久久 | 国产日韩精品一区二区在线观看播放 | 97精品久久人人爽人人爽 | 国产精品99久久久久人中文网介绍 | 欧美日韩精 | 97在线影院 | 麻豆一二| 尤物97国产精品久久精品国产 | 中文字幕视频 | 久久国产精品免费视频 | 欧美日韩精品二区第二页 | 中文字幕视频在线播放 | 少妇av片 | 国产精品久久三 | 日韩精品在线一区 | 日本韩国欧美在线观看 | 国产精品九九热 | 免费在线电影网址大全 | 五月婷婷综合激情 | 91麻豆视频网站 | 992tv人人草| 国产精品色婷婷视频 | 色婷婷亚洲综合 | 夜夜操天天干, | 午夜在线观看影院 | 丝袜网站在线观看 | 欧美日韩三级 | 在线黄色国产 | 免费看一级特黄a大片 | 亚洲成aⅴ人片久久青草影院 | 精品国产成人 | 四虎影视国产精品免费久久 | 日本久久久亚洲精品 | 日本少妇久久久 | 日日精品| 国产精品18p | 丝袜美腿亚洲 | 四虎免费在线观看视频 | 午夜精品成人一区二区三区 | 成人高清av在线 | 顶级欧美色妇4khd | 天天草天天爽 | 亚洲视频久久久久 | 日本精品久久久一区二区三区 | 亚洲乱码国产乱码精品天美传媒 | 一区 在线 影院 | 成人av在线播放网站 | a天堂在线看| 精品国产乱码久久久久久1区二区 | 免费视频二区 | 激情中文字幕 | 国产高清在线免费 | 国产精品久久久久久久婷婷 | 成人av地址 | 亚洲美女在线一区 | 国产一区二区三区免费视频 | 国产日韩欧美在线播放 | 99视频免费播放 | 天堂av在线网 | 伊人色综合网 | 黄网站色视频 | 天天综合狠狠精品 | 日韩资源视频 | 丁香5月婷婷| 韩国精品在线观看 | 91丨九色丨高潮丰满 | 探花视频在线观看免费 | 国产91在线观看 | 深爱激情婷婷网 | 91在线日韩 | 天天操天天射天天 | 国产永久免费观看 | 91国内产香蕉 | 国产精品18久久久久vr手机版特色 | 黄色性av| 九九综合九九 | 五月情婷婷 | 国产精品第二十页 | 人人狠狠综合久久亚洲 | 91亚洲精品久久久蜜桃借种 | 婷婷久久久久 | 高清av免费看 | 99久久久国产精品免费99 | 国产又粗又猛又黄 | 青青河边草免费视频 | 97国产超碰在线 | 日本韩国在线不卡 | 欧美国产日韩一区二区三区 | 97超碰在线免费观看 | 狠狠色狠狠色综合日日92 | 黄免费在线观看 | 国产91精品在线播放 | 天天射综合 | 色婷婷综合久久久久 | 黄网站大全 | 这里只有精品视频在线观看 | 欧美一区二区三区四区夜夜大片 | a√天堂中文在线 | 日韩高清免费电影 | av夜夜操 | 色橹橹欧美在线观看视频高清 | 国产精品久久久久9999吃药 | 欧美精品一区二区三区一线天视频 | 亚洲精品久久久久久久蜜桃 | 午夜免费福利片 | 91视频在线看 | 国产一区二区观看 | 一区二区 久久 | 17videosex性欧美| 亚洲综合网 | 黄色福利网 | 最新av中文字幕 | 黄色成人免费电影 | 91久久偷偷做嫩草影院 | 久久99国产精品 | 日韩成人免费在线电影 | 一区二区三区手机在线观看 | 日韩av进入| 99这里只有精品99 | 四虎影视精品成人 | 天天干一干 | 久久免费一级片 | 日韩一区二区三区免费电影 | 久久久久亚洲最大xxxx | 免费黄在线观看 | 在线国产精品视频 | 深爱激情开心 | 亚洲精品在线二区 | 国产精品a成v人在线播放 | 99久久夜色精品国产亚洲 | 黄色免费高清视频 | 日韩aⅴ视频 | 午夜视频在线观看欧美 | 天天射综合网视频 | 91亚洲综合| 四虎小视频 | 91av资源网| 久草久| 国产精品成人品 | av网址在线播放 | 美女视频久久 | 久久精品欧美日韩精品 | 这里有精品在线视频 | av字幕在线 | 亚洲午夜av久久乱码 | 天天操夜夜叫 | 日韩精品一区二区三区在线视频 | 亚洲综合视频在线 | 日韩高清www| 久久免费的视频 | 丁香婷婷综合五月 | 成人蜜桃视频 | 91麻豆精品国产91久久久久久 | 免费毛片aaaaaa | 怡红院av| 中文字幕在线电影 | 91精品一区二区三区蜜臀 | 国色综合 | 国产91综合一区在线观看 | 午夜精品电影 | 欧美一区二区在线刺激视频 | 99爱国产精品 | 成人在线免费观看视视频 | 国产一级不卡毛片 | 国产精品porn | 亚洲午夜久久久久久久久久久 | aⅴ精品av导航| 成年性视频 | 欧美成人性战久久 | 中文乱幕日产无线码1区 | 国产99视频在线观看 | av免费高清观看 | 欧美福利片在线观看 | 成人视屏免费看 | 狠狠狠狠狠色综合 | 久久国产免| 91丨九色丨蝌蚪丨老版 | 亚洲日本va午夜在线影院 | 在线成人中文字幕 | av 一区二区三区四区 | 国产高清综合 | 日韩在线网址 | 狠狠色丁香九九婷婷综合五月 | 亚洲精品乱码白浆高清久久久久久 | 精品久久久久一区二区国产 | 片黄色毛片黄色毛片 | 97香蕉视频| 免费看色的网站 | 一区二区精品在线 | 亚洲欧洲成人精品av97 | 怡春院av| 韩国av永久免费 | 高清av中文字幕 | 四虎精品成人免费网站 | 超碰在线日本 | 国内久久久久久 | 日韩网站在线 | 在线观看免费色 | 人人搞人人爽 | 五月天综合色 | 精品亚洲免费视频 | 麻豆视频在线看 | 69av视频在线观看 | 欧美一级专区免费大片 | 亚洲国产成人久久综合 | 久草免费电影 | 久久夜色精品国产欧美乱 | 国产v在线播放 | 亚洲精品国产精品国自产观看 | 国产免费黄视频在线观看 | 91女人18片女毛片60分钟 | 日韩精品久久久久久久电影竹菊 | 麻豆mv在线观看 | 欧美午夜精品久久久久久孕妇 | 美女视频a美女大全免费下载蜜臀 | 久久国产乱 | 午夜精品久久久久久久久久久久久久 | 天天天天综合 | 91精品视频在线免费观看 | 91片网 | 久草免费在线 | 久久乱码卡一卡2卡三卡四 五月婷婷久 | 欧美日韩中文字幕视频 | 成年人视频在线观看免费 | 夜夜高潮夜夜爽国产伦精品 | 欧美日韩一区二区三区免费视频 | 91视频免费网站 | 国产一级特黄毛片在线毛片 | 天堂网中文在线 | 久久精视频 | 亚洲综合在线发布 | 色综合久久88色综合天天免费 | 青青河边草免费直播 | 一区二区三区 亚洲 | 九九热精品国产 | 欧美在线视频日韩 | 欧美视频xxx | 日韩欧美国产精品 | 在线观看成人小视频 | 久久欧美在线电影 | 日本不卡123区 | 成年人精品 | 天天草天天 | 欧美在线18 | 欧美肥妇free | 亚洲更新最快 | 91丨九色丨高潮 | 国产精品人人做人人爽人人添 | 国产一级黄色免费看 | www.久艹| 亚洲国产视频a | 久久精品91久久久久久再现 | 国产永久免费观看 | 国产精品久久久久婷婷 | 成人免费观看a | 91麻豆精品国产91久久久更新时间 | 麻豆视传媒官网免费观看 | 狠狠撸电影 | 免费99精品国产自在在线 | 男女免费视频观看 | 深夜国产在线 | av天天草| 中文字幕在线观看完整版 | 在线观看视频免费播放 | 91视频免费| 成人毛片在线视频 | 亚洲闷骚少妇在线观看网站 | 国产精品网站一区二区三区 | 日本h在线播放 | 天天操天天操天天操天天操天天操天天操 | 夜夜操天天干, | 91传媒在线播放 | 国产一级电影在线 | 日韩 国产 | 久久中文字幕导航 | 久久99精品国产99久久6尤 | 免费看片黄色 | 香蕉视频在线网站 | 中文字幕在线视频一区 | 九九爱免费视频 | 99热在线看 | 久久久久久久久久久免费 | 久久久美女 | 四季av综合网站 | 日韩成人欧美 | 人人玩人人爽 | 女人高潮一级片 | 狠狠色丁香久久婷婷综合_中 | 99亚洲精品视频 | 亚洲最新av在线网址 | 日韩午夜剧场 | 色在线网站 | 97国产大学生情侣酒店的特点 | 国内精品美女在线观看 | 人人澡人人爽 | 99精品国产高清在线观看 | 99久久久成人国产精品 | 国产成人精品亚洲日本在线观看 | 操久在线 | 在线免费观看的av网站 | 91成年人在线观看 | 久久艹在线观看 | 91超国产 | 97免费中文视频在线观看 | 最近中文字幕大全中文字幕免费 | 国产免费午夜 | 免费国产ww| 一级片观看 | 亚洲成av人片一区二区梦乃 | av中文在线 | 91爱爱电影| 精品五月天 | 国产中的精品av小宝探花 | 免费国产黄线在线观看视频 | 国产高清黄色 | 亚洲国产中文字幕在线观看 | 国产精品久久久久久久久久妇女 | 色婷婷a | 91在线视频免费观看 | 丁香六月欧美 | 在线看毛片网站 | 91av视频在线观看免费 | 日韩精品久久中文字幕 | 97精品一区 | 婷婷色综合 | 91最新网址 | 操操操日日日干干干 | 91福利视频一区 | 日韩va在线观看 | 免费精品视频在线 | 天天操天天曰 | 激情小说网站亚洲综合网 | 高潮久久久久久久久 | 国产精品久久久久久久久久久久午夜片 | 国产美女视频一区 | 成人黄色国产 | 探花在线观看 | 美女久久久久久 | 中文字幕高清在线播放 | 亚洲欧洲国产精品 | 69欧美视频 | 日本久久久亚洲精品 | www.777奇米| 日韩欧美高清不卡 | 国产一级电影网 | 在线观看理论 | 色网站国产精品 | 亚洲传媒在线 | av短片在线观看 | 久久在线播放 | 日日夜夜草| 色综合久久久 | 国产视频在线一区二区 | 国产福利网站 | 96精品高清视频在线观看软件特色 | av丝袜天堂 | 免费国产在线视频 | 亚洲国产伊人 | 亚洲电影网站 | 尤物一区二区三区 | 国产精品嫩草在线 | 日日干夜夜草 | 免费在线观看亚洲视频 | 91黄色在线观看 | 最新成人av | 久久久久久久久精 | 欧美中文字幕久久 | 国产在线观看高清视频 | 91看片淫黄大片一级在线观看 | 综合五月| 免费成人在线观看视频 | www欧美日韩| 国产美腿白丝袜足在线av | 国产成人99久久亚洲综合精品 | 欧美日韩电影在线播放 | 麻豆国产露脸在线观看 | 蜜桃av久久久亚洲精品 | 美女免费网站 | www.香蕉视频在线观看 | 日韩网站一区 | 韩日三级在线 | www中文在线 | 精品二区视频 | 一级黄色大片在线观看 | 久久久久久久久久网站 | 久久久国产精品亚洲一区 | 亚洲精品国产精品乱码在线观看 | 全黄网站 | 九九亚洲视频 | 色综合中文字幕 | 亚洲欧美精品一区 | 最新中文字幕在线观看视频 | 日韩精品一卡 | 91九色porny在线 | 日本公乱妇视频 | 久影院 | 免费观看版 | 在线观看你懂的网址 | 日韩av影视 | 性色视频在线 | 中文字幕成人在线 | 欧美精品xxx| 日日日日 | 日韩v欧美v日本v亚洲v国产v | 在线视频精品 | 欧美日韩在线观看一区二区三区 | 亚洲精品日韩在线观看 | 狠狠做深爱婷婷综合一区 | 狠狠色丁香婷婷 | 久草视频在线新免费 | 国产精品一区二区三区四区在线观看 | 人人狠狠综合久久亚洲婷 | 色播亚洲婷婷 | 99视频网站| 黄色av成人在线 | 欧美日韩三级在线观看 | 久久婷婷国产色一区二区三区 | 麻豆av一区二区三区在线观看 | av网站在线观看播放 | 二区三区在线视频 | 国产高清av免费在线观看 | 国产精品久久久久久久久免费看 | 午夜手机电影 | 国产精品 中文在线 | 国产在线观看你懂得 | 黄色资源在线观看 | 91福利社区在线观看 | 久久久免费电影 | 久久网址 | 日韩一二区在线 | 婷婷视频导航 | 天天曰夜夜爽 | 日韩三级.com | 日韩欧美高清 | 久久国产精品第一页 | 国产乱码精品一区二区蜜臀 | 又黄又刺激的视频 | 久久影院亚洲 | 欧美精品在线一区二区 | 69视频永久免费观看 | 人人爱人人做人人爽 | 天堂在线一区二区 | 日韩黄色中文字幕 | 中文字幕一区二区三区乱码在线 | 婷婷丁香激情五月 | 日韩电影在线看 | 在线观看黄色小视频 | 国产一区二区三区网站 | 国产五月天婷婷 | 日日干天天插 | 午夜久久福利视频 | 天天综合在线观看 | 国产精品免费久久久久久久久久中文 | 久久综合久久八八 | 91av在线视频免费观看 | 免费视频黄色 | 69国产成人综合久久精品欧美 | 五月婷在线观看 | www.成人精品 | 日本一区二区高清不卡 | 亚洲黄色影院 | 奇米影视在线99精品 | 国产精品视频免费在线观看 | 69av视频在线 | 欧美国产日韩一区二区三区 | 亚洲视频在线看 | 中文字幕中文字幕在线中文字幕三区 | 99r在线 | 亚洲欧美日韩不卡 | 国产亚洲aⅴaaaaaa毛片 | 亚洲欧美一区二区三区孕妇写真 | 久久不射电影网 | 色噜噜日韩精品欧美一区二区 | 久久国产精品久久w女人spa | 色播亚洲婷婷 | 天天综合色 | 欧美日韩在线视频免费 | 青草视频在线免费 | 中文字幕 国产视频 | 91在线视频在线观看 | 香蕉视频色 | 国产精品久久久久免费 | 在线天堂中文www视软件 | 丁香六月av| 久久草在线视频国产 | 国产精品美女久久久久久网站 | 久久久久 免费视频 | 激情开心 | 人人草在线视频 | 免费看黄网站在线 | 四虎成人av | 久久深夜福利免费观看 | 欧美日韩免费在线观看视频 | 国产视频首页 | 色婷婷成人网 | 天天干人人干 | a色视频| 日日干日日 | 人人爽人人爽人人爽学生一级 | 久久国产精品免费一区二区三区 | 久草视频在线资源 | 日韩在线观看不卡 | 黄色亚洲大片免费在线观看 | 天天干.com | 天天色宗合 | av大全在线看 | 精品视频123区在线观看 | 日本aaaa级毛片在线看 | 欧美韩日精品 | 日韩免费在线观看视频 | 超碰在线公开免费 | 国产视频一区在线免费观看 | 国产一线二线三线性视频 | 日韩三区在线 | 在线不卡a | av经典在线 | 西西444www大胆高清图片 | 涩涩网站免费 | 精品欧美一区二区精品久久 | 精品久久久国产 | 亚洲视频分类 | 国产精品乱码久久久 | 91久久久久久久一区二区 | 国产精品久久久久影视 | 日韩大片在线免费观看 | 精品在线播放 | 五月婷影院 | 国产在线观看你懂得 | 免费在线观看av片 | 午夜黄色一级片 | 日韩精品一区二区不卡 | 色婷婷88av视频一二三区 | 一级成人免费 | 91在线视频免费91 | 国产亚洲在线观看 | 人人澡人人舔 | 亚洲精品午夜国产va久久成人 | 91视频观看免费 | 在线一区观看 | 中文字幕在线电影 | 国产亚洲成人网 | 99久久久国产精品美女 | 免费看日韩片 | 欧美最猛性xxxxx亚洲精品 | 国产成人在线播放 | 亚洲欧美日韩一区二区三区在线观看 | 在线观看激情av | 怡红院久久 | 视频在线91 | 精品极品在线 | 中文字幕av有码 | 成人午夜在线电影 | av成人在线网站 | 国产中文字幕网 | 4hu视频 | 国产群p | 91精品一区在线观看 | 日韩av成人在线 | 国产区网址 | 国产午夜三级一区二区三桃花影视 | 探花视频网站 | 久久久亚洲精华液 | 中文字幕 国产 一区 | 成人在线视频网 | 青草视频在线 | 香蕉视频在线观看免费 | 中文字幕一二 | 在线免费观看黄色av | 一区二区三区 中文字幕 | 黄色三级网站在线观看 | 中文久草| 免费看的av片 | 国产欧美精品一区二区三区四区 | 日韩精品一区二区三区水蜜桃 | 女人18精品一区二区三区 | 日本少妇高清做爰视频 | 色人久久| 伊人一级 | 91在线看视频免费 | 91免费试看 | 国产成人av在线 | 日韩免费观看一区二区 | 狠狠干2018 | 黄色片网站免费 | 韩国av免费在线观看 | 999久久国产 | 免费a视频在线观看 | 亚洲精区二区三区四区麻豆 | 国产亚洲视频在线观看 | 久久不射电影院 | 欧美精品一区在线发布 | 欧美日韩在线免费观看视频 | 欧美小视频在线观看 | 日韩久久久久久 | 九九视频在线播放 | 国产亚洲人成网站在线观看 | 国产成人99av超碰超爽 | 91网免费观看 | 国产精品美女久久久久久2018 | 一级免费黄视频 | 九九热久久久 | 五月天色中色 | 日本免费一二三区 | 国产精品九色 | 日韩理论电影在线观看 | 成人av资源网站 | 日韩网站免费观看 | 精品免费一区二区三区 | 在线性视频日韩欧美 | 色综合天 | 色国产精品一区在线观看 | 免费三级大片 | 97视频在线观看视频免费视频 | 色婷婷播放 | 黄色资源在线观看 | 欧美视频日韩视频 | 精品一区二区免费 | 亚洲综合色视频在线观看 | 久久综合九色综合网站 | 国产精品女人网站 | 久久国产精品第一页 | 97超碰在线播放 | 永久免费观看视频 | 在线观看免费黄视频 | 最新日韩中文字幕 | av日韩中文| 九九久久久久久久久激情 | 麻豆传媒视频在线播放 | 日韩高清一二三区 | 中文字幕在线观看免费观看 | 欧美精品久久久久久久 | 精品国产一区二区三区四区vr | 国产一区二区三区久久久 | 国产一二区视频 | 免费av片在线 | 黄色福利网 | 日韩一区二区三免费高清在线观看 | 黄色大片日本免费大片 | 精品黄色在线观看 | 国产免费又黄又爽 | 日韩精品中文字幕在线 | 一级特黄aaa大片在线观看 | 欧美日韩国产一区二区三区 | 久久精品国产亚洲精品2020 | 激情综合五月 | 日韩精品不卡在线 | 亚洲一二视频 | 欧美日韩视频在线观看免费 | 国产精品一区二区62 | 波多野结衣在线中文字幕 | 国产毛片久久久 | 久久99精品国产一区二区三区 | 亚洲一区二区三区精品在线观看 | 欧美一区二区三区在线视频观看 | 国产美女精品视频免费观看 | 麻豆传媒视频观看 | 热精品 | 国产一二三四在线观看视频 | 国产成本人视频在线观看 | 99色网站 | 一区二区三区国 | 中文字幕av在线不卡 | 综合黄色网| 成人国产综合 | 在线观看中文字幕第一页 | 五月婷婷丁香在线观看 | 日韩在线观看的 | 视频在线观看99 | 国产精品免费一区二区三区 | 中文在线字幕免费观看 | 在线免费色 | 国产精品黄网站在线观看 | 欧美成人69av| 国产色在线 | 日本精品视频免费 | 中文字幕之中文字幕 | 亚洲一区二区三区四区精品 | 一级片免费观看视频 | 国产精品9999久久久久仙踪林 | 国产精品日韩久久久久 | 亚洲激情p | 中文字幕资源在线 | 国产成人91 | 97电影手机版| 在线观看黄网 | 久久99网 | 成人a级免费视频 | 久久久国产精品网站 | 色www永久免费| 国产91精品一区二区 | 国产精品毛片 | 日韩剧| 欧美精品在线观看免费 | 亚洲国产午夜 | 日韩色综合 | 亚洲1级片 | av高清一区 | 国产亚洲永久域名 | 国产精品黄网站在线观看 | 久久97精品| 天天天干天天天操 | 欧洲av不卡| 色婷婷狠狠干 | 91片黄在线观看 | 免费观看黄色12片一级视频 | 久久五月婷婷丁香社区 | 天天干天天干天天 | 国产在线精品观看 | 国产高清视频在线播放 | 亚洲区二区 | 国色天香在线观看 | 色多多视频在线观看 | 91视视频在线直接观看在线看网页在线看 | 亚洲黄色免费在线 | 成年人在线播放视频 | 欧美日韩一区二区免费在线观看 | 久久精品久久久久 | 亚洲欧美视频网站 | 成人黄色av网站 | 久草精品视频在线播放 | 免费精品在线 | 亚洲精品视频中文字幕 | 国产理论免费 | 久草色在线观看 | 亚洲一二三久久 | 91麻豆高清视频 | 久久国产精彩视频 | 国产a网站 | 99色免费视频 | 国产精品99久久久久 | 久久国产免费视频 | 国产亚洲人成网站在线观看 | 91av99| 久久久免费观看视频 | 日本中文字幕一二区观 | 91丨精品丨蝌蚪丨白丝jk | 91 | 日韩av一区二区在线 | 亚洲成人一二三 | 在线日韩精品视频 | 日韩乱码中文字幕 | 日韩视频免费在线 | 久久99精品久久久久久秒播蜜臀 | 国产精品亚州 | 亚洲精品久久激情国产片 | 午夜精品久久久久久中宇69 | 久久精品视频免费播放 | 免费精品视频在线观看 | 国产精品国产三级国产不产一地 | 中文视频在线播放 | 在线电影中文字幕 | 国产视频久久久久 | 国产一级黄色av | 综合伊人久久 | 欧美日韩精品在线观看 | 精品久久1| 波多野结衣一区 | 天天操天天操天天操天天 | 国产精品婷婷午夜在线观看 | 天天干天天干天天射 | 午夜久久久久久久久久影院 | 欧美综合在线观看 | 在线观看视频你懂的 | 超碰公开在线观看 | 国产精品国内免费一区二区三区 | 亚洲精品中文在线观看 | 国产99久久精品 | 免费午夜网站 | 亚洲不卡av一区二区三区 | 国产精品黄色av | 99久久精品国产一区二区三区 | 特级毛片在线观看 | 最新国产视频 | 亚洲一级黄色 | 久久精品国产一区二区三区 | 91精品在线免费视频 | 久久香蕉一区 | 中文字幕av在线 | 五月天久久久久 | 91精品视频在线观看免费 | 精品视频在线免费 | 在线国产不卡 | 97国产精品免费 | 欧美精品一区二区免费 | 欧美亚洲另类在线视频 | 伊人电影在线观看 | 中文字幕观看av | 成人97视频 | 黄色成人免费电影 | 国产精品黄色影片导航在线观看 | 久影院| 久久69精品久久久久久久电影好 | 国产午夜精品久久久久久久久久 | a在线观看视频 | 人人插人人插 | 日韩欧美一区二区在线播放 | 天天躁日日躁狠狠躁av中文 | 伊人天天色 | www·22com天天操 | 成人av手机在线 | 国产精品正在播放 | 日韩电影在线观看一区二区三区 | 黄色视屏免费在线观看 | 久久伊人五月天 | 麻豆极品 | 亚洲综合色激情五月 | 久久久电影网站 | 最新三级在线 | 亚洲精品乱码久久 | 中文一区在线 | 久草热视频| 免费亚洲视频 | 欧美亚洲国产一卡 | 亚洲一区二区天堂 | 婷婷丁香激情五月 | 中文字幕人成不卡一区 | 久草精品视频在线看网站免费 | 808电影免费观看三年 | 亚洲黄色影院 | 中文字幕精品一区 | 亚洲影视九九影院在线观看 | 黄色免费av | av片一区| 婷婷综合成人 | 在线视频在线观看 | 成人av一区二区兰花在线播放 | 91视频观看免费 | 天天色成人 | 日韩av不卡在线观看 | 97电影院在线观看 | 亚洲精品在线二区 | 黄色精品国产 | 成人av在线网址 | 日韩二区三区在线 | 久久看视频 |