Loading cmds/stagefright/stagefright.cpp +1 −1 Original line number Diff line number Diff line Loading @@ -893,7 +893,7 @@ int main(int argc, char **argv) { VideoFrame *frame = (VideoFrame *)mem->pointer(); CHECK_EQ(writeJpegFile("/sdcard/out.jpg", (uint8_t *)frame + sizeof(VideoFrame), frame->getFlattenedData(), frame->mWidth, frame->mHeight), 0); } Loading include/private/media/VideoFrame.h +17 −105 Original line number Diff line number Diff line Loading @@ -25,96 +25,34 @@ namespace android { // Represents a color converted (RGB-based) video frame // with bitmap pixels stored in FrameBuffer // Represents a color converted (RGB-based) video frame with bitmap // pixels stored in FrameBuffer. // In a VideoFrame struct stored in IMemory, frame data and ICC data // come after the VideoFrame structure. Their locations can be retrieved // by getFlattenedData() and getFlattenedIccData(); class VideoFrame { public: // Construct a VideoFrame object with the specified parameters, // will allocate frame buffer if |allocate| is set to true, will // allocate buffer to hold ICC data if |iccData| and |iccSize| // indicate its presence. // will calculate frame buffer size if |hasData| is set to true. VideoFrame(uint32_t width, uint32_t height, uint32_t displayWidth, uint32_t displayHeight, uint32_t angle, uint32_t bpp, bool allocate, const void *iccData, size_t iccSize): uint32_t angle, uint32_t bpp, bool hasData, size_t iccSize): mWidth(width), mHeight(height), mDisplayWidth(displayWidth), mDisplayHeight(displayHeight), mRotationAngle(angle), mBytesPerPixel(bpp), mRowBytes(bpp * width), mSize(0), mIccSize(0), mReserved(0), mData(0), mIccData(0) { if (allocate) { mSize = mRowBytes * mHeight; mData = new uint8_t[mSize]; if (mData == NULL) { mSize = 0; } } if (iccData != NULL && iccSize > 0) { mIccSize = iccSize; mIccData = new uint8_t[iccSize]; if (mIccData != NULL) { memcpy(mIccData, iccData, iccSize); } else { mIccSize = 0; } } mSize(hasData ? (bpp * width * height) : 0), mIccSize(iccSize), mReserved(0) { } // Deep copy of both the information fields and the frame data VideoFrame(const VideoFrame& copy) { copyInfoOnly(copy); mSize = copy.mSize; mData = NULL; // initialize it first if (mSize > 0 && copy.mData != NULL) { mData = new uint8_t[mSize]; if (mData != NULL) { memcpy(mData, copy.mData, mSize); } else { mSize = 0; } } mIccSize = copy.mIccSize; mIccData = NULL; // initialize it first if (mIccSize > 0 && copy.mIccData != NULL) { mIccData = new uint8_t[mIccSize]; if (mIccData != NULL) { memcpy(mIccData, copy.mIccData, mIccSize); void init(const VideoFrame& copy, const void* iccData, size_t iccSize) { *this = copy; if (mIccSize == iccSize && iccSize > 0 && iccData != NULL) { memcpy(getFlattenedIccData(), iccData, iccSize); } else { mIccSize = 0; } } } ~VideoFrame() { if (mData != 0) { delete[] mData; } if (mIccData != 0) { delete[] mIccData; } } // Copy |copy| to a flattened VideoFrame in IMemory, 'this' must point to // a chunk of memory back by IMemory of size at least getFlattenedSize() // of |copy|. void copyFlattened(const VideoFrame& copy) { copyInfoOnly(copy); mSize = copy.mSize; mData = NULL; // initialize it first if (copy.mSize > 0 && copy.mData != NULL) { memcpy(getFlattenedData(), copy.mData, copy.mSize); } mIccSize = copy.mIccSize; mIccData = NULL; // initialize it first if (copy.mIccSize > 0 && copy.mIccData != NULL) { memcpy(getFlattenedIccData(), copy.mIccData, copy.mIccSize); } } // Calculate the flattened size to put it in IMemory size_t getFlattenedSize() const { Loading @@ -139,35 +77,9 @@ public: int32_t mRotationAngle; // Rotation angle, clockwise, should be multiple of 90 uint32_t mBytesPerPixel; // Number of bytes per pixel uint32_t mRowBytes; // Number of bytes per row before rotation uint32_t mSize; // Number of bytes in mData uint32_t mIccSize; // Number of bytes in mIccData uint32_t mSize; // Number of bytes of frame data uint32_t mIccSize; // Number of bytes of ICC data uint32_t mReserved; // (padding to make mData 64-bit aligned) // mData should be 64-bit aligned to prevent additional padding uint8_t* mData; // Actual binary data // pad structure so it's the same size on 64-bit and 32-bit char mPadding[8 - sizeof(mData)]; // mIccData should be 64-bit aligned to prevent additional padding uint8_t* mIccData; // Actual binary data // pad structure so it's the same size on 64-bit and 32-bit char mIccPadding[8 - sizeof(mIccData)]; private: // // Utility methods used only within VideoFrame struct // // Copy the information fields only void copyInfoOnly(const VideoFrame& copy) { mWidth = copy.mWidth; mHeight = copy.mHeight; mDisplayWidth = copy.mDisplayWidth; mDisplayHeight = copy.mDisplayHeight; mRotationAngle = copy.mRotationAngle; mBytesPerPixel = copy.mBytesPerPixel; mRowBytes = copy.mRowBytes; } }; }; // namespace android Loading media/libmedia/include/media/MediaMetadataRetrieverInterface.h +6 −6 Original line number Diff line number Diff line Loading @@ -43,12 +43,12 @@ public: virtual status_t setDataSource(int fd, int64_t offset, int64_t length) = 0; virtual status_t setDataSource(const sp<DataSource>& source, const char *mime) = 0; virtual VideoFrame* getFrameAtTime( virtual sp<IMemory> getFrameAtTime( int64_t timeUs, int option, int colorFormat, bool metaOnly) = 0; virtual VideoFrame* getImageAtIndex( virtual sp<IMemory> getImageAtIndex( int index, int colorFormat, bool metaOnly, bool thumbnail) = 0; virtual status_t getFrameAtIndex( std::vector<VideoFrame*>* frames, std::vector<sp<IMemory> >* frames, int frameIndex, int numFrames, int colorFormat, bool metaOnly) = 0; virtual MediaAlbumArt* extractAlbumArt() = 0; virtual const char* extractMetadata(int keyCode) = 0; Loading @@ -61,14 +61,14 @@ public: MediaMetadataRetrieverInterface() {} virtual ~MediaMetadataRetrieverInterface() {} virtual VideoFrame* getFrameAtTime( virtual sp<IMemory> getFrameAtTime( int64_t /*timeUs*/, int /*option*/, int /*colorFormat*/, bool /*metaOnly*/) { return NULL; } virtual VideoFrame* getImageAtIndex( virtual sp<IMemory> getImageAtIndex( int /*index*/, int /*colorFormat*/, bool /*metaOnly*/, bool /*thumbnail*/) { return NULL; } virtual status_t getFrameAtIndex( std::vector<VideoFrame*>* /*frames*/, std::vector<sp<IMemory> >* /*frames*/, int /*frameIndex*/, int /*numFrames*/, int /*colorFormat*/, bool /*metaOnly*/) { return ERROR_UNSUPPORTED; } virtual MediaAlbumArt* extractAlbumArt() { return NULL; } Loading media/libmediaplayerservice/MetadataRetrieverClient.cpp +6 −28 Original line number Diff line number Diff line Loading @@ -194,25 +194,6 @@ status_t MetadataRetrieverClient::setDataSource( Mutex MetadataRetrieverClient::sLock; static sp<IMemory> getThumbnail(VideoFrame* frame) { std::unique_ptr<VideoFrame> frameDeleter(frame); size_t size = frame->getFlattenedSize(); sp<MemoryHeapBase> heap = new MemoryHeapBase(size, 0, "MetadataRetrieverClient"); if (heap == NULL) { ALOGE("failed to create MemoryDealer"); return NULL; } sp<IMemory> thrumbnail = new MemoryBase(heap, 0, size); if (thrumbnail == NULL) { ALOGE("not enough memory for VideoFrame size=%zu", size); return NULL; } VideoFrame *frameCopy = static_cast<VideoFrame *>(thrumbnail->pointer()); frameCopy->copyFlattened(*frame); return thrumbnail; } sp<IMemory> MetadataRetrieverClient::getFrameAtTime( int64_t timeUs, int option, int colorFormat, bool metaOnly) { Loading @@ -225,12 +206,12 @@ sp<IMemory> MetadataRetrieverClient::getFrameAtTime( ALOGE("retriever is not initialized"); return NULL; } VideoFrame *frame = mRetriever->getFrameAtTime(timeUs, option, colorFormat, metaOnly); sp<IMemory> frame = mRetriever->getFrameAtTime(timeUs, option, colorFormat, metaOnly); if (frame == NULL) { ALOGE("failed to capture a video frame"); return NULL; } return getThumbnail(frame); return frame; } sp<IMemory> MetadataRetrieverClient::getImageAtIndex( Loading @@ -244,12 +225,12 @@ sp<IMemory> MetadataRetrieverClient::getImageAtIndex( ALOGE("retriever is not initialized"); return NULL; } VideoFrame *frame = mRetriever->getImageAtIndex(index, colorFormat, metaOnly, thumbnail); sp<IMemory> frame = mRetriever->getImageAtIndex(index, colorFormat, metaOnly, thumbnail); if (frame == NULL) { ALOGE("failed to extract image"); return NULL; } return getThumbnail(frame); return frame; } status_t MetadataRetrieverClient::getFrameAtIndex( Loading @@ -264,15 +245,12 @@ status_t MetadataRetrieverClient::getFrameAtIndex( return INVALID_OPERATION; } std::vector<VideoFrame*> videoFrames; status_t err = mRetriever->getFrameAtIndex( &videoFrames, frameIndex, numFrames, colorFormat, metaOnly); frames, frameIndex, numFrames, colorFormat, metaOnly); if (err != OK) { frames->clear(); return err; } for (size_t i = 0; i < videoFrames.size(); i++) { frames->push_back(getThumbnail(videoFrames[i])); } return OK; } Loading media/libstagefright/FrameDecoder.cpp +37 −19 Original line number Diff line number Diff line Loading @@ -17,12 +17,11 @@ //#define LOG_NDEBUG 0 #define LOG_TAG "FrameDecoder" #include <inttypes.h> #include <utils/Log.h> #include <gui/Surface.h> #include "include/FrameDecoder.h" #include <binder/MemoryBase.h> #include <binder/MemoryHeapBase.h> #include <gui/Surface.h> #include <inttypes.h> #include <media/ICrypto.h> #include <media/IMediaSource.h> #include <media/MediaCodecBuffer.h> Loading @@ -36,6 +35,7 @@ #include <media/stagefright/MediaErrors.h> #include <media/stagefright/Utils.h> #include <private/media/VideoFrame.h> #include <utils/Log.h> namespace android { Loading @@ -43,7 +43,7 @@ static const int64_t kBufferTimeOutUs = 30000ll; // 30 msec static const size_t kRetryCount = 20; // must be >0 //static VideoFrame *allocVideoFrame(const sp<MetaData> &trackMeta, sp<IMemory> allocVideoFrame(const sp<MetaData>& trackMeta, int32_t width, int32_t height, int32_t dstBpp, bool metaOnly = false) { int32_t rotationAngle; if (!trackMeta->findInt32(kKeyRotation, &rotationAngle)) { Loading Loading @@ -74,8 +74,24 @@ VideoFrame *allocVideoFrame(const sp<MetaData> &trackMeta, displayHeight = height; } return new VideoFrame(width, height, displayWidth, displayHeight, rotationAngle, dstBpp, !metaOnly, iccData, iccSize); VideoFrame frame(width, height, displayWidth, displayHeight, rotationAngle, dstBpp, !metaOnly, iccSize); size_t size = frame.getFlattenedSize(); sp<MemoryHeapBase> heap = new MemoryHeapBase(size, 0, "MetadataRetrieverClient"); if (heap == NULL) { ALOGE("failed to create MemoryDealer"); return NULL; } sp<IMemory> frameMem = new MemoryBase(heap, 0, size); if (frameMem == NULL) { ALOGE("not enough memory for VideoFrame size=%zu", size); return NULL; } VideoFrame* frameCopy = static_cast<VideoFrame*>(frameMem->pointer()); frameCopy->init(frame, iccData, iccSize); return frameMem; } //static Loading @@ -92,7 +108,7 @@ bool findThumbnailInfo( } //static VideoFrame* FrameDecoder::getMetadataOnly( sp<IMemory> FrameDecoder::getMetadataOnly( const sp<MetaData> &trackMeta, int colorFormat, bool thumbnail) { OMX_COLOR_FORMATTYPE dstFormat; int32_t dstBpp; Loading Loading @@ -146,7 +162,7 @@ bool FrameDecoder::getDstColorFormat( return false; } VideoFrame* FrameDecoder::extractFrame( sp<IMemory> FrameDecoder::extractFrame( int64_t frameTimeUs, int option, int colorFormat) { if (!getDstColorFormat( (android_pixel_format_t)colorFormat, &mDstFormat, &mDstBpp)) { Loading @@ -158,12 +174,12 @@ VideoFrame* FrameDecoder::extractFrame( return NULL; } return mFrames.size() > 0 ? mFrames[0].release() : NULL; return mFrames.size() > 0 ? mFrames[0] : NULL; } status_t FrameDecoder::extractFrames( int64_t frameTimeUs, size_t numFrames, int option, int colorFormat, std::vector<VideoFrame*>* frames) { std::vector<sp<IMemory> >* frames) { if (!getDstColorFormat( (android_pixel_format_t)colorFormat, &mDstFormat, &mDstBpp)) { return ERROR_UNSUPPORTED; Loading @@ -175,7 +191,7 @@ status_t FrameDecoder::extractFrames( } for (size_t i = 0; i < mFrames.size(); i++) { frames->push_back(mFrames[i].release()); frames->push_back(mFrames[i]); } return OK; } Loading Loading @@ -468,12 +484,13 @@ status_t VideoFrameDecoder::onOutputReceived( crop_bottom = height - 1; } VideoFrame *frame = allocVideoFrame( sp<IMemory> frameMem = allocVideoFrame( trackMeta(), (crop_right - crop_left + 1), (crop_bottom - crop_top + 1), dstBpp()); addFrame(frame); addFrame(frameMem); VideoFrame* frame = static_cast<VideoFrame*>(frameMem->pointer()); int32_t srcFormat; CHECK(outputFormat->findInt32("color-format", &srcFormat)); Loading @@ -485,7 +502,7 @@ status_t VideoFrameDecoder::onOutputReceived( (const uint8_t *)videoFrameBuffer->data(), width, height, crop_left, crop_top, crop_right, crop_bottom, frame->mData, frame->getFlattenedData(), frame->mWidth, frame->mHeight, crop_left, crop_top, crop_right, crop_bottom); Loading Loading @@ -596,9 +613,10 @@ status_t ImageDecoder::onOutputReceived( } if (mFrame == NULL) { mFrame = allocVideoFrame(trackMeta(), imageWidth, imageHeight, dstBpp()); sp<IMemory> frameMem = allocVideoFrame(trackMeta(), imageWidth, imageHeight, dstBpp()); mFrame = static_cast<VideoFrame*>(frameMem->pointer()); addFrame(mFrame); addFrame(frameMem); } int32_t srcFormat; Loading Loading @@ -639,7 +657,7 @@ status_t ImageDecoder::onOutputReceived( (const uint8_t *)videoFrameBuffer->data(), width, height, crop_left, crop_top, crop_right, crop_bottom, mFrame->mData, mFrame->getFlattenedData(), mFrame->mWidth, mFrame->mHeight, dstLeft, dstTop, dstRight, dstBottom); Loading Loading
cmds/stagefright/stagefright.cpp +1 −1 Original line number Diff line number Diff line Loading @@ -893,7 +893,7 @@ int main(int argc, char **argv) { VideoFrame *frame = (VideoFrame *)mem->pointer(); CHECK_EQ(writeJpegFile("/sdcard/out.jpg", (uint8_t *)frame + sizeof(VideoFrame), frame->getFlattenedData(), frame->mWidth, frame->mHeight), 0); } Loading
include/private/media/VideoFrame.h +17 −105 Original line number Diff line number Diff line Loading @@ -25,96 +25,34 @@ namespace android { // Represents a color converted (RGB-based) video frame // with bitmap pixels stored in FrameBuffer // Represents a color converted (RGB-based) video frame with bitmap // pixels stored in FrameBuffer. // In a VideoFrame struct stored in IMemory, frame data and ICC data // come after the VideoFrame structure. Their locations can be retrieved // by getFlattenedData() and getFlattenedIccData(); class VideoFrame { public: // Construct a VideoFrame object with the specified parameters, // will allocate frame buffer if |allocate| is set to true, will // allocate buffer to hold ICC data if |iccData| and |iccSize| // indicate its presence. // will calculate frame buffer size if |hasData| is set to true. VideoFrame(uint32_t width, uint32_t height, uint32_t displayWidth, uint32_t displayHeight, uint32_t angle, uint32_t bpp, bool allocate, const void *iccData, size_t iccSize): uint32_t angle, uint32_t bpp, bool hasData, size_t iccSize): mWidth(width), mHeight(height), mDisplayWidth(displayWidth), mDisplayHeight(displayHeight), mRotationAngle(angle), mBytesPerPixel(bpp), mRowBytes(bpp * width), mSize(0), mIccSize(0), mReserved(0), mData(0), mIccData(0) { if (allocate) { mSize = mRowBytes * mHeight; mData = new uint8_t[mSize]; if (mData == NULL) { mSize = 0; } } if (iccData != NULL && iccSize > 0) { mIccSize = iccSize; mIccData = new uint8_t[iccSize]; if (mIccData != NULL) { memcpy(mIccData, iccData, iccSize); } else { mIccSize = 0; } } mSize(hasData ? (bpp * width * height) : 0), mIccSize(iccSize), mReserved(0) { } // Deep copy of both the information fields and the frame data VideoFrame(const VideoFrame& copy) { copyInfoOnly(copy); mSize = copy.mSize; mData = NULL; // initialize it first if (mSize > 0 && copy.mData != NULL) { mData = new uint8_t[mSize]; if (mData != NULL) { memcpy(mData, copy.mData, mSize); } else { mSize = 0; } } mIccSize = copy.mIccSize; mIccData = NULL; // initialize it first if (mIccSize > 0 && copy.mIccData != NULL) { mIccData = new uint8_t[mIccSize]; if (mIccData != NULL) { memcpy(mIccData, copy.mIccData, mIccSize); void init(const VideoFrame& copy, const void* iccData, size_t iccSize) { *this = copy; if (mIccSize == iccSize && iccSize > 0 && iccData != NULL) { memcpy(getFlattenedIccData(), iccData, iccSize); } else { mIccSize = 0; } } } ~VideoFrame() { if (mData != 0) { delete[] mData; } if (mIccData != 0) { delete[] mIccData; } } // Copy |copy| to a flattened VideoFrame in IMemory, 'this' must point to // a chunk of memory back by IMemory of size at least getFlattenedSize() // of |copy|. void copyFlattened(const VideoFrame& copy) { copyInfoOnly(copy); mSize = copy.mSize; mData = NULL; // initialize it first if (copy.mSize > 0 && copy.mData != NULL) { memcpy(getFlattenedData(), copy.mData, copy.mSize); } mIccSize = copy.mIccSize; mIccData = NULL; // initialize it first if (copy.mIccSize > 0 && copy.mIccData != NULL) { memcpy(getFlattenedIccData(), copy.mIccData, copy.mIccSize); } } // Calculate the flattened size to put it in IMemory size_t getFlattenedSize() const { Loading @@ -139,35 +77,9 @@ public: int32_t mRotationAngle; // Rotation angle, clockwise, should be multiple of 90 uint32_t mBytesPerPixel; // Number of bytes per pixel uint32_t mRowBytes; // Number of bytes per row before rotation uint32_t mSize; // Number of bytes in mData uint32_t mIccSize; // Number of bytes in mIccData uint32_t mSize; // Number of bytes of frame data uint32_t mIccSize; // Number of bytes of ICC data uint32_t mReserved; // (padding to make mData 64-bit aligned) // mData should be 64-bit aligned to prevent additional padding uint8_t* mData; // Actual binary data // pad structure so it's the same size on 64-bit and 32-bit char mPadding[8 - sizeof(mData)]; // mIccData should be 64-bit aligned to prevent additional padding uint8_t* mIccData; // Actual binary data // pad structure so it's the same size on 64-bit and 32-bit char mIccPadding[8 - sizeof(mIccData)]; private: // // Utility methods used only within VideoFrame struct // // Copy the information fields only void copyInfoOnly(const VideoFrame& copy) { mWidth = copy.mWidth; mHeight = copy.mHeight; mDisplayWidth = copy.mDisplayWidth; mDisplayHeight = copy.mDisplayHeight; mRotationAngle = copy.mRotationAngle; mBytesPerPixel = copy.mBytesPerPixel; mRowBytes = copy.mRowBytes; } }; }; // namespace android Loading
media/libmedia/include/media/MediaMetadataRetrieverInterface.h +6 −6 Original line number Diff line number Diff line Loading @@ -43,12 +43,12 @@ public: virtual status_t setDataSource(int fd, int64_t offset, int64_t length) = 0; virtual status_t setDataSource(const sp<DataSource>& source, const char *mime) = 0; virtual VideoFrame* getFrameAtTime( virtual sp<IMemory> getFrameAtTime( int64_t timeUs, int option, int colorFormat, bool metaOnly) = 0; virtual VideoFrame* getImageAtIndex( virtual sp<IMemory> getImageAtIndex( int index, int colorFormat, bool metaOnly, bool thumbnail) = 0; virtual status_t getFrameAtIndex( std::vector<VideoFrame*>* frames, std::vector<sp<IMemory> >* frames, int frameIndex, int numFrames, int colorFormat, bool metaOnly) = 0; virtual MediaAlbumArt* extractAlbumArt() = 0; virtual const char* extractMetadata(int keyCode) = 0; Loading @@ -61,14 +61,14 @@ public: MediaMetadataRetrieverInterface() {} virtual ~MediaMetadataRetrieverInterface() {} virtual VideoFrame* getFrameAtTime( virtual sp<IMemory> getFrameAtTime( int64_t /*timeUs*/, int /*option*/, int /*colorFormat*/, bool /*metaOnly*/) { return NULL; } virtual VideoFrame* getImageAtIndex( virtual sp<IMemory> getImageAtIndex( int /*index*/, int /*colorFormat*/, bool /*metaOnly*/, bool /*thumbnail*/) { return NULL; } virtual status_t getFrameAtIndex( std::vector<VideoFrame*>* /*frames*/, std::vector<sp<IMemory> >* /*frames*/, int /*frameIndex*/, int /*numFrames*/, int /*colorFormat*/, bool /*metaOnly*/) { return ERROR_UNSUPPORTED; } virtual MediaAlbumArt* extractAlbumArt() { return NULL; } Loading
media/libmediaplayerservice/MetadataRetrieverClient.cpp +6 −28 Original line number Diff line number Diff line Loading @@ -194,25 +194,6 @@ status_t MetadataRetrieverClient::setDataSource( Mutex MetadataRetrieverClient::sLock; static sp<IMemory> getThumbnail(VideoFrame* frame) { std::unique_ptr<VideoFrame> frameDeleter(frame); size_t size = frame->getFlattenedSize(); sp<MemoryHeapBase> heap = new MemoryHeapBase(size, 0, "MetadataRetrieverClient"); if (heap == NULL) { ALOGE("failed to create MemoryDealer"); return NULL; } sp<IMemory> thrumbnail = new MemoryBase(heap, 0, size); if (thrumbnail == NULL) { ALOGE("not enough memory for VideoFrame size=%zu", size); return NULL; } VideoFrame *frameCopy = static_cast<VideoFrame *>(thrumbnail->pointer()); frameCopy->copyFlattened(*frame); return thrumbnail; } sp<IMemory> MetadataRetrieverClient::getFrameAtTime( int64_t timeUs, int option, int colorFormat, bool metaOnly) { Loading @@ -225,12 +206,12 @@ sp<IMemory> MetadataRetrieverClient::getFrameAtTime( ALOGE("retriever is not initialized"); return NULL; } VideoFrame *frame = mRetriever->getFrameAtTime(timeUs, option, colorFormat, metaOnly); sp<IMemory> frame = mRetriever->getFrameAtTime(timeUs, option, colorFormat, metaOnly); if (frame == NULL) { ALOGE("failed to capture a video frame"); return NULL; } return getThumbnail(frame); return frame; } sp<IMemory> MetadataRetrieverClient::getImageAtIndex( Loading @@ -244,12 +225,12 @@ sp<IMemory> MetadataRetrieverClient::getImageAtIndex( ALOGE("retriever is not initialized"); return NULL; } VideoFrame *frame = mRetriever->getImageAtIndex(index, colorFormat, metaOnly, thumbnail); sp<IMemory> frame = mRetriever->getImageAtIndex(index, colorFormat, metaOnly, thumbnail); if (frame == NULL) { ALOGE("failed to extract image"); return NULL; } return getThumbnail(frame); return frame; } status_t MetadataRetrieverClient::getFrameAtIndex( Loading @@ -264,15 +245,12 @@ status_t MetadataRetrieverClient::getFrameAtIndex( return INVALID_OPERATION; } std::vector<VideoFrame*> videoFrames; status_t err = mRetriever->getFrameAtIndex( &videoFrames, frameIndex, numFrames, colorFormat, metaOnly); frames, frameIndex, numFrames, colorFormat, metaOnly); if (err != OK) { frames->clear(); return err; } for (size_t i = 0; i < videoFrames.size(); i++) { frames->push_back(getThumbnail(videoFrames[i])); } return OK; } Loading
media/libstagefright/FrameDecoder.cpp +37 −19 Original line number Diff line number Diff line Loading @@ -17,12 +17,11 @@ //#define LOG_NDEBUG 0 #define LOG_TAG "FrameDecoder" #include <inttypes.h> #include <utils/Log.h> #include <gui/Surface.h> #include "include/FrameDecoder.h" #include <binder/MemoryBase.h> #include <binder/MemoryHeapBase.h> #include <gui/Surface.h> #include <inttypes.h> #include <media/ICrypto.h> #include <media/IMediaSource.h> #include <media/MediaCodecBuffer.h> Loading @@ -36,6 +35,7 @@ #include <media/stagefright/MediaErrors.h> #include <media/stagefright/Utils.h> #include <private/media/VideoFrame.h> #include <utils/Log.h> namespace android { Loading @@ -43,7 +43,7 @@ static const int64_t kBufferTimeOutUs = 30000ll; // 30 msec static const size_t kRetryCount = 20; // must be >0 //static VideoFrame *allocVideoFrame(const sp<MetaData> &trackMeta, sp<IMemory> allocVideoFrame(const sp<MetaData>& trackMeta, int32_t width, int32_t height, int32_t dstBpp, bool metaOnly = false) { int32_t rotationAngle; if (!trackMeta->findInt32(kKeyRotation, &rotationAngle)) { Loading Loading @@ -74,8 +74,24 @@ VideoFrame *allocVideoFrame(const sp<MetaData> &trackMeta, displayHeight = height; } return new VideoFrame(width, height, displayWidth, displayHeight, rotationAngle, dstBpp, !metaOnly, iccData, iccSize); VideoFrame frame(width, height, displayWidth, displayHeight, rotationAngle, dstBpp, !metaOnly, iccSize); size_t size = frame.getFlattenedSize(); sp<MemoryHeapBase> heap = new MemoryHeapBase(size, 0, "MetadataRetrieverClient"); if (heap == NULL) { ALOGE("failed to create MemoryDealer"); return NULL; } sp<IMemory> frameMem = new MemoryBase(heap, 0, size); if (frameMem == NULL) { ALOGE("not enough memory for VideoFrame size=%zu", size); return NULL; } VideoFrame* frameCopy = static_cast<VideoFrame*>(frameMem->pointer()); frameCopy->init(frame, iccData, iccSize); return frameMem; } //static Loading @@ -92,7 +108,7 @@ bool findThumbnailInfo( } //static VideoFrame* FrameDecoder::getMetadataOnly( sp<IMemory> FrameDecoder::getMetadataOnly( const sp<MetaData> &trackMeta, int colorFormat, bool thumbnail) { OMX_COLOR_FORMATTYPE dstFormat; int32_t dstBpp; Loading Loading @@ -146,7 +162,7 @@ bool FrameDecoder::getDstColorFormat( return false; } VideoFrame* FrameDecoder::extractFrame( sp<IMemory> FrameDecoder::extractFrame( int64_t frameTimeUs, int option, int colorFormat) { if (!getDstColorFormat( (android_pixel_format_t)colorFormat, &mDstFormat, &mDstBpp)) { Loading @@ -158,12 +174,12 @@ VideoFrame* FrameDecoder::extractFrame( return NULL; } return mFrames.size() > 0 ? mFrames[0].release() : NULL; return mFrames.size() > 0 ? mFrames[0] : NULL; } status_t FrameDecoder::extractFrames( int64_t frameTimeUs, size_t numFrames, int option, int colorFormat, std::vector<VideoFrame*>* frames) { std::vector<sp<IMemory> >* frames) { if (!getDstColorFormat( (android_pixel_format_t)colorFormat, &mDstFormat, &mDstBpp)) { return ERROR_UNSUPPORTED; Loading @@ -175,7 +191,7 @@ status_t FrameDecoder::extractFrames( } for (size_t i = 0; i < mFrames.size(); i++) { frames->push_back(mFrames[i].release()); frames->push_back(mFrames[i]); } return OK; } Loading Loading @@ -468,12 +484,13 @@ status_t VideoFrameDecoder::onOutputReceived( crop_bottom = height - 1; } VideoFrame *frame = allocVideoFrame( sp<IMemory> frameMem = allocVideoFrame( trackMeta(), (crop_right - crop_left + 1), (crop_bottom - crop_top + 1), dstBpp()); addFrame(frame); addFrame(frameMem); VideoFrame* frame = static_cast<VideoFrame*>(frameMem->pointer()); int32_t srcFormat; CHECK(outputFormat->findInt32("color-format", &srcFormat)); Loading @@ -485,7 +502,7 @@ status_t VideoFrameDecoder::onOutputReceived( (const uint8_t *)videoFrameBuffer->data(), width, height, crop_left, crop_top, crop_right, crop_bottom, frame->mData, frame->getFlattenedData(), frame->mWidth, frame->mHeight, crop_left, crop_top, crop_right, crop_bottom); Loading Loading @@ -596,9 +613,10 @@ status_t ImageDecoder::onOutputReceived( } if (mFrame == NULL) { mFrame = allocVideoFrame(trackMeta(), imageWidth, imageHeight, dstBpp()); sp<IMemory> frameMem = allocVideoFrame(trackMeta(), imageWidth, imageHeight, dstBpp()); mFrame = static_cast<VideoFrame*>(frameMem->pointer()); addFrame(mFrame); addFrame(frameMem); } int32_t srcFormat; Loading Loading @@ -639,7 +657,7 @@ status_t ImageDecoder::onOutputReceived( (const uint8_t *)videoFrameBuffer->data(), width, height, crop_left, crop_top, crop_right, crop_bottom, mFrame->mData, mFrame->getFlattenedData(), mFrame->mWidth, mFrame->mHeight, dstLeft, dstTop, dstRight, dstBottom); Loading