OSDN Git Service

c5aee188a80e99bc873d9ebd0cdd01d5afb7d6ad
[lamexp/LameXP.git] / src / Encoder_AAC_QAAC.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // LameXP - Audio Encoder Front-End
3 // Copyright (C) 2004-2015 LoRd_MuldeR <MuldeR2@GMX.de>
4 //
5 // This program is free software; you can redistribute it and/or modify
6 // it under the terms of the GNU General Public License as published by
7 // the Free Software Foundation; either version 2 of the License, or
8 // (at your option) any later version, but always including the *additional*
9 // restrictions defined in the "License.txt" file.
10 //
11 // This program is distributed in the hope that it will be useful,
12 // but WITHOUT ANY WARRANTY; without even the implied warranty of
13 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14 // GNU General Public License for more details.
15 //
16 // You should have received a copy of the GNU General Public License along
17 // with this program; if not, write to the Free Software Foundation, Inc.,
18 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
19 //
20 // http://www.gnu.org/licenses/gpl-2.0.txt
21 ///////////////////////////////////////////////////////////////////////////////
22
23 #include "Encoder_AAC_QAAC.h"
24
25 //Internal
26 #include "Global.h"
27 #include "Model_Settings.h"
28
29 //MUtils
30 #include <MUtils/Global.h>
31
32 //StdLib
33 #include <math.h>
34
35 //Qt
36 #include <QProcess>
37 #include <QDir>
38 #include <QCoreApplication>
39
40 static int index2bitrate(const int index)
41 {
42         return (index < 32) ? ((index + 1) * 8) : ((index - 15) * 16);
43 }
44
45 static const int g_qaacVBRQualityLUT[16] = {0 ,9, 18, 27, 36, 45, 54, 63, 73, 82, 91, 100, 109, 118, 127, INT_MAX};
46
47 ///////////////////////////////////////////////////////////////////////////////
48 // Encoder Info
49 ///////////////////////////////////////////////////////////////////////////////
50
51 class QAACEncoderInfo : public AbstractEncoderInfo
52 {
53         virtual bool isModeSupported(int mode) const
54         {
55                 switch(mode)
56                 {
57                 case SettingsModel::VBRMode:
58                 case SettingsModel::CBRMode:
59                 case SettingsModel::ABRMode:
60                         return true;
61                         break;
62                 default:
63                         MUTILS_THROW("Bad RC mode specified!");
64                 }
65         }
66
67         virtual int valueCount(int mode) const
68         {
69                 switch(mode)
70                 {
71                 case SettingsModel::VBRMode:
72                         return 15;
73                         break;
74                 case SettingsModel::ABRMode:
75                 case SettingsModel::CBRMode:
76                         return 52;
77                         break;
78                 default:
79                         MUTILS_THROW("Bad RC mode specified!");
80                 }
81         }
82
83         virtual int valueAt(int mode, int index) const
84         {
85                 switch(mode)
86                 {
87                 case SettingsModel::VBRMode:
88                         return g_qaacVBRQualityLUT[qBound(0, index , 14)];
89                         break;
90                 case SettingsModel::ABRMode:
91                 case SettingsModel::CBRMode:
92                         return qBound(8, index2bitrate(index), 576);
93                         break;
94                 default:
95                         MUTILS_THROW("Bad RC mode specified!");
96                 }
97         }
98
99         virtual int valueType(int mode) const
100         {
101                 switch(mode)
102                 {
103                 case SettingsModel::VBRMode:
104                         return TYPE_QUALITY_LEVEL_INT;
105                         break;
106                 case SettingsModel::ABRMode:
107                         return TYPE_APPROX_BITRATE;
108                         break;
109                 case SettingsModel::CBRMode:
110                         return TYPE_BITRATE;
111                         break;
112                 default:
113                         MUTILS_THROW("Bad RC mode specified!");
114                 }
115         }
116
117         virtual const char *description(void) const
118         {
119                 static const char* s_description = "QAAC/QuickTime (\x0C2\x0A9 Apple Inc.)";
120                 return s_description;
121         }
122
123         virtual const char *extension(void) const
124         {
125                 static const char* s_extension = "mp4";
126                 return s_extension;
127         }
128 }
129 static const g_qaacEncoderInfo;
130
131 ///////////////////////////////////////////////////////////////////////////////
132 // Encoder implementation
133 ///////////////////////////////////////////////////////////////////////////////
134
135 QAACEncoder::QAACEncoder(void)
136 :
137         m_binary_qaac32(lamexp_tools_lookup("qaac.exe")),
138         m_binary_qaac64(lamexp_tools_lookup("qaac64.exe"))
139 {
140         if(m_binary_qaac32.isEmpty() && m_binary_qaac64.isEmpty())
141         {
142                 MUTILS_THROW("Error initializing QAAC. Tool 'qaac.exe' is not registred!");
143         }
144
145         m_configProfile = 0;
146 }
147
148 QAACEncoder::~QAACEncoder(void)
149 {
150 }
151
152 bool QAACEncoder::encode(const QString &sourceFile, const AudioFileModel_MetaInfo &metaInfo, const unsigned int duration, const QString &outputFile, volatile bool *abortFlag)
153 {
154         const QString qaac_bin = m_binary_qaac64.isEmpty() ? m_binary_qaac32 : m_binary_qaac64;
155
156         QProcess process;
157         QStringList args;
158
159         process.setWorkingDirectory(QFileInfo(outputFile).canonicalPath());
160
161         QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
162         env.insert("PATH", QDir::toNativeSeparators(QString("%1;%1/QTfiles;%2").arg(QDir(QCoreApplication::applicationDirPath()).canonicalPath(), MUtils::temp_folder())));
163         process.setProcessEnvironment(env);
164
165         if(m_configRCMode != SettingsModel::VBRMode)
166         {
167                 switch(m_configProfile)
168                 {
169                 case 2:
170                 case 3:
171                         args << "--he"; //Forces use of HE AAC profile (there is no explicit HEv2 switch for QAAC)
172                         break;
173                 }
174         }
175
176         switch(m_configRCMode)
177         {
178         case SettingsModel::CBRMode:
179                 args << "--cbr" << QString::number(qBound(8, index2bitrate(m_configBitrate), 576));
180                 break;
181         case SettingsModel::ABRMode:
182                 args << "--abr" << QString::number(qBound(8, index2bitrate(m_configBitrate), 576));
183                 break;
184         case SettingsModel::VBRMode:
185                 args << "--tvbr" << QString::number(g_qaacVBRQualityLUT[qBound(0, m_configBitrate , 14)]);
186                 break;
187         default:
188                 MUTILS_THROW("Bad rate-control mode!");
189                 break;
190         }
191
192         if(!m_configCustomParams.isEmpty()) args << m_configCustomParams.split(" ", QString::SkipEmptyParts);
193
194         if(!metaInfo.title().isEmpty())   args << "--title"   << cleanTag(metaInfo.title());
195         if(!metaInfo.artist().isEmpty())  args << "--artist"  << cleanTag(metaInfo.artist());
196         if(!metaInfo.album().isEmpty())   args << "--album"   << cleanTag(metaInfo.album());
197         if(!metaInfo.genre().isEmpty())   args << "--genre"   << cleanTag(metaInfo.genre());
198         if(!metaInfo.comment().isEmpty()) args << "--comment" << cleanTag( metaInfo.comment());
199         if(metaInfo.year())               args << "--date"    << QString::number(metaInfo.year());
200         if(metaInfo.position())           args << "--track"   << QString::number(metaInfo.position());
201         if(!metaInfo.cover().isEmpty())   args << "--artwork" << metaInfo.cover();
202
203         args << "-d" << ".";
204         args << "-o" << QDir::toNativeSeparators(outputFile);
205         args << QDir::toNativeSeparators(sourceFile);
206
207         if(!startProcess(process, qaac_bin, args))
208         {
209                 return false;
210         }
211
212         bool bTimeout = false;
213         bool bAborted = false;
214         int prevProgress = -1;
215
216         QRegExp regExp("\\[(\\d+)\\.(\\d)%\\]");
217
218         while(process.state() != QProcess::NotRunning)
219         {
220                 if(*abortFlag)
221                 {
222                         process.kill();
223                         bAborted = true;
224                         emit messageLogged("\nABORTED BY USER !!!");
225                         break;
226                 }
227                 process.waitForReadyRead(m_processTimeoutInterval);
228                 if(!process.bytesAvailable() && process.state() == QProcess::Running)
229                 {
230                         process.kill();
231                         qWarning("QAAC process timed out <-- killing!");
232                         emit messageLogged("\nPROCESS TIMEOUT !!!");
233                         bTimeout = true;
234                         break;
235                 }
236                 while(process.bytesAvailable() > 0)
237                 {
238                         QByteArray line = process.readLine();
239                         QString text = QString::fromUtf8(line.constData()).simplified();
240                         if(regExp.lastIndexIn(text) >= 0)
241                         {
242                                 bool ok = false;
243                                 int progress = regExp.cap(1).toInt(&ok);
244                                 if(ok && (progress > prevProgress))
245                                 {
246                                         emit statusUpdated(progress);
247                                         prevProgress = qMin(progress + 2, 99);
248                                 }
249                         }
250                         else if(!text.isEmpty())
251                         {
252                                 emit messageLogged(text);
253                         }
254                 }
255         }
256
257         process.waitForFinished();
258         if(process.state() != QProcess::NotRunning)
259         {
260                 process.kill();
261                 process.waitForFinished(-1);
262         }
263         
264         emit statusUpdated(100);
265         emit messageLogged(QString().sprintf("\nExited with code: 0x%04X", process.exitCode()));
266
267         if(bTimeout || bAborted || process.exitCode() != EXIT_SUCCESS)
268         {
269                 return false;
270         }
271
272         return true;
273 }
274
275 bool QAACEncoder::isFormatSupported(const QString &containerType, const QString &containerProfile, const QString &formatType, const QString &formatProfile, const QString &formatVersion)
276 {
277         if(containerType.compare("Wave", Qt::CaseInsensitive) == 0)
278         {
279                 if(formatType.compare("PCM", Qt::CaseInsensitive) == 0)
280                 {
281                         return true;
282                 }
283         }
284
285         return false;
286 }
287
288 void QAACEncoder::setProfile(int profile)
289 {
290         m_configProfile = profile;
291 }
292
293 const AbstractEncoderInfo *QAACEncoder::getEncoderInfo(void)
294 {
295         return &g_qaacEncoderInfo;
296 }