Update contrib.
2 * Copyright (c) 1999-2009 Nokia Corporation and/or its subsidiary(-ies).
4 * This component and the accompanying materials are made available
5 * under the terms of "Eclipse Public License v1.0"
6 * which accompanies this distribution, and is available
7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
9 * Initial Contributors:
10 * Nokia Corporation - initial contribution.
21 #include "OstTraceDefinitions.h"
22 #ifdef OST_TRACE_COMPILER_IN_USE
23 #include "ParseLstTraces.h"
28 //++ sort out definitive hash includes
31 CParserList::CParserItem::CParserItem(MParser* aParser, const TUint aTagIndex)
38 // Panic the process with UikParse as the category.
39 void CParserList::Panic(TParserListPanic aPanic) const
41 OstTraceExt2( TRACE_FATAL, DUP1_CPARSERLIST_PANIC, "CParserList::Panic;this=%x;aPanic=%x", (unsigned int)this, ( TUint )&( aPanic ) );
42 _LIT(panicStr, "ParseLst");
43 User::Panic(panicStr, aPanic);
47 CParserList::CParserList()
55 CParserList::~CParserList()
57 iParserList.ResetAndDestroy(); // Reset the list and destroy the CParserItems
58 iParserList.Close(); // but leave the parsers and free the resources
62 // Add parser to appropriate list
63 void CParserList::ActivateAParserL(MParser* aParser, const TBool aDefaultParser)
65 // Check the index isn't rolling over
66 // N.B. This is somewhat unlikely. Since index is effectively a 29-bit
67 // uint (I'm using the top 3 bits as flags) this allows 268435456 parsers
68 // to be installed. (Assuming we deinstall almost as many, as we went
69 // along, to avoid OOM.) At 50 per sec that's 20 years continuous action!
70 if (iHighestIndex >= EMaxParserIndex)
72 OstTrace0( TRACE_DUMP, CPARSERLIST_ACTIVATEAPARSERL, "EParserIndexRollover" );
74 __ASSERT_DEBUG((iHighestIndex < EMaxParserIndex), Panic(EParserIndexRollover));
77 OstTrace0( TRACE_DUMP, DUP1_CPARSERLIST_ACTIVATEAPARSERL, "EParserNullPtr" );
79 __ASSERT_DEBUG(aParser, Panic(EParserNullPtr));
80 TUint parserIndex = iHighestIndex + 1;
81 // Adjust flags to describe parser
82 if (aParser->ReformatOnRecognise())
83 parserIndex |= static_cast<TUint>( EReformatOnRecogniseFlag );
84 if (aParser->ReformatOnRollover())
85 parserIndex |= EReformatOnRolloverFlag;
87 parserIndex |= EDefaultParserFlag;
88 // Create a parser item
90 CParserItem* parserItem = new (ELeave) CParserItem(aParser, parserIndex);
94 CleanupStack::PushL(parserItem);
95 result = iParserList.Append(parserItem);
96 if (result) // We couldn't get it in the main list
102 // Look for the end of the specific parsers and the start of the default set
104 for (count = 0; count < iNumberInList; count++)
106 if (DefaultParser(iParserList[count]->TagIndex()))
109 CleanupStack::PushL(parserItem);
110 result = iParserList.Insert(parserItem, count);
111 if (result) // We couldn't get it in the main list
119 // Activate an individual parser
120 void CParserList::ActivateParserL(MParser* aParser)
122 ActivateAParserL(aParser, EFalse);
126 // N.B. We can't just delete CParserList and zero TLS when we get back
127 // to a list with no specific or default parsers because there could be
128 // an EText left with a local ptr to it.
129 void CParserList::DeactivateParser(MParser* aParser)
133 OstTrace0( TRACE_DUMP, CPARSERLIST_DEACTIVATEPARSER, "EParserNoneActive" );
135 __ASSERT_DEBUG(iNumberInList, Panic(EParserNoneActive));
138 OstTrace0( TRACE_DUMP, DUP1_CPARSERLIST_DEACTIVATEPARSER, "EParserNullPtr" );
140 __ASSERT_DEBUG(aParser, Panic(EParserNullPtr));
141 // Run thru list till find entry we need
143 for (count = 0; count < iNumberInList; count++)
145 if (iParserList[count]->Parser() == aParser)
147 delete iParserList[count];
148 iParserList.Remove(count);
152 if (count >= iNumberInList)
154 OstTrace0( TRACE_DUMP, DUP2_CPARSERLIST_DEACTIVATEPARSER, "EParserInstanceNotActive" );
156 __ASSERT_DEBUG((count < iNumberInList), Panic(EParserInstanceNotActive));
161 // Activate a parser as one of the default set
162 void CParserList::ActivateDefaultParserL(MParser* aParser)
164 ActivateAParserL(aParser, ETrue);
168 //++ Put comment here
169 void CParserList::DeactivateParserDefaults()
173 // Take them out of the list
174 while (iNumberInList && DefaultParser(iParserList[iNumberInList - 1]->TagIndex()))
176 // Tell the parser to free itself
177 iParserList[iNumberInList - 1]->Parser()->Release();
178 // Delete the item that refers to it
179 delete iParserList[iNumberInList - 1];
180 // Remove the entry from the list
181 iParserList.Remove(iNumberInList - 1);
188 // Called by EText to scan an area of text
189 TBool CParserList::ParseThisText(CRichText& aTextObj,TInt aStartScan,TInt aScanLength,TInt& aStartOfTags,TInt& aLength)
191 TBool foundSomething = EFalse;
193 TCharFormatXMask varies;
197 // Scan either side of the range in case part of some tagged text was deleted.
203 if (aStartScan + aScanLength < aTextObj.DocumentLength())
206 if (iNumberInList && aScanLength)
208 aStartOfTags = aStartScan + aScanLength;
210 for (TInt count = 0; count < iNumberInList; count++)
212 // For each parser in the list
213 TUint tagIndex = iParserList[count]->TagIndex();
215 MParser* parser = iParserList[count]->Parser();
216 TInt localStartScan = aStartScan;
217 TInt localScanLength = aScanLength;
218 // Start by removing existing tags for this parser. This ensures
219 // that old tags that have been invalidated by subsequent editing
220 // are removed. Any that are still valid will be replaced.
221 aTextObj.GetExtendedCharFormat(format, varies, localStartScan, aScanLength);
223 if (format.iParserTag || varies.AttribIsSet(EAttParserTag))
225 // At least some of the object contains a non-zero tag - go through it
226 // Are we starting part way through a tag?
227 aTextObj.GetExtendedCharFormat(format, varies, localStartScan, 1);
228 if (format.iParserTag == tagIndex)
230 // The first char of this range has the current parsers tag
231 // so we need to check backwards for the start of that tag
232 for (pos = localStartScan; pos > 0; pos--)
234 aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
235 if (format.iParserTag != tagIndex)
238 // Since we are going to remove a tag starting from here
239 // we need to allow this area to be rescanned
240 localScanLength += localStartScan - pos;
241 localStartScan = pos;
243 // What about off the end?
244 aTextObj.GetExtendedCharFormat(format, varies, localStartScan + localScanLength - 1, 1);
245 if (format.iParserTag == tagIndex)
247 // The last char of this range has the current parsers tag
248 // so we need to check forwards for the end of that tag
249 pos = localStartScan + localScanLength;
250 TInt end = aTextObj.DocumentLength();
253 aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
254 if (format.iParserTag != tagIndex)
258 // Adjust scan length
259 localScanLength = pos - localStartScan;
261 pos = localStartScan;
262 while (pos < localStartScan + localScanLength)
264 // Run along the scan range
265 aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
266 if (format.iParserTag == tagIndex)
269 format.iParserTag = 0;
271 varies.SetAttrib(EAttParserTag);
272 TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, pos, ptr.Length()));
273 if (leaveCode==KErrNone)
274 foundSomething = ETrue; // We are removing a tag
277 if (pos < aStartOfTags)
279 aLength += aStartOfTags - pos;
282 if (pos + ptr.Length() > aStartOfTags + aLength)
283 aLength = pos + ptr.Length() - aStartOfTags;
288 aLength = ptr.Length();
294 endRange = localStartScan + localScanLength;
296 // For this parser, run through text looking for changes
297 TBool allowBack = ETrue;
298 for (;;) // Run forever
300 TInt localStartTag = aTextObj.DocumentLength();
301 TInt localTagLength = 0;
302 TInt result = parser->ParseThisText(aTextObj, allowBack, localStartScan, localScanLength, localStartTag, localTagLength);
305 if (!allowBack && (localStartTag < localStartScan))
307 OstTrace0( TRACE_DUMP, CPARSERLIST_PARSETHISTEXT, "EParserIgnoringAllowFlag" );
309 __ASSERT_DEBUG(allowBack || (localStartTag >= localStartScan), Panic(EParserIgnoringAllowFlag));
310 TInt startNewTag = localStartTag;
311 TInt lengthNewTag = localTagLength;
312 aTextObj.GetExtendedCharFormat(format, varies, localStartTag, localTagLength);
313 if (format.iParserTag || varies.AttribIsSet(EAttParserTag))
315 // At least some of this area contains a non-zero tag - go through it
317 TBool higher = EFalse;
318 while (pos < localStartTag + localTagLength)
320 aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
321 if (format.iParserTag && (MaskedTag(format.iParserTag) < MaskedTag(tagIndex)))
323 // A higher precedence tag is already here so we can't
324 // insert our tag - let's see how far it goes
325 TUint tag = format.iParserTag; // Stash tag before overwriting it
326 TInt len = aTextObj.DocumentLength();
329 aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
330 if (format.iParserTag != tag)
339 // If there aren't any higher precedence tags in here then this
340 // will save us having to go through again if there aren't any
341 // lower precedence tags either
342 if (MaskedTag(format.iParserTag) >= MaskedTag(tagIndex))
348 // There are lower or equal precedence tags in this range
349 // Do they extend back off the start?
350 aTextObj.GetExtendedCharFormat(format, varies, localStartTag, 1);
351 if (format.iParserTag)
353 // need to check backwards
354 TUint tag = format.iParserTag; // Stash tag before overwriting it
355 for (pos = localStartTag; pos > 0; pos--)
357 aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
358 if (format.iParserTag != tag)
361 localTagLength += localStartTag - pos;
364 // What about off the end?
365 pos = localStartTag + localTagLength;
366 aTextObj.GetExtendedCharFormat(format, varies, pos - 1, 1);
367 if (format.iParserTag)
369 // need to check forwards
370 TUint tag = format.iParserTag; // Stash tag before overwriting it
371 TInt len = aTextObj.DocumentLength();
374 aTextObj.GetTextAndExtendedFormat(ptr, format, pos);
375 if (format.iParserTag != tag)
379 localTagLength = pos - localStartTag;
382 // Remove all tags in this area - they all have lower precedence
383 format.iParserTag = 0;
385 varies.SetAttrib(EAttCharLanguage);
386 TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, localStartTag, localTagLength));
387 if (leaveCode==KErrNone)
388 foundSomething = ETrue; // We are removing a tag
394 // Format tag this area with tagIndex
395 format.iParserTag = tagIndex;
397 varies.SetAttrib(EAttParserTag);
398 // Use the original length, not the possibly expanded version
399 TRAPD(leaveCode, aTextObj.ApplyExtendedCharFormatL(format, varies, startNewTag, lengthNewTag));
400 if (leaveCode==KErrNone)
401 foundSomething = ETrue; // We are applying a tag
404 if (localStartTag < aStartOfTags)
406 aLength += aStartOfTags - localStartTag;
407 aStartOfTags = localStartTag;
409 if (localStartTag + localTagLength > aStartOfTags + aLength)
410 aLength = localStartTag + localTagLength - aStartOfTags;
414 aStartOfTags = localStartTag;
415 aLength = localTagLength;
419 localScanLength -= startNewTag + lengthNewTag - localStartScan;
420 localStartScan = startNewTag + lengthNewTag; // Adjust start of next scan run
421 if (localStartScan >= endRange) // Have we reached the end of the range yet?
428 return foundSomething;
432 // given a tag, fetch a ptr to the parser - or null
433 MParser* CParserList::ParserWithThisTag(const TUint aTagIndex) const
435 MParser* parser = NULL;
436 for (TInt count = 0; count < iNumberInList; count++)
438 if (aTagIndex == iParserList[count]->TagIndex())
440 parser = iParserList[count]->Parser();
447 // given a ptr to a parser, fetch a tag - or zero
448 TUint CParserList::TagForThisParser(const MParser *const aParser) const
451 for (TInt count = 0; count < iNumberInList; count++)
453 if (aParser == iParserList[count]->Parser())
455 tagIndex = iParserList[count]->TagIndex();
461 OstTrace0( TRACE_DUMP, CPARSERLIST_TAGFORTHISPARSER, "EParserNoSuchTag" );
463 __ASSERT_DEBUG(tagIndex, Panic(EParserNoSuchTag));
468 CParserData::CParserData(TInt aEndParse):
473 iActiveParserList = (CParserList*)Dll::Tls();
474 iLastKnownCursor = -1;
475 if (iActiveParserList)
476 iActiveParserList->iRefCount++;
480 CParserData::~CParserData()
482 if (iActiveParserList)
484 iActiveParserList->iRefCount--;
485 if ((iActiveParserList->iRefCount == 0) && (iActiveParserList->iNumberInList == 0))
488 delete iActiveParserList;
494 // Merge the specified range, which may have changed length, into the current range.
495 // aOldLength is the # of chars deleted and to be removed from the range
496 // aNewLength is the # of chars inserted and to be added to the range
497 void CParserData::MergeRange(TInt aStart,TInt aOldLength,TInt aNewLength)
499 if (iStartParse == -1) // no current range
501 iStartParse = aStart;
502 iEndParse = aStart + aNewLength;
506 if (aStart < iStartParse)
507 iStartParse = aStart;
508 if (aStart + aOldLength > iEndParse)
509 iEndParse = aStart + aOldLength;
510 iEndParse += aNewLength - aOldLength;