Horde3D
http://horde3d.org/forums/

Asynchronous file loading
http://horde3d.org/forums/viewtopic.php?f=8&t=969
Page 1 of 1

Author:  marciano [ 04.10.2009, 09:06 ]
Post subject:  Asynchronous file loading

This is a small test we did once: non-blocking/asynchronous file loading. While a new resource is read from disk, at the same time the resource that was loaded before is parsed. In theory this should bring some speedups, especially when reading from slow media. However, I did not really notice any improvement when loading a limited amount of data from HDD (HDDs are much faster than optical media). Furthermore, the async loading code is very platform dependent and only works on Windows right now (Linux has some special functions as well). So we decided to not put it into the h3dutLoadResourcesFromDisk function, to keep the code there simple. Anyway, maybe the code is useful for someone, so I post it here:

Code:
#ifdef PLATFORM_WIN  // Do asynchronous file loading on Windows

   HANDLE hFile = INVALID_HANDLE_VALUE;
   OVERLAPPED overlapped;
   char *data0 = 0x0, *data1 = 0x0;
   int size0 = 0, size1 = 0;
   int res0 = 0, res1 = 0;
   
   while( h3dQueryUnloadedResource( 0 ) != 0 )
   {
      // Wait until previous file reading request has finished (if any)
      if( hFile != INVALID_HANDLE_VALUE )
      {
         if( !HasOverlappedIoCompleted( &overlapped ) ) continue;
         CloseHandle( hFile ); hFile = INVALID_HANDLE_VALUE;
      }

      // Start reading next resource
      res1 = h3dQueryUnloadedResource( res0 == 0 ? 0 : 1 );
      if( res1 != 0 )
      {
         // Loop over search paths and try to open files
         for( unsigned int i = 0; i < dirs.size(); ++i )
         {
            string fileName = dirs[i] + resourcePaths[h3dGetResType( res1 )] + "/" + h3dGetResName( res1 );
            hFile = CreateFile( fileName.c_str(), GENERIC_READ, 0, 0, OPEN_EXISTING,
                                FILE_ATTRIBUTE_NORMAL | FILE_FLAG_OVERLAPPED | FILE_FLAG_SEQUENTIAL_SCAN, 0 );
            if( hFile != INVALID_HANDLE_VALUE ) break;
         }
         
         if( hFile != INVALID_HANDLE_VALUE )
         {
            // Start asynchronous reading
            ZeroMemory( &overlapped, sizeof( OVERLAPPED ) );
            size1 = GetFileSize( hFile, 0x0 );
            data1 = new char[size1 + 1];
            ReadFile( hFile, data1, size1, 0x0, &overlapped );
         }
         else
         {
            // Tell engine that file could not be found
            h3dLoadResource( res1, 0x0, 0 );
            result = false;
            data1 = 0x0; size1 = 0; res1 = 0;
         }
      }

      // Load current resource
      if( res0 != 0 )
      {
         data0[size0] = '\0';  // NULL-terminate for XML
         result &= h3dLoadResource( res0, data0, size0 + 1 );
         delete[] data0;
      }

      // Set next resource as current resource
      data0 = data1; size0 = size1; res0 = res1;
   }

#else  // Standard (blocking) file loading

Author:  swiftcoder [ 04.10.2009, 12:44 ]
Post subject:  Re: Asynchronous file loading

Non-blocking file I/O on Linux/Mac is just a matter of putting the file descriptor in non-blocking mode (using fnctl), and then calling read() until it stops returning EWOULDBLOCK.

While you probably won't see any performance gains over blocking I/O, the important part is that you can continue rendering while loading, all without spawning a huge number of threads.

Author:  swiftcoder [ 15.10.2009, 16:12 ]
Post subject:  Re: Asynchronous file loading

Thinking about it, a simpler way to perform asynchronous I/O on Mac/linux is to open a file descriptor as normal, and use select() to determine when it is ready to read from (just as you would do with a socket).

Author:  marciano [ 17.10.2009, 10:36 ]
Post subject:  Re: Asynchronous file loading

Thanks for the hints swiftcoder!

If someone wants to provide an asynchronous version for linux/mac we can integrate it. However, as stated before, it does not bring a huge advantage when just loading from HDD.

Page 1 of 1 All times are UTC + 1 hour
Powered by phpBB® Forum Software © phpBB Group
https://www.phpbb.com/