| /* |
| * 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 |