os/ossrv/lowlevellibsandfws/pluginfw/Framework/DriveInfoTest/t_driveinfo.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2005-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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // This file contains code to test the EcomCachedDriveInfo class.
    15 // 
    16 //
    17 
    18 #include <e32test.h>
    19 #include <f32file.h>
    20 #include <ecom/ecomerrorcodes.h>
    21 #include "EComPatchDataConstantv2.h"
    22 #include "DriveInfo.h"
    23 #include "EComInternalErrorCodes.h"
    24 #define UNUSED_VAR(a) (a = a)
    25 
    26 LOCAL_D RTest TheTest(_L("t_driveinfo"));
    27 LOCAL_D RFs	TheFs;
    28 static TInt IteratorPanicTest(TAny* aFuncCode);
    29 
    30 const TInt KPanicIndexOutOfBound = 133;
    31 _LIT(KTestFolder,			"C:\\TestTemp\\");
    32 
    33 enum TIteratorFunctionToTest
    34 	{
    35 	EIterFunctionDriveUnit,
    36 	EIterFunctionDriveNumber,
    37 	EIterFunctionIsReadOnlyInternal,
    38 	EIterFunctionIsRemovable
    39 	};
    40 
    41 class TDriveInfo_StateAccessor 
    42 	{
    43 public:
    44 	static CEComCachedDriveInfo* GetCachedDriveInfoL(RFs& aFs, TUint32 aDrvMask);
    45 	static void EComDrvFlagsL(TInt aDriveNum,
    46 							  TUint32& aDrvFlags,
    47 							  const CEComCachedDriveInfo& aCachedDriveInfo);
    48 	};
    49 
    50 /**
    51 Because this class is friend of CEComCachedDriveInfo, it can call the
    52 private constructor of CEComCachedDriveInfo to make object instance
    53 with drive disabled mask different from the patchable constant.
    54 @param aFs Connected RFs session.
    55 @param aDrvMask The discovery disabled drive mask to pass to
    56 	CEComCachedDriveInfo.
    57 @return fully constructed CEComCachedDriveInfo. Caller owns the pointer.
    58 @leave KErrNoMemory if system out of memory.
    59 */
    60 CEComCachedDriveInfo* TDriveInfo_StateAccessor::GetCachedDriveInfoL(
    61 	RFs&	aFs,
    62 	TUint32 aDrvMask)
    63 	{
    64 	// Set this bool to false otherwise ConstructL will do nothing.
    65 	CEComCachedDriveInfo::iInitialized = EFalse;
    66 
    67 	CEComCachedDriveInfo* ptr = new (ELeave) CEComCachedDriveInfo();
    68 	CleanupStack::PushL(ptr);
    69 	ptr->ConstructL(aFs, aDrvMask);
    70 	CleanupStack::Pop();
    71 	return ptr;
    72 	}
    73 
    74 /**
    75 static method
    76 
    77 Retrieve the flag word stored by CEComCachedDriveInfo about a given drive.
    78 @param aDriveNum the drive of interest
    79 @param aDrvFlags output parameter to return the drive attributes
    80 @param aCachedDriveInfo the object instance to access.
    81 @leave KErrNotFound if no such drive
    82 */
    83 void TDriveInfo_StateAccessor::EComDrvFlagsL(TInt aDriveNum,
    84 											TUint32& aDrvFlags,
    85 											const CEComCachedDriveInfo& aCachedDriveInfo)
    86 	{
    87 	for (TInt i = 0; i <= aCachedDriveInfo.iLastIndex; i++)
    88 		{
    89 		if (aCachedDriveInfo.iDriveAttr[i].iDrvNumber == aDriveNum)
    90 			{
    91 			aDrvFlags = aCachedDriveInfo.iDriveAttr[i].iFlags;
    92 			return;
    93 			}
    94 		}
    95 
    96 	User::Leave(KErrNotFound);
    97 	}
    98 
    99 //Test macroses and functions
   100 LOCAL_C void CheckL(TInt aValue, TInt aLine)
   101 	{
   102 	if(!aValue)
   103 		{
   104 		TheTest(EFalse, aLine);
   105 		}
   106 	}
   107 #define TESTL(arg) ::CheckL((arg), __LINE__)
   108 
   109 
   110 /** Check CEComCachedDriveInfo has the correct attributes for the given drive
   111 and the iterator will return the drive if discovery is not disabled.
   112 */
   113 LOCAL_C void VerifyDrvAttributeL(const TInt aDriveNum, 
   114 						  		TUint32 aDisableMask,
   115 							  	const CEComCachedDriveInfo& aCachedDriveInfo)
   116 	{
   117 	TEComCachedDriveInfoIterator iter(aCachedDriveInfo);
   118 
   119 	TDriveInfo driveInfo;
   120 	User::LeaveIfError(TheFs.Drive(driveInfo, aDriveNum));
   121 
   122 	if (0 == driveInfo.iDriveAtt)
   123 		{
   124 		// Drive not exist, i.e. drive letter not in-used
   125 
   126 		TESTL( !iter.SetPos(aDriveNum) );
   127 		return;
   128 		}
   129 
   130 	TUint32 expectedAttr = 0;
   131 	if ((driveInfo.iDriveAtt & KDriveAttInternal) &&
   132 		(driveInfo.iMediaAtt & KMediaAttWriteProtected))
   133 		{
   134 		// Drive is ROnly internal which cannot be disabled.
   135 		expectedAttr = EEComDrvAttrReadOnlyInternal;
   136 		}
   137 	else
   138 		{
   139 		TUint32 drvBitMask = 1;
   140 		if ((drvBitMask << aDriveNum) & aDisableMask ||
   141 			driveInfo.iDriveAtt & KDriveAttSubsted ||
   142 			driveInfo.iDriveAtt & KDriveAttRemote)
   143 			{
   144 			expectedAttr |= EEComDrvAttrNoDiscovery;
   145 			}
   146 		
   147 		if (driveInfo.iDriveAtt & KDriveAttRemovable)
   148 			{
   149 			expectedAttr |= EEComDrvAttrRemovable;
   150 			}
   151 		
   152 		if (0 == (driveInfo.iDriveAtt & KDriveAttRom))
   153 			{
   154 			expectedAttr |= EEComDrvAttrWritable;
   155 			}
   156 		}
   157 
   158 	// Test iterator does not return disabled drives.
   159 	TBool found = EFalse;
   160 	for (iter.First(); iter.InRange() && !found; iter.Next())
   161 		{
   162 		if (iter.DriveNumber() == aDriveNum)
   163 			{
   164 			found = ETrue;
   165 			}
   166 		}
   167 
   168 	TBool expectedFound = !(expectedAttr & EEComDrvAttrNoDiscovery);
   169 	if (found != expectedFound)
   170 		{
   171 		TheTest.Printf(_L("Error drive %d, expected att 0x%X, iter found %d"), aDriveNum, expectedAttr, found);
   172 		}
   173 	TESTL(expectedFound == found);
   174 
   175 	// verify drive attributes
   176 	TUint32 actualAttr = 0;
   177 	TDriveInfo_StateAccessor::EComDrvFlagsL(aDriveNum, actualAttr,
   178 		aCachedDriveInfo);
   179 	if (actualAttr != expectedAttr)
   180 		{
   181 		TheTest.Printf(_L("Error drive %d, expected att 0x%X, got 0x%X"), aDriveNum, expectedAttr, actualAttr);
   182 		}
   183 	TESTL(actualAttr == expectedAttr);
   184 	}
   185 
   186 /**
   187 @SYMTestCaseID			SYSLIB-ECOM-UT-3536
   188 @SYMTestCaseDesc		Disable/enable each drive and verify CEComCachedDriveInfo
   189 	has correct attribute for each drive and the iterator will not return
   190 	drives that are disabled, subst or remote.
   191 @SYMTestPriority		High
   192 @SYMTestActions			Instantiate CEComCachedDriveInfo with each drive
   193 	disabled in turn. Verify the attribute and iterator operation on the drive.
   194 	Instantiate CEComCachedDriveInfo with all drive enable. Verify
   195 	attribute and iterator operation on each drive.
   196 @SYMTestExpectedResults	CEComCachedDriveInfo has the expected attributes for
   197 	each drive whethe the drive is enabled or disabled. The iterator will only
   198 	return the drive is the drive is enabled.
   199 @SYMCR					CR1049
   200 */
   201 LOCAL_C void DriveMaskTestL()
   202 	{
   203 	CEComCachedDriveInfo* cachedDriveInfo;
   204 
   205 	// Disable each drive in turn to check that disable works as expected.
   206 	TInt i;
   207 	for (i = EDriveA; i <= EDriveZ; i++)
   208 		{
   209 		TUint32 disableMask = 1 << i;
   210 		cachedDriveInfo = TDriveInfo_StateAccessor::GetCachedDriveInfoL(TheFs, disableMask);
   211 		CleanupStack::PushL(cachedDriveInfo);
   212 
   213 		// Check CEComCachedDriveInfo has the expected value.
   214 		VerifyDrvAttributeL(i, disableMask, *cachedDriveInfo);
   215 
   216 		// Test CEComCachedDriveInfo::DriveIsReadOnlyInternalL and 
   217 		// DriveIsRemovableL leaving. They should be used on drives that
   218 		// are known to be valid, e.g. drive extracted from the path
   219 		// of a discovered DLL.
   220 		// Since C: is disabled, CEComCachedDriveInfo will leave instead
   221 		// of answering true or false (because the answer is misleading).
   222 		if (i == EDriveC)
   223 			{
   224 			TRAPD(err, cachedDriveInfo->DriveIsReadOnlyInternalL(i) );
   225 			TESTL(err == KEComErrDriveNotFound);
   226 
   227 			TRAP(err, cachedDriveInfo->DriveIsRemovableL(i) );
   228 			TESTL(err == KEComErrDriveNotFound);
   229 			}
   230 
   231 		CleanupStack::PopAndDestroy(cachedDriveInfo);
   232 		}
   233 
   234 	// Test enable case.
   235 	// Make sure the disable mask is zero.
   236 	TESTL(KDiscoveryDisabledDriveList == 0);
   237 
   238 	cachedDriveInfo = TDriveInfo_StateAccessor::GetCachedDriveInfoL(TheFs,0);
   239 	CleanupStack::PushL(cachedDriveInfo);
   240 
   241 	for (i = EDriveA; i < KMaxDrives; i++)
   242 		{
   243 		VerifyDrvAttributeL(i, 0, *cachedDriveInfo);
   244 		}
   245 
   246 	CleanupStack::PopAndDestroy(cachedDriveInfo);
   247 	}
   248 
   249 /**
   250 @SYMTestCaseID			SYSLIB-ECOM-UT-3537
   251 @SYMTestCaseDesc		Test the CEComCachedDriveInfo and its iterator classes
   252 	handles substituted drives correctly.
   253 @SYMTestPriority		High
   254 @SYMTestActions			Create a substituted drive, instantiate the cached drive info and verify that 
   255 						the sustitued drive is not in the valid drive list.
   256 @SYMTestExpectedResults	Substituted drive is not in the valid drive list.
   257 @SYMCR					CR1049
   258 */
   259 LOCAL_C void SubstitutedDriveTestL()
   260 	{
   261 	//Create c:\TestTemp folder
   262 	TInt err = TheFs.MkDir(KTestFolder);
   263 	//Create substituted drive L:, it maps to C:\TestTemp\ folder
   264 	err = TheFs.SetSubst(KTestFolder,EDriveL);
   265 	TESTL(err==KErrNone);
   266 
   267 	//Verify that L Drive is not in the valid list.	
   268 	CEComCachedDriveInfo* cachedDriveInfo =
   269 		TDriveInfo_StateAccessor::GetCachedDriveInfoL(TheFs,0);
   270 	CleanupStack::PushL(cachedDriveInfo);
   271 
   272 	VerifyDrvAttributeL(EDriveL, 0, *cachedDriveInfo);
   273 
   274 	CleanupStack::PopAndDestroy(cachedDriveInfo);
   275 
   276 	// undo subst
   277 	err = TheFs.SetSubst(KNullDesC, EDriveL);
   278 	TESTL(err==KErrNone);
   279 
   280 	err = TheFs.RmDir(KTestFolder);
   281 	TESTL(err==KErrNone);
   282 	}
   283 
   284 /**
   285 @SYMTestCaseID			SYSLIB-ECOM-UT-3539
   286 @SYMTestCaseDesc		Test the various methods exposed by TEComCachedDriveInfoIterator class.
   287 @SYMTestPriority		High
   288 @SYMTestActions			For each drive returned by the iterator, call all the
   289 	getter methods.
   290 @SYMTestExpectedResults	No leave or panic occur.
   291 @SYMCR					CR1049
   292 */
   293 LOCAL_C void ExerciseIterator()
   294 	{
   295 	CEComCachedDriveInfo* cachedDriveInfo = CEComCachedDriveInfo::NewL(TheFs);
   296 	CleanupStack::PushL(cachedDriveInfo);
   297 
   298 	TEComCachedDriveInfoIterator iter(*cachedDriveInfo);
   299 
   300 	for (iter.Last(); iter.InRange(); iter.Prev())
   301 		{
   302 		TDriveNumber drvNum = iter.DriveNumber();
   303 		TDriveUnit drvUnit = iter.DriveUnit();
   304 
   305 		// A trivial test just to use the returned objects.
   306 		TESTL(drvNum == drvUnit);
   307 
   308 		TBool b = iter.DriveIsReadOnlyInternal();
   309 		UNUSED_VAR(b);
   310 
   311 		b = iter.DriveIsRemovable();
   312 		UNUSED_VAR(b);
   313 		}
   314 
   315 	CleanupStack::PopAndDestroy(cachedDriveInfo);
   316 	}
   317 
   318 /**
   319 Intended Usage	: Capture the PANIC that occurs in the thread.
   320 @param			: aName The name to be assigned to this thread. 
   321 @param			: aFunction The function which causes the panic.
   322 */
   323 LOCAL_C void ThreadPanicTest(const TDesC& aName,TThreadFunction aFunction)
   324 	{
   325 	TheTest.Next(aName);
   326 	TRequestStatus threadStatus;
   327 	RThread thread;
   328 	TBool jit;
   329 	jit=User::JustInTime();
   330 	User::SetJustInTime(EFalse);
   331 
   332 	for (TInt i = EIterFunctionDriveUnit; i <= EIterFunctionIsRemovable; i++)
   333 		{
   334 		TIteratorFunctionToTest func = static_cast<TIteratorFunctionToTest>(i);
   335 		TInt err=thread.Create(aName,aFunction,KDefaultStackSize,KMinHeapSize,KMinHeapSize, &func);
   336 		TESTL(err==KErrNone);
   337 	
   338 		thread.Logon(threadStatus);
   339 		thread.Resume();
   340 		User::WaitForRequest(threadStatus);
   341 
   342 		//Now check why the thread Exit
   343 		TExitType exitType = thread.ExitType();
   344 		TInt exitReason = thread.ExitReason();
   345 		TheTest.Printf(_L("PanicTest: exitType %d, reason %d\n"), exitType, exitReason);
   346 
   347 		TESTL(exitType == EExitPanic);
   348 		TESTL(exitReason == KPanicIndexOutOfBound);
   349 		thread.Close();
   350 		}
   351 
   352 	User::SetJustInTime(jit);
   353 	}
   354 
   355 /**
   356 @SYMTestCaseID			SYSLIB-ECOM-UT-3540
   357 @SYMTestCaseDesc		Test the TEComCachedDriveInfoIterator class will panic if object instance of this class access out of bound elements.
   358 @SYMTestPriority		High
   359 @SYMTestActions			Cause the iterator to panic by accessing out of range
   360 	cell.
   361 @SYMTestExpectedResults	Panic occur.
   362 @SYMCR					CR1049
   363 */
   364 LOCAL_C void DoIteratorPanicTestL(TIteratorFunctionToTest aFuncCode)
   365 	{
   366 	CEComCachedDriveInfo* cachedDriveInfo = CEComCachedDriveInfo::NewL(TheFs);
   367 	CleanupStack::PushL(cachedDriveInfo);
   368 
   369 	TEComCachedDriveInfoIterator iter(*cachedDriveInfo);
   370 	// Run the iterator out of range.
   371 	for (iter.Last(); iter.InRange(); iter.Prev())
   372 		{
   373 		}
   374 
   375 	// Access the getter function to trigger panic.
   376 	switch (aFuncCode)
   377 		{
   378 		case EIterFunctionDriveUnit:
   379 			{
   380 			TDriveUnit d = iter.DriveUnit();
   381 			break;
   382 			}
   383 		case EIterFunctionDriveNumber:
   384 			{
   385 			TDriveNumber d = iter.DriveNumber();
   386 			break;
   387 			}
   388 		case EIterFunctionIsReadOnlyInternal:
   389 			{
   390 			TBool f = iter.DriveIsReadOnlyInternal();
   391 			break;
   392 			}
   393 		case EIterFunctionIsRemovable:
   394 			{
   395 			TBool f = iter.DriveIsRemovable();
   396 			break;
   397 			}
   398 		default:
   399 			// do nothing and the test will fail.
   400 			break;
   401 		}
   402 
   403 	CleanupStack::PopAndDestroy(cachedDriveInfo);
   404 	}
   405 
   406 /**
   407 @SYMTestCaseID			SYSLIB-ECOM-CT-1485
   408 @SYMTestCaseDesc		Tests EcomCachedDriveInfo class.
   409 @SYMTestPriority			High
   410 @SYMTestActions			Test the various methods exposed by EcomCachedDriveInfo,
   411 						 ensuring that the returned values are correct.
   412 @SYMTestExpectedResults	The test must not fail.
   413 @SYMDEF				 DEF073919
   414 */
   415 LOCAL_C void SystemDriveTestL()
   416 	{
   417 	CEComCachedDriveInfo* cachedDriveInfo = CEComCachedDriveInfo::NewL(TheFs);
   418 	CleanupStack::PushL(cachedDriveInfo);
   419 
   420 	TDriveNumber driveSys = RFs::GetSystemDrive();
   421 	TESTL( !cachedDriveInfo->DriveIsReadOnlyInternalL(driveSys) );
   422 	TESTL(  cachedDriveInfo->DriveIsWritableL(driveSys) );
   423 
   424 	// The old EcomCachedDriveInfo class has DriveIsRemoteL and
   425 	// DriveIsSubstL methods. To verify that system drive is neither
   426 	// remote nor subst, use TEComCachedDriveInfoIterator::SetPos().
   427 	
   428 	TEComCachedDriveInfoIterator iter(*cachedDriveInfo);
   429 	TESTL( iter.SetPos(driveSys) );
   430 
   431 	// Test Z: drive property
   432 	TESTL( cachedDriveInfo->DriveIsReadOnlyInternalL(EDriveZ) );
   433 	
   434 	CleanupStack::PopAndDestroy(cachedDriveInfo);
   435 	}
   436 
   437 /** Setup the TRAP harness to invoke DoIteratorPanicTestL
   438 */
   439 TInt IteratorPanicTest(TAny* aFuncCode)
   440 	{
   441 	CTrapCleanup* threadcleanup = CTrapCleanup::New();
   442 	TIteratorFunctionToTest* funcCode = static_cast<TIteratorFunctionToTest*>(aFuncCode);
   443 	TRAPD(ret, DoIteratorPanicTestL(*funcCode)); 
   444 
   445 	delete threadcleanup;
   446 	return ret;
   447 	}
   448 
   449 typedef void (*ClassFuncPtrL) (void);
   450 /**
   451 Test under OOM conditions.
   452 This is a wrapper function to call all test functions.
   453 @param	aTestFunctionL	 Pointer to test function.
   454 @param		aTestDesc test function name
   455 */
   456 LOCAL_C void DoOOMTestL(ClassFuncPtrL aTestFunctionL, const TDesC& aTestDesc)
   457 	{
   458 	TheTest.Next(aTestDesc);
   459 	TInt err = KErrNone;
   460 	TInt tryCount = 0;
   461 	do
   462 		{
   463 		__UHEAP_MARK;
   464 
   465 		// find out the number of open handles
   466 		TInt startProcessHandleCount;
   467 		TInt startThreadHandleCount;
   468 		RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
   469 
   470 		__UHEAP_SETFAIL(RHeap::EDeterministic, ++tryCount);
   471 	
   472 		TRAP(err, aTestFunctionL());
   473 
   474 		__UHEAP_SETFAIL(RHeap::ENone, 0);
   475 		
   476 		// check that no handles have leaked
   477 		TInt endProcessHandleCount;
   478 		TInt endThreadHandleCount; 
   479 		RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
   480 		TESTL(startProcessHandleCount == endProcessHandleCount);
   481 		TESTL(startThreadHandleCount  == endThreadHandleCount);
   482 
   483 		__UHEAP_MARKEND;
   484 		} while(err == KErrNoMemory);
   485 
   486  	TESTL(err==KErrNone);
   487 	TheTest.Printf(_L("- succeeded at heap failure rate of %i\n"), tryCount);
   488 	}
   489 
   490 /**
   491 Wrapper function to call all test functions
   492 @param		aTestFunctionL pointer to test function
   493 @param		aTestDesc test function name
   494 */
   495 LOCAL_C void DoBasicTestL(ClassFuncPtrL aTestFunctionL, const TDesC& aTestDesc)
   496 	{
   497 	TheTest.Next(aTestDesc);
   498 	__UHEAP_MARK;
   499   	// find out the number of open handles
   500 	TInt startProcessHandleCount;
   501 	TInt startThreadHandleCount;
   502 	RThread().HandleCount(startProcessHandleCount, startThreadHandleCount);
   503 
   504 	aTestFunctionL();
   505 	
   506 	// check that no handles have leaked
   507 	TInt endProcessHandleCount;
   508 	TInt endThreadHandleCount;
   509 	RThread().HandleCount(endProcessHandleCount, endThreadHandleCount);
   510 
   511 	TESTL(startProcessHandleCount == endProcessHandleCount);
   512 	TESTL(startThreadHandleCount  == endThreadHandleCount);
   513 
   514 	__UHEAP_MARKEND;
   515 	}
   516 	
   517 LOCAL_C void DoTestsL()
   518 	{
   519 	__UHEAP_MARK;
   520 
   521 	User::LeaveIfError(TheFs.Connect());
   522 	CleanupClosePushL(TheFs);
   523 
   524 	// normal tests
   525 	DoBasicTestL(&DriveMaskTestL, _L("Drive Mask Test."));
   526 	DoBasicTestL(&SubstitutedDriveTestL, _L("Substituted Drive Test."));
   527 	DoBasicTestL(&ExerciseIterator, _L("Getter Test."));
   528 	DoBasicTestL(&SystemDriveTestL, _L("System Drive Test."));
   529 
   530 	// panic tests
   531 	ThreadPanicTest(_L("Iterator Panic Testing"),IteratorPanicTest);
   532 	
   533 	// OOM tests
   534 	DoOOMTestL(&DriveMaskTestL, _L("OOM Test for Drive Mask Test."));
   535 	DoOOMTestL(&ExerciseIterator, _L("OOM Test for Getter."));
   536 	DoOOMTestL(&SystemDriveTestL, _L("OOM Test for System Drive Test."));
   537 	
   538 	CleanupStack::PopAndDestroy();
   539 	
   540 	__UHEAP_MARKEND;
   541 	}
   542 
   543 
   544 GLDEF_C TInt E32Main()
   545 	{
   546 	__UHEAP_MARK;
   547 		
   548 	TheTest.Title();
   549 	TheTest.Start(_L("Start Drive Info Tests."));
   550 	
   551 	User::LeaveIfError (TheFs.Connect ());
   552 	
   553 	CTrapCleanup* cleanup = CTrapCleanup::New();
   554 	CActiveScheduler* scheduler = new(ELeave)CActiveScheduler;
   555 	CActiveScheduler::Install(scheduler);
   556 
   557 	TRAPD(err,DoTestsL());
   558 	TESTL(err==KErrNone);
   559 	
   560 	delete scheduler;
   561 	delete cleanup;
   562 	
   563 	TheFs.Close();
   564 	
   565 	TheTest.End();
   566 	TheTest.Close();
   567 	
   568 	__UHEAP_MARKEND;
   569 	return(0);
   570 	}