/* ** Copyright 2008, Google Inc. ** Copyright (c) 2009-2012, The Linux Foundation. All rights reserved. ** ** 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_HARDWARE_QCAMERA_STREAM_H #define ANDROID_HARDWARE_QCAMERA_STREAM_H #include <utils/threads.h> #include <binder/MemoryBase.h> #include <binder/MemoryHeapBase.h> #include <utils/threads.h> #include "QCameraHWI.h" #include "QCameraHWI_Mem.h" #include "QCamera_Intf.h" extern "C" { #include <mm_camera_interface2.h> #define DEFAULT_STREAM_WIDTH 320 #define DEFAULT_STREAM_HEIGHT 240 #define DEFAULT_LIVESHOT_WIDTH 2592 #define DEFAULT_LIVESHOT_HEIGHT 1944 #define MM_CAMERA_CH_PREVIEW_MASK (0x01 << MM_CAMERA_CH_PREVIEW) #define MM_CAMERA_CH_VIDEO_MASK (0x01 << MM_CAMERA_CH_VIDEO) #define MM_CAMERA_CH_SNAPSHOT_MASK (0x01 << MM_CAMERA_CH_SNAPSHOT) } /* extern C*/ typedef struct snap_hdr_record_t_ { bool hdr_on; int num_frame; int num_raw_received; /*in terms of 2^(n/6), e.g -6 means (1/2)x, while 12 is 4x*/ int exp[MAX_HDR_EXP_FRAME_NUM]; mm_camera_ch_data_buf_t *recvd_frame[MAX_HDR_EXP_FRAME_NUM]; } snap_hdr_record_t; namespace android { class QCameraHardwareInterface; class StreamQueue { private: Mutex mQueueLock; Condition mQueueWait; bool mInitialized; //Vector<struct msm_frame *> mContainer; Vector<void *> mContainer; public: StreamQueue(); virtual ~StreamQueue(); bool enqueue(void *element); void flush(); void* dequeue(); void init(); void deinit(); bool isInitialized(); bool isEmpty(); }; class QCameraStream { //: public virtual RefBase{ public: bool mInit; bool mActive; virtual status_t init(); virtual status_t start(); virtual void stop(); virtual void release(); status_t setFormat(uint8_t ch_type_mask, cam_format_t previewFmt); status_t setMode(int enable); virtual void setHALCameraControl(QCameraHardwareInterface* ctrl); //static status_t openChannel(mm_camera_t *, mm_camera_channel_type_t ch_type); virtual status_t initChannel(int cameraId, uint32_t ch_type_mask); virtual status_t deinitChannel(int cameraId, mm_camera_channel_type_t ch_type); virtual void releaseRecordingFrame(const void *opaque) { ; } #if 0 // mzhu virtual status_t getBufferInfo(sp<IMemory>& Frame, size_t *alignedSize) { return NO_ERROR; } #endif // mzhu virtual void prepareHardware() { ; } virtual sp<IMemoryHeap> getHeap() const{return NULL;} virtual status_t initDisplayBuffers(){return NO_ERROR;} virtual status_t initPreviewOnlyBuffers(){return NO_ERROR;} virtual sp<IMemoryHeap> getRawHeap() const {return NULL;} virtual void *getLastQueuedFrame(void){return NULL;} virtual status_t takePictureZSL(void){return NO_ERROR;} virtual status_t takeLiveSnapshot(){return NO_ERROR;} virtual status_t takePictureLiveshot(mm_camera_ch_data_buf_t* recvd_frame){return NO_ERROR;} virtual void setModeLiveSnapshot(bool){;} virtual status_t initSnapshotBuffers(cam_ctrl_dimension_t *dim, int num_of_buf){return NO_ERROR;} virtual void setFullSizeLiveshot(bool){}; /* Set the ANativeWindow */ virtual int setPreviewWindow(preview_stream_ops_t* window) {return NO_ERROR;} virtual void notifyROIEvent(fd_roi_t roi) {;} virtual void notifyWDenoiseEvent(cam_ctrl_status_t status, void * cookie) {}; virtual void resetSnapshotCounters(void ){}; virtual void InitHdrInfoForSnapshot(bool HDR_on, int number_frames, int *exp ) {}; virtual void notifyHdrEvent(cam_ctrl_status_t status, void * cookie) {}; QCameraStream(); QCameraStream(int, camera_mode_t); virtual ~QCameraStream(); QCameraHardwareInterface* mHalCamCtrl; mm_camera_ch_crop_t mCrop; int mCameraId; camera_mode_t myMode; mutable Mutex mStopCallbackLock; private: StreamQueue mBusyQueue; StreamQueue mFreeQueue; public: friend void liveshot_callback(mm_camera_ch_data_buf_t *frame,void *user_data); }; /* * Record Class */ class QCameraStream_record : public QCameraStream { public: status_t init(); status_t start() ; void stop() ; void release() ; static QCameraStream* createInstance(int cameraId, camera_mode_t); static void deleteInstance(QCameraStream *p); QCameraStream_record() {}; virtual ~QCameraStream_record(); status_t processRecordFrame(void *data); status_t initEncodeBuffers(); status_t getBufferInfo(sp<IMemory>& Frame, size_t *alignedSize); //sp<IMemoryHeap> getHeap() const; void releaseRecordingFrame(const void *opaque); void debugShowVideoFPS() const; status_t takeLiveSnapshot(); private: QCameraStream_record(int, camera_mode_t); void releaseEncodeBuffer(); cam_ctrl_dimension_t dim; bool mDebugFps; mm_camera_reg_buf_t mRecordBuf; //int record_frame_len; //static const int maxFrameCnt = 16; //camera_memory_t *mCameraMemoryPtr[maxFrameCnt]; //int mNumRecordFrames; //sp<PmemPool> mRecordHeap[maxFrameCnt]; struct msm_frame *recordframes; //uint32_t record_offset[VIDEO_BUFFER_COUNT]; mm_camera_ch_data_buf_t mRecordedFrames[MM_CAMERA_MAX_NUM_FRAMES]; //Mutex mRecordFreeQueueLock; //Vector<mm_camera_ch_data_buf_t> mRecordFreeQueue; int mJpegMaxSize; QCameraStream *mStreamSnap; }; class QCameraStream_preview : public QCameraStream { public: status_t init(); status_t start() ; void stop() ; void release() ; static QCameraStream* createInstance(int, camera_mode_t); static void deleteInstance(QCameraStream *p); QCameraStream_preview() {}; virtual ~QCameraStream_preview(); void *getLastQueuedFrame(void); /*init preview buffers with display case*/ status_t initDisplayBuffers(); /*init preview buffers without display case*/ status_t initPreviewOnlyBuffers(); status_t processPreviewFrame(mm_camera_ch_data_buf_t *frame); /*init preview buffers with display case*/ status_t processPreviewFrameWithDisplay(mm_camera_ch_data_buf_t *frame); /*init preview buffers without display case*/ status_t processPreviewFrameWithOutDisplay(mm_camera_ch_data_buf_t *frame); int setPreviewWindow(preview_stream_ops_t* window); void notifyROIEvent(fd_roi_t roi); friend class QCameraHardwareInterface; private: QCameraStream_preview(int cameraId, camera_mode_t); /*allocate and free buffers with display case*/ status_t getBufferFromSurface(); status_t putBufferToSurface(); /*allocate and free buffers without display case*/ status_t getBufferNoDisplay(); status_t freeBufferNoDisplay(); void dumpFrameToFile(struct msm_frame* newFrame); bool mFirstFrameRcvd; int8_t my_id; mm_camera_op_mode_type_t op_mode; cam_ctrl_dimension_t dim; struct msm_frame *mLastQueuedFrame; mm_camera_reg_buf_t mDisplayBuf; mm_cameara_stream_buf_t mDisplayStreamBuf; Mutex mDisplayLock; preview_stream_ops_t *mPreviewWindow; static const int kPreviewBufferCount = PREVIEW_BUFFER_COUNT; mm_camera_ch_data_buf_t mNotifyBuffer[16]; int8_t mNumFDRcvd; int mVFEOutputs; int mHFRFrameCnt; int mHFRFrameSkip; }; /* Snapshot Class - handle data flow*/ class QCameraStream_Snapshot : public QCameraStream { public: status_t init(); status_t start(); void stop(); void release(); void prepareHardware(); static QCameraStream* createInstance(int cameraId, camera_mode_t); static void deleteInstance(QCameraStream *p); status_t takePictureZSL(void); status_t takePictureLiveshot(mm_camera_ch_data_buf_t* recvd_frame); status_t receiveRawPicture(mm_camera_ch_data_buf_t* recvd_frame); void receiveCompleteJpegPicture(jpeg_event_t event); void jpegErrorHandler(jpeg_event_t event); void receiveJpegFragment(uint8_t *ptr, uint32_t size); void deInitBuffer(void); sp<IMemoryHeap> getRawHeap() const; int getSnapshotState(); /*Temp: to be removed once event handling is enabled in mm-camera*/ void runSnapshotThread(void *data); bool isZSLMode(); void setFullSizeLiveshot(bool); void notifyWDenoiseEvent(cam_ctrl_status_t status, void * cookie); friend void liveshot_callback(mm_camera_ch_data_buf_t *frame,void *user_data); void resetSnapshotCounters(void ); void InitHdrInfoForSnapshot(bool HDR_on, int number_frames, int *exp ); void notifyHdrEvent(cam_ctrl_status_t status, void * cookie); private: QCameraStream_Snapshot(int, camera_mode_t); virtual ~QCameraStream_Snapshot(); /* snapshot related private members */ status_t initJPEGSnapshot(int num_of_snapshots); status_t initRawSnapshot(int num_of_snapshots); status_t initZSLSnapshot(void); status_t initFullLiveshot(void); status_t cancelPicture(); void notifyShutter(common_crop_t *crop, bool play_shutter_sound); status_t initSnapshotBuffers(cam_ctrl_dimension_t *dim, int num_of_buf); status_t initRawSnapshotBuffers(cam_ctrl_dimension_t *dim, int num_of_buf); status_t deinitRawSnapshotBuffers(void); status_t deinitSnapshotBuffers(void); status_t initRawSnapshotChannel(cam_ctrl_dimension_t* dim, int num_snapshots); status_t initSnapshotFormat(cam_ctrl_dimension_t *dim); status_t takePictureRaw(void); status_t takePictureJPEG(void); status_t startStreamZSL(void); void deinitSnapshotChannel(mm_camera_channel_type_t); status_t configSnapshotDimension(cam_ctrl_dimension_t* dim); status_t encodeData(mm_camera_ch_data_buf_t* recvd_frame, common_crop_t *crop_info, int frame_len, bool enqueued); status_t encodeDisplayAndSave(mm_camera_ch_data_buf_t* recvd_frame, bool enqueued); status_t setZSLChannelAttribute(void); void handleError(); void setSnapshotState(int state); void setModeLiveSnapshot(bool); bool isLiveSnapshot(void); void stopPolling(void); bool isFullSizeLiveshot(void); status_t doWaveletDenoise(mm_camera_ch_data_buf_t* frame); status_t sendWDenoiseStartMsg(mm_camera_ch_data_buf_t * frame); void lauchNextWDenoiseFromQueue(); status_t doHdrProcessing( ); /* Member variables */ int mSnapshotFormat; int mPictureWidth; int mPictureHeight; cam_format_t mPictureFormat; int mPostviewWidth; int mPostviewHeight; int mThumbnailWidth; int mThumbnailHeight; cam_format_t mThumbnailFormat; int mJpegOffset; int mSnapshotState; int mNumOfSnapshot; int mNumOfRecievedJPEG; bool mModeLiveSnapshot; bool mBurstModeFlag; int mActualPictureWidth; int mActualPictureHeight; bool mJpegDownscaling; sp<AshmemPool> mJpegHeap; /*TBD:Bikas: This is defined in HWI too.*/ #ifdef USE_ION sp<IonPool> mDisplayHeap; sp<IonPool> mPostviewHeap; #else sp<PmemPool> mDisplayHeap; sp<PmemPool> mPostviewHeap; #endif mm_camera_ch_data_buf_t *mCurrentFrameEncoded; mm_cameara_stream_buf_t mSnapshotStreamBuf; mm_cameara_stream_buf_t mPostviewStreamBuf; StreamQueue mSnapshotQueue; static const int mMaxSnapshotBufferCount = 16; int mSnapshotBufferNum; int mMainfd[mMaxSnapshotBufferCount]; int mThumbfd[mMaxSnapshotBufferCount]; int mMainSize; int mThumbSize; camera_memory_t *mCameraMemoryPtrMain[mMaxSnapshotBufferCount]; camera_memory_t *mCameraMemoryPtrThumb[mMaxSnapshotBufferCount]; int mJpegSessionId; int dump_fd; bool mFullLiveshot; StreamQueue mWDNQueue; // queue to hold frames while one frame is sent out for WDN bool mIsDoingWDN; // flag to indicate if WDN is going on (one frame is sent out for WDN) bool mDropThumbnail; int mJpegQuality; snap_hdr_record_t mHdrInfo; int hdrRawCount; int hdrJpegCount; }; // QCameraStream_Snapshot }; // namespace android #endif