/* * Copyright (C) 2007 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifndef ANDROID_SF_SURFACE_H #define ANDROID_SF_SURFACE_H #include <stdint.h> #include <sys/types.h> #include <utils/KeyedVector.h> #include <utils/RefBase.h> #include <utils/threads.h> #include <ui/PixelFormat.h> #include <ui/Region.h> #include <ui/egl/android_natives.h> #include <surfaceflinger/ISurface.h> #include <surfaceflinger/ISurfaceComposerClient.h> #define ANDROID_VIEW_SURFACE_JNI_ID "mNativeSurface" namespace android { // --------------------------------------------------------------------------- class GraphicBuffer; class GraphicBufferMapper; class IOMX; class Rect; class Surface; class SurfaceComposerClient; class SharedClient; class SharedBufferClient; class SurfaceClient; // --------------------------------------------------------------------------- class SurfaceControl : public RefBase { public: static bool isValid(const sp<SurfaceControl>& surface) { return (surface != 0) && surface->isValid(); } bool isValid() { return mToken>=0 && mClient!=0; } static bool isSameSurface( const sp<SurfaceControl>& lhs, const sp<SurfaceControl>& rhs); uint32_t getFlags() const { return mFlags; } uint32_t getIdentity() const { return mIdentity; } // release surface data from java void clear(); status_t setLayer(int32_t layer); status_t setPosition(int32_t x, int32_t y); status_t setSize(uint32_t w, uint32_t h); status_t hide(); status_t show(int32_t layer = -1); status_t freeze(); status_t unfreeze(); status_t setFlags(uint32_t flags, uint32_t mask); status_t setTransparentRegionHint(const Region& transparent); status_t setAlpha(float alpha=1.0f); status_t setMatrix(float dsdx, float dtdx, float dsdy, float dtdy); status_t setFreezeTint(uint32_t tint); static status_t writeSurfaceToParcel( const sp<SurfaceControl>& control, Parcel* parcel); sp<Surface> getSurface() const; private: // can't be copied SurfaceControl& operator = (SurfaceControl& rhs); SurfaceControl(const SurfaceControl& rhs); friend class SurfaceComposerClient; // camera and camcorder need access to the ISurface binder interface for preview friend class CameraService; friend class MediaRecorder; // mediaplayer needs access to ISurface for display friend class MediaPlayer; // for testing friend class Test; // videoEditor preview classes friend class VideoEditorPreviewController; const sp<ISurface>& getISurface() const { return mSurface; } friend class Surface; SurfaceControl( const sp<SurfaceComposerClient>& client, const sp<ISurface>& surface, const ISurfaceComposerClient::surface_data_t& data, uint32_t w, uint32_t h, PixelFormat format, uint32_t flags); ~SurfaceControl(); status_t validate() const; void destroy(); sp<SurfaceComposerClient> mClient; sp<ISurface> mSurface; SurfaceID mToken; uint32_t mIdentity; uint32_t mWidth; uint32_t mHeight; PixelFormat mFormat; uint32_t mFlags; mutable Mutex mLock; mutable sp<Surface> mSurfaceData; }; // --------------------------------------------------------------------------- class Surface : public EGLNativeBase<ANativeWindow, Surface, RefBase> { public: struct SurfaceInfo { uint32_t w; uint32_t h; uint32_t s; uint32_t usage; PixelFormat format; void* bits; uint32_t reserved[2]; }; static status_t writeToParcel( const sp<Surface>& control, Parcel* parcel); static sp<Surface> readFromParcel(const Parcel& data); static bool isValid(const sp<Surface>& surface) { return (surface != 0) && surface->isValid(); } bool isValid(); uint32_t getFlags() const { return mFlags; } uint32_t getIdentity() const { return mIdentity; } // the lock/unlock APIs must be used from the same thread status_t lock(SurfaceInfo* info, bool blocking = true); status_t lock(SurfaceInfo* info, Region* dirty, bool blocking = true); status_t unlockAndPost(); // setSwapRectangle() is intended to be used by GL ES clients void setSwapRectangle(const Rect& r); private: /* * Android frameworks friends * (eventually this should go away and be replaced by proper APIs) */ // camera and camcorder need access to the ISurface binder interface for preview friend class CameraService; friend class MediaRecorder; // MediaPlayer needs access to ISurface for display friend class MediaPlayer; friend class IOMX; friend class SoftwareRenderer; // this is just to be able to write some unit tests friend class Test; // videoEditor preview classes friend class VideoEditorPreviewController; friend class PreviewRenderer; private: friend class SurfaceComposerClient; friend class SurfaceControl; // can't be copied Surface& operator = (Surface& rhs); Surface(const Surface& rhs); Surface(const sp<SurfaceControl>& control); Surface(const Parcel& data, const sp<IBinder>& ref); ~Surface(); /* * ANativeWindow hooks */ static int setSwapInterval(ANativeWindow* window, int interval); static int dequeueBuffer(ANativeWindow* window, android_native_buffer_t** buffer); static int cancelBuffer(ANativeWindow* window, android_native_buffer_t* buffer); static int lockBuffer(ANativeWindow* window, android_native_buffer_t* buffer); static int queueBuffer(ANativeWindow* window, android_native_buffer_t* buffer); static int query(ANativeWindow* window, int what, int* value); static int perform(ANativeWindow* window, int operation, ...); int dequeueBuffer(android_native_buffer_t** buffer); int lockBuffer(android_native_buffer_t* buffer); int queueBuffer(android_native_buffer_t* buffer); int cancelBuffer(android_native_buffer_t* buffer); int query(int what, int* value); int perform(int operation, va_list args); void dispatch_setUsage(va_list args); int dispatch_connect(va_list args); int dispatch_disconnect(va_list args); int dispatch_crop(va_list args); int dispatch_set_buffer_count(va_list args); int dispatch_set_buffers_geometry(va_list args); int dispatch_set_buffers_transform(va_list args); void setUsage(uint32_t reqUsage); int connect(int api); int disconnect(int api); int crop(Rect const* rect); int setBufferCount(int bufferCount); int setBuffersGeometry(int w, int h, int format); int setBuffersTransform(int transform); /* * private stuff... */ void init(); status_t validate(bool inCancelBuffer = false) const; sp<ISurface> getISurface() const; // When the buffer pool is a fixed size we want to make sure SurfaceFlinger // won't stall clients, so we require an extra buffer. enum { MIN_UNDEQUEUED_BUFFERS = 2 }; inline const GraphicBufferMapper& getBufferMapper() const { return mBufferMapper; } inline GraphicBufferMapper& getBufferMapper() { return mBufferMapper; } status_t getBufferLocked(int index, uint32_t w, uint32_t h, uint32_t format, uint32_t usage); int getBufferIndex(const sp<GraphicBuffer>& buffer) const; int getConnectedApi() const; bool needNewBuffer(int bufIdx, uint32_t *pWidth, uint32_t *pHeight, uint32_t *pFormat, uint32_t *pUsage) const; static void cleanCachedSurfacesLocked(); class BufferInfo { uint32_t mWidth; uint32_t mHeight; uint32_t mFormat; uint32_t mUsage; mutable uint32_t mDirty; enum { GEOMETRY = 0x01 }; public: BufferInfo(); void set(uint32_t w, uint32_t h, uint32_t format); void set(uint32_t usage); void get(uint32_t *pWidth, uint32_t *pHeight, uint32_t *pFormat, uint32_t *pUsage) const; bool validateBuffer(const sp<GraphicBuffer>& buffer) const; }; // constants GraphicBufferMapper& mBufferMapper; SurfaceClient& mClient; SharedBufferClient* mSharedBufferClient; status_t mInitCheck; sp<ISurface> mSurface; uint32_t mIdentity; PixelFormat mFormat; uint32_t mFlags; // protected by mSurfaceLock Rect mSwapRectangle; int mConnected; Rect mNextBufferCrop; uint32_t mNextBufferTransform; BufferInfo mBufferInfo; // protected by mSurfaceLock. These are also used from lock/unlock // but in that case, they must be called form the same thread. mutable Region mDirtyRegion; // must be used from the lock/unlock thread sp<GraphicBuffer> mLockedBuffer; sp<GraphicBuffer> mPostedBuffer; mutable Region mOldDirtyRegion; bool mReserved; // only used from dequeueBuffer() Vector< sp<GraphicBuffer> > mBuffers; // query() must be called from dequeueBuffer() thread uint32_t mWidth; uint32_t mHeight; // Inherently thread-safe mutable Mutex mSurfaceLock; mutable Mutex mApiLock; // A cache of Surface objects that have been deserialized into this process. static Mutex sCachedSurfacesLock; static DefaultKeyedVector<wp<IBinder>, wp<Surface> > sCachedSurfaces; }; }; // namespace android #endif // ANDROID_SF_SURFACE_H