Update contrib.
1 // Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
19 #include <ecom/test_bed/datalogger.h>
21 // KLogBufferSize is the maximum line length allowed by flogger
22 const TInt KMaxTBLogEntrySize = KLogBufferSize;
24 // Common string literals for all output formats
25 _LIT(KTimeFormatStr,"%J%:1%T%:2%S%.%*C3");
26 // Above string will give time in format HH:MM:SS.SSS therefore max length = 12
27 const TInt KTimeBufLength = 12;
29 // Define some string literals for HTML formatting
30 _LIT(KHTMLDocumentStart,"<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\"><HTML><HEAD><META NAME=\"robots\" CONTENT=\"noindex\"><META NAME=\"author\" CONTENT=\"Symbian RTestBed log generator\"><TITLE>");
31 // Insert title string here
32 _LIT(KHTMLContentStart,"</TITLE></HEAD><BODY><P><FONT=3>");
33 // Insert time string here
34 _LIT(KHTMLCommentStart,"<P>");
35 // Insert content here
36 _LIT(KHTMLCommentEnd,"</P>");
37 _LIT(KHTMLDocumentEnd,"</FONT></P></BODY></HTML>");
39 //___________________________________________________________________________
40 // Define the overflow handling classes for any log formatting methods
41 // Simply record the overflow...
44 NONSHARABLE_CLASS(TLogMessageOverflow) : public TDes16Overflow
48 TLogMessageOverflow();
50 void Overflow(TDes16&);
55 TLogMessageOverflow::TLogMessageOverflow()
61 void TLogMessageOverflow::Overflow(TDes16&)
64 iError = KErrOverflow;
68 NONSHARABLE_CLASS(TLogMessageOverflow8) : public TDes8Overflow
72 TLogMessageOverflow8();
74 void Overflow(TDes8&);
79 TLogMessageOverflow8::TLogMessageOverflow8()
85 void TLogMessageOverflow8::Overflow(TDes8&)
88 iError = KErrOverflow;
91 //___________________________________________________________________________
93 //___________________________________________________________________________
95 CDataLogger::~CDataLogger()
97 if(iLogOutput != NULL)
99 if(iLogStyle != EText && iLogFormat.iDocumentEnd)
100 Log(iLogOutput, *(iLogFormat.iDocumentEnd));
104 if(iReportOutput != NULL)
106 if(iLogStyle != EText && iLogFormat.iDocumentEnd)
107 Log(iReportOutput, *(iLogFormat.iDocumentEnd));
108 iReportOutput->Close();
113 delete iDefaultLogOutput;
114 delete iDefaultReportOutput;
118 CDataLogger::CDataLogger()
122 CDataLogger* CDataLogger::NewLC(TLoggingInfo* aLogInfo)
124 CDataLogger* self = new (ELeave) CDataLogger();
125 CleanupStack::PushL(self);
126 self->ConstructL(aLogInfo);
130 CDataLogger* CDataLogger::NewL(TLoggingInfo* aLogInfo)
132 CDataLogger* self = NewLC(aLogInfo);
138 void CDataLogger::ConstructL(TLoggingInfo* aLogInfo)
140 iFormatBuf = HBufC::NewL(KMaxTBLogEntrySize);
142 SetupLoggingL(aLogInfo);
145 iReportOutput->OpenL();
147 // Record additional information : Time
149 time.UniversalTime();
150 TBuf<KTimeBufLength> timeBuf;
151 time.FormatL(timeBuf,KTimeFormatStr);
152 if(iLogStyle != EText)
154 // Use the log format
155 Log(iLogOutput, *(iLogFormat.iDocumentStart));
157 Log(iLogOutput, *(aLogInfo->iTitle));
158 Log(iLogOutput, *(iLogFormat.iContentStart));
159 Log(iLogOutput, timeBuf);
161 Log(iReportOutput, *(iLogFormat.iDocumentStart));
163 Log(iReportOutput, *(aLogInfo->iTitle));
164 Log(iReportOutput, *(iLogFormat.iContentStart));
165 Log(iReportOutput, timeBuf);
169 if(aLogInfo && aLogInfo->iTitle)
171 Log(iLogOutput, *(aLogInfo->iTitle));
172 Log(iReportOutput, *(aLogInfo->iTitle));
174 Log(iLogOutput, timeBuf);
175 Log(iReportOutput, timeBuf);
180 EXPORT_C void CDataLogger::DumpMemoryBlock(const TUint8* aAddress, TInt aLength)
182 const TInt KBytesPerRow = 16;
183 const TInt KLowestAsciiPrint = 32;
184 const TInt KHighestAsciiPrint = 127;
186 // The required descriptors for outputting the data
187 _LIT(KDumpLineStart, "%04x : ");
188 _LIT(KDumpHexOutput, "%02x ");
189 _LIT(KDumpHexBlank, " ");
190 _LIT(KDumpHexEnd, ": ");
191 _LIT(KDumpCharOutput, "%c");
192 _LIT(KDumpCharUnknown, ".");
193 _LIT(KDumpCharBlank, " ");
195 TPtrC8 theData(aAddress, aLength);
197 // Iterate the supplied block of data in blocks of 16 bytes
199 TBuf<KMaxTBLogEntrySize> logLine;
200 TBuf<KMaxTBLogEntrySize> anEntry;
201 while (pos < theData.Length())
205 anEntry.Format(KDumpLineStart, pos);
206 logLine.Append(anEntry);
209 for (offset = 0; offset < KBytesPerRow; ++offset)
211 if (pos + offset < theData.Length())
213 TInt nextByte = theData[pos + offset];
214 anEntry.Format(KDumpHexOutput, nextByte);
215 logLine.Append(anEntry);
219 anEntry.Format(KDumpHexBlank);
220 logLine.Append(anEntry);
223 anEntry.Format(KDumpHexEnd);
224 logLine.Append(anEntry);
227 for (offset = 0; offset < KBytesPerRow; ++offset)
229 if (pos + offset < theData.Length())
231 TInt nextByte = theData[pos + offset];
232 if ((nextByte >= KLowestAsciiPrint) && (nextByte <= KHighestAsciiPrint))
234 anEntry.Format(KDumpCharOutput, nextByte);
235 logLine.Append(anEntry);
239 anEntry.Format(KDumpCharUnknown);
240 logLine.Append(anEntry);
245 anEntry.Format(KDumpCharBlank);
246 logLine.Append(anEntry);
250 //Log this line to the file
251 if(iLogStyle != EText)
253 Log(iLogOutput, *(iLogFormat.iCommentStart));
254 Log(iLogOutput, logLine);
255 Log(iLogOutput, *(iLogFormat.iCommentEnd));
258 Log(iLogOutput, logLine);
262 // Advance to next segment of size 'KBytesPerRow'
269 EXPORT_C void CDataLogger::LogInformation(const TDesC16& aComment)
271 if(iLogStyle != EText)
273 Log(iLogOutput, *(iLogFormat.iCommentStart));
274 Log(iLogOutput, aComment);
275 Log(iLogOutput, *(iLogFormat.iCommentEnd));
278 Log(iLogOutput, aComment);
279 iDebug->Print(aComment);
282 EXPORT_C void CDataLogger::LogInformation(const TDesC8& aComment)
284 // Create a wide descriptor to copy aComment into
285 HBufC* message = HBufC::NewMax(aComment.Length());
287 // If the allocation failed then do nothing
290 TPtr message16 = message->Des();
292 message16.Copy(aComment);
293 LogInformation(message16);
298 EXPORT_C void CDataLogger::LogInformationWithParameters(TRefByValue<const TDesC16> aFormat, ...)
300 // Prepare the message
301 // coverity [var_decl]
302 // VA_LIST is initialized in VA_START
304 VA_START(list,aFormat);
306 TPtr message = iFormatBuf->Des();
308 // Catch the overflow if formatting
309 TLogMessageOverflow overflowHandler;
310 message.AppendFormatList(aFormat,list,&overflowHandler);
312 if(overflowHandler.iError == KErrNone)
314 // Ok formatted correctly so...
315 // Wrap the logging level as the first parameter
316 if(iLogStyle != EText)
318 Log(iLogOutput, *(iLogFormat.iCommentStart));
319 Log(iLogOutput, message);
320 Log(iLogOutput, *(iLogFormat.iCommentEnd));
323 Log(iLogOutput, message);
324 iDebug->Print(message);
327 // Clear the message buffer
331 EXPORT_C void CDataLogger::LogInformationWithParameters(TRefByValue<const TDesC8> aFormat, ...)
333 // Create an 8 bit descriptor to copy aFormat into
334 HBufC8* message8 = HBufC8::New(KMaxTBLogEntrySize);
337 // Prepare the message
338 // coverity [var_decl]
339 // VA_LIST is initialized in VA_START
341 VA_START(list,aFormat);
342 TPtr8 messagePtr8 = message8->Des();
344 // Catch the overflow if formatting
345 TLogMessageOverflow8 overflowHandler;
346 messagePtr8.AppendFormatList(aFormat,list,&overflowHandler);
348 if(overflowHandler.iError == KErrNone)
350 TPtr message = iFormatBuf->Des();
351 // Copy over the fromatted message into the 16 bit descriptor
352 message.Copy(messagePtr8);
354 // Ok formatted correctly so...
355 // Wrap the logging level as the first parameter
356 if(iLogStyle != EText)
358 Log(iLogOutput, *(iLogFormat.iCommentStart));
359 Log(iLogOutput, message);
360 Log(iLogOutput, *(iLogFormat.iCommentEnd));
363 Log(iLogOutput, message);
364 iDebug->Print(message);
366 // Clear the message buffer
373 EXPORT_C void CDataLogger::ReportInformation(const TDesC& aComment)
375 if(iLogStyle != EText)
377 Log(iReportOutput, *(iLogFormat.iCommentStart));
378 Log(iReportOutput, aComment);
379 Log(iReportOutput, *(iLogFormat.iCommentEnd));
382 Log(iReportOutput, aComment);
385 EXPORT_C void CDataLogger::ReportInformationWithParameters(TRefByValue<const TDesC> aFormat, ...)
387 // Prepare the message
388 // coverity [var_decl]
389 // VA_LIST is initialized in VA_START
391 VA_START(list,aFormat);
393 TPtr message = iFormatBuf->Des();
395 // Catch the overflow if formatting
396 TLogMessageOverflow overflowHandler;
397 message.AppendFormatList(aFormat,list,&overflowHandler);
399 if(overflowHandler.iError == KErrNone)
401 // Ok formatted correctly so...
402 // Wrap the logging level as the first parameter
403 if(iLogStyle != EText)
405 Log(iReportOutput, *(iLogFormat.iCommentStart));
406 Log(iReportOutput, message);
407 Log(iReportOutput, *(iLogFormat.iCommentEnd));
410 Log(iReportOutput, message);
413 // Clear the message buffer
418 void CDataLogger::SetupRDebugL(TBool aRequest)
424 iDebug = new(ELeave) TDebugPrint; // Print to RDebug
426 iDebug = new(ELeave) TNullDebugPrint; // Ignore prints
429 void CDataLogger::TDebugPrint::Print(const TDesC& aMessage)
431 _LIT(KRDebugFormatStr,"%S");
432 RDebug::Print(KRDebugFormatStr, &aMessage);
435 void CDataLogger::Log(MLogOutput* aLogOutput, const TDesC16& aMessage)
437 // If the message is short enough then log it in one go
438 if(aMessage.Length() < KMaxTBLogEntrySize)
439 aLogOutput->Write(aMessage);
442 // Start at the beginning and log out short blocks until finished
444 while(outIndex < aMessage.Length())
446 if((outIndex+KMaxTBLogEntrySize) > aMessage.Length())
448 aLogOutput->Write(aMessage.Right(aMessage.Length() - outIndex));
449 outIndex = aMessage.Length();
453 // The -1 is required to ensure that the submessage is not too long
454 TPtrC subMessage = aMessage.Mid(outIndex, KMaxTBLogEntrySize - 1);
455 // Find the space nearest the end for a convenient break point
456 TInt spaceLoc = subMessage.LocateReverse(TChar(' '));
457 if(spaceLoc != KErrNotFound)
460 outIndex = KMaxTBLogEntrySize - 1;
461 aLogOutput->Write(subMessage.Left(++outIndex));
467 void CDataLogger::SetupLoggingL(TLoggingInfo* aLogInfo)
469 // The possible log filenames
470 _LIT(KTestBedLogName, "RTestBed.log");
471 _LIT(KTestBedHtmlLogName, "TestBedLog.html");
472 // The possible report file names
473 _LIT(KTestBedReportName, "RTestBed.rep");
474 _LIT(KTestBedHtmlReportName, "TestBedReport.html");
478 iLogStyle = aLogInfo->iStyle;
480 if(aLogInfo->iLogOutput)
481 iLogOutput = aLogInfo->iLogOutput;
485 iDefaultLogOutput = new(ELeave) CDefaultLogOutput(KTestBedHtmlLogName);
487 iDefaultLogOutput = new(ELeave) CDefaultLogOutput(KTestBedLogName);
489 iLogOutput = iDefaultLogOutput;
492 if(aLogInfo->iReportOutput)
493 iReportOutput = aLogInfo->iReportOutput;
497 iDefaultReportOutput = new(ELeave) CDefaultLogOutput(KTestBedHtmlReportName);
499 iDefaultReportOutput = new(ELeave) CDefaultLogOutput(KTestBedReportName);
501 iReportOutput = iDefaultReportOutput;
504 SetupRDebugL(aLogInfo->iUseRDebug);
509 iDefaultLogOutput = new(ELeave) CDefaultLogOutput(KTestBedLogName);
510 iLogOutput = iDefaultLogOutput;
511 iDefaultReportOutput = new(ELeave) CDefaultLogOutput(KTestBedReportName);
512 iReportOutput = iDefaultReportOutput;
517 // If the user has specified a custom logging style then use their LogFormat
518 if(iLogStyle == ECustom)
519 iLogFormat = *(aLogInfo->iLogFormat);
520 else if(iLogStyle == EHtml)
523 iLogFormat.iDocumentStart = &(KHTMLDocumentStart());
524 iLogFormat.iContentStart = &(KHTMLContentStart());
525 iLogFormat.iCommentStart = &(KHTMLCommentStart());
526 iLogFormat.iCommentEnd = &(KHTMLCommentEnd());
527 iLogFormat.iDocumentEnd = &(KHTMLDocumentEnd());