|  | @@ -42,21 +42,39 @@
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  #define MAX_MAC_PATH_LONG     2048
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | +//Cache handling functions
 | 
	
		
			
				|  |  | +//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | +bool isCachePath(const char* path)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	//TODO: implement isCachePath
 | 
	
		
			
				|  |  | +	if (!path || !*path)
 | 
	
		
			
				|  |  | +	      return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	return false;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool Platform::fileDelete(const char * name)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -   Con::warnf("Platform::FileDelete() - Not supported on android.");
 | 
	
		
			
				|  |  | +	if(!name )
 | 
	
		
			
				|  |  | +	  return(false);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   if (dStrlen(name) > MAX_MAC_PATH_LONG)
 | 
	
		
			
				|  |  | +	  Con::warnf("Platform::FileDelete() - Filename length is pretty long...");
 | 
	
		
			
				|  |  |     
 | 
	
		
			
				|  |  | -   return false;
 | 
	
		
			
				|  |  | +   return(remove(name) == 0); // remove returns 0 on success
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool dFileTouch(const char *path)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -	Con::warnf("Platform::dFileTouch() - Not supported on android.");
 | 
	
		
			
				|  |  | +	if (!path || !*path)
 | 
	
		
			
				|  |  | +	   return false;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	return true;
 | 
	
		
			
				|  |  | +    // set file at path's modification and access times to now.
 | 
	
		
			
				|  |  | +    return( utimes( path, NULL) == 0); // utimes returns 0 on success.
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
	
		
			
				|  | @@ -73,6 +91,7 @@ File::File()
 | 
	
		
			
				|  |  |     buffer = NULL;
 | 
	
		
			
				|  |  |     size = 0;
 | 
	
		
			
				|  |  |     filePointer = 0;
 | 
	
		
			
				|  |  | +   handle = NULL;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
	
		
			
				|  | @@ -84,8 +103,10 @@ File::File()
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::~File()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -	if (buffer != NULL)
 | 
	
		
			
				|  |  | +	if (buffer != NULL || handle != NULL)
 | 
	
		
			
				|  |  |  		close();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	handle = NULL;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -99,6 +120,66 @@ File::~File()
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::open(const char *filename, const AccessMode openMode)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	//If its a cache path then we need to open it using C methods not AssetManager
 | 
	
		
			
				|  |  | +   if (isCachePath(filename))
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  if (dStrlen(filename) > MAX_MAC_PATH_LONG)
 | 
	
		
			
				|  |  | +	   	  Con::warnf("File::open: Filename length is pretty long...");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // Close the file if it was already open...
 | 
	
		
			
				|  |  | +	  if (currentStatus != Closed)
 | 
	
		
			
				|  |  | +	  close();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // create the appropriate type of file...
 | 
	
		
			
				|  |  | +	  switch (openMode)
 | 
	
		
			
				|  |  | +	  {
 | 
	
		
			
				|  |  | +	  case Read:
 | 
	
		
			
				|  |  | +		 handle = (void *)fopen(filename, "rb"); // read only
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  case Write:
 | 
	
		
			
				|  |  | +		 handle = (void *)fopen(filename, "wb"); // write only
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  case ReadWrite:
 | 
	
		
			
				|  |  | +		 handle = (void *)fopen(filename, "ab+"); // write(append) and read
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  case WriteAppend:
 | 
	
		
			
				|  |  | +		 handle = (void *)fopen(filename, "ab"); // write(append) only
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  default:
 | 
	
		
			
				|  |  | +		 AssertFatal(false, "File::open: bad access mode");
 | 
	
		
			
				|  |  | +	  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // handle not created successfully
 | 
	
		
			
				|  |  | +	  if (handle == NULL)
 | 
	
		
			
				|  |  | +	  return setStatus();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // successfully created file, so set the file capabilities...
 | 
	
		
			
				|  |  | +	  switch (openMode)
 | 
	
		
			
				|  |  | +	  {
 | 
	
		
			
				|  |  | +	  case Read:
 | 
	
		
			
				|  |  | +		 capability = FileRead;
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  case Write:
 | 
	
		
			
				|  |  | +	  case WriteAppend:
 | 
	
		
			
				|  |  | +		 capability = FileWrite;
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  case ReadWrite:
 | 
	
		
			
				|  |  | +		 capability = FileRead | FileWrite;
 | 
	
		
			
				|  |  | +		 break;
 | 
	
		
			
				|  |  | +	  default:
 | 
	
		
			
				|  |  | +		 AssertFatal(false, "File::open: bad access mode");
 | 
	
		
			
				|  |  | +	  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // must set the file status before setting the position.
 | 
	
		
			
				|  |  | +	  currentStatus = Ok;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  if (openMode == ReadWrite)
 | 
	
		
			
				|  |  | +	  setPosition(0);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // success!
 | 
	
		
			
				|  |  | +	  return currentStatus;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     if (dStrlen(filename) > MAX_MAC_PATH_LONG)
 | 
	
		
			
				|  |  |        Con::warnf("File::open: Filename length is pretty long...");
 | 
	
		
			
				|  |  |     
 | 
	
	
		
			
				|  | @@ -146,6 +227,14 @@ File::Status File::open(const char *filename, const AccessMode openMode)
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  U32 File::getPosition() const
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	    AssertFatal(currentStatus != Closed , "File::getPosition: file closed");
 | 
	
		
			
				|  |  | +	    AssertFatal(handle != NULL, "File::getPosition: invalid file handle");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	    return ftell((FILE*)handle);
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     AssertFatal(currentStatus != Closed , "File::getPosition: file closed");
 | 
	
		
			
				|  |  |     AssertFatal(buffer != NULL, "File::getPosition: invalid file buffer");
 | 
	
		
			
				|  |  |     
 | 
	
	
		
			
				|  | @@ -166,6 +255,45 @@ U32 File::getPosition() const
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::setPosition(S32 position, bool absolutePos)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +   if (handle != NULL)
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  AssertFatal(Closed != currentStatus, "File::setPosition: file closed");
 | 
	
		
			
				|  |  | +	  AssertFatal(handle != NULL, "File::setPosition: invalid file handle");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  if (currentStatus != Ok && currentStatus != EOS )
 | 
	
		
			
				|  |  | +		 return currentStatus;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  U32 finalPos;
 | 
	
		
			
				|  |  | +	  if(absolutePos)
 | 
	
		
			
				|  |  | +	  {
 | 
	
		
			
				|  |  | +		 // absolute position
 | 
	
		
			
				|  |  | +		 AssertFatal(0 <= position, "File::setPosition: negative absolute position");
 | 
	
		
			
				|  |  | +		 // position beyond EOS is OK
 | 
	
		
			
				|  |  | +		 fseek((FILE*)handle, position, SEEK_SET);
 | 
	
		
			
				|  |  | +		 finalPos = ftell((FILE*)handle);
 | 
	
		
			
				|  |  | +	  }
 | 
	
		
			
				|  |  | +	  else
 | 
	
		
			
				|  |  | +	  {
 | 
	
		
			
				|  |  | +		 // relative position
 | 
	
		
			
				|  |  | +		 AssertFatal((getPosition() + position) >= 0, "File::setPosition: negative relative position");
 | 
	
		
			
				|  |  | +		 // position beyond EOS is OK
 | 
	
		
			
				|  |  | +		 fseek((FILE*)handle, position, SEEK_CUR);
 | 
	
		
			
				|  |  | +		 finalPos = ftell((FILE*)handle);
 | 
	
		
			
				|  |  | +	  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // ftell returns -1 on error. set error status
 | 
	
		
			
				|  |  | +	  if (0xffffffff == finalPos)
 | 
	
		
			
				|  |  | +		 return setStatus();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // success, at end of file
 | 
	
		
			
				|  |  | +	  else if (finalPos >= getSize())
 | 
	
		
			
				|  |  | +		 return currentStatus = EOS;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // success!
 | 
	
		
			
				|  |  | +	  else
 | 
	
		
			
				|  |  | +		 return currentStatus = Ok;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     AssertFatal(Closed != currentStatus, "File::setPosition: file closed");
 | 
	
		
			
				|  |  |     AssertFatal(buffer != NULL, "File::setPosition: invalid file buffer");
 | 
	
		
			
				|  |  |     
 | 
	
	
		
			
				|  | @@ -210,6 +338,26 @@ File::Status File::setPosition(S32 position, bool absolutePos)
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  U32 File::getSize() const
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +   if (handle != NULL)
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  AssertWarn(Closed != currentStatus, "File::getSize: file closed");
 | 
	
		
			
				|  |  | +	  AssertFatal(handle != NULL, "File::getSize: invalid file handle");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  if (Ok == currentStatus || EOS == currentStatus)
 | 
	
		
			
				|  |  | +	  {
 | 
	
		
			
				|  |  | +		 struct stat statData;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		 if(fstat(fileno((FILE*)handle), &statData) != 0)
 | 
	
		
			
				|  |  | +			return 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		 // return the size in bytes
 | 
	
		
			
				|  |  | +		 return statData.st_size;
 | 
	
		
			
				|  |  | +	  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  return 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     AssertWarn(Closed != currentStatus, "File::getSize: file closed");
 | 
	
		
			
				|  |  |     AssertFatal(buffer != NULL, "File::getSize: invalid file buffer");
 | 
	
		
			
				|  |  |     
 | 
	
	
		
			
				|  | @@ -228,6 +376,17 @@ U32 File::getSize() const
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::flush()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   AssertFatal(Closed != currentStatus, "File::flush: file closed");
 | 
	
		
			
				|  |  | +	   AssertFatal(handle != NULL, "File::flush: invalid file handle");
 | 
	
		
			
				|  |  | +	   AssertFatal(true == hasCapability(FileWrite), "File::flush: cannot flush a read-only file");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   if (fflush((FILE*)handle) != 0)
 | 
	
		
			
				|  |  | +		  return setStatus();
 | 
	
		
			
				|  |  | +	   else
 | 
	
		
			
				|  |  | +		  return currentStatus = Ok;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  |     AssertFatal(Closed != currentStatus, "File::flush: file closed");
 | 
	
		
			
				|  |  |     AssertFatal(buffer != NULL, "File::flush: invalid file buffer");
 | 
	
		
			
				|  |  |     AssertFatal(true == hasCapability(FileWrite), "File::flush: cannot flush a read-only file");
 | 
	
	
		
			
				|  | @@ -242,6 +401,22 @@ File::Status File::flush()
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::close()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   // check if it's already closed...
 | 
	
		
			
				|  |  | +	   if (Closed == currentStatus)
 | 
	
		
			
				|  |  | +		  return currentStatus;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // it's not, so close it...
 | 
	
		
			
				|  |  | +	   if (handle != NULL)
 | 
	
		
			
				|  |  | +	   {
 | 
	
		
			
				|  |  | +		  if (fclose((FILE*)handle) != 0)
 | 
	
		
			
				|  |  | +			 return setStatus();
 | 
	
		
			
				|  |  | +	   }
 | 
	
		
			
				|  |  | +	   handle = NULL;
 | 
	
		
			
				|  |  | +	   return currentStatus = Closed;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     // check if it's already closed...
 | 
	
		
			
				|  |  |     if (Closed == currentStatus)
 | 
	
		
			
				|  |  |        return currentStatus;
 | 
	
	
		
			
				|  | @@ -271,6 +446,24 @@ File::Status File::getStatus() const
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::setStatus()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   switch (errno)
 | 
	
		
			
				|  |  | +	   {
 | 
	
		
			
				|  |  | +		  case EACCES:   // permission denied
 | 
	
		
			
				|  |  | +			 currentStatus = IOError;
 | 
	
		
			
				|  |  | +			 break;
 | 
	
		
			
				|  |  | +		  case EBADF:   // Bad File Pointer
 | 
	
		
			
				|  |  | +		  case EINVAL:   // Invalid argument
 | 
	
		
			
				|  |  | +		  case ENOENT:   // file not found
 | 
	
		
			
				|  |  | +		  case ENAMETOOLONG:
 | 
	
		
			
				|  |  | +		  default:
 | 
	
		
			
				|  |  | +			 currentStatus = UnknownError;
 | 
	
		
			
				|  |  | +	   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   return currentStatus;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     currentStatus = UnknownError;
 | 
	
		
			
				|  |  |     return currentStatus;
 | 
	
		
			
				|  |  |  }
 | 
	
	
		
			
				|  | @@ -291,6 +484,32 @@ File::Status File::setStatus(File::Status status)
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::read(U32 _size, char *dst, U32 *bytesRead)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   AssertFatal(Closed != currentStatus, "File::read: file closed");
 | 
	
		
			
				|  |  | +	   AssertFatal(handle != NULL, "File::read: invalid file handle");
 | 
	
		
			
				|  |  | +	   AssertFatal(NULL != dst, "File::read: NULL destination pointer");
 | 
	
		
			
				|  |  | +	   AssertFatal(true == hasCapability(FileRead), "File::read: file lacks capability");
 | 
	
		
			
				|  |  | +	   AssertWarn(0 != size, "File::read: size of zero");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   if (Ok != currentStatus || 0 == size)
 | 
	
		
			
				|  |  | +		  return currentStatus;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // read from stream
 | 
	
		
			
				|  |  | +	   U32 nBytes = fread(dst, 1, size, (FILE*)handle);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // did we hit the end of the stream?
 | 
	
		
			
				|  |  | +	   if( nBytes != size)
 | 
	
		
			
				|  |  | +		  currentStatus = EOS;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // if bytesRead is a valid pointer, send number of bytes read there.
 | 
	
		
			
				|  |  | +	   if(bytesRead)
 | 
	
		
			
				|  |  | +		  *bytesRead = nBytes;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // successfully read size bytes
 | 
	
		
			
				|  |  | +	   return currentStatus;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     AssertFatal(Closed != currentStatus, "File::read: file closed");
 | 
	
		
			
				|  |  |     AssertFatal(buffer != NULL, "File::read: invalid file buffer");
 | 
	
		
			
				|  |  |     AssertFatal(NULL != dst, "File::read: NULL destination pointer");
 | 
	
	
		
			
				|  | @@ -341,6 +560,33 @@ File::Status File::read(U32 _size, char *dst, U32 *bytesRead)
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  File::Status File::write(U32 size, const char *src, U32 *bytesWritten)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (handle != NULL)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   AssertFatal(Closed != currentStatus, "File::write: file closed");
 | 
	
		
			
				|  |  | +	   AssertFatal(handle != NULL, "File::write: invalid file handle");
 | 
	
		
			
				|  |  | +	   AssertFatal(NULL != src, "File::write: NULL source pointer");
 | 
	
		
			
				|  |  | +	   AssertFatal(true == hasCapability(FileWrite), "File::write: file lacks capability");
 | 
	
		
			
				|  |  | +	   AssertWarn(0 != size, "File::write: size of zero");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   if ((Ok != currentStatus && EOS != currentStatus) || 0 == size)
 | 
	
		
			
				|  |  | +		  return currentStatus;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // write bytes to the stream
 | 
	
		
			
				|  |  | +	   U32 nBytes = fwrite(src, 1, size,(FILE*)handle);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // if we couldn't write everything, we've got a problem. set error status.
 | 
	
		
			
				|  |  | +	   if(nBytes != size)
 | 
	
		
			
				|  |  | +		  setStatus();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // if bytesWritten is a valid pointer, put number of bytes read there.
 | 
	
		
			
				|  |  | +	   if(bytesWritten)
 | 
	
		
			
				|  |  | +		  *bytesWritten = nBytes;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   // return current File status, whether good or ill.
 | 
	
		
			
				|  |  | +	   return currentStatus;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     AssertFatal(0, "File::write: Not supported on Android.");
 | 
	
		
			
				|  |  |     return setStatus();
 | 
	
		
			
				|  |  |  }
 | 
	
	
		
			
				|  | @@ -374,21 +620,73 @@ bool Platform::getFileTimes(const char *path, FileTime *createTime, FileTime *mo
 | 
	
		
			
				|  |  |     // and UNIX does not keep a record of a file's creation time anywhere.
 | 
	
		
			
				|  |  |     // So instead of creation time we return changed time,
 | 
	
		
			
				|  |  |     // just like the Linux platform impl does.
 | 
	
		
			
				|  |  | +  if (!path || !*path)
 | 
	
		
			
				|  |  | +	 return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  struct stat statData;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if (stat(path, &statData) == -1)
 | 
	
		
			
				|  |  | +	 return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if(createTime)
 | 
	
		
			
				|  |  | +	 *createTime = statData.st_ctime;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  if(modifyTime)
 | 
	
		
			
				|  |  | +	 *modifyTime = statData.st_mtime;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  return true;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     
 | 
	
		
			
				|  |  | -   if (!path || !*path) 
 | 
	
		
			
				|  |  | -      return false;
 | 
	
		
			
				|  |  | -   
 | 
	
		
			
				|  |  | -   Con::warnf("Platform::getFileTimes - Not supported on android.");
 | 
	
		
			
				|  |  | -   
 | 
	
		
			
				|  |  | -   return true;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool Platform::createPath(const char *file)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -   Con::warnf("Platform::createPath() - Not supported on android.");
 | 
	
		
			
				|  |  | -   return false;
 | 
	
		
			
				|  |  | +   //<Mat> needless console noise
 | 
	
		
			
				|  |  | +   //Con::warnf("creating path %s",file);
 | 
	
		
			
				|  |  | +   // if the path exists, we're done.
 | 
	
		
			
				|  |  | +   struct stat statData;
 | 
	
		
			
				|  |  | +   if( stat(file, &statData) == 0 )
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  return true;               // exists, rejoice.
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // get the parent path.
 | 
	
		
			
				|  |  | +   // we're not using basename because it's not thread safe.
 | 
	
		
			
				|  |  | +   const U32 len = dStrlen(file) + 1;
 | 
	
		
			
				|  |  | +   char parent[len];
 | 
	
		
			
				|  |  | +   bool isDirPath = false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   dSprintf(parent, len, "%s", file);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   if(parent[len - 2] == '/')
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  parent[len - 2] = '\0';    // cut off the trailing slash, if there is one
 | 
	
		
			
				|  |  | +	  isDirPath = true;          // we got a trailing slash, so file is a directory.
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // recusively create the parent path.
 | 
	
		
			
				|  |  | +   // only recurse if newpath has a slash that isn't a leading slash.
 | 
	
		
			
				|  |  | +   char *slash = dStrrchr(parent,'/');
 | 
	
		
			
				|  |  | +   if( slash && slash != parent)
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  // snip the path just after the last slash.
 | 
	
		
			
				|  |  | +	  slash[1] = '\0';
 | 
	
		
			
				|  |  | +	  // recusively create the parent path. fail if parent path creation failed.
 | 
	
		
			
				|  |  | +	  if(!Platform::createPath(parent))
 | 
	
		
			
				|  |  | +		 return false;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // create *file if it is a directory path.
 | 
	
		
			
				|  |  | +   if(isDirPath)
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  // try to create the directory
 | 
	
		
			
				|  |  | +	  if( mkdir(file, 0777) != 0) // app may reside in global apps dir, and so must be writable to all.
 | 
	
		
			
				|  |  | +		 return false;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   return true;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -475,6 +773,21 @@ bool Platform::isFile(const char *path)
 | 
	
		
			
				|  |  |     if (!path || !*path) 
 | 
	
		
			
				|  |  |        return false;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +   if (isCachePath(path))
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  // make sure we can stat the file
 | 
	
		
			
				|  |  | +	  struct stat statData;
 | 
	
		
			
				|  |  | +	  if( stat(path, &statData) < 0 )
 | 
	
		
			
				|  |  | +		 return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // now see if it's a regular file
 | 
	
		
			
				|  |  | +	  if( (statData.st_mode & S_IFMT) == S_IFREG)
 | 
	
		
			
				|  |  | +		 return true;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  return false;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     return android_IsFile(path);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -485,6 +798,20 @@ bool Platform::isDirectory(const char *path)
 | 
	
		
			
				|  |  |     if (!path || !*path) 
 | 
	
		
			
				|  |  |        return false;
 | 
	
		
			
				|  |  |     
 | 
	
		
			
				|  |  | +   if (isCachePath(path))
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  // make sure we can stat the file
 | 
	
		
			
				|  |  | +	  struct stat statData;
 | 
	
		
			
				|  |  | +	  if( stat(path, &statData) < 0 )
 | 
	
		
			
				|  |  | +		 return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // now see if it's a directory
 | 
	
		
			
				|  |  | +	  if( (statData.st_mode & S_IFMT) == S_IFDIR)
 | 
	
		
			
				|  |  | +		 return true;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  return false;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     return android_IsDir(path);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -494,6 +821,16 @@ S32 Platform::getFileSize(const char* pFilePath)
 | 
	
		
			
				|  |  |     if (!pFilePath || !*pFilePath) 
 | 
	
		
			
				|  |  |        return 0;
 | 
	
		
			
				|  |  |     
 | 
	
		
			
				|  |  | +   if (isCachePath(pFilePath))
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +	  struct stat statData;
 | 
	
		
			
				|  |  | +	  if( stat(pFilePath, &statData) < 0 )
 | 
	
		
			
				|  |  | +		 return 0;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	  // and return it's size in bytes
 | 
	
		
			
				|  |  | +	  return (S32)statData.st_size;
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     return android_GetFileSize(pFilePath);
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -525,9 +862,44 @@ inline bool isGoodDirectory(const char* path)
 | 
	
		
			
				|  |  |             && !Platform::isExcludedDirectory(name)); // not excluded
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +inline bool isGoodDirectoryCache(dirent* entry)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +   return (entry->d_type == DT_DIR                          // is a dir
 | 
	
		
			
				|  |  | +           && dStrcmp(entry->d_name,".") != 0                 // not here
 | 
	
		
			
				|  |  | +           && dStrcmp(entry->d_name,"..") != 0                // not parent
 | 
	
		
			
				|  |  | +           && !Platform::isExcludedDirectory(entry->d_name)); // not excluded
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool Platform::hasSubDirectory(const char *path) 
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (isCachePath(path))
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   DIR *dir;
 | 
	
		
			
				|  |  | +	   dirent *entry;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   dir = opendir(path);
 | 
	
		
			
				|  |  | +	   if(!dir)
 | 
	
		
			
				|  |  | +		  return false; // we got a bad path, so no, it has no subdirectory.
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   while( true )
 | 
	
		
			
				|  |  | +	   {
 | 
	
		
			
				|  |  | +		   entry = readdir(dir);
 | 
	
		
			
				|  |  | +		   if ( entry == NULL )
 | 
	
		
			
				|  |  | +			   break;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		  if(isGoodDirectoryCache(entry) )
 | 
	
		
			
				|  |  | +		  {
 | 
	
		
			
				|  |  | +			 closedir(dir);
 | 
	
		
			
				|  |  | +			 return true; // we have a subdirectory, that isnt on the exclude list.
 | 
	
		
			
				|  |  | +		  }
 | 
	
		
			
				|  |  | +	   }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   closedir(dir);
 | 
	
		
			
				|  |  | +	   return false; // either this dir had no subdirectories, or they were all on the exclude list.
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  	android_InitDirList(path);
 | 
	
		
			
				|  |  |  	char dir[80];
 | 
	
		
			
				|  |  |  	char pdir[255];
 | 
	
	
		
			
				|  | @@ -614,9 +986,107 @@ bool recurseDumpDirectories(const char *basePath, const char *path, Vector<Strin
 | 
	
		
			
				|  |  |       return true;
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | +bool recurseDumpDirectoriesCache(const char *basePath, const char *path, Vector<StringTableEntry> &directoryVector, S32 depth, bool noBasePath)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +   DIR *dir;
 | 
	
		
			
				|  |  | +   dirent *entry;
 | 
	
		
			
				|  |  | +   const U32 len = dStrlen(basePath) + dStrlen(path) + 2;
 | 
	
		
			
				|  |  | +   char pathbuf[len];
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // construct the file path
 | 
	
		
			
				|  |  | +   dSprintf(pathbuf, len, "%s/%s", basePath, path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // be sure it opens.
 | 
	
		
			
				|  |  | +   dir = opendir(pathbuf);
 | 
	
		
			
				|  |  | +   if(!dir)
 | 
	
		
			
				|  |  | +      return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // look inside the current directory
 | 
	
		
			
				|  |  | +   while( true )
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +       entry = readdir(dir);
 | 
	
		
			
				|  |  | +       if ( entry == NULL )
 | 
	
		
			
				|  |  | +           break;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // we just want directories.
 | 
	
		
			
				|  |  | +      if(!isGoodDirectoryCache(entry))
 | 
	
		
			
				|  |  | +         continue;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // TODO: better unicode file name handling
 | 
	
		
			
				|  |  | +      //      // Apple's file system stores unicode file names in decomposed form.
 | 
	
		
			
				|  |  | +      //      // ATSUI will not reliably draw out just the accent character by itself,
 | 
	
		
			
				|  |  | +      //      // so our text renderer has no chance of rendering decomposed form unicode.
 | 
	
		
			
				|  |  | +      //      // We have to convert the entry name to precomposed normalized form.
 | 
	
		
			
				|  |  | +      //      CFStringRef cfdname = CFStringCreateWithCString(NULL,entry->d_name,kCFStringEncodingUTF8);
 | 
	
		
			
				|  |  | +      //      CFMutableStringRef cfentryName = CFStringCreateMutableCopy(NULL,0,cfdname);
 | 
	
		
			
				|  |  | +      //      CFStringNormalize(cfentryName,kCFStringNormalizationFormC);
 | 
	
		
			
				|  |  | +      //
 | 
	
		
			
				|  |  | +      //      U32 entryNameLen = CFStringGetLength(cfentryName) * 4 + 1;
 | 
	
		
			
				|  |  | +      //      char entryName[entryNameLen];
 | 
	
		
			
				|  |  | +      //      CFStringGetCString(cfentryName, entryName, entryNameLen, kCFStringEncodingUTF8);
 | 
	
		
			
				|  |  | +      //      entryName[entryNameLen-1] = NULL; // sometimes, CFStringGetCString() doesn't null terminate.
 | 
	
		
			
				|  |  | +      //      CFRelease(cfentryName);
 | 
	
		
			
				|  |  | +      //      CFRelease(cfdname);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // construct the new path string, we'll need this below.
 | 
	
		
			
				|  |  | +      const U32 newpathlen = dStrlen(path) + dStrlen(entry->d_name) + 2;
 | 
	
		
			
				|  |  | +      char newpath[newpathlen];
 | 
	
		
			
				|  |  | +      if(dStrlen(path) > 0)
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +          dSprintf(newpath, newpathlen, "%s/%s", path, entry->d_name);
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      else
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +         dSprintf(newpath, newpathlen, "%s", entry->d_name);
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // we have a directory, add it to the list.
 | 
	
		
			
				|  |  | +      if( noBasePath )
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +         directoryVector.push_back(StringTable->insert(newpath));
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      else
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +         const U32 fullpathlen = dStrlen(basePath) + dStrlen(newpath) + 2;
 | 
	
		
			
				|  |  | +         char fullpath[fullpathlen];
 | 
	
		
			
				|  |  | +         dSprintf(fullpath, fullpathlen, "%s/%s",basePath,newpath);
 | 
	
		
			
				|  |  | +         directoryVector.push_back(StringTable->insert(fullpath));
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // and recurse into it, unless we've run out of depth
 | 
	
		
			
				|  |  | +      if( depth != 0) // passing a val of -1 as the recurse depth means go forever
 | 
	
		
			
				|  |  | +         recurseDumpDirectoriesCache(basePath, newpath, directoryVector, depth-1, noBasePath);
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +   closedir(dir);
 | 
	
		
			
				|  |  | +   return true;
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool Platform::dumpDirectories(const char *path, Vector<StringTableEntry> &directoryVector, S32 depth, bool noBasePath)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (isCachePath(path))
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +	   PROFILE_START(dumpDirectories);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   ResourceManager->initExcludedDirectories();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   const S32 len = dStrlen(path)+1;
 | 
	
		
			
				|  |  | +	   char newpath[len];
 | 
	
		
			
				|  |  | +	   dSprintf(newpath, len, "%s", path);
 | 
	
		
			
				|  |  | +	   if(newpath[len - 1] == '/')
 | 
	
		
			
				|  |  | +		  newpath[len - 1] = '\0'; // cut off the trailing slash, if there is one
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		// Insert base path to follow what Windows does.
 | 
	
		
			
				|  |  | +		if ( !noBasePath )
 | 
	
		
			
				|  |  | +			directoryVector.push_back(StringTable->insert(newpath));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		bool ret = recurseDumpDirectoriesCache(newpath, "", directoryVector, depth, noBasePath);
 | 
	
		
			
				|  |  | +	   PROFILE_END();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   return ret;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     PROFILE_START(dumpDirectories);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |     ResourceManager->initExcludedDirectories();
 | 
	
	
		
			
				|  | @@ -712,10 +1182,82 @@ static bool recurseDumpPath(const char* curPath, Vector<Platform::FileInfo>& fil
 | 
	
		
			
				|  |  |     
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +//-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  | +static bool recurseDumpPathCache(const char* curPath, Vector<Platform::FileInfo>& fileVector, U32 depth)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +   DIR *dir;
 | 
	
		
			
				|  |  | +   dirent *entry;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // be sure it opens.
 | 
	
		
			
				|  |  | +   dir = opendir(curPath);
 | 
	
		
			
				|  |  | +   if(!dir)
 | 
	
		
			
				|  |  | +      return false;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +   // look inside the current directory
 | 
	
		
			
				|  |  | +   while( true )
 | 
	
		
			
				|  |  | +   {
 | 
	
		
			
				|  |  | +       entry = readdir(dir);
 | 
	
		
			
				|  |  | +       if ( entry == NULL )
 | 
	
		
			
				|  |  | +           break;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // construct the full file path. we need this to get the file size and to recurse
 | 
	
		
			
				|  |  | +      const U32 len = dStrlen(curPath) + dStrlen(entry->d_name) + 2;
 | 
	
		
			
				|  |  | +      char pathbuf[len];
 | 
	
		
			
				|  |  | +      dSprintf( pathbuf, len, "%s/%s", curPath, entry->d_name);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      // ok, deal with directories and files seperately.
 | 
	
		
			
				|  |  | +      if( entry->d_type == DT_DIR )
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +         if( depth == 0)
 | 
	
		
			
				|  |  | +            continue;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +         // filter out dirs we dont want.
 | 
	
		
			
				|  |  | +         if( !isGoodDirectoryCache(entry) )
 | 
	
		
			
				|  |  | +            continue;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +         // recurse into the dir
 | 
	
		
			
				|  |  | +         recurseDumpPathCache( pathbuf, fileVector, depth-1);
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +      else
 | 
	
		
			
				|  |  | +      {
 | 
	
		
			
				|  |  | +         //add the file entry to the list
 | 
	
		
			
				|  |  | +         // unlike recurseDumpDirectories(), we need to return more complex info here.
 | 
	
		
			
				|  |  | +          //<Mat> commented this out in case we ever want a dir file printout again
 | 
	
		
			
				|  |  | +          //printf( "File Name: %s ", entry->d_name );
 | 
	
		
			
				|  |  | +         const U32 fileSize = Platform::getFileSize(pathbuf);
 | 
	
		
			
				|  |  | +         fileVector.increment();
 | 
	
		
			
				|  |  | +         Platform::FileInfo& rInfo = fileVector.last();
 | 
	
		
			
				|  |  | +         rInfo.pFullPath = StringTable->insert(curPath);
 | 
	
		
			
				|  |  | +         rInfo.pFileName = StringTable->insert(entry->d_name);
 | 
	
		
			
				|  |  | +         rInfo.fileSize  = fileSize;
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +   }
 | 
	
		
			
				|  |  | +   closedir(dir);
 | 
	
		
			
				|  |  | +   return true;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  //-----------------------------------------------------------------------------
 | 
	
		
			
				|  |  |  bool Platform::dumpPath(const char *path, Vector<Platform::FileInfo>& fileVector, S32 depth)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | +	if (isCachePath(path))
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		PROFILE_START(dumpPath);
 | 
	
		
			
				|  |  | +		const S32 len = dStrlen(path) + 1;
 | 
	
		
			
				|  |  | +	   char newpath[len];
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +		dSprintf(newpath, len, "%s", path);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   if(newpath[len - 2] == '/')
 | 
	
		
			
				|  |  | +		  newpath[len - 2] = '\0'; // cut off the trailing slash, if there is one
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   bool ret = recurseDumpPathCache( newpath, fileVector, depth);
 | 
	
		
			
				|  |  | +	   PROFILE_END();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	   return ret;
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |     PROFILE_START(dumpPath);
 | 
	
		
			
				|  |  |     char apath[80];
 | 
	
		
			
				|  |  |     if (path[0] == '/')
 |