/* * * Copyright 2010, 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_MEDIAPROFILES_H #define ANDROID_MEDIAPROFILES_H #include #include #include namespace android { enum camcorder_quality { CAMCORDER_QUALITY_LIST_START = 0, CAMCORDER_QUALITY_LOW = 0, CAMCORDER_QUALITY_HIGH = 1, CAMCORDER_QUALITY_QCIF = 2, CAMCORDER_QUALITY_CIF = 3, CAMCORDER_QUALITY_480P = 4, CAMCORDER_QUALITY_720P = 5, CAMCORDER_QUALITY_1080P = 6, CAMCORDER_QUALITY_QVGA = 7, CAMCORDER_QUALITY_2160P = 8, CAMCORDER_QUALITY_VGA = 9, CAMCORDER_QUALITY_4KDCI = 10, CAMCORDER_QUALITY_QHD = 11, CAMCORDER_QUALITY_2K = 12, CAMCORDER_QUALITY_8KUHD = 13, CAMCORDER_QUALITY_LIST_END = 13, CAMCORDER_QUALITY_TIME_LAPSE_LIST_START = 1000, CAMCORDER_QUALITY_TIME_LAPSE_LOW = 1000, CAMCORDER_QUALITY_TIME_LAPSE_HIGH = 1001, CAMCORDER_QUALITY_TIME_LAPSE_QCIF = 1002, CAMCORDER_QUALITY_TIME_LAPSE_CIF = 1003, CAMCORDER_QUALITY_TIME_LAPSE_480P = 1004, CAMCORDER_QUALITY_TIME_LAPSE_720P = 1005, CAMCORDER_QUALITY_TIME_LAPSE_1080P = 1006, CAMCORDER_QUALITY_TIME_LAPSE_QVGA = 1007, CAMCORDER_QUALITY_TIME_LAPSE_2160P = 1008, CAMCORDER_QUALITY_TIME_LAPSE_VGA = 1009, CAMCORDER_QUALITY_TIME_LAPSE_4KDCI = 1010, CAMCORDER_QUALITY_TIME_LAPSE_QHD = 1011, CAMCORDER_QUALITY_TIME_LAPSE_2K = 1012, CAMCORDER_QUALITY_TIME_LAPSE_8KUHD = 1013, CAMCORDER_QUALITY_TIME_LAPSE_LIST_END = 1013, CAMCORDER_QUALITY_HIGH_SPEED_LIST_START = 2000, CAMCORDER_QUALITY_HIGH_SPEED_LOW = 2000, CAMCORDER_QUALITY_HIGH_SPEED_HIGH = 2001, CAMCORDER_QUALITY_HIGH_SPEED_480P = 2002, CAMCORDER_QUALITY_HIGH_SPEED_720P = 2003, CAMCORDER_QUALITY_HIGH_SPEED_1080P = 2004, CAMCORDER_QUALITY_HIGH_SPEED_2160P = 2005, CAMCORDER_QUALITY_HIGH_SPEED_CIF = 2006, CAMCORDER_QUALITY_HIGH_SPEED_VGA = 2007, CAMCORDER_QUALITY_HIGH_SPEED_4KDCI = 2008, CAMCORDER_QUALITY_HIGH_SPEED_LIST_END = 2008, }; enum video_decoder { VIDEO_DECODER_WMV, }; enum audio_decoder { AUDIO_DECODER_WMA, }; enum chroma_subsampling { CHROMA_SUBSAMPLING_YUV_420, CHROMA_SUBSAMPLING_YUV_422, CHROMA_SUBSAMPLING_YUV_444, }; enum hdr_format { HDR_FORMAT_NONE, HDR_FORMAT_HLG, HDR_FORMAT_HDR10, HDR_FORMAT_HDR10PLUS, HDR_FORMAT_DOLBY_VISION, }; class MediaProfiles { public: /** * Returns the singleton instance for subsequence queries or NULL if error. * * If property media.settings.xml is set, getInstance() will attempt to read * from file path in media.settings.xml. Otherwise, getInstance() will * search through the list of preset XML file paths. * * If the search is unsuccessful, the default instance will be created * instead. * * TODO: After validation is added, getInstance() should handle validation * failure properly. */ static MediaProfiles* getInstance(); /** * Configuration for a video encoder. */ struct VideoCodec { public: /** * Constructs a video encoder configuration. * * @param codec codec type * @param bitrate bitrate in bps * @param frameWidth frame width in pixels * @param frameHeight frame height in pixels * @param frameRate frame rate in fps * @param profile codec profile (for MediaCodec) or -1 for none */ VideoCodec(video_encoder codec, int bitrate, int frameWidth, int frameHeight, int frameRate, int profile = -1, chroma_subsampling chroma = CHROMA_SUBSAMPLING_YUV_420, int bitDepth = 8, hdr_format hdr = HDR_FORMAT_NONE) : mCodec(codec), mBitRate(bitrate), mFrameWidth(frameWidth), mFrameHeight(frameHeight), mFrameRate(frameRate), mProfile(profile), mChromaSubsampling(chroma), mBitDepth(bitDepth), mHdrFormat(hdr) { } VideoCodec(const VideoCodec&) = default; ~VideoCodec() {} /** Returns the codec type. */ video_encoder getCodec() const { return mCodec; } /** Returns the bitrate in bps. */ int getBitrate() const { return mBitRate; } /** Returns the frame width in pixels. */ int getFrameWidth() const { return mFrameWidth; } /** Returns the frame height in pixels. */ int getFrameHeight() const { return mFrameHeight; } /** Returns the frame rate in fps. */ int getFrameRate() const { return mFrameRate; } /** Returns the codec profile (or -1 for no profile). */ int getProfile() const { return mProfile; } /** Returns the chroma subsampling. */ chroma_subsampling getChromaSubsampling() const { return mChromaSubsampling; } /** Returns the bit depth. */ int getBitDepth() const { return mBitDepth; } /** Returns the chroma subsampling. */ hdr_format getHdrFormat() const { return mHdrFormat; } private: video_encoder mCodec; int mBitRate; int mFrameWidth; int mFrameHeight; int mFrameRate; int mProfile; chroma_subsampling mChromaSubsampling; int mBitDepth; hdr_format mHdrFormat; friend class MediaProfiles; }; /** * Configuration for an audio encoder. */ struct AudioCodec { public: /** * Constructs an audio encoder configuration. * * @param codec codec type * @param bitrate bitrate in bps * @param sampleRate sample rate in Hz * @param channels number of channels * @param profile codec profile (for MediaCodec) or -1 for none */ AudioCodec(audio_encoder codec, int bitrate, int sampleRate, int channels, int profile = -1) : mCodec(codec), mBitRate(bitrate), mSampleRate(sampleRate), mChannels(channels), mProfile(profile) { } AudioCodec(const AudioCodec&) = default; ~AudioCodec() {} /** Returns the codec type. */ audio_encoder getCodec() const { return mCodec; } /** Returns the bitrate in bps. */ int getBitrate() const { return mBitRate; } /** Returns the sample rate in Hz. */ int getSampleRate() const { return mSampleRate; } /** Returns the number of channels. */ int getChannels() const { return mChannels; } /** Returns the codec profile (or -1 for no profile). */ int getProfile() const { return mProfile; } private: audio_encoder mCodec; int mBitRate; int mSampleRate; int mChannels; int mProfile; friend class MediaProfiles; }; /** * Configuration for a camcorder profile/encoder profiles object. */ struct CamcorderProfile { /** * Returns on ordered list of the video codec configurations in * decreasing preference. The returned object is only valid * during the lifetime of this object. */ std::vector getVideoCodecs() const; /** * Returns on ordered list of the audio codec configurations in * decreasing preference. The returned object is only valid * during the lifetime of this object. */ std::vector getAudioCodecs() const; /** Returns the default duration in seconds. */ int getDuration() const { return mDuration; } /** Returns the preferred file format. */ int getFileFormat() const { return mFileFormat; } CamcorderProfile(const CamcorderProfile& copy) = default; ~CamcorderProfile() = default; private: /** * Constructs an empty object with no audio/video profiles. */ CamcorderProfile() : mCameraId(0), mFileFormat(OUTPUT_FORMAT_THREE_GPP), mQuality(CAMCORDER_QUALITY_HIGH), mDuration(0) {} int mCameraId; output_format mFileFormat; camcorder_quality mQuality; int mDuration; std::vector mVideoCodecs; std::vector mAudioCodecs; friend class MediaProfiles; }; /** * Returns the CamcorderProfile object for the given camera at * the given quality level, or null if it does not exist. */ const CamcorderProfile *getCamcorderProfile( int cameraId, camcorder_quality quality) const; /** * Returns the value for the given param name for the given camera at * the given quality level, or -1 if error. * * Supported param name are: * duration - the recording duration. * file.format - output file format. see mediarecorder.h for details * vid.codec - video encoder. see mediarecorder.h for details. * aud.codec - audio encoder. see mediarecorder.h for details. * vid.width - video frame width * vid.height - video frame height * vid.fps - video frame rate * vid.bps - video bit rate * aud.bps - audio bit rate * aud.hz - audio sample rate * aud.ch - number of audio channels */ int getCamcorderProfileParamByName(const char *name, int cameraId, camcorder_quality quality) const; /** * Returns true if a profile for the given camera at the given quality exists, * or false if not. */ bool hasCamcorderProfile(int cameraId, camcorder_quality quality) const; /** * Returns the output file formats supported. */ Vector getOutputFileFormats() const; /** * Returns the video encoders supported. */ Vector getVideoEncoders() const; /** * Returns the value for the given param name for the given video encoder * returned from getVideoEncoderByIndex or -1 if error. * * Supported param name are: * enc.vid.width.min - min video frame width * enc.vid.width.max - max video frame width * enc.vid.height.min - min video frame height * enc.vid.height.max - max video frame height * enc.vid.bps.min - min bit rate in bits per second * enc.vid.bps.max - max bit rate in bits per second * enc.vid.fps.min - min frame rate in frames per second * enc.vid.fps.max - max frame rate in frames per second */ int getVideoEncoderParamByName(const char *name, video_encoder codec) const; /** * Returns the audio encoders supported. */ Vector getAudioEncoders() const; /** * Returns the value for the given param name for the given audio encoder * returned from getAudioEncoderByIndex or -1 if error. * * Supported param name are: * enc.aud.ch.min - min number of channels * enc.aud.ch.max - max number of channels * enc.aud.bps.min - min bit rate in bits per second * enc.aud.bps.max - max bit rate in bits per second * enc.aud.hz.min - min sample rate in samples per second * enc.aud.hz.max - max sample rate in samples per second */ int getAudioEncoderParamByName(const char *name, audio_encoder codec) const; /** * Returns the video decoders supported. */ Vector getVideoDecoders() const; /** * Returns the audio decoders supported. */ Vector getAudioDecoders() const; /** * Returns the number of image encoding quality levels supported. */ Vector getImageEncodingQualityLevels(int cameraId) const; /** * Returns the start time offset (in ms) for the given camera Id. * If the given camera Id does not exist, -1 will be returned. */ int getStartTimeOffsetMs(int cameraId) const; private: enum { // Camcorder profiles (high/low) and timelapse profiles (high/low) kNumRequiredProfiles = 4, }; MediaProfiles& operator=(const MediaProfiles&); // Don't call me MediaProfiles(const MediaProfiles&); // Don't call me MediaProfiles() {} // Dummy default constructor ~MediaProfiles(); // Don't delete me struct VideoEncoderCap { // Ugly constructor VideoEncoderCap(video_encoder codec, int minBitRate, int maxBitRate, int minFrameWidth, int maxFrameWidth, int minFrameHeight, int maxFrameHeight, int minFrameRate, int maxFrameRate) : mCodec(codec), mMinBitRate(minBitRate), mMaxBitRate(maxBitRate), mMinFrameWidth(minFrameWidth), mMaxFrameWidth(maxFrameWidth), mMinFrameHeight(minFrameHeight), mMaxFrameHeight(maxFrameHeight), mMinFrameRate(minFrameRate), mMaxFrameRate(maxFrameRate) {} ~VideoEncoderCap() {} video_encoder mCodec; int mMinBitRate, mMaxBitRate; int mMinFrameWidth, mMaxFrameWidth; int mMinFrameHeight, mMaxFrameHeight; int mMinFrameRate, mMaxFrameRate; }; struct AudioEncoderCap { // Ugly constructor AudioEncoderCap(audio_encoder codec, int minBitRate, int maxBitRate, int minSampleRate, int maxSampleRate, int minChannels, int maxChannels) : mCodec(codec), mMinBitRate(minBitRate), mMaxBitRate(maxBitRate), mMinSampleRate(minSampleRate), mMaxSampleRate(maxSampleRate), mMinChannels(minChannels), mMaxChannels(maxChannels) {} ~AudioEncoderCap() {} audio_encoder mCodec; int mMinBitRate, mMaxBitRate; int mMinSampleRate, mMaxSampleRate; int mMinChannels, mMaxChannels; }; struct VideoDecoderCap { VideoDecoderCap(video_decoder codec): mCodec(codec) {} ~VideoDecoderCap() {} video_decoder mCodec; }; struct AudioDecoderCap { AudioDecoderCap(audio_decoder codec): mCodec(codec) {} ~AudioDecoderCap() {} audio_decoder mCodec; }; struct NameToTagMap { const char* name; int tag; }; struct ImageEncodingQualityLevels { int mCameraId; Vector mLevels; }; int getCamcorderProfileIndex(int cameraId, camcorder_quality quality) const; void initRequiredProfileRefs(const Vector& cameraIds); int getRequiredProfileRefIndex(int cameraId); // Debug static void logVideoCodec(const VideoCodec& codec); static void logAudioCodec(const AudioCodec& codec); static void logVideoEncoderCap(const VideoEncoderCap& cap); static void logAudioEncoderCap(const AudioEncoderCap& cap); static void logVideoDecoderCap(const VideoDecoderCap& cap); static void logAudioDecoderCap(const AudioDecoderCap& cap); // Returns true if xmlFile exists. // TODO: Add runtime validation. static bool checkXmlFile(const char* xmlFile); // If the xml configuration file does exist, use the settings // from the xml static MediaProfiles* createInstanceFromXmlFile(const char *xml); static output_format createEncoderOutputFileFormat(const char **atts, size_t natts); static void createVideoCodec(const char **atts, size_t natts, MediaProfiles *profiles); static void createAudioCodec(const char **atts, size_t natts, MediaProfiles *profiles); static AudioDecoderCap* createAudioDecoderCap(const char **atts, size_t natts); static VideoDecoderCap* createVideoDecoderCap(const char **atts, size_t natts); static VideoEncoderCap* createVideoEncoderCap(const char **atts, size_t natts); static AudioEncoderCap* createAudioEncoderCap(const char **atts, size_t natts); static CamcorderProfile* createCamcorderProfile( int cameraId, const char **atts, size_t natts, Vector& cameraIds); static int getCameraId(const char **atts, size_t natts); void addStartTimeOffset(int cameraId, const char **atts, size_t natts); ImageEncodingQualityLevels* findImageEncodingQualityLevels(int cameraId) const; void addImageEncodingQualityLevel(int cameraId, const char** atts, size_t natts); // Customized element tag handler for parsing the xml configuration file. static void startElementHandler(void *userData, const char *name, const char **atts); // If the xml configuration file does not exist, use hard-coded values static MediaProfiles* createDefaultInstance(); static CamcorderProfile *createDefaultCamcorderQcifProfile(camcorder_quality quality); static CamcorderProfile *createDefaultCamcorderCifProfile(camcorder_quality quality); static void createDefaultCamcorderLowProfiles( MediaProfiles::CamcorderProfile **lowProfile, MediaProfiles::CamcorderProfile **lowSpecificProfile); static void createDefaultCamcorderHighProfiles( MediaProfiles::CamcorderProfile **highProfile, MediaProfiles::CamcorderProfile **highSpecificProfile); static CamcorderProfile *createDefaultCamcorderTimeLapseQcifProfile(camcorder_quality quality); static CamcorderProfile *createDefaultCamcorderTimeLapse480pProfile(camcorder_quality quality); static void createDefaultCamcorderTimeLapseLowProfiles( MediaProfiles::CamcorderProfile **lowTimeLapseProfile, MediaProfiles::CamcorderProfile **lowSpecificTimeLapseProfile); static void createDefaultCamcorderTimeLapseHighProfiles( MediaProfiles::CamcorderProfile **highTimeLapseProfile, MediaProfiles::CamcorderProfile **highSpecificTimeLapseProfile); static void createDefaultCamcorderProfiles(MediaProfiles *profiles); static void createDefaultVideoEncoders(MediaProfiles *profiles); static void createDefaultAudioEncoders(MediaProfiles *profiles); static void createDefaultVideoDecoders(MediaProfiles *profiles); static void createDefaultAudioDecoders(MediaProfiles *profiles); static void createDefaultEncoderOutputFileFormats(MediaProfiles *profiles); static void createDefaultImageEncodingQualityLevels(MediaProfiles *profiles); static void createDefaultImageDecodingMaxMemory(MediaProfiles *profiles); static VideoEncoderCap* createDefaultH263VideoEncoderCap(); static VideoEncoderCap* createDefaultM4vVideoEncoderCap(); static AudioEncoderCap* createDefaultAmrNBEncoderCap(); static int findTagForName(const NameToTagMap *map, size_t nMappings, const char *name); /** * Finds the string representation for an integer enum tag. * * This is the reverse for findTagForName * * @param map the name-to-tag map to search * @param nMappings the number of mappings in |map| * @param tag the enum value to find * @param def_ the return value if the enum is not found * * @return the string name corresponding to |tag| or |def_| if not found. */ static const char *findNameForTag( const NameToTagMap *map, size_t nMappings, int tag, const char *def_ = "(unknown)"); /** * Updates the chroma subsampling, bit-depth and hdr-format for * advanced codec profiles. * * @param codec the video codec type * @param profile the MediaCodec profile * @param chroma pointer to the chroma subsampling output * @param bitDepth pointer to the bit depth output * @param hdr pointer to the hdr format output * * @return true, if the profile fully determined chroma, bit-depth and hdr-format, false * otherwise. */ static bool detectAdvancedVideoProfile( video_encoder codec, int profile, chroma_subsampling *chroma, int *bitDepth, hdr_format *hdr); /** * Check on existing profiles with the following criteria: * 1. Low quality profile must have the lowest video * resolution product (width x height) * 2. High quality profile must have the highest video * resolution product (width x height) * * and add required low/high quality camcorder/timelapse * profiles if they are not found. This allows to remove * duplicate profile definitions in the media_profiles.xml * file. */ void checkAndAddRequiredProfilesIfNecessary(); // Mappings from name (for instance, codec name) to enum value static const NameToTagMap sVideoEncoderNameMap[]; static const NameToTagMap sChromaSubsamplingNameMap[]; static const NameToTagMap sHdrFormatNameMap[]; static const NameToTagMap sAudioEncoderNameMap[]; static const NameToTagMap sFileFormatMap[]; static const NameToTagMap sVideoDecoderNameMap[]; static const NameToTagMap sAudioDecoderNameMap[]; static const NameToTagMap sCamcorderQualityNameMap[]; static bool sIsInitialized; static MediaProfiles *sInstance; static Mutex sLock; int mCurrentCameraId; Vector mCamcorderProfiles; Vector mAudioEncoders; Vector mVideoEncoders; Vector mAudioDecoders; Vector mVideoDecoders; Vector mEncoderOutputFileFormats; Vector mImageEncodingQualityLevels; KeyedVector mStartTimeOffsets; typedef struct { bool mHasRefProfile; // Refers to an existing profile int mRefProfileIndex; // Reference profile index int mResolutionProduct; // width x height } RequiredProfileRefInfo; // Required low and high profiles typedef struct { RequiredProfileRefInfo mRefs[kNumRequiredProfiles]; int mCameraId; } RequiredProfiles; RequiredProfiles *mRequiredProfileRefs; Vector mCameraIds; }; }; // namespace android #endif // ANDROID_MEDIAPROFILES_H