Android 7.1 GUI系统-BufferQueue 的管理(三)

BufferQueue 的管理


1),对buffer的管理符合“生产者-消费者”模型,这里有以下几个相关的类:

BufferQueueProducer.cppIGraphicBufferProducer对应服务端的实现,负责缓冲区的申请。

BufferQueueConsumer.cppIGraphicBufferProducer对应的服务端的实现,是buffer消费者。

BufferQueueCore.cpp,可以看作是Buffer的中枢,一方面BufferqueueProducerbuffer的申请也是通过BufferQueueCore来实现的,具体是BufferQueueCore中的GraphicBufferAlloc。另一方面,当有可以消费的buffer时,是通过BufferQueueCore通知到BufferQueueConsumer,具体是通过ConsumerListener来实现的。

GraphicBufferAlloc.cppIGraphicBufferAlloc对应的服务端的实现,负责GraphicBuffer的申请。

既然说BufferQueueCore中的GraphicBufferAlloc负责GraphicBuffer的申请,先来看下谁生成的GraphicBufferAlloc实例。

BufferQueueCore::BufferQueueCore(const sp& allocator) :
    	mAllocator(allocator)@BufferQueueCore.cpp{
	if (allocator == NULL) {
//这里返回的 ISurfaceComposer类型的对象 composer,实际就是SurfaceFinger,最终 IGraphicBufferAlloc的实例生成还是由SurfaceFinger来完成的。 ComposerService类是SurfaceComposerClient.cpp的内部类。
		sp composer(ComposerService::getComposerService());
		mAllocator = composer->createGraphicBufferAlloc();
	}
}

可以看到这里创建的是一个GraphicBufferAlloc对象。

sp SurfaceFlinger::createGraphicBufferAlloc()@SurfaceFlinger.cpp
{
    sp gba(new GraphicBufferAlloc());
    return gba;
}

BufferQueueCore的初始化列表中虽然给IGraphicBufferAlloc做了赋值:mAllocator(allocator),但实际这个值是null的,因为它是由BufferQueue.cpp的默认参数传过来的,而这默认参数初始值是null的,所以需要在构造函数中重新生成IGraphicBufferAlloc实例。

BufferQueueCore中有一个重要的成员数组BufferQueueDefs::SlotsTypemSlots;

前面在说的surface中也有一个同名的成员数组BufferSlotmSlots[NUM_BUFFER_SLOTS];都是BufferSlot类型,只是定义有差别。


这是Surface中的BufferSlot的定义,是一个结构体。

Surface.h
    struct BufferSlot {
        sp buffer;
        Region dirtyRegion;
    };
BufferSlot mSlots[NUM_BUFFER_SLOTS];

下面是BufferQueueCore中的Bufferslot中的定义:

BufferSlot.h
struct BufferSlot {
// mGraphicBuffer指向一个slot所分配的buffer,如果buffer还没分配,值为null。
	sp mGraphicBuffer;
//标示一个buffer slot的当前状态。
	BufferState mBufferState;
}

一个buffer可以有5中状态。

struct BufferState {


mShared

mDequeueCount

mQueueCount

mAcquireCount

FREE

false

0

0

0

DEQUEUED

False

1

0

0

QUEUED

False

0

1

0

ACQUIRED

False

0

0

1

SHARED

True

Any

Any

any

FREE表明缓冲区当前可用,可以被生产者dequeued,这个slotsownerBufferQueue,当dequeueBuffer后会转变为DEQUEUED

DEQUEUED表明这个buffer已经被生产者dequeued,但是还没被queued或者canceled,生产者可以修改这个buffer的内容,现在bufferownerproducer(应用程序),可以通过调用queueBufferattachBuffer转成QUEUED,也可以通过调用cancelBufferdetachBuffer转成FREE

QUEUED表明这个buffer已经被Producer填充了数据,可以被consumer使用,现在bufferownerBufferQueue,可以通过调用acquireBuffer转成ACQUIRED

ACQUIRED表明这个buffer已经被consumeracquired了,现在的ownerconsumer,通过releasebuffer可以转成FREE

SHARED表明这个buffer被用于sharedbuffer mode。在同一时间,它可以是除FREE外的其他的状态的集合。

}


通过上面buffer的各个状态的介绍,对buffer的管理主要有BufferQueueProducerConsumer,三个对象。BufferQueue的实现者就是BufferQueueCore,它是一个中心,Producer想要获取一个buffer,是通过BufferQueueCore完成的,同样的当有一个buffer可以被consumer消费时,也是通过BufferQueueCore来通知到到consumer的。

Producer通常是指应用程序,应用程序不断刷新UI,把产生的显示数据写到buffer中,当Producer需要一块buffer时,先向bufferQueue发起dequeue申请,然后才能对制定的缓冲区进行填充操作,当填充完成后,通过调用queueBuffer把这个buffer入队到Bufferqueue中,由BufferQueuecore通知consumer有可以别消费的buffer了。

具体怎么通知到consumer的,还要介绍一个类ConsumerListener,介绍了这个类的几个主要接口,就知道是怎么通知到consumer的了。

class ConsumerListener : public virtual RefBase {
//当一帧对消费者来说可用时就会调用这个函数,
	virtual void onFrameAvailable(const BufferItem& item) = 0; /* Asynchronous */
//这个函数会去通知consumer,BufferQueue已经释放了一个或多个GraphicBuffers的引用,consumer应该调用getReleasedBuffers去检索这个buffer列表。
	virtual void onBuffersReleased() = 0;
}

2),BufferQueue缓冲区的分配。

分析缓冲区的分配,就是分析BufferSlot数组中的指针mGraphicBuffer的空间分配。这个指针一开始是null,下面看什么情况下为mSlots数组中的一个slot分配实际空间。

整个需求的起点是ProducerbufferdequeueBuffer,所以从分析这个函数开始:

BufferQueueProducer.cpp
status_t BufferQueueProducer::dequeueBuffer(int *outSlot,sp *outFence, 
	uint32_t width,uint32_t height, QueueBufferOutput *output) @BufferQueueProducer.cpp{
//step1,这个while循环,查找一个可用的slot,具体是通过 waitForFreeSlotThenRelock函数实现,也就是found的有效值。
	int found = BufferItem::INVALID_BUFFER_SLOT;
	while (found == BufferItem::INVALID_BUFFER_SLOT) {
		status_t status = waitForFreeSlotThenRelock(FreeSlotCaller::Dequeue,&found);
//如果不允许分配新的buffers, waitForFreeSlotThenRelock还必须返回一个包含buffer的slot,如果这个buffer需要充分配才能满足要求,就要先把它free了,然后尝试再获取一个。
		const sp& buffer(mSlots[found].mGraphicBuffer);
		if (!mCore->mAllowAllocation) {
			if (buffer->needsReallocation(width, height, format, usage)) {
				mCore->mFreeSlots.insert(found);
				mCore->clearBufferSlotLocked(found);
				found = BufferItem::INVALID_BUFFER_SLOT;
				continue;
			}
		}
	}
//step2,成功找到slot序号,接下来对这个bufferslot对象做初始化。还要判断要不要为这个slot重新分配空间。
	*outSlot = found;
	mSlots[found].mBufferState.dequeue();
	if ((buffer == NULL) ||
		buffer->needsReallocation(width, height, format, usage)){
		mSlots[found].mAcquireCalled = false;
		mSlots[found].mGraphicBuffer = NULL;
		mSlots[found].mFence = Fence::NO_FENCE;
		mCore->mIsAllocating = true;
		returnFlags |= BUFFER_NEEDS_REALLOCATION;
	}
//step3,上一步的判断结果需要重新分配空间,调用 createGraphicBuffer分配空间。
	if (returnFlags & BUFFER_NEEDS_REALLOCATION) {
		sp graphicBuffer(mCore->mAllocator->createGraphicBuffer(
			width, height, format, usage,{mConsumerName.string(), 
			mConsumerName.size()}	, &error));
//分配的空间,赋值给了slot,这里的mSlots实质是BufferQueueCore中的变量。
		mSlots[*outSlot].mGraphicBuffer = graphicBuffer;
	}
}

对其中关键步骤做详细分析:

step1,怎么找到一个可用的slotfound指的是这个BufferslotmSlots数组中的序号。

status_t BufferQueueProducer::waitForFreeSlotThenRelock(
	FreeSlotCaller caller,int* found) const {
	bool tryAgain = true;
	while (tryAgain) {
// mActiveBuffers这个集合中包含所有非free的buffer。然后分别统计dequeued的个数,acquired的个数。
		for (int s : mCore->mActiveBuffers) {
			if (mSlots[s].mBufferState.isDequeued()) {
				++dequeuedCount;
			}
			if (mSlots[s].mBufferState.isAcquired()) {
				++acquiredCount;
			}
		}
//不允许Producersdequeue的数量的超过 mMaxDequeuedBufferCount,这个判断只在buffer已经被queued的情况下。 mMaxDequeuedBufferCount是Producer一次可以dequeue的buffers的数量,默认是1,可以通过setMaxDequeuedBufferCount来修改。
		if (mCore->mBufferHasBeenQueued &&
			dequeuedCount >= mCore->mMaxDequeuedBufferCount) {
			return INVALID_OPERATION;
		}
//如果在shared buffer mode,并且有一个shared buffer存在,就返回这个slot。
		if (mCore->mSharedBufferMode && mCore->mSharedBufferSlot !=
			BufferQueueCore::INVALID_BUFFER_SLOT) {
			*found = mCore->mSharedBufferSlot;
		}else {
//slot的获取,主要来自两个集合,mFreeSlots和mFreeBuffers,
//mFreeSlots包含的是所有状态是free,并且没有attached的slots,
//mFreeBuffers包含所有状态是free,但是已经attached的slots。
			if (caller == FreeSlotCaller::Dequeue) {
//如果调用来自与dequeue,优先选择mFreeBuffers中的。
				int slot = getFreeBufferLocked();
				if (slot != BufferQueueCore::INVALID_BUFFER_SLOT) {
					*found = slot;
				}else{
					*found = getFreeSlotLocked();
				}
			}else{
//如果调用来自于attache,优先选择mFreeSlots中的。
				int slot = getFreeSlotLocked();
				if (slot != BufferQueueCore::INVALID_BUFFER_SLOT) {
					*found = slot;
				}else{
					*found = getFreeBufferLocked();
				}
			}
		}
//如果没有找到符合要求的buffer,或者queue的buffers还有太多没有完成,就要等待这个buffer被acquired或者released,或者修改最大的buffer数量。这里使用条件变量mDequeueCondition进入wait,当有buffer被释放时,这个条件就会满足,然后继续查找。
		if (tryAgain) {
			mCore->mDequeueCondition.wait(mCore->mMutex);
		}
	}//while(tryAgain)
	return NO_ERROR;
}


Step2,找到可用的slot,还要判断是否要重新分配空间,即needsReallocation,如果buffer的属性跟要求的属性不符(主要指宽,高,格式,usage),就要重新分配,最后的返回值会加上returnFlags|= BUFFER_NEEDS_REALLOCATION;标记。

Step3,调用IGraphicBufferAlloc的方法createGraphicBuffer方法分配GraphicBufferIGraphicBufferAlloc对应的服务端的实现是GraphicBufferAlloc.cpp,所以这个方法的实现就到了GraphicBufferAlloc.cpp中,这里new了一个GraphicBuffer,这个GraphicBuffer会赋值给BufferQueueCore中的变量mSlots中的slotmGraphicBuffer

sp GraphicBufferAlloc::createGraphicBuffer(uint32_t width,
        uint32_t height, PixelFormat format, uint32_t usage,
        std::string requestorName, status_t* error) @GraphicBufferAlloc.cpp{
    sp graphicBuffer(new GraphicBuffer(
            width, height, format, usage, std::move(requestorName)));
    status_t err = graphicBuffer->initCheck();
    *error = err;
    if (err != 0 || graphicBuffer->handle == 0) {
        if (err == NO_MEMORY) {
            GraphicBuffer::dumpAllocationsToSystemLog();
        }
        return 0;
    }
    return graphicBuffer;
}

BufferQueueProducer.cpp中的dequeueBuffer的源头是在surface中调用的dequeueBuffer,所以BufferQueueProducer.cpp这边执行完会返回到Surface.cpp中,

int Surface::dequeueBuffer(android_native_buffer_t** buffer, int* fenceFd) @Surface.cpp{
//BufferQueueProducer.cpp的dequeueBuffer是从这里开始的。
	status_t result = mGraphicBufferProducer->dequeueBuffer(&buf, &fence,
            reqWidth, reqHeight, reqFormat, reqUsage);
//如果返回值带有BUFFER_NEEDS_REALLOCATION,还要调用 requestBuffer。
	if ((result & IGraphicBufferProducer::BUFFER_NEEDS_REALLOCATION) || gbuf == 0) {
		result = mGraphicBufferProducer->requestBuffer(buf, &gbuf);
	}
}

调用requestBuffer是获取最新的buffer地址,BufferQueueProducer中分配的buffer是在服务端进程空间的,而surface需要的buffer是要在客户端进程,也就是应用进程空间使用的,这两块buffer应该指向同一块物理内存,怎么实现让两块buffer指向同一块物理内存的,这就是requestbuffer的作用。


//这个方法是BpGraphicBufferProducer中的方法:

virtual status_t requestBuffer(int bufferIdx, sp* buf) 	@IGraphicBufferProducer.cpp{
	Parcel data, reply;
//只传了在mSlots中的序号 bufferIdx,第二个参数* buf实际是一个出参。
	data.writeInt32(bufferIdx);
//开始跨进程操作。
	status_t result =remote()->transact(REQUEST_BUFFER, data, &reply);
//读取服务端返回的信息,在执行跨进程操作时,client端会被挂起,等服务端处理完成返回结果时,在唤醒client继续执行。
	bool nonNull = reply.readInt32();
	if (nonNull) {
//这里生成一个GraphicBuffer实例,实际是一个空壳,要通过服务端的返回值来填充。
		*buf = new GraphicBuffer();
//使用服务端的返回的结果填充**buf ,这个**buf具体指的什么?从surface.cpp传入的参数说起:
//sp& gbuf(mSlots[buf].buffer);这个gbuf 是一个指向 mSlots[buf].buffer的引用,//传入requestBuffer中的实参是&gbuf,也就是说*buf指针指向的是 gbuf 的地址,所以**buf指向的就是 gbuf代表的内容,也就是 mSlots[buf].buffer。
		result = reply.read(**buf);
	}
	result = reply.readInt32();
}

相应client请求的代码如下:

status_t BnGraphicBufferProducer::onTransact(
	uint32_t code, const Parcel& data, Parcel* reply, uint32_t flags){
	switch(code) {
//匹配请求的业务码 REQUEST_BUFFER。
		case REQUEST_BUFFER: {
//读取要处理的slot序号。
			int bufferIdx   = data.readInt32();
			sp buffer;
			int result = requestBuffer(bufferIdx, &buffer);
//写入本地端是否成功获取到了对应的GraphicBuffer,这个GraphicBuffer就是dequeuebuffer中申请//的,也即是BufferQueueCore中的mSlots[slot].mGraphicBuffer。
			reply->writeInt32(buffer != 0);
			if (buffer != 0) {
//把 mGraphicBuffer写入reply,等待客户端读取。
				reply->write(*buffer);
			}
			reply->writeInt32(result);
			return NO_ERROR;
		}
	}
}

上面的实现中,清晰的描述了一个怎样跨进程传递一个对象。

能够通过Binder跨进程传递的对象,在java层的通常要继承Parcelable,如Bundle.java,需要实现的接口是writeToParcelreadFromParcel

native层通常要继承Flattenable,如这里的GraphicBuffer.cpp,需要实现的接口是flattenunflatten

只要看下GraphicBufferflattenunflatten的函数实现,就可以清楚客户端、服务端的两块buffer是不是指向的同一块物理空间了。

status_t GraphicBuffer::flatten(void*& buffer, size_t& size, int*& fds, size_t& count) const 	@GraphicBuffer.cpp{
	int32_t* buf = static_cast(buffer);
//handle的类型是buffer_handle_t* handle,它是在生成GraphicBuffer实例时,//由GraphicBufferAllocator在申请真实的物理内存时赋值的,代表了代表了一块共享缓冲区的句柄。
	if (handle) {
//写入缓冲区相关的数据,client端也会按相同的格式读取。
		buf[9] = handle->numFds;
		buf[10] = handle->numInts;
		memcpy(fds, handle->data,
			static_cast(handle->numFds) * sizeof(int));
		memcpy(&buf[11], handle->data + handle->numFds,
		static_cast(handle->numInts) * sizeof(int));
	}
}

status_t GraphicBuffer::unflatten(void const*& buffer, size_t& size, 
	int const*& fds, size_t& count) @GraphicBuffer.cpp{
	const size_t numFds  = static_cast(buf[9]);
	const size_t numInts = static_cast(buf[10]);
	if (numFds || numInts) {
//通过生成一个本地层 native_handle实例,把相关的数据复制过来。这块物理空间跟BufferQueueCore中指向的物理空间是一致的。这就实现了client和服务端之间缓冲区的共享。
		native_handle* h = native_handle_create(
			static_cast(numFds), static_cast(numInts));
		memcpy(h->data, fds, numFds * sizeof(int));
		memcpy(h->data + numFds, &buf[11], numInts * sizeof(int));
	}
}

surface执行dequeuebuffer的过程中,如果需要重新分配空间,就要调用requestbuffer就是来更新clientmSlots中相应的缓冲区信息的。这样就保证了SurfacebufferQueuecore之前在数据缓冲区上的一致。

3)应用程序跟BufferQueue的关系。

应用程序不会直接使用BufferQueue,接下来分析应用程序跟bufferQueue之间怎么配合完成图形绘制的。

分析这个过程需要一个切入点,可以从一个常见的应用程序的启动入手,经过ViewrootImpl.javasetView,到WindowManagerService.javarelayoutWindow,然后转到createSurface的分析。这个过程要从java层进过jni到本地层相对繁琐,不利于直观跟BufferQueue的关系。所以借助系统的一段本地层的测试代码来分析,为什么要用这段代码?我们知道surfaceflinger是借助于opengles来完成绘图的,很多应用程序也是借助于opengles来完成UI显示的,比如图库的图片展示,相机的预览界面等,当然也有应用程序是借助于skia这个图形库。下面这段代码不仅展示了应用程序跟BufferQueue的通信过程,还展示了openglesegl的使用流程。

frameworks/native/libs/gui/tests/GLTest.cpp
void GLTest::SetUp() {
//获取默认的物理屏幕。
	mEglDisplay = eglGetDisplay(EGL_DEFAULT_DISPLAY);
//初始化。
	eglInitialize(mEglDisplay, &majorVersion, &minorVersion);
//选取最佳的config。
	eglChooseConfig(mEglDisplay, getConfigAttribs(), &mGlConfig, 1, &numConfigs);
//生成一个SurfaceComposerClient实例,它是应用程序与surfaceflinger之间的通信桥梁,不过SurfaceComposerClient只是一个封装,实质是通过内部的ISurfaceComposerClient来执行的,ISurfaceComposerClient对应的服务端实现是Client。
	mComposerClient = new SurfaceComposerClient;
	mComposerClient→initCheck();
//生成一个surface,surface是由surfaceControl来管理的,所以这里返回的是SurfaceControl。
	mSurfaceControl = mComposerClient->createSurface(String8("Test Surface"), 
		getSurfaceWidth(), getSurfaceHeight(),PIXEL_FORMAT_RGB_888, 0);
//设置layer的层级,层级越大越靠近用户。
	mSurfaceControl->setLayer(0x7FFFFFFF);
	mSurfaceControl→show();
//从surfacecontrol中取出surface,得到了egl需要的本地窗口。
	sp window = mSurfaceControl→getSurface();
//通过本地窗口window创建surface。
	mEglSurface = createWindowSurface(mEglDisplay, mGlConfig, window);
//创建、设置当前context环境。
	mEglContext = eglCreateContext(mEglDisplay, mGlConfig, EGL_NO_CONTEXT,
		getContextAttribs());
	eglMakeCurrent(mEglDisplay, mEglSurface, mEglSurface,mEglContext));
}

SurfaceComposerClient是一个封装,具体功能实现是通过其内部的ISurfaceComposerClient来实现的,ISurfaceComposerClient对应服务端的实现是Client.cpp,所以上面的createSurface实现就在Client中。这里涉及ipc,具体细节就不列出代码了。

SurfaceComposerClient.cpp

sp SurfaceComposerClient::createSurface(const String8& name,uint32_t w,
	uint32_t h,PixelFormat format,uint32_t flags)@SurfaceComposerClient.cpp{
	sp handle;
	sp gbp;
//调用服务端的实现,实质是创建了一个layer,通过这个layer获取 IGraphicBufferProducer。
	status_t err = mClient->createSurface(name, w, h, format, flags,&handle, &gbp);
//生成一个 SurfaceControl,由他管理surface。
	sur = new SurfaceControl(this, handle, gbp);
	return sur;
}

接着看服务端的实现,createSurface这个方法只能在GL线程中调用,这样它才能范文GL的上下文环境

Client.cpp

status_t Client::createSurface(const String8& name,uint32_t w, uint32_t h, 
	PixelFormat format, uint32_t flags,sp* handle,sp* gbp){
	class MessageCreateLayer : public MessageBase {
	        SurfaceFlinger* flinger;
//这个client表明,这个消息来自于那个client,
       		Client* client;
//与这个layer对应的handle和gbp。
        	sp* handle;
       		sp* gbp;
        	status_t result;
        	const String8& name;
        	uint32_t w, h;
        	PixelFormat format;
       		uint32_t flags;

		virtual bool handler() {
//这里是由surfaceFlinger来执行操作。
			result = flinger->createLayer(name, client, w, h, format, flags,handle, gbp);
			return true;
		}
	}
//把请求通过一个同步函数把msg放入队列,等到有结果再返回。
	sp msg = new MessageCreateLayer(mFlinger.get(),
		name, this, w, h, format, flags, handle, gbp);
	mFlinger->postMessageSync(msg);
	return static_cast( msg.get() )->getResult();
}

SurfaceFlinger.cpp
status_t SurfaceFlinger::createLayer(const String8& name,const sp& client,
	uint32_t w, uint32_t h, PixelFormat format, uint32_t flags,
	sp* handle, sp* gbp)@SurfaceFlinger.cpp{
	sp layer;
	switch (flags & ISurfaceComposerClient::eFXSurfaceMask) {
//根据flag,创建不同的layer,这里是创建普通的layer,代表了一个画面。
		case IsurfaceComposerClient::eFXSurfaceNormal:
			result = createNormalLayer(client,name, w, h, flags, format,
				handle, gbp, &layer);
		break;
	}	
//将layer,gbp添加到全局的管理中。
	result = addClientLayer(client, *handle, *gbp, layer);
}

SurfaceFlinger.cpp
status_t SurfaceFlinger::createNormalLayer(const sp& client,
	const String8& name, uint32_t w, uint32_t h, uint32_t flags, PixelFormat& format,
	sp* handle, sp* gbp, sp* outLayer)@SurfaceFlinger.cpp{
//创建一个layer,同时给他设置缓冲区。
	*outLayer = new Layer(this, client, name, w, h, flags);
	status_t err = (*outLayer)->setBuffers(w, h, format, flags);
//通过layer获取IGraphicBufferProducer。
	*handle = (*outLayer)→getHandle();
	*gbp = (*outLayer)->getProducer();
}

//addClientLayer函数的实现

status_t SurfaceFlinger::addClientLayer(const sp& client,const sp& handle,
	const sp& gbc,const sp& lbc)@SurfaceFlinger.cpp{
//把layer添加到当前state的列表中。把gbc添加到全局变量中。
	mCurrentState.layersSortedByZ.add(lbc);
	mGraphicBufferProducerList.add(IInterface::asBinder(gbc));
//让layer与client产生关联,也就是添加到client的mLayers成员变量中。
	client->attachLayer(handle, lbc);
}

//再回头看下*handle= (*outLayer)→getHandle();其中的handle是什么类型,又起了什么作用呢?

Layer.cpp

//创建了一个handle对象,并传入的surfaceflinger

sp Layer::getHandle() @Layer.cpp{
	return new Handle(mFlinger, this);
}

//这个handle只是传给client端的一个BBbinder,当客户端释放了这个引用时,会调用其析构函数,那么它的析构函数做了什么事情呢Handle本身没写析构函数,所以要做的事情就在其父类LayerCleaner的析构函数中

class Layer::Handle : public BBinder, public LayerCleaner {
	Handle(const sp& flinger, const sp& layer)
		: LayerCleaner(flinger, layer), owner(layer) {}
};

//LayerCleaner的析构函数,调用了surfaceflingeronLayerDestroyed,把layersurfaceflinger的全局变量mCurrentState.layersSortedByZ中移除


Layer::LayerCleaner::~LayerCleaner() {
	mFlinger->onLayerDestroyed(mLayer);
}

//Layer中很多变量的实例化,都是在onFirstRef中完成的,当Layer实例第一次被引用时会执行onFirstRef函数。

Layer.cpp
void Layer::onFirstRef() {
	sp producer;
	sp consumer;
//通过BufferQueue创建BufferQueueProducer,BufferQueueConsumer,这样标识了 IGraphicBufferProducer对应的服务端是BufferQueueProducer, IGraphicBufferConsumer对应的服务端是 BufferQueueConsumer,同时BufferQueueProducer、 BufferQueueConsumer的构造函数中都有一个BufferQueueCore实例,生成者,消费者的对缓冲区的操作都会经过BufferQueueCore。
	BufferQueue::createBufferQueue(&producer, &consumer);
//这里返回给client端的并不是 IGraphicBufferProducer,而是对它的封装类 MonitoredProducer, //MonitoredProducer并没有增加新的功能,只是在 IGraphicBufferProducer被销毁时,通知   
// Surfaceflinger做些操作。
	mProducer = new MonitoredProducer(producer, mFlinger);
	mSurfaceFlingerConsumer = new SurfaceFlingerConsumer(consumer, mTextureName,this);
}

//我们只关注MonitoredProducer的构造函数,看在IGraphicBufferProducer被销毁时,它要做什么。

MonitoredProducer.cpp

MonitoredProducer::~MonitoredProducer() @MonitoredProducer.cpp{
//这里使用了跟Client.cpp中调用surfaceflinger创建layer类似的方式,不同的是这里是异步的消息机制。
	class MessageCleanUpList : public MessageBase {
		virtual bool handler() {
//把IGraphicBufferProducer从surfaceflinger的列表中删除,这个操作是异步执行的,因为不知道会从哪里调用这个析构函数。还记得 mProducer是什么时候添加到这个列表中的吗?是在surfaceflinger的createLayer中,当创建一个layer之后,调用addClientLayer来添加的。
			mFlinger->mGraphicBufferProducerList.remove(mProducer);
		}
	}
	mFlinger->postMessageAsync(new MessageCleanUpList(mFlinger, asBinder(this)));
}

Layer中还有一个重要的变量SurfaceFlingerConsumer

SurfaceFlingerConsumerGLConsumer的简单的包装类,把IGraphicBufferConsumer实例传给了GLConsumer

SurfaceFlingerConsumer.cpp
class SurfaceFlingerConsumer : public GLConsumer {
	SurfaceFlingerConsumer(const sp& consumer,
		uint32_t tex, const Layer* layer)
		: GLConsumer(consumer, tex, GLConsumer::TEXTURE_EXTERNAL, false, false),
		mPrevReleaseFence(Fence::NO_FENCE), mLayer(layer){}
}

class GLConsumer :public ConsumerBase {}

class ConsumerBase :public virtual RefBase,protected ConsumerListener {}

GLConsumer有继承自ConsumerBase,间接继承了ConsumerListener

接下来逐层分析逐个继承关系。

GLConsumer会消费来自BufferQueue的缓冲区或者叫图形数据,并把他们转做textureopengl使用。一个典型的使用方式是使用所需要的选项设置GLConsumer,然后当有一帧需要时,调用updateTexImage,这个函数会获取最近queuedbuffer,然后绑定这个bufferGLtexture,如果这一帧是可用的,texture会被更新,如果不可用,前一帧的内容会被保留。

ConsumerBaseBufferQueue消费者的基类,它处理一些通用任务,比如到BufferQueue的连接,buffer池的管理,其中的成员变量spmConsumer;就是SurfaceFlingerConsumer传过来的,ConsumerBase是通过IGraphicBufferConsumer对象来对BufferQueue执行操作的。


简单做个总结:

1)应用程序通过BufferQueue执行绘图,surfaceflinger会把系统中所有应用程序最终的绘图结果进行混合,然后送到物理屏幕显示。应用程序跟surfaceflinger之间的通信桥梁是IsurfaceComposrClient

这个类被封装在SurfaceComposerClient中。

2)应用程序通过SurfaceComposerClient调用CreateSurface的过程中,除了得到IGraphicBufferProducer实例外,还创建了一个layer。把控制这个layerhandleIGraphicBufferProducer对象,连同SurfaceComposerClient自身都保存在应用端的Surfacecontrol中。

3Surface,应用程序通过SurfacecontrolgetSurface获取,其内部持有IGraphicBufferProducer对象,即BufferQueueProducer的实现接口。如果egl想通过Surface这个本地窗口完成某个功能,Surface实际上是利用IGraphicBufferProducer取得远程端的服务,完成egl的请求。

4)应用程序跟layer应该是一对多的关系,因为createSurface是可以被调用多次的,并且,一个应用程序申请的多个layer可以通过addClientLayer添加到一个全局变量mLayers中。应用程序申请的layer,一方面要记录都各个Client的内部变量mLayers中,另一方面还要告知SurfaceFlinger,这个操作也是在addClientLayer函数中完成的,即把layer添加到全局变量mCurrentState.layerSortByZSurfaceflinger会对这个列表中的所有layer排序,排序的结果直接影响了屏幕显示的画面。

5)每个layer对应一个BufferQueue,所以一个应用程序可能对应多个BufferQueuelayer并没有直接持有BufferQueue对象,而是通过其内部的IGraphicBufferProducermSurfaceFlingerConsumer来管理。



你可能感兴趣的:(Android 7.1 GUI系统-BufferQueue 的管理(三))