blob: 47f92582cdb835d354f1dc9e4e1411f53861b905 [file] [log] [blame]
James Dong1d7491b2010-01-19 17:45:38 -08001/*
2**
3** Copyright 2010, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9** http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
18
19//#define LOG_NDEBUG 0
20#define LOG_TAG "MediaProfiles"
21
22#include <stdlib.h>
23#include <utils/Log.h>
24#include <utils/Vector.h>
25#include <cutils/properties.h>
Elliott Hughese8057dd2012-09-09 14:58:14 -070026#include <libexpat/expat.h>
James Dong1d7491b2010-01-19 17:45:38 -080027#include <media/MediaProfiles.h>
James Dongf1d5aa12012-02-06 23:46:37 -080028#include <media/stagefright/foundation/ADebug.h>
James Dong6c6b4d02012-03-12 14:37:53 -070029#include <OMX_Video.h>
James Dong1d7491b2010-01-19 17:45:38 -080030
31namespace android {
32
33Mutex MediaProfiles::sLock;
34bool MediaProfiles::sIsInitialized = false;
35MediaProfiles *MediaProfiles::sInstance = NULL;
36
37const MediaProfiles::NameToTagMap MediaProfiles::sVideoEncoderNameMap[] = {
38 {"h263", VIDEO_ENCODER_H263},
39 {"h264", VIDEO_ENCODER_H264},
40 {"m4v", VIDEO_ENCODER_MPEG_4_SP}
41};
42
43const MediaProfiles::NameToTagMap MediaProfiles::sAudioEncoderNameMap[] = {
Dave Burkeaeb8fd42012-04-19 00:14:27 -070044 {"amrnb", AUDIO_ENCODER_AMR_NB},
45 {"amrwb", AUDIO_ENCODER_AMR_WB},
46 {"aac", AUDIO_ENCODER_AAC},
Dave Burkef60c6602012-04-28 21:58:22 -070047 {"heaac", AUDIO_ENCODER_HE_AAC},
48 {"aaceld", AUDIO_ENCODER_AAC_ELD}
James Dong1d7491b2010-01-19 17:45:38 -080049};
50
51const MediaProfiles::NameToTagMap MediaProfiles::sFileFormatMap[] = {
52 {"3gp", OUTPUT_FORMAT_THREE_GPP},
53 {"mp4", OUTPUT_FORMAT_MPEG_4}
54};
55
56const MediaProfiles::NameToTagMap MediaProfiles::sVideoDecoderNameMap[] = {
57 {"wmv", VIDEO_DECODER_WMV}
58};
59
60const MediaProfiles::NameToTagMap MediaProfiles::sAudioDecoderNameMap[] = {
61 {"wma", AUDIO_DECODER_WMA}
62};
63
64const MediaProfiles::NameToTagMap MediaProfiles::sCamcorderQualityNameMap[] = {
Nipun Kwatrac0a84782010-09-06 15:59:02 -070065 {"low", CAMCORDER_QUALITY_LOW},
James Dong1d7491b2010-01-19 17:45:38 -080066 {"high", CAMCORDER_QUALITY_HIGH},
Nipun Kwatrac0a84782010-09-06 15:59:02 -070067 {"qcif", CAMCORDER_QUALITY_QCIF},
Nipun Kwatra9783ed82010-09-10 15:45:57 -070068 {"cif", CAMCORDER_QUALITY_CIF},
Nipun Kwatrac0a84782010-09-06 15:59:02 -070069 {"480p", CAMCORDER_QUALITY_480P},
70 {"720p", CAMCORDER_QUALITY_720P},
71 {"1080p", CAMCORDER_QUALITY_1080P},
Zhijun He5f6af1a2014-06-10 08:26:33 -070072 {"2160p", CAMCORDER_QUALITY_2160P},
James Dong669012d2011-09-19 16:27:31 -070073 {"qvga", CAMCORDER_QUALITY_QVGA},
Nipun Kwatrac0a84782010-09-06 15:59:02 -070074
75 {"timelapselow", CAMCORDER_QUALITY_TIME_LAPSE_LOW},
76 {"timelapsehigh", CAMCORDER_QUALITY_TIME_LAPSE_HIGH},
77 {"timelapseqcif", CAMCORDER_QUALITY_TIME_LAPSE_QCIF},
Nipun Kwatra9783ed82010-09-10 15:45:57 -070078 {"timelapsecif", CAMCORDER_QUALITY_TIME_LAPSE_CIF},
Nipun Kwatrac0a84782010-09-06 15:59:02 -070079 {"timelapse480p", CAMCORDER_QUALITY_TIME_LAPSE_480P},
80 {"timelapse720p", CAMCORDER_QUALITY_TIME_LAPSE_720P},
James Dong669012d2011-09-19 16:27:31 -070081 {"timelapse1080p", CAMCORDER_QUALITY_TIME_LAPSE_1080P},
Zhijun He5f6af1a2014-06-10 08:26:33 -070082 {"timelapse2160p", CAMCORDER_QUALITY_TIME_LAPSE_2160P},
James Dong669012d2011-09-19 16:27:31 -070083 {"timelapseqvga", CAMCORDER_QUALITY_TIME_LAPSE_QVGA},
Zhijun Hee0790972014-07-23 15:17:26 -070084
85 {"highspeedlow", CAMCORDER_QUALITY_HIGH_SPEED_LOW},
86 {"highspeedhigh", CAMCORDER_QUALITY_HIGH_SPEED_HIGH},
87 {"highspeed480p", CAMCORDER_QUALITY_HIGH_SPEED_480P},
88 {"highspeed720p", CAMCORDER_QUALITY_HIGH_SPEED_720P},
89 {"highspeed1080p", CAMCORDER_QUALITY_HIGH_SPEED_1080P},
Zhijun He9520aa62014-09-09 16:18:31 -070090 {"highspeed2160p", CAMCORDER_QUALITY_HIGH_SPEED_2160P},
James Dong1d7491b2010-01-19 17:45:38 -080091};
92
Glenn Kasten80520382014-01-31 16:49:31 -080093#if LOG_NDEBUG
94#define UNUSED __unused
95#else
96#define UNUSED
97#endif
98
James Dong1d7491b2010-01-19 17:45:38 -080099/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800100MediaProfiles::logVideoCodec(const MediaProfiles::VideoCodec& codec UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800101{
Steve Block3856b092011-10-20 11:56:00 +0100102 ALOGV("video codec:");
103 ALOGV("codec = %d", codec.mCodec);
104 ALOGV("bit rate: %d", codec.mBitRate);
105 ALOGV("frame width: %d", codec.mFrameWidth);
106 ALOGV("frame height: %d", codec.mFrameHeight);
107 ALOGV("frame rate: %d", codec.mFrameRate);
James Dong1d7491b2010-01-19 17:45:38 -0800108}
109
110/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800111MediaProfiles::logAudioCodec(const MediaProfiles::AudioCodec& codec UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800112{
Steve Block3856b092011-10-20 11:56:00 +0100113 ALOGV("audio codec:");
114 ALOGV("codec = %d", codec.mCodec);
115 ALOGV("bit rate: %d", codec.mBitRate);
116 ALOGV("sample rate: %d", codec.mSampleRate);
117 ALOGV("number of channels: %d", codec.mChannels);
James Dong1d7491b2010-01-19 17:45:38 -0800118}
119
120/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800121MediaProfiles::logVideoEncoderCap(const MediaProfiles::VideoEncoderCap& cap UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800122{
Steve Block3856b092011-10-20 11:56:00 +0100123 ALOGV("video encoder cap:");
124 ALOGV("codec = %d", cap.mCodec);
125 ALOGV("bit rate: min = %d and max = %d", cap.mMinBitRate, cap.mMaxBitRate);
126 ALOGV("frame width: min = %d and max = %d", cap.mMinFrameWidth, cap.mMaxFrameWidth);
127 ALOGV("frame height: min = %d and max = %d", cap.mMinFrameHeight, cap.mMaxFrameHeight);
128 ALOGV("frame rate: min = %d and max = %d", cap.mMinFrameRate, cap.mMaxFrameRate);
James Dong1d7491b2010-01-19 17:45:38 -0800129}
130
131/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800132MediaProfiles::logAudioEncoderCap(const MediaProfiles::AudioEncoderCap& cap UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800133{
Steve Block3856b092011-10-20 11:56:00 +0100134 ALOGV("audio encoder cap:");
135 ALOGV("codec = %d", cap.mCodec);
136 ALOGV("bit rate: min = %d and max = %d", cap.mMinBitRate, cap.mMaxBitRate);
137 ALOGV("sample rate: min = %d and max = %d", cap.mMinSampleRate, cap.mMaxSampleRate);
138 ALOGV("number of channels: min = %d and max = %d", cap.mMinChannels, cap.mMaxChannels);
James Dong1d7491b2010-01-19 17:45:38 -0800139}
140
141/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800142MediaProfiles::logVideoDecoderCap(const MediaProfiles::VideoDecoderCap& cap UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800143{
Steve Block3856b092011-10-20 11:56:00 +0100144 ALOGV("video decoder cap:");
145 ALOGV("codec = %d", cap.mCodec);
James Dong1d7491b2010-01-19 17:45:38 -0800146}
147
148/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800149MediaProfiles::logAudioDecoderCap(const MediaProfiles::AudioDecoderCap& cap UNUSED)
James Dong1d7491b2010-01-19 17:45:38 -0800150{
Steve Block3856b092011-10-20 11:56:00 +0100151 ALOGV("audio codec cap:");
152 ALOGV("codec = %d", cap.mCodec);
James Dong1d7491b2010-01-19 17:45:38 -0800153}
154
Hong Tengcabd5f82011-07-06 18:33:09 -0700155/*static*/ void
Glenn Kasten80520382014-01-31 16:49:31 -0800156MediaProfiles::logVideoEditorCap(const MediaProfiles::VideoEditorCap& cap UNUSED)
Hong Tengcabd5f82011-07-06 18:33:09 -0700157{
Steve Block3856b092011-10-20 11:56:00 +0100158 ALOGV("videoeditor cap:");
159 ALOGV("mMaxInputFrameWidth = %d", cap.mMaxInputFrameWidth);
160 ALOGV("mMaxInputFrameHeight = %d", cap.mMaxInputFrameHeight);
161 ALOGV("mMaxOutputFrameWidth = %d", cap.mMaxOutputFrameWidth);
162 ALOGV("mMaxOutputFrameHeight = %d", cap.mMaxOutputFrameHeight);
Hong Tengcabd5f82011-07-06 18:33:09 -0700163}
164
James Dong1d7491b2010-01-19 17:45:38 -0800165/*static*/ int
Glenn Kastenb187de12014-12-30 08:18:15 -0800166MediaProfiles::findTagForName(const MediaProfiles::NameToTagMap *map, size_t nMappings,
167 const char *name)
James Dong1d7491b2010-01-19 17:45:38 -0800168{
169 int tag = -1;
170 for (size_t i = 0; i < nMappings; ++i) {
171 if (!strcmp(map[i].name, name)) {
172 tag = map[i].tag;
173 break;
174 }
175 }
176 return tag;
177}
178
179/*static*/ MediaProfiles::VideoCodec*
180MediaProfiles::createVideoCodec(const char **atts, MediaProfiles *profiles)
181{
182 CHECK(!strcmp("codec", atts[0]) &&
183 !strcmp("bitRate", atts[2]) &&
184 !strcmp("width", atts[4]) &&
185 !strcmp("height", atts[6]) &&
186 !strcmp("frameRate", atts[8]));
187
188 const size_t nMappings = sizeof(sVideoEncoderNameMap)/sizeof(sVideoEncoderNameMap[0]);
189 const int codec = findTagForName(sVideoEncoderNameMap, nMappings, atts[1]);
190 CHECK(codec != -1);
191
192 MediaProfiles::VideoCodec *videoCodec =
193 new MediaProfiles::VideoCodec(static_cast<video_encoder>(codec),
194 atoi(atts[3]), atoi(atts[5]), atoi(atts[7]), atoi(atts[9]));
195 logVideoCodec(*videoCodec);
196
197 size_t nCamcorderProfiles;
198 CHECK((nCamcorderProfiles = profiles->mCamcorderProfiles.size()) >= 1);
199 profiles->mCamcorderProfiles[nCamcorderProfiles - 1]->mVideoCodec = videoCodec;
200 return videoCodec;
201}
202
203/*static*/ MediaProfiles::AudioCodec*
204MediaProfiles::createAudioCodec(const char **atts, MediaProfiles *profiles)
205{
206 CHECK(!strcmp("codec", atts[0]) &&
207 !strcmp("bitRate", atts[2]) &&
208 !strcmp("sampleRate", atts[4]) &&
209 !strcmp("channels", atts[6]));
210 const size_t nMappings = sizeof(sAudioEncoderNameMap)/sizeof(sAudioEncoderNameMap[0]);
211 const int codec = findTagForName(sAudioEncoderNameMap, nMappings, atts[1]);
212 CHECK(codec != -1);
213
214 MediaProfiles::AudioCodec *audioCodec =
215 new MediaProfiles::AudioCodec(static_cast<audio_encoder>(codec),
216 atoi(atts[3]), atoi(atts[5]), atoi(atts[7]));
217 logAudioCodec(*audioCodec);
218
219 size_t nCamcorderProfiles;
220 CHECK((nCamcorderProfiles = profiles->mCamcorderProfiles.size()) >= 1);
221 profiles->mCamcorderProfiles[nCamcorderProfiles - 1]->mAudioCodec = audioCodec;
222 return audioCodec;
223}
224/*static*/ MediaProfiles::AudioDecoderCap*
225MediaProfiles::createAudioDecoderCap(const char **atts)
226{
227 CHECK(!strcmp("name", atts[0]) &&
228 !strcmp("enabled", atts[2]));
229
230 const size_t nMappings = sizeof(sAudioDecoderNameMap)/sizeof(sAudioDecoderNameMap[0]);
231 const int codec = findTagForName(sAudioDecoderNameMap, nMappings, atts[1]);
232 CHECK(codec != -1);
233
234 MediaProfiles::AudioDecoderCap *cap =
235 new MediaProfiles::AudioDecoderCap(static_cast<audio_decoder>(codec));
236 logAudioDecoderCap(*cap);
237 return cap;
238}
239
240/*static*/ MediaProfiles::VideoDecoderCap*
241MediaProfiles::createVideoDecoderCap(const char **atts)
242{
243 CHECK(!strcmp("name", atts[0]) &&
244 !strcmp("enabled", atts[2]));
245
246 const size_t nMappings = sizeof(sVideoDecoderNameMap)/sizeof(sVideoDecoderNameMap[0]);
247 const int codec = findTagForName(sVideoDecoderNameMap, nMappings, atts[1]);
248 CHECK(codec != -1);
249
250 MediaProfiles::VideoDecoderCap *cap =
251 new MediaProfiles::VideoDecoderCap(static_cast<video_decoder>(codec));
252 logVideoDecoderCap(*cap);
253 return cap;
254}
255
256/*static*/ MediaProfiles::VideoEncoderCap*
257MediaProfiles::createVideoEncoderCap(const char **atts)
258{
259 CHECK(!strcmp("name", atts[0]) &&
260 !strcmp("enabled", atts[2]) &&
261 !strcmp("minBitRate", atts[4]) &&
262 !strcmp("maxBitRate", atts[6]) &&
263 !strcmp("minFrameWidth", atts[8]) &&
264 !strcmp("maxFrameWidth", atts[10]) &&
265 !strcmp("minFrameHeight", atts[12]) &&
266 !strcmp("maxFrameHeight", atts[14]) &&
267 !strcmp("minFrameRate", atts[16]) &&
268 !strcmp("maxFrameRate", atts[18]));
269
270 const size_t nMappings = sizeof(sVideoEncoderNameMap)/sizeof(sVideoEncoderNameMap[0]);
271 const int codec = findTagForName(sVideoEncoderNameMap, nMappings, atts[1]);
272 CHECK(codec != -1);
273
274 MediaProfiles::VideoEncoderCap *cap =
275 new MediaProfiles::VideoEncoderCap(static_cast<video_encoder>(codec),
276 atoi(atts[5]), atoi(atts[7]), atoi(atts[9]), atoi(atts[11]), atoi(atts[13]),
277 atoi(atts[15]), atoi(atts[17]), atoi(atts[19]));
278 logVideoEncoderCap(*cap);
279 return cap;
280}
281
282/*static*/ MediaProfiles::AudioEncoderCap*
283MediaProfiles::createAudioEncoderCap(const char **atts)
284{
285 CHECK(!strcmp("name", atts[0]) &&
286 !strcmp("enabled", atts[2]) &&
287 !strcmp("minBitRate", atts[4]) &&
288 !strcmp("maxBitRate", atts[6]) &&
289 !strcmp("minSampleRate", atts[8]) &&
290 !strcmp("maxSampleRate", atts[10]) &&
291 !strcmp("minChannels", atts[12]) &&
292 !strcmp("maxChannels", atts[14]));
293
294 const size_t nMappings = sizeof(sAudioEncoderNameMap)/sizeof(sAudioEncoderNameMap[0]);
295 const int codec = findTagForName(sAudioEncoderNameMap, nMappings, atts[1]);
296 CHECK(codec != -1);
297
298 MediaProfiles::AudioEncoderCap *cap =
Glenn Kastenb187de12014-12-30 08:18:15 -0800299 new MediaProfiles::AudioEncoderCap(static_cast<audio_encoder>(codec), atoi(atts[5]),
300 atoi(atts[7]), atoi(atts[9]), atoi(atts[11]), atoi(atts[13]), atoi(atts[15]));
James Dong1d7491b2010-01-19 17:45:38 -0800301 logAudioEncoderCap(*cap);
302 return cap;
303}
304
305/*static*/ output_format
306MediaProfiles::createEncoderOutputFileFormat(const char **atts)
307{
308 CHECK(!strcmp("name", atts[0]));
309
310 const size_t nMappings =sizeof(sFileFormatMap)/sizeof(sFileFormatMap[0]);
311 const int format = findTagForName(sFileFormatMap, nMappings, atts[1]);
312 CHECK(format != -1);
313
314 return static_cast<output_format>(format);
315}
316
James Dong2a7e0a12011-02-28 21:07:39 -0800317static bool isCameraIdFound(int cameraId, const Vector<int>& cameraIds) {
318 for (int i = 0, n = cameraIds.size(); i < n; ++i) {
319 if (cameraId == cameraIds[i]) {
320 return true;
321 }
322 }
323 return false;
324}
325
James Dong1d7491b2010-01-19 17:45:38 -0800326/*static*/ MediaProfiles::CamcorderProfile*
James Dong2a7e0a12011-02-28 21:07:39 -0800327MediaProfiles::createCamcorderProfile(int cameraId, const char **atts, Vector<int>& cameraIds)
James Dong1d7491b2010-01-19 17:45:38 -0800328{
329 CHECK(!strcmp("quality", atts[0]) &&
330 !strcmp("fileFormat", atts[2]) &&
331 !strcmp("duration", atts[4]));
332
Glenn Kastenb187de12014-12-30 08:18:15 -0800333 const size_t nProfileMappings = sizeof(sCamcorderQualityNameMap)/
334 sizeof(sCamcorderQualityNameMap[0]);
James Dong1d7491b2010-01-19 17:45:38 -0800335 const int quality = findTagForName(sCamcorderQualityNameMap, nProfileMappings, atts[1]);
336 CHECK(quality != -1);
337
338 const size_t nFormatMappings = sizeof(sFileFormatMap)/sizeof(sFileFormatMap[0]);
339 const int fileFormat = findTagForName(sFileFormatMap, nFormatMappings, atts[3]);
340 CHECK(fileFormat != -1);
341
342 MediaProfiles::CamcorderProfile *profile = new MediaProfiles::CamcorderProfile;
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800343 profile->mCameraId = cameraId;
James Dong2a7e0a12011-02-28 21:07:39 -0800344 if (!isCameraIdFound(cameraId, cameraIds)) {
345 cameraIds.add(cameraId);
346 }
James Dong1d7491b2010-01-19 17:45:38 -0800347 profile->mFileFormat = static_cast<output_format>(fileFormat);
348 profile->mQuality = static_cast<camcorder_quality>(quality);
349 profile->mDuration = atoi(atts[5]);
350 return profile;
351}
352
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800353MediaProfiles::ImageEncodingQualityLevels*
354MediaProfiles::findImageEncodingQualityLevels(int cameraId) const
355{
356 int n = mImageEncodingQualityLevels.size();
357 for (int i = 0; i < n; i++) {
358 ImageEncodingQualityLevels *levels = mImageEncodingQualityLevels[i];
359 if (levels->mCameraId == cameraId) {
360 return levels;
361 }
362 }
363 return NULL;
364}
365
366void MediaProfiles::addImageEncodingQualityLevel(int cameraId, const char** atts)
James Dongf5a83852010-02-23 17:21:44 -0800367{
368 CHECK(!strcmp("quality", atts[0]));
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800369 int quality = atoi(atts[1]);
Glenn Kasten90bebef2012-01-27 15:24:38 -0800370 ALOGV("%s: cameraId=%d, quality=%d", __func__, cameraId, quality);
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800371 ImageEncodingQualityLevels *levels = findImageEncodingQualityLevels(cameraId);
372
373 if (levels == NULL) {
374 levels = new ImageEncodingQualityLevels();
375 levels->mCameraId = cameraId;
376 mImageEncodingQualityLevels.add(levels);
377 }
378
379 levels->mLevels.add(quality);
380}
381
382/*static*/ int
383MediaProfiles::getCameraId(const char** atts)
384{
385 if (!atts[0]) return 0; // default cameraId = 0
386 CHECK(!strcmp("cameraId", atts[0]));
James Dongf5a83852010-02-23 17:21:44 -0800387 return atoi(atts[1]);
388}
389
James Dong0f056292011-05-09 18:49:31 -0700390void MediaProfiles::addStartTimeOffset(int cameraId, const char** atts)
391{
Eric Laurentb1eb1a02012-10-22 17:44:24 -0700392 int offsetTimeMs = 1000;
James Dong0f056292011-05-09 18:49:31 -0700393 if (atts[2]) {
394 CHECK(!strcmp("startOffsetMs", atts[2]));
395 offsetTimeMs = atoi(atts[3]);
396 }
397
Steve Block3856b092011-10-20 11:56:00 +0100398 ALOGV("%s: cameraId=%d, offset=%d ms", __func__, cameraId, offsetTimeMs);
James Dong0f056292011-05-09 18:49:31 -0700399 mStartTimeOffsets.replaceValueFor(cameraId, offsetTimeMs);
400}
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -0700401/*static*/ MediaProfiles::ExportVideoProfile*
402MediaProfiles::createExportVideoProfile(const char **atts)
403{
404 CHECK(!strcmp("name", atts[0]) &&
405 !strcmp("profile", atts[2]) &&
406 !strcmp("level", atts[4]));
James Dong0f056292011-05-09 18:49:31 -0700407
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -0700408 const size_t nMappings =
409 sizeof(sVideoEncoderNameMap)/sizeof(sVideoEncoderNameMap[0]);
410 const int codec = findTagForName(sVideoEncoderNameMap, nMappings, atts[1]);
411 CHECK(codec != -1);
412
413 MediaProfiles::ExportVideoProfile *profile =
414 new MediaProfiles::ExportVideoProfile(
415 codec, atoi(atts[3]), atoi(atts[5]));
416
417 return profile;
418}
Hong Tengcabd5f82011-07-06 18:33:09 -0700419/*static*/ MediaProfiles::VideoEditorCap*
420MediaProfiles::createVideoEditorCap(const char **atts, MediaProfiles *profiles)
421{
422 CHECK(!strcmp("maxInputFrameWidth", atts[0]) &&
423 !strcmp("maxInputFrameHeight", atts[2]) &&
424 !strcmp("maxOutputFrameWidth", atts[4]) &&
Hong Teng3a9cefe2011-11-10 14:54:26 -0800425 !strcmp("maxOutputFrameHeight", atts[6]) &&
426 !strcmp("maxPrefetchYUVFrames", atts[8]));
Hong Tengcabd5f82011-07-06 18:33:09 -0700427
428 MediaProfiles::VideoEditorCap *pVideoEditorCap =
429 new MediaProfiles::VideoEditorCap(atoi(atts[1]), atoi(atts[3]),
Hong Teng3a9cefe2011-11-10 14:54:26 -0800430 atoi(atts[5]), atoi(atts[7]), atoi(atts[9]));
Hong Tengcabd5f82011-07-06 18:33:09 -0700431
432 logVideoEditorCap(*pVideoEditorCap);
433 profiles->mVideoEditorCap = pVideoEditorCap;
434
435 return pVideoEditorCap;
436}
437
James Dong1d7491b2010-01-19 17:45:38 -0800438/*static*/ void
439MediaProfiles::startElementHandler(void *userData, const char *name, const char **atts)
440{
441 MediaProfiles *profiles = (MediaProfiles *) userData;
442 if (strcmp("Video", name) == 0) {
443 createVideoCodec(atts, profiles);
444 } else if (strcmp("Audio", name) == 0) {
445 createAudioCodec(atts, profiles);
446 } else if (strcmp("VideoEncoderCap", name) == 0 &&
447 strcmp("true", atts[3]) == 0) {
448 profiles->mVideoEncoders.add(createVideoEncoderCap(atts));
449 } else if (strcmp("AudioEncoderCap", name) == 0 &&
450 strcmp("true", atts[3]) == 0) {
451 profiles->mAudioEncoders.add(createAudioEncoderCap(atts));
452 } else if (strcmp("VideoDecoderCap", name) == 0 &&
453 strcmp("true", atts[3]) == 0) {
454 profiles->mVideoDecoders.add(createVideoDecoderCap(atts));
455 } else if (strcmp("AudioDecoderCap", name) == 0 &&
456 strcmp("true", atts[3]) == 0) {
457 profiles->mAudioDecoders.add(createAudioDecoderCap(atts));
458 } else if (strcmp("EncoderOutputFileFormat", name) == 0) {
459 profiles->mEncoderOutputFileFormats.add(createEncoderOutputFileFormat(atts));
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800460 } else if (strcmp("CamcorderProfiles", name) == 0) {
461 profiles->mCurrentCameraId = getCameraId(atts);
James Dong0f056292011-05-09 18:49:31 -0700462 profiles->addStartTimeOffset(profiles->mCurrentCameraId, atts);
James Dong1d7491b2010-01-19 17:45:38 -0800463 } else if (strcmp("EncoderProfile", name) == 0) {
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800464 profiles->mCamcorderProfiles.add(
James Dong2a7e0a12011-02-28 21:07:39 -0800465 createCamcorderProfile(profiles->mCurrentCameraId, atts, profiles->mCameraIds));
James Dongf5a83852010-02-23 17:21:44 -0800466 } else if (strcmp("ImageEncoding", name) == 0) {
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800467 profiles->addImageEncodingQualityLevel(profiles->mCurrentCameraId, atts);
Hong Tengcabd5f82011-07-06 18:33:09 -0700468 } else if (strcmp("VideoEditorCap", name) == 0) {
469 createVideoEditorCap(atts, profiles);
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -0700470 } else if (strcmp("ExportVideoProfile", name) == 0) {
471 profiles->mVideoEditorExportProfiles.add(createExportVideoProfile(atts));
James Dong1d7491b2010-01-19 17:45:38 -0800472 }
473}
474
James Dong2a7e0a12011-02-28 21:07:39 -0800475static bool isCamcorderProfile(camcorder_quality quality) {
476 return quality >= CAMCORDER_QUALITY_LIST_START &&
477 quality <= CAMCORDER_QUALITY_LIST_END;
478}
479
480static bool isTimelapseProfile(camcorder_quality quality) {
481 return quality >= CAMCORDER_QUALITY_TIME_LAPSE_LIST_START &&
482 quality <= CAMCORDER_QUALITY_TIME_LAPSE_LIST_END;
483}
484
Zhijun Hee0790972014-07-23 15:17:26 -0700485static bool isHighSpeedProfile(camcorder_quality quality) {
486 return quality >= CAMCORDER_QUALITY_HIGH_SPEED_LIST_START &&
487 quality <= CAMCORDER_QUALITY_HIGH_SPEED_LIST_END;
488}
489
James Dong2a7e0a12011-02-28 21:07:39 -0800490void MediaProfiles::initRequiredProfileRefs(const Vector<int>& cameraIds) {
Mark Salyzyn34fb2962014-06-18 16:30:56 -0700491 ALOGV("Number of camera ids: %zu", cameraIds.size());
James Dong2a7e0a12011-02-28 21:07:39 -0800492 CHECK(cameraIds.size() > 0);
493 mRequiredProfileRefs = new RequiredProfiles[cameraIds.size()];
494 for (size_t i = 0, n = cameraIds.size(); i < n; ++i) {
495 mRequiredProfileRefs[i].mCameraId = cameraIds[i];
496 for (size_t j = 0; j < kNumRequiredProfiles; ++j) {
497 mRequiredProfileRefs[i].mRefs[j].mHasRefProfile = false;
498 mRequiredProfileRefs[i].mRefs[j].mRefProfileIndex = -1;
499 if ((j & 1) == 0) { // low resolution
500 mRequiredProfileRefs[i].mRefs[j].mResolutionProduct = 0x7FFFFFFF;
501 } else { // high resolution
502 mRequiredProfileRefs[i].mRefs[j].mResolutionProduct = 0;
503 }
504 }
505 }
506}
507
508int MediaProfiles::getRequiredProfileRefIndex(int cameraId) {
509 for (size_t i = 0, n = mCameraIds.size(); i < n; ++i) {
510 if (mCameraIds[i] == cameraId) {
511 return i;
512 }
513 }
514 return -1;
515}
516
517void MediaProfiles::checkAndAddRequiredProfilesIfNecessary() {
518 if (sIsInitialized) {
519 return;
520 }
521
522 initRequiredProfileRefs(mCameraIds);
523
524 for (size_t i = 0, n = mCamcorderProfiles.size(); i < n; ++i) {
525 int product = mCamcorderProfiles[i]->mVideoCodec->mFrameWidth *
526 mCamcorderProfiles[i]->mVideoCodec->mFrameHeight;
527
528 camcorder_quality quality = mCamcorderProfiles[i]->mQuality;
529 int cameraId = mCamcorderProfiles[i]->mCameraId;
530 int index = -1;
531 int refIndex = getRequiredProfileRefIndex(cameraId);
532 CHECK(refIndex != -1);
533 RequiredProfileRefInfo *info;
534 camcorder_quality refQuality;
535 VideoCodec *codec = NULL;
536
Zhijun Hee0790972014-07-23 15:17:26 -0700537 // Check high and low from either camcorder profile, timelapse profile
538 // or high speed profile, but not all of them. Default, check camcorder profile
James Dong2a7e0a12011-02-28 21:07:39 -0800539 size_t j = 0;
Bernhard Rosenkraenzer3c8889e2012-03-29 11:38:59 +0200540 size_t o = 2;
James Dong2a7e0a12011-02-28 21:07:39 -0800541 if (isTimelapseProfile(quality)) {
542 // Check timelapse profile instead.
543 j = 2;
Bernhard Rosenkraenzer3c8889e2012-03-29 11:38:59 +0200544 o = kNumRequiredProfiles;
Zhijun Hee0790972014-07-23 15:17:26 -0700545 } else if (isHighSpeedProfile(quality)) {
546 // Skip the check for high speed profile.
547 continue;
James Dong2a7e0a12011-02-28 21:07:39 -0800548 } else {
549 // Must be camcorder profile.
550 CHECK(isCamcorderProfile(quality));
551 }
Bernhard Rosenkraenzer3c8889e2012-03-29 11:38:59 +0200552 for (; j < o; ++j) {
James Dong2a7e0a12011-02-28 21:07:39 -0800553 info = &(mRequiredProfileRefs[refIndex].mRefs[j]);
554 if ((j % 2 == 0 && product > info->mResolutionProduct) || // low
555 (j % 2 != 0 && product < info->mResolutionProduct)) { // high
556 continue;
557 }
558 switch (j) {
559 case 0:
560 refQuality = CAMCORDER_QUALITY_LOW;
561 break;
562 case 1:
563 refQuality = CAMCORDER_QUALITY_HIGH;
564 break;
565 case 2:
566 refQuality = CAMCORDER_QUALITY_TIME_LAPSE_LOW;
567 break;
568 case 3:
569 refQuality = CAMCORDER_QUALITY_TIME_LAPSE_HIGH;
570 break;
571 default:
572 CHECK(!"Should never reach here");
573 }
574
575 if (!info->mHasRefProfile) {
576 index = getCamcorderProfileIndex(cameraId, refQuality);
577 }
578 if (index == -1) {
579 // New high or low quality profile is found.
580 // Update its reference.
581 info->mHasRefProfile = true;
582 info->mRefProfileIndex = i;
583 info->mResolutionProduct = product;
584 }
585 }
586 }
587
588 for (size_t cameraId = 0; cameraId < mCameraIds.size(); ++cameraId) {
589 for (size_t j = 0; j < kNumRequiredProfiles; ++j) {
590 int refIndex = getRequiredProfileRefIndex(cameraId);
591 CHECK(refIndex != -1);
592 RequiredProfileRefInfo *info =
593 &mRequiredProfileRefs[refIndex].mRefs[j];
594
595 if (info->mHasRefProfile) {
596
597 CamcorderProfile *profile =
598 new CamcorderProfile(
599 *mCamcorderProfiles[info->mRefProfileIndex]);
600
601 // Overwrite the quality
602 switch (j % kNumRequiredProfiles) {
603 case 0:
604 profile->mQuality = CAMCORDER_QUALITY_LOW;
605 break;
606 case 1:
607 profile->mQuality = CAMCORDER_QUALITY_HIGH;
608 break;
609 case 2:
610 profile->mQuality = CAMCORDER_QUALITY_TIME_LAPSE_LOW;
611 break;
612 case 3:
613 profile->mQuality = CAMCORDER_QUALITY_TIME_LAPSE_HIGH;
614 break;
615 default:
616 CHECK(!"Should never come here");
617 }
618
619 int index = getCamcorderProfileIndex(cameraId, profile->mQuality);
620 if (index != -1) {
Mark Salyzyn34fb2962014-06-18 16:30:56 -0700621 ALOGV("Profile quality %d for camera %zu already exists",
James Dong2a7e0a12011-02-28 21:07:39 -0800622 profile->mQuality, cameraId);
623 CHECK(index == refIndex);
624 continue;
625 }
626
627 // Insert the new profile
Mark Salyzyn34fb2962014-06-18 16:30:56 -0700628 ALOGV("Add a profile: quality %d=>%d for camera %zu",
James Dong2a7e0a12011-02-28 21:07:39 -0800629 mCamcorderProfiles[info->mRefProfileIndex]->mQuality,
630 profile->mQuality, cameraId);
631
632 mCamcorderProfiles.add(profile);
633 }
634 }
635 }
636}
637
James Dong1d7491b2010-01-19 17:45:38 -0800638/*static*/ MediaProfiles*
639MediaProfiles::getInstance()
640{
Steve Block3856b092011-10-20 11:56:00 +0100641 ALOGV("getInstance");
James Dong1d7491b2010-01-19 17:45:38 -0800642 Mutex::Autolock lock(sLock);
643 if (!sIsInitialized) {
644 char value[PROPERTY_VALUE_MAX];
645 if (property_get("media.settings.xml", value, NULL) <= 0) {
646 const char *defaultXmlFile = "/etc/media_profiles.xml";
647 FILE *fp = fopen(defaultXmlFile, "r");
648 if (fp == NULL) {
Steve Block5ff1dd52012-01-05 23:22:43 +0000649 ALOGW("could not find media config xml file");
James Dong1d7491b2010-01-19 17:45:38 -0800650 sInstance = createDefaultInstance();
651 } else {
652 fclose(fp); // close the file first.
653 sInstance = createInstanceFromXmlFile(defaultXmlFile);
654 }
655 } else {
656 sInstance = createInstanceFromXmlFile(value);
657 }
James Dong2a7e0a12011-02-28 21:07:39 -0800658 CHECK(sInstance != NULL);
659 sInstance->checkAndAddRequiredProfilesIfNecessary();
660 sIsInitialized = true;
James Dong1d7491b2010-01-19 17:45:38 -0800661 }
662
663 return sInstance;
664}
665
666/*static*/ MediaProfiles::VideoEncoderCap*
667MediaProfiles::createDefaultH263VideoEncoderCap()
668{
669 return new MediaProfiles::VideoEncoderCap(
670 VIDEO_ENCODER_H263, 192000, 420000, 176, 352, 144, 288, 1, 20);
671}
672
673/*static*/ MediaProfiles::VideoEncoderCap*
674MediaProfiles::createDefaultM4vVideoEncoderCap()
675{
676 return new MediaProfiles::VideoEncoderCap(
677 VIDEO_ENCODER_MPEG_4_SP, 192000, 420000, 176, 352, 144, 288, 1, 20);
678}
679
680
681/*static*/ void
682MediaProfiles::createDefaultVideoEncoders(MediaProfiles *profiles)
683{
684 profiles->mVideoEncoders.add(createDefaultH263VideoEncoderCap());
685 profiles->mVideoEncoders.add(createDefaultM4vVideoEncoderCap());
686}
687
688/*static*/ MediaProfiles::CamcorderProfile*
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700689MediaProfiles::createDefaultCamcorderTimeLapseQcifProfile(camcorder_quality quality)
Nipun Kwatrac0a84782010-09-06 15:59:02 -0700690{
691 MediaProfiles::VideoCodec *videoCodec =
692 new MediaProfiles::VideoCodec(VIDEO_ENCODER_H263, 1000000, 176, 144, 20);
693
694 AudioCodec *audioCodec = new AudioCodec(AUDIO_ENCODER_AMR_NB, 12200, 8000, 1);
695 CamcorderProfile *profile = new MediaProfiles::CamcorderProfile;
696 profile->mCameraId = 0;
697 profile->mFileFormat = OUTPUT_FORMAT_THREE_GPP;
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700698 profile->mQuality = quality;
Nipun Kwatrac0a84782010-09-06 15:59:02 -0700699 profile->mDuration = 60;
700 profile->mVideoCodec = videoCodec;
701 profile->mAudioCodec = audioCodec;
702 return profile;
703}
704
705/*static*/ MediaProfiles::CamcorderProfile*
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700706MediaProfiles::createDefaultCamcorderTimeLapse480pProfile(camcorder_quality quality)
James Dong1d7491b2010-01-19 17:45:38 -0800707{
708 MediaProfiles::VideoCodec *videoCodec =
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700709 new MediaProfiles::VideoCodec(VIDEO_ENCODER_H263, 20000000, 720, 480, 20);
James Dong1d7491b2010-01-19 17:45:38 -0800710
711 AudioCodec *audioCodec = new AudioCodec(AUDIO_ENCODER_AMR_NB, 12200, 8000, 1);
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800712 CamcorderProfile *profile = new MediaProfiles::CamcorderProfile;
713 profile->mCameraId = 0;
James Dong1d7491b2010-01-19 17:45:38 -0800714 profile->mFileFormat = OUTPUT_FORMAT_THREE_GPP;
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700715 profile->mQuality = quality;
James Dong1d7491b2010-01-19 17:45:38 -0800716 profile->mDuration = 60;
717 profile->mVideoCodec = videoCodec;
718 profile->mAudioCodec = audioCodec;
719 return profile;
720}
721
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700722/*static*/ void
723MediaProfiles::createDefaultCamcorderTimeLapseLowProfiles(
724 MediaProfiles::CamcorderProfile **lowTimeLapseProfile,
725 MediaProfiles::CamcorderProfile **lowSpecificTimeLapseProfile) {
Glenn Kastenb187de12014-12-30 08:18:15 -0800726 *lowTimeLapseProfile = createDefaultCamcorderTimeLapseQcifProfile(
727 CAMCORDER_QUALITY_TIME_LAPSE_LOW);
728 *lowSpecificTimeLapseProfile = createDefaultCamcorderTimeLapseQcifProfile(
729 CAMCORDER_QUALITY_TIME_LAPSE_QCIF);
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700730}
731
732/*static*/ void
733MediaProfiles::createDefaultCamcorderTimeLapseHighProfiles(
734 MediaProfiles::CamcorderProfile **highTimeLapseProfile,
735 MediaProfiles::CamcorderProfile **highSpecificTimeLapseProfile) {
Glenn Kastenb187de12014-12-30 08:18:15 -0800736 *highTimeLapseProfile = createDefaultCamcorderTimeLapse480pProfile(
737 CAMCORDER_QUALITY_TIME_LAPSE_HIGH);
738 *highSpecificTimeLapseProfile = createDefaultCamcorderTimeLapse480pProfile(
739 CAMCORDER_QUALITY_TIME_LAPSE_480P);
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700740}
741
James Dong1d7491b2010-01-19 17:45:38 -0800742/*static*/ MediaProfiles::CamcorderProfile*
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700743MediaProfiles::createDefaultCamcorderQcifProfile(camcorder_quality quality)
James Dong1d7491b2010-01-19 17:45:38 -0800744{
745 MediaProfiles::VideoCodec *videoCodec =
746 new MediaProfiles::VideoCodec(VIDEO_ENCODER_H263, 192000, 176, 144, 20);
747
748 MediaProfiles::AudioCodec *audioCodec =
749 new MediaProfiles::AudioCodec(AUDIO_ENCODER_AMR_NB, 12200, 8000, 1);
750
751 MediaProfiles::CamcorderProfile *profile = new MediaProfiles::CamcorderProfile;
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800752 profile->mCameraId = 0;
James Dong1d7491b2010-01-19 17:45:38 -0800753 profile->mFileFormat = OUTPUT_FORMAT_THREE_GPP;
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700754 profile->mQuality = quality;
James Dong1d7491b2010-01-19 17:45:38 -0800755 profile->mDuration = 30;
756 profile->mVideoCodec = videoCodec;
757 profile->mAudioCodec = audioCodec;
758 return profile;
759}
760
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700761/*static*/ MediaProfiles::CamcorderProfile*
762MediaProfiles::createDefaultCamcorderCifProfile(camcorder_quality quality)
763{
764 MediaProfiles::VideoCodec *videoCodec =
765 new MediaProfiles::VideoCodec(VIDEO_ENCODER_H263, 360000, 352, 288, 20);
766
767 AudioCodec *audioCodec = new AudioCodec(AUDIO_ENCODER_AMR_NB, 12200, 8000, 1);
768 CamcorderProfile *profile = new MediaProfiles::CamcorderProfile;
769 profile->mCameraId = 0;
770 profile->mFileFormat = OUTPUT_FORMAT_THREE_GPP;
771 profile->mQuality = quality;
772 profile->mDuration = 60;
773 profile->mVideoCodec = videoCodec;
774 profile->mAudioCodec = audioCodec;
775 return profile;
776}
777
778/*static*/ void
779MediaProfiles::createDefaultCamcorderLowProfiles(
780 MediaProfiles::CamcorderProfile **lowProfile,
781 MediaProfiles::CamcorderProfile **lowSpecificProfile) {
782 *lowProfile = createDefaultCamcorderQcifProfile(CAMCORDER_QUALITY_LOW);
783 *lowSpecificProfile = createDefaultCamcorderQcifProfile(CAMCORDER_QUALITY_QCIF);
784}
785
786/*static*/ void
787MediaProfiles::createDefaultCamcorderHighProfiles(
788 MediaProfiles::CamcorderProfile **highProfile,
789 MediaProfiles::CamcorderProfile **highSpecificProfile) {
790 *highProfile = createDefaultCamcorderCifProfile(CAMCORDER_QUALITY_HIGH);
791 *highSpecificProfile = createDefaultCamcorderCifProfile(CAMCORDER_QUALITY_CIF);
792}
793
James Dong1d7491b2010-01-19 17:45:38 -0800794/*static*/ void
795MediaProfiles::createDefaultCamcorderProfiles(MediaProfiles *profiles)
796{
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700797 // low camcorder profiles.
798 MediaProfiles::CamcorderProfile *lowProfile, *lowSpecificProfile;
799 createDefaultCamcorderLowProfiles(&lowProfile, &lowSpecificProfile);
800 profiles->mCamcorderProfiles.add(lowProfile);
801 profiles->mCamcorderProfiles.add(lowSpecificProfile);
802
803 // high camcorder profiles.
804 MediaProfiles::CamcorderProfile* highProfile, *highSpecificProfile;
805 createDefaultCamcorderHighProfiles(&highProfile, &highSpecificProfile);
806 profiles->mCamcorderProfiles.add(highProfile);
807 profiles->mCamcorderProfiles.add(highSpecificProfile);
808
809 // low camcorder time lapse profiles.
810 MediaProfiles::CamcorderProfile *lowTimeLapseProfile, *lowSpecificTimeLapseProfile;
811 createDefaultCamcorderTimeLapseLowProfiles(&lowTimeLapseProfile, &lowSpecificTimeLapseProfile);
812 profiles->mCamcorderProfiles.add(lowTimeLapseProfile);
813 profiles->mCamcorderProfiles.add(lowSpecificTimeLapseProfile);
814
815 // high camcorder time lapse profiles.
816 MediaProfiles::CamcorderProfile *highTimeLapseProfile, *highSpecificTimeLapseProfile;
Glenn Kastenb187de12014-12-30 08:18:15 -0800817 createDefaultCamcorderTimeLapseHighProfiles(&highTimeLapseProfile,
818 &highSpecificTimeLapseProfile);
Nipun Kwatrad5672bc2010-09-16 22:25:23 -0700819 profiles->mCamcorderProfiles.add(highTimeLapseProfile);
820 profiles->mCamcorderProfiles.add(highSpecificTimeLapseProfile);
James Dong8031ec72011-03-16 14:09:50 -0700821
822 // For emulator and other legacy devices which does not have a
823 // media_profiles.xml file, We assume that the default camera id
824 // is 0 and that is the only camera available.
825 profiles->mCameraIds.push(0);
James Dong1d7491b2010-01-19 17:45:38 -0800826}
827
828/*static*/ void
829MediaProfiles::createDefaultAudioEncoders(MediaProfiles *profiles)
830{
831 profiles->mAudioEncoders.add(createDefaultAmrNBEncoderCap());
832}
833
834/*static*/ void
835MediaProfiles::createDefaultVideoDecoders(MediaProfiles *profiles)
836{
837 MediaProfiles::VideoDecoderCap *cap =
838 new MediaProfiles::VideoDecoderCap(VIDEO_DECODER_WMV);
839
840 profiles->mVideoDecoders.add(cap);
841}
842
843/*static*/ void
844MediaProfiles::createDefaultAudioDecoders(MediaProfiles *profiles)
845{
846 MediaProfiles::AudioDecoderCap *cap =
847 new MediaProfiles::AudioDecoderCap(AUDIO_DECODER_WMA);
848
849 profiles->mAudioDecoders.add(cap);
850}
851
852/*static*/ void
853MediaProfiles::createDefaultEncoderOutputFileFormats(MediaProfiles *profiles)
854{
855 profiles->mEncoderOutputFileFormats.add(OUTPUT_FORMAT_THREE_GPP);
856 profiles->mEncoderOutputFileFormats.add(OUTPUT_FORMAT_MPEG_4);
857}
858
859/*static*/ MediaProfiles::AudioEncoderCap*
860MediaProfiles::createDefaultAmrNBEncoderCap()
861{
862 return new MediaProfiles::AudioEncoderCap(
863 AUDIO_ENCODER_AMR_NB, 5525, 12200, 8000, 8000, 1, 1);
864}
865
James Dongf5a83852010-02-23 17:21:44 -0800866/*static*/ void
867MediaProfiles::createDefaultImageEncodingQualityLevels(MediaProfiles *profiles)
868{
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +0800869 ImageEncodingQualityLevels *levels = new ImageEncodingQualityLevels();
870 levels->mCameraId = 0;
871 levels->mLevels.add(70);
872 levels->mLevels.add(80);
873 levels->mLevels.add(90);
874 profiles->mImageEncodingQualityLevels.add(levels);
James Dongf5a83852010-02-23 17:21:44 -0800875}
876
Hong Tengcabd5f82011-07-06 18:33:09 -0700877/*static*/ void
878MediaProfiles::createDefaultVideoEditorCap(MediaProfiles *profiles)
879{
880 profiles->mVideoEditorCap =
881 new MediaProfiles::VideoEditorCap(
882 VIDEOEDITOR_DEFAULT_MAX_INPUT_FRAME_WIDTH,
883 VIDEOEDITOR_DEFUALT_MAX_INPUT_FRAME_HEIGHT,
884 VIDEOEDITOR_DEFAULT_MAX_OUTPUT_FRAME_WIDTH,
Hong Teng3a9cefe2011-11-10 14:54:26 -0800885 VIDEOEDITOR_DEFUALT_MAX_OUTPUT_FRAME_HEIGHT,
886 VIDEOEDITOR_DEFAULT_MAX_PREFETCH_YUV_FRAMES);
Hong Tengcabd5f82011-07-06 18:33:09 -0700887}
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -0700888/*static*/ void
889MediaProfiles::createDefaultExportVideoProfiles(MediaProfiles *profiles)
890{
891 // Create default video export profiles
892 profiles->mVideoEditorExportProfiles.add(
893 new ExportVideoProfile(VIDEO_ENCODER_H263,
894 OMX_VIDEO_H263ProfileBaseline, OMX_VIDEO_H263Level10));
895 profiles->mVideoEditorExportProfiles.add(
896 new ExportVideoProfile(VIDEO_ENCODER_MPEG_4_SP,
897 OMX_VIDEO_MPEG4ProfileSimple, OMX_VIDEO_MPEG4Level1));
898 profiles->mVideoEditorExportProfiles.add(
899 new ExportVideoProfile(VIDEO_ENCODER_H264,
900 OMX_VIDEO_AVCProfileBaseline, OMX_VIDEO_AVCLevel13));
901}
Hong Tengcabd5f82011-07-06 18:33:09 -0700902
James Dong1d7491b2010-01-19 17:45:38 -0800903/*static*/ MediaProfiles*
904MediaProfiles::createDefaultInstance()
905{
906 MediaProfiles *profiles = new MediaProfiles;
907 createDefaultCamcorderProfiles(profiles);
908 createDefaultVideoEncoders(profiles);
909 createDefaultAudioEncoders(profiles);
910 createDefaultVideoDecoders(profiles);
911 createDefaultAudioDecoders(profiles);
912 createDefaultEncoderOutputFileFormats(profiles);
James Dongf5a83852010-02-23 17:21:44 -0800913 createDefaultImageEncodingQualityLevels(profiles);
Hong Tengcabd5f82011-07-06 18:33:09 -0700914 createDefaultVideoEditorCap(profiles);
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -0700915 createDefaultExportVideoProfiles(profiles);
James Dong1d7491b2010-01-19 17:45:38 -0800916 return profiles;
917}
918
919/*static*/ MediaProfiles*
920MediaProfiles::createInstanceFromXmlFile(const char *xml)
921{
922 FILE *fp = NULL;
923 CHECK((fp = fopen(xml, "r")));
924
925 XML_Parser parser = ::XML_ParserCreate(NULL);
926 CHECK(parser != NULL);
927
928 MediaProfiles *profiles = new MediaProfiles();
929 ::XML_SetUserData(parser, profiles);
930 ::XML_SetElementHandler(parser, startElementHandler, NULL);
931
932 /*
933 FIXME:
934 expat is not compiled with -DXML_DTD. We don't have DTD parsing support.
935
936 if (!::XML_SetParamEntityParsing(parser, XML_PARAM_ENTITY_PARSING_ALWAYS)) {
Steve Block29357bc2012-01-06 19:20:56 +0000937 ALOGE("failed to enable DTD support in the xml file");
James Dong1d7491b2010-01-19 17:45:38 -0800938 return UNKNOWN_ERROR;
939 }
940
941 */
942
943 const int BUFF_SIZE = 512;
944 for (;;) {
945 void *buff = ::XML_GetBuffer(parser, BUFF_SIZE);
946 if (buff == NULL) {
Steve Block29357bc2012-01-06 19:20:56 +0000947 ALOGE("failed to in call to XML_GetBuffer()");
James Dong1d7491b2010-01-19 17:45:38 -0800948 delete profiles;
949 profiles = NULL;
950 goto exit;
951 }
952
953 int bytes_read = ::fread(buff, 1, BUFF_SIZE, fp);
954 if (bytes_read < 0) {
Steve Block29357bc2012-01-06 19:20:56 +0000955 ALOGE("failed in call to read");
James Dong1d7491b2010-01-19 17:45:38 -0800956 delete profiles;
957 profiles = NULL;
958 goto exit;
959 }
960
961 CHECK(::XML_ParseBuffer(parser, bytes_read, bytes_read == 0));
962
963 if (bytes_read == 0) break; // done parsing the xml file
964 }
965
966exit:
967 ::XML_ParserFree(parser);
968 ::fclose(fp);
James Dong1d7491b2010-01-19 17:45:38 -0800969 return profiles;
970}
971
972Vector<output_format> MediaProfiles::getOutputFileFormats() const
973{
974 return mEncoderOutputFileFormats; // copy out
975}
976
977Vector<video_encoder> MediaProfiles::getVideoEncoders() const
978{
979 Vector<video_encoder> encoders;
980 for (size_t i = 0; i < mVideoEncoders.size(); ++i) {
981 encoders.add(mVideoEncoders[i]->mCodec);
982 }
983 return encoders; // copy out
984}
985
986int MediaProfiles::getVideoEncoderParamByName(const char *name, video_encoder codec) const
987{
Steve Block3856b092011-10-20 11:56:00 +0100988 ALOGV("getVideoEncoderParamByName: %s for codec %d", name, codec);
James Dong1d7491b2010-01-19 17:45:38 -0800989 int index = -1;
990 for (size_t i = 0, n = mVideoEncoders.size(); i < n; ++i) {
991 if (mVideoEncoders[i]->mCodec == codec) {
992 index = i;
993 break;
994 }
995 }
996 if (index == -1) {
Steve Block29357bc2012-01-06 19:20:56 +0000997 ALOGE("The given video encoder %d is not found", codec);
James Dong1d7491b2010-01-19 17:45:38 -0800998 return -1;
999 }
1000
1001 if (!strcmp("enc.vid.width.min", name)) return mVideoEncoders[index]->mMinFrameWidth;
1002 if (!strcmp("enc.vid.width.max", name)) return mVideoEncoders[index]->mMaxFrameWidth;
1003 if (!strcmp("enc.vid.height.min", name)) return mVideoEncoders[index]->mMinFrameHeight;
1004 if (!strcmp("enc.vid.height.max", name)) return mVideoEncoders[index]->mMaxFrameHeight;
1005 if (!strcmp("enc.vid.bps.min", name)) return mVideoEncoders[index]->mMinBitRate;
1006 if (!strcmp("enc.vid.bps.max", name)) return mVideoEncoders[index]->mMaxBitRate;
1007 if (!strcmp("enc.vid.fps.min", name)) return mVideoEncoders[index]->mMinFrameRate;
1008 if (!strcmp("enc.vid.fps.max", name)) return mVideoEncoders[index]->mMaxFrameRate;
1009
Steve Block29357bc2012-01-06 19:20:56 +00001010 ALOGE("The given video encoder param name %s is not found", name);
James Dong1d7491b2010-01-19 17:45:38 -08001011 return -1;
1012}
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -07001013int MediaProfiles::getVideoEditorExportParamByName(
1014 const char *name, int codec) const
1015{
Steve Block3856b092011-10-20 11:56:00 +01001016 ALOGV("getVideoEditorExportParamByName: name %s codec %d", name, codec);
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -07001017 ExportVideoProfile *exportProfile = NULL;
1018 int index = -1;
1019 for (size_t i =0; i < mVideoEditorExportProfiles.size(); i++) {
1020 exportProfile = mVideoEditorExportProfiles[i];
1021 if (exportProfile->mCodec == codec) {
1022 index = i;
1023 break;
1024 }
1025 }
1026 if (index == -1) {
Steve Block29357bc2012-01-06 19:20:56 +00001027 ALOGE("The given video decoder %d is not found", codec);
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -07001028 return -1;
1029 }
1030 if (!strcmp("videoeditor.export.profile", name))
1031 return exportProfile->mProfile;
1032 if (!strcmp("videoeditor.export.level", name))
1033 return exportProfile->mLevel;
James Dong1d7491b2010-01-19 17:45:38 -08001034
Steve Block29357bc2012-01-06 19:20:56 +00001035 ALOGE("The given video editor export param name %s is not found", name);
Rajneesh Chowdury8f74b712011-08-12 16:43:37 -07001036 return -1;
1037}
Hong Tengcabd5f82011-07-06 18:33:09 -07001038int MediaProfiles::getVideoEditorCapParamByName(const char *name) const
1039{
Steve Block3856b092011-10-20 11:56:00 +01001040 ALOGV("getVideoEditorCapParamByName: %s", name);
Hong Tengcabd5f82011-07-06 18:33:09 -07001041
1042 if (mVideoEditorCap == NULL) {
Steve Block29357bc2012-01-06 19:20:56 +00001043 ALOGE("The mVideoEditorCap is not created, then create default cap.");
Hong Tengcabd5f82011-07-06 18:33:09 -07001044 createDefaultVideoEditorCap(sInstance);
1045 }
1046
1047 if (!strcmp("videoeditor.input.width.max", name))
1048 return mVideoEditorCap->mMaxInputFrameWidth;
1049 if (!strcmp("videoeditor.input.height.max", name))
1050 return mVideoEditorCap->mMaxInputFrameHeight;
1051 if (!strcmp("videoeditor.output.width.max", name))
1052 return mVideoEditorCap->mMaxOutputFrameWidth;
1053 if (!strcmp("videoeditor.output.height.max", name))
1054 return mVideoEditorCap->mMaxOutputFrameHeight;
Hong Teng3a9cefe2011-11-10 14:54:26 -08001055 if (!strcmp("maxPrefetchYUVFrames", name))
1056 return mVideoEditorCap->mMaxPrefetchYUVFrames;
Hong Tengcabd5f82011-07-06 18:33:09 -07001057
Steve Block29357bc2012-01-06 19:20:56 +00001058 ALOGE("The given video editor param name %s is not found", name);
Hong Tengcabd5f82011-07-06 18:33:09 -07001059 return -1;
1060}
1061
James Dong1d7491b2010-01-19 17:45:38 -08001062Vector<audio_encoder> MediaProfiles::getAudioEncoders() const
1063{
1064 Vector<audio_encoder> encoders;
1065 for (size_t i = 0; i < mAudioEncoders.size(); ++i) {
1066 encoders.add(mAudioEncoders[i]->mCodec);
1067 }
1068 return encoders; // copy out
1069}
1070
1071int MediaProfiles::getAudioEncoderParamByName(const char *name, audio_encoder codec) const
1072{
Steve Block3856b092011-10-20 11:56:00 +01001073 ALOGV("getAudioEncoderParamByName: %s for codec %d", name, codec);
James Dong1d7491b2010-01-19 17:45:38 -08001074 int index = -1;
1075 for (size_t i = 0, n = mAudioEncoders.size(); i < n; ++i) {
1076 if (mAudioEncoders[i]->mCodec == codec) {
1077 index = i;
1078 break;
1079 }
1080 }
1081 if (index == -1) {
Steve Block29357bc2012-01-06 19:20:56 +00001082 ALOGE("The given audio encoder %d is not found", codec);
James Dong1d7491b2010-01-19 17:45:38 -08001083 return -1;
1084 }
1085
1086 if (!strcmp("enc.aud.ch.min", name)) return mAudioEncoders[index]->mMinChannels;
1087 if (!strcmp("enc.aud.ch.max", name)) return mAudioEncoders[index]->mMaxChannels;
1088 if (!strcmp("enc.aud.bps.min", name)) return mAudioEncoders[index]->mMinBitRate;
1089 if (!strcmp("enc.aud.bps.max", name)) return mAudioEncoders[index]->mMaxBitRate;
1090 if (!strcmp("enc.aud.hz.min", name)) return mAudioEncoders[index]->mMinSampleRate;
1091 if (!strcmp("enc.aud.hz.max", name)) return mAudioEncoders[index]->mMaxSampleRate;
1092
Steve Block29357bc2012-01-06 19:20:56 +00001093 ALOGE("The given audio encoder param name %s is not found", name);
James Dong1d7491b2010-01-19 17:45:38 -08001094 return -1;
1095}
1096
1097Vector<video_decoder> MediaProfiles::getVideoDecoders() const
1098{
1099 Vector<video_decoder> decoders;
1100 for (size_t i = 0; i < mVideoDecoders.size(); ++i) {
1101 decoders.add(mVideoDecoders[i]->mCodec);
1102 }
1103 return decoders; // copy out
1104}
1105
1106Vector<audio_decoder> MediaProfiles::getAudioDecoders() const
1107{
1108 Vector<audio_decoder> decoders;
1109 for (size_t i = 0; i < mAudioDecoders.size(); ++i) {
1110 decoders.add(mAudioDecoders[i]->mCodec);
1111 }
1112 return decoders; // copy out
1113}
1114
Nipun Kwatra8bb56032010-09-09 16:25:08 -07001115int MediaProfiles::getCamcorderProfileIndex(int cameraId, camcorder_quality quality) const
James Dong1d7491b2010-01-19 17:45:38 -08001116{
James Dong1d7491b2010-01-19 17:45:38 -08001117 int index = -1;
1118 for (size_t i = 0, n = mCamcorderProfiles.size(); i < n; ++i) {
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +08001119 if (mCamcorderProfiles[i]->mCameraId == cameraId &&
1120 mCamcorderProfiles[i]->mQuality == quality) {
James Dong1d7491b2010-01-19 17:45:38 -08001121 index = i;
1122 break;
1123 }
1124 }
Nipun Kwatra8bb56032010-09-09 16:25:08 -07001125 return index;
1126}
1127
1128int MediaProfiles::getCamcorderProfileParamByName(const char *name,
1129 int cameraId,
1130 camcorder_quality quality) const
1131{
Steve Block3856b092011-10-20 11:56:00 +01001132 ALOGV("getCamcorderProfileParamByName: %s for camera %d, quality %d",
Glenn Kastene53b9ea2012-03-12 16:29:55 -07001133 name, cameraId, quality);
Nipun Kwatra8bb56032010-09-09 16:25:08 -07001134
1135 int index = getCamcorderProfileIndex(cameraId, quality);
James Dong1d7491b2010-01-19 17:45:38 -08001136 if (index == -1) {
Steve Block29357bc2012-01-06 19:20:56 +00001137 ALOGE("The given camcorder profile camera %d quality %d is not found",
Glenn Kastene53b9ea2012-03-12 16:29:55 -07001138 cameraId, quality);
James Dong1d7491b2010-01-19 17:45:38 -08001139 return -1;
1140 }
1141
James Dongf5a83852010-02-23 17:21:44 -08001142 if (!strcmp("duration", name)) return mCamcorderProfiles[index]->mDuration;
James Dong1d7491b2010-01-19 17:45:38 -08001143 if (!strcmp("file.format", name)) return mCamcorderProfiles[index]->mFileFormat;
1144 if (!strcmp("vid.codec", name)) return mCamcorderProfiles[index]->mVideoCodec->mCodec;
1145 if (!strcmp("vid.width", name)) return mCamcorderProfiles[index]->mVideoCodec->mFrameWidth;
1146 if (!strcmp("vid.height", name)) return mCamcorderProfiles[index]->mVideoCodec->mFrameHeight;
1147 if (!strcmp("vid.bps", name)) return mCamcorderProfiles[index]->mVideoCodec->mBitRate;
1148 if (!strcmp("vid.fps", name)) return mCamcorderProfiles[index]->mVideoCodec->mFrameRate;
1149 if (!strcmp("aud.codec", name)) return mCamcorderProfiles[index]->mAudioCodec->mCodec;
1150 if (!strcmp("aud.bps", name)) return mCamcorderProfiles[index]->mAudioCodec->mBitRate;
1151 if (!strcmp("aud.ch", name)) return mCamcorderProfiles[index]->mAudioCodec->mChannels;
1152 if (!strcmp("aud.hz", name)) return mCamcorderProfiles[index]->mAudioCodec->mSampleRate;
1153
Steve Block29357bc2012-01-06 19:20:56 +00001154 ALOGE("The given camcorder profile param id %d name %s is not found", cameraId, name);
James Dong1d7491b2010-01-19 17:45:38 -08001155 return -1;
1156}
1157
Nipun Kwatra8bb56032010-09-09 16:25:08 -07001158bool MediaProfiles::hasCamcorderProfile(int cameraId, camcorder_quality quality) const
1159{
1160 return (getCamcorderProfileIndex(cameraId, quality) != -1);
1161}
1162
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +08001163Vector<int> MediaProfiles::getImageEncodingQualityLevels(int cameraId) const
James Dongf5a83852010-02-23 17:21:44 -08001164{
Chih-Chung Chang3eaa4e92010-06-22 20:50:55 +08001165 Vector<int> result;
1166 ImageEncodingQualityLevels *levels = findImageEncodingQualityLevels(cameraId);
1167 if (levels != NULL) {
1168 result = levels->mLevels; // copy out
1169 }
1170 return result;
James Dongf5a83852010-02-23 17:21:44 -08001171}
1172
James Dong0f056292011-05-09 18:49:31 -07001173int MediaProfiles::getStartTimeOffsetMs(int cameraId) const {
1174 int offsetTimeMs = -1;
1175 ssize_t index = mStartTimeOffsets.indexOfKey(cameraId);
1176 if (index >= 0) {
1177 offsetTimeMs = mStartTimeOffsets.valueFor(cameraId);
1178 }
Steve Block3856b092011-10-20 11:56:00 +01001179 ALOGV("offsetTime=%d ms and cameraId=%d", offsetTimeMs, cameraId);
James Dong0f056292011-05-09 18:49:31 -07001180 return offsetTimeMs;
1181}
1182
James Dong1d7491b2010-01-19 17:45:38 -08001183MediaProfiles::~MediaProfiles()
1184{
1185 CHECK("destructor should never be called" == 0);
1186#if 0
1187 for (size_t i = 0; i < mAudioEncoders.size(); ++i) {
1188 delete mAudioEncoders[i];
1189 }
1190 mAudioEncoders.clear();
1191
1192 for (size_t i = 0; i < mVideoEncoders.size(); ++i) {
1193 delete mVideoEncoders[i];
1194 }
1195 mVideoEncoders.clear();
1196
1197 for (size_t i = 0; i < mVideoDecoders.size(); ++i) {
1198 delete mVideoDecoders[i];
1199 }
1200 mVideoDecoders.clear();
1201
1202 for (size_t i = 0; i < mAudioDecoders.size(); ++i) {
1203 delete mAudioDecoders[i];
1204 }
1205 mAudioDecoders.clear();
1206
1207 for (size_t i = 0; i < mCamcorderProfiles.size(); ++i) {
1208 delete mCamcorderProfiles[i];
1209 }
1210 mCamcorderProfiles.clear();
1211#endif
1212}
1213} // namespace android