git-svn-id: https://svn.code.sf.net/p/flaim/code/trunk@585 0109f412-320b-0410-ab79-c3e0c5ffbbe6
1821 lines
41 KiB
C++
1821 lines
41 KiB
C++
//------------------------------------------------------------------------------
|
|
// Desc: Contains the methods for the F_FileHdl class for UNIX.
|
|
//
|
|
// Tabs: 3
|
|
//
|
|
// Copyright (c) 1999-2006 Novell, Inc. All Rights Reserved.
|
|
//
|
|
// This program is free software; you can redistribute it and/or
|
|
// modify it under the terms of version 2 of the GNU General Public
|
|
// License as published by the Free Software Foundation.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program; if not, contact Novell, Inc.
|
|
//
|
|
// To contact Novell about this file by physical or electronic mail,
|
|
// you may find current contact information at www.novell.com
|
|
//
|
|
// $Id: fposix.cpp 3123 2006-01-24 17:19:50 -0700 (Tue, 24 Jan 2006) ahodgkinson $
|
|
//------------------------------------------------------------------------------
|
|
|
|
#include "ftksys.h"
|
|
|
|
#if defined( FLM_UNIX) || defined( FLM_LIBC_NLM)
|
|
|
|
#ifdef FLM_AIX
|
|
#ifndef _LARGE_FILES
|
|
#define _LARGE_FILES
|
|
#endif
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#include <sys/types.h>
|
|
#if !defined( FLM_OSX) && !defined( FLM_LIBC_NLM)
|
|
#include <aio.h>
|
|
#endif
|
|
|
|
#include <fcntl.h>
|
|
|
|
#if defined( FLM_SOLARIS)
|
|
#include <sys/statvfs.h>
|
|
#elif defined( FLM_LINUX)
|
|
#include <sys/vfs.h>
|
|
#elif defined( FLM_OSF)
|
|
|
|
// Tru64 4.0 does not have this declaration. Tru64 5.0 renames statfs
|
|
// in vague ways, so we put these declarations before including
|
|
// <sys/stat.h>
|
|
|
|
// DSS NOTE: statfs declaration below conflicts with one found in
|
|
// sys/mount.h header file, so I commented it out. This was when I
|
|
// compiled using the GNU compiler.
|
|
|
|
struct statfs;
|
|
#include <sys/mount.h>
|
|
#elif defined( FLM_LIBC_NLM)
|
|
#define pread pread64
|
|
#define pwrite pwrite64
|
|
#define ftruncate ftruncate64
|
|
#endif
|
|
|
|
#ifdef FLM_LINUX
|
|
static FLMUINT gv_uiLinuxMajorVer = 0;
|
|
static FLMUINT gv_uiLinuxMinorVer = 0;
|
|
static FLMUINT gv_uiLinuxRevision = 0;
|
|
#endif
|
|
|
|
static pthread_mutex_t gv_atomicMutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
/******************************************************************************
|
|
Desc:
|
|
*******************************************************************************/
|
|
F_FileHdl::F_FileHdl()
|
|
{
|
|
m_bFileOpened = FALSE;
|
|
m_bDeleteOnRelease = FALSE;
|
|
m_bOpenedReadOnly = FALSE;
|
|
m_pszFileName = NULL;
|
|
|
|
m_fd = -1;
|
|
m_uiExtendSize = 0;
|
|
m_uiBytesPerSector = 0;
|
|
m_ui64NotOnSectorBoundMask = 0;
|
|
m_ui64GetSectorBoundMask = 0;
|
|
m_uiExtendSize = 0;
|
|
m_ui64CurrentPos = 0;
|
|
m_bDoDirectIO = FALSE;
|
|
m_bOpenedInAsyncMode = FALSE;
|
|
m_pucAlignedBuff = NULL;
|
|
m_uiAlignedBuffSize = 0;
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc:
|
|
******************************************************************************/
|
|
F_FileHdl::~F_FileHdl()
|
|
{
|
|
if( m_bFileOpened)
|
|
{
|
|
(void)close();
|
|
}
|
|
|
|
if( m_pucAlignedBuff)
|
|
{
|
|
free( m_pucAlignedBuff);
|
|
}
|
|
|
|
if (m_pszFileName)
|
|
{
|
|
f_free( &m_pszFileName);
|
|
}
|
|
}
|
|
|
|
/***************************************************************************
|
|
Desc: Open or create a file.
|
|
***************************************************************************/
|
|
RCODE F_FileHdl::openOrCreate(
|
|
const char * pszFileName,
|
|
FLMUINT uiAccess,
|
|
FLMBOOL bCreateFlag)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMBOOL bDoDirectIO = FALSE;
|
|
char szSaveFileName[ F_PATH_MAX_SIZE];
|
|
int openFlags = O_RDONLY;
|
|
IF_FileSystem * pFileSystem = f_getFileSysPtr();
|
|
|
|
#if defined( FLM_LINUX) || defined( FLM_SOLARIS) || defined( FLM_OSX)
|
|
bDoDirectIO = (uiAccess & FLM_IO_DIRECT) ? TRUE : FALSE;
|
|
#endif
|
|
|
|
// HPUX needs this defined to access files larger than 2 GB. The Linux
|
|
// man pages *say* it's needed although as of Suse 9.1 it actually
|
|
// isn't. Including this flag on Linux anyway just it case...
|
|
|
|
#if defined( FLM_HPUX) || defined( FLM_LINUX)
|
|
openFlags |= O_LARGEFILE;
|
|
#endif
|
|
|
|
// Save the file name in case we have to create the directory
|
|
|
|
if( bCreateFlag && (uiAccess & FLM_IO_CREATE_DIR))
|
|
{
|
|
f_strcpy( szSaveFileName, pszFileName);
|
|
}
|
|
|
|
if( bCreateFlag)
|
|
{
|
|
openFlags |= O_CREAT;
|
|
if( uiAccess & FLM_IO_EXCL)
|
|
{
|
|
openFlags |= O_EXCL;
|
|
}
|
|
else
|
|
{
|
|
openFlags |= O_TRUNC;
|
|
}
|
|
}
|
|
|
|
if( !(uiAccess & FLM_IO_RDONLY))
|
|
{
|
|
openFlags |= O_RDWR;
|
|
}
|
|
|
|
if( !(uiAccess & FLM_IO_RDONLY))
|
|
{
|
|
openFlags |= O_RDWR;
|
|
}
|
|
|
|
// If doing direct IO, need to get the sector size.
|
|
|
|
if( bDoDirectIO)
|
|
{
|
|
// Can't do direct IO if the block size isn't a multiple of
|
|
// the sector size.
|
|
|
|
#if defined( FLM_LINUX)
|
|
{
|
|
FLMUINT uiMajor;
|
|
FLMUINT uiMinor;
|
|
FLMUINT uiRevision;
|
|
|
|
f_getLinuxKernelVersion( &uiMajor, &uiMinor, &uiRevision);
|
|
|
|
if( uiMajor > 2 || (uiMajor == 2 && uiMinor > 6) ||
|
|
(uiMajor == 2 && uiMinor == 6 && uiRevision >= 5))
|
|
{
|
|
openFlags |= O_DIRECT;
|
|
if( pFileSystem->canDoAsync())
|
|
{
|
|
m_bOpenedInAsyncMode = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bDoDirectIO = FALSE;
|
|
}
|
|
}
|
|
#elif defined( FLM_SOLARIS) || defined( FLM_OSX)
|
|
if( pFileSystem->canDoAsync())
|
|
{
|
|
m_bOpenedInAsyncMode = TRUE;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
Retry_Create:
|
|
|
|
// Try to create or open the file
|
|
|
|
if ((m_fd = ::open( pszFileName, openFlags, 0600)) == -1)
|
|
{
|
|
if ((errno == ENOENT) && (uiAccess & FLM_IO_CREATE_DIR))
|
|
{
|
|
char szTemp[ F_PATH_MAX_SIZE];
|
|
char szIoDirPath[ F_PATH_MAX_SIZE];
|
|
|
|
uiAccess &= ~FLM_IO_CREATE_DIR;
|
|
|
|
// Remove the file name for which we are creating the directory
|
|
|
|
if( RC_OK( pFileSystem->pathReduce( szSaveFileName,
|
|
szIoDirPath, szTemp)))
|
|
{
|
|
if( RC_OK( rc = pFileSystem->createDir( szIoDirPath)))
|
|
{
|
|
goto Retry_Create;
|
|
}
|
|
else
|
|
{
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
#ifdef FLM_LINUX
|
|
else if( errno == EINVAL && bDoDirectIO)
|
|
{
|
|
openFlags &= ~O_DIRECT;
|
|
bDoDirectIO = FALSE;
|
|
m_bOpenedInAsyncMode = FALSE;
|
|
goto Retry_Create;
|
|
}
|
|
#endif
|
|
|
|
rc = f_mapPlatformError( errno, NE_FLM_OPENING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
#if defined( FLM_SOLARIS)
|
|
if( bDoDirectIO)
|
|
{
|
|
directio( m_fd, DIRECTIO_ON);
|
|
}
|
|
#endif
|
|
|
|
#if defined( FLM_SOLARIS) || defined( FLM_LINUX)
|
|
m_uiBytesPerSector = DEV_BSIZE;
|
|
#else
|
|
{
|
|
struct stat filestats;
|
|
|
|
if( fstat( m_fd, &filestats) != 0)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_OPENING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
m_uiBytesPerSector = (FLMUINT)filestats.st_blksize;
|
|
}
|
|
#endif
|
|
|
|
m_ui64NotOnSectorBoundMask = m_uiBytesPerSector - 1;
|
|
m_ui64GetSectorBoundMask = ~m_ui64NotOnSectorBoundMask;
|
|
m_bDoDirectIO = bDoDirectIO;
|
|
|
|
Exit:
|
|
|
|
if( RC_BAD( rc))
|
|
{
|
|
m_fd = -1;
|
|
m_bDoDirectIO = FALSE;
|
|
m_bOpenedInAsyncMode = FALSE;
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Create a file
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::create(
|
|
const char * pszFileName,
|
|
FLMUINT uiIoFlags)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
f_assert( !m_bFileOpened);
|
|
|
|
if( m_bDeleteOnRelease)
|
|
{
|
|
// This file handle had better not been used for another file
|
|
// before. Otherwise, we will get a memory leak.
|
|
|
|
f_assert( m_pszFileName == NULL);
|
|
|
|
// Note: 'OpenOrCreate' will set m_pszFileName
|
|
|
|
if( RC_BAD( rc = f_alloc( F_PATH_MAX_SIZE, &m_pszFileName)))
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
f_strcpy( m_pszFileName, pszFileName);
|
|
}
|
|
|
|
if( RC_BAD( rc = openOrCreate( pszFileName, uiIoFlags, TRUE)))
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
m_bFileOpened = TRUE;
|
|
m_ui64CurrentPos = 0;
|
|
m_bOpenedExclusive = (uiIoFlags & FLM_IO_SH_DENYRW) ? TRUE : FALSE;
|
|
|
|
Exit:
|
|
|
|
if( RC_BAD( rc) && m_bDeleteOnRelease && m_pszFileName)
|
|
{
|
|
f_free( &m_pszFileName);
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc:
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::createUnique(
|
|
char * pszDirName,
|
|
const char * pszFileExtension,
|
|
FLMUINT uiIoFlags)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
char * pszTmp;
|
|
FLMBOOL bModext = TRUE;
|
|
FLMUINT uiBaseTime = 0;
|
|
FLMBYTE ucHighByte = 0;
|
|
char szFileName[ F_FILENAME_SIZE];
|
|
char * pszDirPath;
|
|
char szDirPath[ F_PATH_MAX_SIZE];
|
|
char szTmpPath[ F_PATH_MAX_SIZE];
|
|
FLMUINT uiCount;
|
|
IF_FileSystem * pFileSystem = f_getFileSysPtr();
|
|
|
|
f_assert( !m_bFileOpened);
|
|
f_memset( szFileName, 0, sizeof( szFileName));
|
|
|
|
if( m_bDeleteOnRelease)
|
|
{
|
|
// This file handle had better not been used for another file
|
|
// before. Otherwise, we will get a memory leak.
|
|
|
|
f_assert( !m_pszFileName);
|
|
}
|
|
|
|
if( !pszDirName || pszDirName[ 0] == '\0')
|
|
{
|
|
f_strcpy( szDirPath, "./");
|
|
}
|
|
else
|
|
{
|
|
f_strcpy( szDirPath, pszDirName);
|
|
}
|
|
pszDirPath = &szDirPath [0];
|
|
|
|
// Search backwards replacing trailing spaces with NULLs.
|
|
|
|
pszTmp = pszDirPath;
|
|
pszTmp += (f_strlen( pszTmp) - 1);
|
|
while (*pszTmp == ' ' && pszTmp >= pszDirPath)
|
|
{
|
|
*pszTmp = 0;
|
|
pszTmp--;
|
|
}
|
|
|
|
// Append a slash if one isn't already there
|
|
|
|
if (pszTmp >= pszDirPath && *pszTmp != '/')
|
|
{
|
|
pszTmp++;
|
|
*pszTmp++ = '/';
|
|
}
|
|
else
|
|
{
|
|
pszTmp++;
|
|
}
|
|
*pszTmp = 0;
|
|
|
|
if( pszFileExtension && f_strlen( pszFileExtension) >= 3)
|
|
{
|
|
bModext = FALSE;
|
|
}
|
|
|
|
uiCount = 0;
|
|
do
|
|
{
|
|
pFileSystem->pathCreateUniqueName( &uiBaseTime, szFileName,
|
|
pszFileExtension,
|
|
&ucHighByte, bModext);
|
|
|
|
f_strcpy( szTmpPath, pszDirPath);
|
|
pFileSystem->pathAppend( szTmpPath, szFileName);
|
|
if( m_pszFileName)
|
|
{
|
|
f_free( &m_pszFileName);
|
|
}
|
|
|
|
rc = create( szTmpPath, uiIoFlags | FLM_IO_EXCL);
|
|
|
|
if (rc == NE_FLM_IO_DISK_FULL)
|
|
{
|
|
pFileSystem->deleteFile( pszDirPath);
|
|
goto Exit;
|
|
}
|
|
|
|
if( rc == NE_FLM_IO_PATH_NOT_FOUND || rc == NE_FLM_IO_INVALID_PASSWORD)
|
|
{
|
|
goto Exit;
|
|
}
|
|
} while ((rc != NE_FLM_OK) && (uiCount++ < 10));
|
|
|
|
// Check if the path was created
|
|
|
|
if( uiCount >= 10 && rc != NE_FLM_OK)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_PATH_CREATE_FAILURE);
|
|
goto Exit;
|
|
}
|
|
|
|
m_bFileOpened = TRUE;
|
|
m_bOpenedExclusive = (uiIoFlags & FLM_IO_SH_DENYRW) ? TRUE : FALSE;
|
|
|
|
// Created file name needs to be returned.
|
|
|
|
f_strcpy( pszDirName, szTmpPath);
|
|
|
|
Exit:
|
|
|
|
if( RC_BAD( rc) && m_pszFileName)
|
|
{
|
|
f_free( &m_pszFileName);
|
|
m_pszFileName = NULL;
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Open a file
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::open(
|
|
const char * pszFileName,
|
|
FLMUINT uiIoFlags)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
f_assert( !m_bFileOpened);
|
|
|
|
if( m_bDeleteOnRelease)
|
|
{
|
|
// This file handle had better not been used for another file
|
|
// before. Otherwise, we will get a memory leak.
|
|
|
|
f_assert( !m_pszFileName);
|
|
|
|
if( RC_BAD( rc = f_alloc( F_PATH_MAX_SIZE, &m_pszFileName)))
|
|
{
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
if( RC_BAD( rc = openOrCreate( pszFileName, uiIoFlags, FALSE)))
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
m_bFileOpened = TRUE;
|
|
m_ui64CurrentPos = 0;
|
|
m_bOpenedReadOnly = (uiIoFlags & FLM_IO_RDONLY) ? TRUE : FALSE;
|
|
m_bOpenedExclusive = (uiIoFlags & FLM_IO_SH_DENYRW) ? TRUE : FALSE;
|
|
|
|
Exit:
|
|
|
|
if( RC_BAD( rc) && m_bDeleteOnRelease && m_pszFileName)
|
|
{
|
|
f_free( &m_pszFileName);
|
|
m_pszFileName = NULL;
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Close a file
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::close( void)
|
|
{
|
|
FLMBOOL bDeleteAllowed = TRUE;
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
if( !m_bFileOpened)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
::close( m_fd);
|
|
|
|
m_fd = -1;
|
|
m_bFileOpened = FALSE;
|
|
m_bOpenedReadOnly = FALSE;
|
|
m_bOpenedExclusive = FALSE;
|
|
|
|
if( m_bDeleteOnRelease)
|
|
{
|
|
f_assert( m_pszFileName);
|
|
|
|
if( bDeleteAllowed)
|
|
{
|
|
f_getFileSysPtr()->deleteFile( m_pszFileName);
|
|
}
|
|
m_bDeleteOnRelease = FALSE;
|
|
|
|
f_free( &m_pszFileName);
|
|
m_pszFileName = NULL;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Make sure all file data is safely on disk
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::flush( void)
|
|
{
|
|
f_assert( m_bFileOpened);
|
|
|
|
#ifdef FLM_SOLARIS
|
|
// Direct I/O on Solaris is ADVISORY, meaning that the
|
|
// operating system may or may not actually honor the
|
|
// option for some or all operations on a given file.
|
|
// Thus, the only way to guarantee that writes are on disk
|
|
// is to call fdatasync.
|
|
//
|
|
// If a process is killed (with SIGKILL or SIGTERM), the
|
|
// dirty cache buffers associated with open files will be discarded unless
|
|
// the process intercepts the signal and properly closes the files.
|
|
//
|
|
// NOTES FROM THE UNIX MAN PAGES ON SIGNALS
|
|
//
|
|
// When killing a process or series of processes, it is common sense
|
|
// to start trying with the least dangerous signal, SIGTERM. That way,
|
|
// programs that care about an orderly shutdown get the chance to follow
|
|
// the procedures that they have been designed to execute when getting
|
|
// the SIGTERM signal, such as cleaning up and closing open files. If you
|
|
// send a SIGKILL to a process, you remove any chance for the process
|
|
// to do a tidy cleanup and shutdown, which might have unfortunate
|
|
// consequences.
|
|
|
|
if( fdatasync( m_fd) != 0)
|
|
{
|
|
return( f_mapPlatformError( errno, NE_FLM_FLUSHING_FILE));
|
|
}
|
|
#else
|
|
if( !m_bDoDirectIO)
|
|
{
|
|
#ifdef FLM_OSX
|
|
if( fsync( m_fd) != 0)
|
|
#else
|
|
if( fdatasync( m_fd) != 0)
|
|
#endif
|
|
{
|
|
return( f_mapPlatformError( errno, NE_FLM_FLUSHING_FILE));
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return( NE_FLM_OK);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Desc: Position and do a single read operation.
|
|
****************************************************************************/
|
|
RCODE F_FileHdl::doOneRead(
|
|
FLMUINT64 ui64ReadOffset,
|
|
FLMUINT uiBytesToRead,
|
|
void * pvReadBuffer,
|
|
FLMUINT * puiBytesRead)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMINT iTmp;
|
|
FLMUINT uiBytesRead = 0;
|
|
|
|
if( !m_bOpenedInAsyncMode)
|
|
{
|
|
if( (iTmp = pread( m_fd, pvReadBuffer,
|
|
uiBytesToRead, ui64ReadOffset)) == -1)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
uiBytesRead = (FLMUINT)iTmp;
|
|
}
|
|
else
|
|
#ifdef FLM_LIBC_NLM
|
|
{
|
|
rc = RC_SET_AND_ASSERT( NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
#else
|
|
{
|
|
struct aiocb tmpAio;
|
|
struct aiocb * ppAioList[ 1];
|
|
|
|
f_memset( &tmpAio, 0, sizeof( struct aiocb));
|
|
tmpAio.aio_lio_opcode = LIO_READ;
|
|
tmpAio.aio_sigevent.sigev_notify = SIGEV_NONE;
|
|
tmpAio.aio_fildes = m_fd;
|
|
tmpAio.aio_offset = ui64ReadOffset;
|
|
tmpAio.aio_nbytes = uiBytesToRead;
|
|
tmpAio.aio_buf = pvReadBuffer;
|
|
|
|
if( aio_read( &tmpAio) != 0)
|
|
{
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
ppAioList[ 0] = &tmpAio;
|
|
|
|
for( ;;)
|
|
{
|
|
FLMINT iAsyncResult;
|
|
|
|
if( aio_suspend( ppAioList, 1, NULL) != 0)
|
|
{
|
|
if( errno == EINTR)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
iAsyncResult = aio_error( &tmpAio);
|
|
|
|
if( !iAsyncResult)
|
|
{
|
|
if( (iAsyncResult = aio_return( &tmpAio)) < 0)
|
|
{
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
uiBytesRead = (FLMUINT)iAsyncResult;
|
|
break;
|
|
}
|
|
|
|
if( iAsyncResult == EINTR || iAsyncResult == EINPROGRESS)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( iAsyncResult, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if( uiBytesRead < uiBytesToRead)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_END_OF_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
if( puiBytesRead)
|
|
{
|
|
*puiBytesRead = uiBytesRead;
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Read from a file
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::directRead(
|
|
FLMUINT64 ui64ReadOffset,
|
|
FLMUINT uiBytesToRead,
|
|
void * pvBuffer,
|
|
FLMUINT * puiBytesRead)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMUINT uiBytesRead;
|
|
FLMBYTE * pucReadBuffer;
|
|
FLMBYTE * pucDestBuffer;
|
|
FLMUINT uiMaxBytesToRead;
|
|
FLMBOOL bHitEOF;
|
|
|
|
f_assert( m_bFileOpened);
|
|
f_assert( m_bDoDirectIO);
|
|
|
|
if( puiBytesRead)
|
|
{
|
|
*puiBytesRead = 0;
|
|
}
|
|
|
|
if( ui64ReadOffset == FLM_IO_CURRENT_POS)
|
|
{
|
|
ui64ReadOffset = m_ui64CurrentPos;
|
|
}
|
|
|
|
// This loop does multiple reads (if necessary) to get all of the
|
|
// data. It uses aligned buffers and reads at sector offsets.
|
|
|
|
pucDestBuffer = (FLMBYTE *)pvBuffer;
|
|
for (;;)
|
|
{
|
|
// See if we are using an aligned buffer. If not, allocate
|
|
// one (if not already allocated), and use it.
|
|
|
|
if( (ui64ReadOffset & m_ui64NotOnSectorBoundMask) ||
|
|
(uiBytesToRead & m_ui64NotOnSectorBoundMask) ||
|
|
(((FLMUINT64)(FLMUINT)pucDestBuffer) & m_ui64NotOnSectorBoundMask))
|
|
{
|
|
if( !m_pucAlignedBuff)
|
|
{
|
|
if( RC_BAD( rc = allocAlignedBuffer()))
|
|
{
|
|
goto Exit;
|
|
}
|
|
}
|
|
pucReadBuffer = m_pucAlignedBuff;
|
|
|
|
// Must read enough bytes to cover all of the sectors that
|
|
// contain the data we are trying to read. The value of
|
|
// (ui64ReadOffset & m_ui64NotOnSectorBoundMask) will give us the
|
|
// number of additional bytes that are in the sector prior to
|
|
// the read offset. We then round that up to the next sector
|
|
// to get the total number of bytes we are going to read.
|
|
|
|
uiMaxBytesToRead = (FLMUINT)roundToNextSector(
|
|
uiBytesToRead + (ui64ReadOffset & m_ui64NotOnSectorBoundMask));
|
|
|
|
// Can't read more than the aligned buffer will hold.
|
|
|
|
if( uiMaxBytesToRead > m_uiAlignedBuffSize)
|
|
{
|
|
uiMaxBytesToRead = m_uiAlignedBuffSize;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uiMaxBytesToRead = (FLMUINT)roundToNextSector( uiBytesToRead);
|
|
f_assert( uiMaxBytesToRead >= uiBytesToRead);
|
|
pucReadBuffer = pucDestBuffer;
|
|
}
|
|
|
|
bHitEOF = FALSE;
|
|
|
|
if( RC_BAD( rc = doOneRead( truncateToPrevSector( ui64ReadOffset),
|
|
uiMaxBytesToRead, pucReadBuffer, &uiBytesRead)))
|
|
{
|
|
if( rc != NE_FLM_IO_END_OF_FILE)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
rc = NE_FLM_OK;
|
|
}
|
|
|
|
if( uiBytesRead < uiMaxBytesToRead)
|
|
{
|
|
bHitEOF = TRUE;
|
|
}
|
|
|
|
// If the offset we want to read from is not on a sector
|
|
// boundary, increment the read buffer pointer to the
|
|
// offset where the data we need starts and decrement the
|
|
// bytes read by the difference between the start of the
|
|
// sector and the actual read offset.
|
|
|
|
if( ui64ReadOffset & m_ui64NotOnSectorBoundMask)
|
|
{
|
|
pucReadBuffer += (ui64ReadOffset & m_ui64NotOnSectorBoundMask);
|
|
f_assert( uiBytesRead >= m_uiBytesPerSector);
|
|
uiBytesRead -= (FLMUINT)(ui64ReadOffset & m_ui64NotOnSectorBoundMask);
|
|
}
|
|
|
|
// If bytes read is more than we actually need, truncate it back
|
|
// so that we only copy what we actually need.
|
|
|
|
if( uiBytesRead > uiBytesToRead)
|
|
{
|
|
uiBytesRead = uiBytesToRead;
|
|
}
|
|
|
|
uiBytesToRead -= uiBytesRead;
|
|
|
|
if( puiBytesRead)
|
|
{
|
|
(*puiBytesRead) += uiBytesRead;
|
|
}
|
|
|
|
m_ui64CurrentPos = ui64ReadOffset + uiBytesRead;
|
|
|
|
// If using a different buffer for reading, copy the
|
|
// data read into the destination buffer.
|
|
|
|
if( pucDestBuffer != pucReadBuffer)
|
|
{
|
|
f_memcpy( pucDestBuffer, pucReadBuffer, uiBytesRead);
|
|
}
|
|
|
|
if( !uiBytesToRead)
|
|
{
|
|
break;
|
|
}
|
|
|
|
// Still more to read - did we hit EOF above?
|
|
|
|
if( bHitEOF)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_END_OF_FILE);
|
|
break;
|
|
}
|
|
|
|
pucDestBuffer += uiBytesRead;
|
|
ui64ReadOffset += uiBytesRead;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Read from a file
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::read(
|
|
FLMUINT64 ui64ReadOffset,
|
|
FLMUINT uiBytesToRead,
|
|
void * pvBuffer,
|
|
FLMUINT * puiBytesRead)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMINT iBytesRead;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
if( m_bDoDirectIO || m_bOpenedInAsyncMode)
|
|
{
|
|
rc = directRead( ui64ReadOffset, uiBytesToRead,
|
|
pvBuffer, puiBytesRead);
|
|
goto Exit;
|
|
}
|
|
|
|
if( ui64ReadOffset == FLM_IO_CURRENT_POS)
|
|
{
|
|
ui64ReadOffset = m_ui64CurrentPos;
|
|
}
|
|
|
|
if( (iBytesRead = pread( m_fd, pvBuffer,
|
|
uiBytesToRead, ui64ReadOffset)) == -1)
|
|
{
|
|
rc = f_mapPlatformError(errno, NE_FLM_READING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
if( puiBytesRead)
|
|
{
|
|
*puiBytesRead = (FLMUINT)iBytesRead;
|
|
}
|
|
|
|
m_ui64CurrentPos = ui64ReadOffset + (FLMUINT)iBytesRead;
|
|
|
|
if( (FLMUINT)iBytesRead < uiBytesToRead)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_END_OF_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Note: This function assumes that the pvBuffer that is passed in is
|
|
a multiple of a the sector size.
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::sectorRead(
|
|
FLMUINT64 ui64ReadOffset,
|
|
FLMUINT uiBytesToRead,
|
|
void * pvBuffer,
|
|
FLMUINT * puiBytesRead)
|
|
{
|
|
if( m_bDoDirectIO || m_bOpenedInAsyncMode)
|
|
{
|
|
return( directRead( ui64ReadOffset, uiBytesToRead,
|
|
pvBuffer, puiBytesRead));
|
|
}
|
|
else
|
|
{
|
|
return( read( ui64ReadOffset, uiBytesToRead, pvBuffer, puiBytesRead));
|
|
}
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Sets current position of file.
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::seek(
|
|
FLMUINT64 ui64Offset,
|
|
FLMINT iWhence,
|
|
FLMUINT64 * pui64NewOffset)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
switch( iWhence)
|
|
{
|
|
case FLM_IO_SEEK_CUR:
|
|
{
|
|
m_ui64CurrentPos += ui64Offset;
|
|
break;
|
|
}
|
|
|
|
case FLM_IO_SEEK_SET:
|
|
{
|
|
m_ui64CurrentPos = ui64Offset;
|
|
break;
|
|
}
|
|
|
|
case FLM_IO_SEEK_END:
|
|
{
|
|
if( RC_BAD( rc = size( &m_ui64CurrentPos)))
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
rc = RC_SET_AND_ASSERT( NE_FLM_NOT_IMPLEMENTED);
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
if( pui64NewOffset)
|
|
{
|
|
*pui64NewOffset = m_ui64CurrentPos;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Return the size of the file
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::size(
|
|
FLMUINT64 * pui64Size)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
struct stat statBuf;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
if( fstat( m_fd, &statBuf) == -1)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_GETTING_FILE_SIZE);
|
|
goto Exit;
|
|
}
|
|
|
|
*pui64Size = statBuf.st_size;
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc:
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::tell(
|
|
FLMUINT64 * pui64Offset)
|
|
{
|
|
f_assert( m_bFileOpened);
|
|
*pui64Offset = m_ui64CurrentPos;
|
|
return( NE_FLM_OK);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Truncate the file to the indicated size
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::truncate(
|
|
FLMUINT64 ui64Size)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
if( ftruncate( m_fd, ui64Size) == -1)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_TRUNCATING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Write to a file
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::write(
|
|
FLMUINT64 ui64WriteOffset,
|
|
FLMUINT uiBytesToWrite,
|
|
const void * pvBuffer,
|
|
FLMUINT * puiBytesWrittenRV)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMINT iBytesWritten = 0;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
if( m_bDoDirectIO || m_bOpenedInAsyncMode)
|
|
{
|
|
rc = directWrite( ui64WriteOffset, uiBytesToWrite, pvBuffer,
|
|
NULL, puiBytesWrittenRV);
|
|
goto Exit;
|
|
}
|
|
|
|
if( ui64WriteOffset == FLM_IO_CURRENT_POS)
|
|
{
|
|
ui64WriteOffset = m_ui64CurrentPos;
|
|
}
|
|
|
|
if( (iBytesWritten = pwrite( m_fd, pvBuffer,
|
|
uiBytesToWrite, ui64WriteOffset)) == -1)
|
|
{
|
|
rc = f_mapPlatformError(errno, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
if( puiBytesWrittenRV)
|
|
{
|
|
*puiBytesWrittenRV = (FLMUINT)iBytesWritten;
|
|
}
|
|
|
|
m_ui64CurrentPos = ui64WriteOffset + (FLMUINT)iBytesWritten;
|
|
|
|
if( (FLMUINT)iBytesWritten < uiBytesToWrite)
|
|
{
|
|
rc = RC_SET_AND_ASSERT( NE_FLM_IO_DISK_FULL);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Allocate an aligned buffer.
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::allocAlignedBuffer( void)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
if( m_pucAlignedBuff)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
// Allocate at least 64K - this will handle most read and write
|
|
// operations and will also be a multiple of the sector size most of
|
|
// the time. The calculation below rounds it up to the next sector
|
|
// boundary if it is not already on one.
|
|
|
|
m_uiAlignedBuffSize = (FLMUINT)roundToNextSector( 64 * 1024);
|
|
f_assert( m_uiAlignedBuffSize >= m_uiBytesPerSector);
|
|
|
|
#if defined( FLM_SOLARIS)
|
|
if( (m_pucAlignedBuff = (FLMBYTE *)memalign(
|
|
sysconf( _SC_PAGESIZE), m_uiAlignedBuffSize)) == NULL)
|
|
#elif defined( FLM_OSX) || defined( FLM_LIBC_NLM)
|
|
if( (m_pucAlignedBuff = (FLMBYTE *)malloc( m_uiAlignedBuffSize)) == NULL)
|
|
#else
|
|
if( posix_memalign( (void **)&m_pucAlignedBuff,
|
|
sysconf( _SC_PAGESIZE), m_uiAlignedBuffSize) == -1)
|
|
#endif
|
|
{
|
|
m_uiAlignedBuffSize = 0;
|
|
rc = f_mapPlatformError( errno, NE_FLM_MEM);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Note: This routine assumes that the size of pvBuffer is a multiple of
|
|
sector size and can be used to write out full sectors. Even if
|
|
uiBytesToWrite does not account for full sectors, data from the
|
|
buffer will still be written out - a partial sector on disk will
|
|
not be preserved.
|
|
******************************************************************************/
|
|
RCODE F_FileHdl::directWrite(
|
|
FLMUINT64 ui64WriteOffset,
|
|
FLMUINT uiBytesToWrite,
|
|
const void * pvBuffer,
|
|
IF_IOBuffer * pBufferObj,
|
|
FLMUINT * puiBytesWrittenRV)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
FLMUINT uiBytesRead = 0;
|
|
FLMBYTE * pucWriteBuffer;
|
|
FLMBYTE * pucSrcBuffer;
|
|
FLMUINT uiTotalBytesToWrite;
|
|
FLMUINT uiBufBytesToWrite;
|
|
FLMUINT uiBytesWritten = 0;
|
|
FLMBOOL bWaitForWrite = (pBufferObj == NULL)
|
|
? TRUE
|
|
: FALSE;
|
|
FLMUINT64 ui64LastWriteOffset;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
if( !bWaitForWrite)
|
|
{
|
|
f_assert( m_bOpenedInAsyncMode);
|
|
}
|
|
|
|
if( puiBytesWrittenRV)
|
|
{
|
|
*puiBytesWrittenRV = 0;
|
|
}
|
|
|
|
if( ui64WriteOffset == FLM_IO_CURRENT_POS)
|
|
{
|
|
ui64WriteOffset = m_ui64CurrentPos;
|
|
}
|
|
|
|
// This loop is for direct IO - must make sure we use
|
|
// aligned buffers.
|
|
|
|
pucSrcBuffer = (FLMBYTE *)pvBuffer;
|
|
for (;;)
|
|
{
|
|
|
|
// See if we are using an aligned buffer. If not, allocate
|
|
// one (if not already allocated), and use it.
|
|
|
|
if ((ui64WriteOffset & m_ui64NotOnSectorBoundMask) ||
|
|
(uiBytesToWrite & m_ui64NotOnSectorBoundMask) ||
|
|
(((FLMUINT)pucSrcBuffer) & m_ui64NotOnSectorBoundMask))
|
|
{
|
|
|
|
// Cannot do an async write if we have to use a temporary buffer
|
|
|
|
bWaitForWrite = TRUE;
|
|
|
|
if (!m_pucAlignedBuff)
|
|
{
|
|
if (RC_BAD( rc = allocAlignedBuffer()))
|
|
{
|
|
goto Exit;
|
|
}
|
|
}
|
|
pucWriteBuffer = m_pucAlignedBuff;
|
|
|
|
// Must write enough bytes to cover all of the sectors that
|
|
// contain the data we are trying to write out. The value of
|
|
// (ui64WriteOffset & m_ui64NotOnSectorBoundMask) will give us the
|
|
// number of additional bytes that are in the sector prior to
|
|
// the read offset. We then round to the next sector to get the
|
|
// total number of bytes we are going to write.
|
|
|
|
uiTotalBytesToWrite = (FLMUINT)roundToNextSector( uiBytesToWrite +
|
|
(ui64WriteOffset & m_ui64NotOnSectorBoundMask));
|
|
|
|
// Can't write more than the aligned buffer will hold.
|
|
|
|
if (uiTotalBytesToWrite > m_uiAlignedBuffSize)
|
|
{
|
|
uiTotalBytesToWrite = m_uiAlignedBuffSize;
|
|
uiBufBytesToWrite = (FLMUINT)(uiTotalBytesToWrite -
|
|
(ui64WriteOffset & m_ui64NotOnSectorBoundMask));
|
|
}
|
|
else
|
|
{
|
|
uiBufBytesToWrite = uiBytesToWrite;
|
|
}
|
|
|
|
// If the write offset is not on a sector boundary, we must
|
|
// read at least the first sector into the buffer.
|
|
|
|
if( (ui64WriteOffset & m_ui64NotOnSectorBoundMask) ||
|
|
uiBufBytesToWrite < m_uiBytesPerSector)
|
|
{
|
|
|
|
// Read the first sector that is to be written out.
|
|
// Read one sector's worth of data - so that we will
|
|
// preserve what is already in the sector before
|
|
// writing it back out again.
|
|
|
|
if (RC_BAD( rc = doOneRead( truncateToPrevSector( ui64WriteOffset),
|
|
m_uiBytesPerSector, pucWriteBuffer, &uiBytesRead)))
|
|
{
|
|
if( rc != NE_FLM_IO_END_OF_FILE)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
rc = NE_FLM_OK;
|
|
f_memset( &pucWriteBuffer[ uiBytesRead], 0,
|
|
m_uiBytesPerSector - uiBytesRead);
|
|
}
|
|
}
|
|
|
|
// If the last sector's worth of data we are writing out is only a
|
|
// partial sector, we must read in this sector as well.
|
|
|
|
if( uiTotalBytesToWrite > m_uiBytesPerSector &&
|
|
((ui64WriteOffset + uiBufBytesToWrite) & m_ui64NotOnSectorBoundMask))
|
|
{
|
|
FLMBYTE * pucReadBuf;
|
|
|
|
pucReadBuf = &pucWriteBuffer[
|
|
uiTotalBytesToWrite - m_uiBytesPerSector];
|
|
|
|
|
|
// Read the last sector that is to be written out.
|
|
// Read one sector's worth of data - so that we will
|
|
// preserve what is already in the sector before
|
|
// writing it back out again.
|
|
|
|
if( RC_BAD( rc = doOneRead(
|
|
truncateToPrevSector( ui64WriteOffset + uiBufBytesToWrite),
|
|
m_uiBytesPerSector, pucReadBuf, &uiBytesRead)))
|
|
{
|
|
if( rc != NE_FLM_IO_END_OF_FILE)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
rc = NE_FLM_OK;
|
|
f_memset( &pucReadBuf[ uiBytesRead], 0,
|
|
m_uiBytesPerSector - uiBytesRead);
|
|
}
|
|
}
|
|
|
|
// Finally, copy the data from the source buffer into the
|
|
// write buffer.
|
|
|
|
f_memcpy( &pucWriteBuffer[ ui64WriteOffset & m_ui64NotOnSectorBoundMask],
|
|
pucSrcBuffer, uiBufBytesToWrite);
|
|
}
|
|
else
|
|
{
|
|
pucWriteBuffer = pucSrcBuffer;
|
|
uiTotalBytesToWrite = uiBytesToWrite;
|
|
uiBufBytesToWrite = uiBytesToWrite;
|
|
}
|
|
|
|
// Position the file to the nearest sector below the write offset.
|
|
|
|
ui64LastWriteOffset = (FLMUINT)truncateToPrevSector( ui64WriteOffset);
|
|
|
|
if( !m_bOpenedInAsyncMode)
|
|
{
|
|
FLMINT iBytesWritten;
|
|
|
|
if( (iBytesWritten = pwrite( m_fd,
|
|
pucWriteBuffer, uiTotalBytesToWrite, ui64LastWriteOffset)) == -1)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
uiBytesWritten = (FLMUINT)iBytesWritten;
|
|
f_assert( uiBytesWritten == uiTotalBytesToWrite);
|
|
}
|
|
else
|
|
#ifdef FLM_LIBC_NLM
|
|
{
|
|
rc = RC_SET_AND_ASSERT( NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
#else
|
|
{
|
|
struct aiocb * pAio;
|
|
struct aiocb tmpAio;
|
|
|
|
if( pBufferObj)
|
|
{
|
|
pAio = ((F_IOBuffer *)pBufferObj)->getAIOStruct();
|
|
}
|
|
else
|
|
{
|
|
pAio = &tmpAio;
|
|
f_assert( bWaitForWrite);
|
|
}
|
|
|
|
f_memset( pAio, 0, sizeof( struct aiocb));
|
|
pAio->aio_lio_opcode = LIO_WRITE;
|
|
pAio->aio_sigevent.sigev_notify = SIGEV_NONE;
|
|
pAio->aio_fildes = m_fd;
|
|
pAio->aio_offset = ui64LastWriteOffset;
|
|
pAio->aio_nbytes = uiTotalBytesToWrite;
|
|
pAio->aio_buf = pucWriteBuffer;
|
|
|
|
if( aio_write( pAio) != 0)
|
|
{
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
if( bWaitForWrite)
|
|
{
|
|
struct aiocb * ppAioList[ 1];
|
|
|
|
ppAioList[ 0] = pAio;
|
|
|
|
for( ;;)
|
|
{
|
|
FLMINT iAsyncResult;
|
|
|
|
if( aio_suspend( ppAioList, 1, NULL) != 0)
|
|
{
|
|
if( errno == EINTR)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
iAsyncResult = aio_error( pAio);
|
|
|
|
if( !iAsyncResult)
|
|
{
|
|
if( (iAsyncResult = aio_return( pAio)) < 0)
|
|
{
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( errno, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
|
|
uiBytesWritten = (FLMUINT)iAsyncResult;
|
|
f_assert( uiBytesWritten == uiTotalBytesToWrite);
|
|
break;
|
|
}
|
|
|
|
if( iAsyncResult == EINTR || iAsyncResult == EINPROGRESS)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
f_assert( 0);
|
|
rc = f_mapPlatformError( iAsyncResult, NE_FLM_WRITING_FILE);
|
|
goto Exit;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pBufferObj->makePending();
|
|
uiBytesWritten = uiTotalBytesToWrite;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if (uiBytesWritten < uiTotalBytesToWrite)
|
|
{
|
|
rc = RC_SET_AND_ASSERT( NE_FLM_IO_DISK_FULL);
|
|
goto Exit;
|
|
}
|
|
|
|
uiBytesToWrite -= uiBufBytesToWrite;
|
|
|
|
if( puiBytesWrittenRV)
|
|
{
|
|
(*puiBytesWrittenRV) += uiBufBytesToWrite;
|
|
}
|
|
|
|
m_ui64CurrentPos = ui64WriteOffset + uiBufBytesToWrite;
|
|
|
|
if (!uiBytesToWrite)
|
|
{
|
|
break;
|
|
}
|
|
|
|
pucSrcBuffer += uiBufBytesToWrite;
|
|
ui64WriteOffset += uiBufBytesToWrite;
|
|
}
|
|
|
|
Exit:
|
|
|
|
if( pBufferObj && !pBufferObj->isPending())
|
|
{
|
|
pBufferObj->notifyComplete( rc);
|
|
}
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Returns flag indicating whether or not we can do async writes.
|
|
******************************************************************************/
|
|
FLMBOOL FLMAPI F_FileHdl::canDoAsync()
|
|
{
|
|
f_assert( m_bFileOpened);
|
|
return( m_bOpenedInAsyncMode);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Attempts to lock byte 0 of the file. This method is used to
|
|
lock byte 0 of the .lck file to ensure that only one process
|
|
has access to a database.
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::lock( void)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
struct flock LockStruct;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
// Lock the first byte in file
|
|
|
|
f_memset( &LockStruct, 0, sizeof( LockStruct));
|
|
LockStruct.l_type = F_WRLCK;
|
|
LockStruct.l_whence = SEEK_SET;
|
|
LockStruct.l_start = 0;
|
|
LockStruct.l_len = 1;
|
|
|
|
if( fcntl( m_fd, F_SETLK, &LockStruct) == -1)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_FILE_LOCK_ERR);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/******************************************************************************
|
|
Desc: Attempts to unlock byte 0 of the file.
|
|
******************************************************************************/
|
|
RCODE FLMAPI F_FileHdl::unlock( void)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
struct flock LockStruct;
|
|
|
|
f_assert( m_bFileOpened);
|
|
|
|
// Unlock the first byte in file
|
|
|
|
f_memset( &LockStruct, 0, sizeof( LockStruct));
|
|
LockStruct.l_type = F_UNLCK;
|
|
LockStruct.l_whence = SEEK_SET;
|
|
LockStruct.l_start = 0;
|
|
LockStruct.l_len = 1;
|
|
|
|
if( fcntl( m_fd, F_SETLK, &LockStruct) == -1)
|
|
{
|
|
rc = RC_SET( NE_FLM_IO_FILE_UNLOCK_ERR);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/***************************************************************************
|
|
Desc: Determines the kernel version of the linux system we are running on
|
|
***************************************************************************/
|
|
#ifdef FLM_LINUX
|
|
void f_getLinuxKernelVersion(
|
|
FLMUINT * puiMajor,
|
|
FLMUINT * puiMinor,
|
|
FLMUINT * puiRevision)
|
|
{
|
|
int fd = -1;
|
|
int iBytesRead;
|
|
char szBuffer [80];
|
|
char * pszVer;
|
|
FLMUINT uiMajorVer = 0;
|
|
FLMUINT uiMinorVer = 0;
|
|
FLMUINT uiRevision = 0;
|
|
|
|
if( gv_uiLinuxMajorVer)
|
|
{
|
|
uiMajorVer = gv_uiLinuxMajorVer;
|
|
uiMinorVer = gv_uiLinuxMinorVer;
|
|
uiRevision = gv_uiLinuxRevision;
|
|
goto Exit;
|
|
}
|
|
|
|
if( (fd = open( "/proc/version", O_RDONLY, 0600)) == -1)
|
|
{
|
|
goto Exit;
|
|
}
|
|
|
|
if( (iBytesRead = read( fd, szBuffer, sizeof( szBuffer))) == -1)
|
|
{
|
|
goto Exit;
|
|
}
|
|
if( (pszVer = f_strstr( szBuffer, "version ")) == NULL)
|
|
{
|
|
goto Exit;
|
|
}
|
|
pszVer += 8;
|
|
|
|
while( *pszVer >= '0' && *pszVer <= '9')
|
|
{
|
|
uiMajorVer *= 10;
|
|
uiMajorVer += (FLMUINT)(*pszVer - '0');
|
|
pszVer++;
|
|
}
|
|
|
|
if( *pszVer == '.')
|
|
{
|
|
pszVer++;
|
|
while (*pszVer >= '0' && *pszVer <= '9')
|
|
{
|
|
uiMinorVer *= 10;
|
|
uiMinorVer += (FLMUINT)(*pszVer - '0');
|
|
pszVer++;
|
|
}
|
|
}
|
|
|
|
if( *pszVer == '.')
|
|
{
|
|
pszVer++;
|
|
while (*pszVer >= '0' && *pszVer <= '9')
|
|
{
|
|
uiRevision *= 10;
|
|
uiRevision += (FLMUINT)(*pszVer - '0');
|
|
pszVer++;
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
|
|
if( fd != -1)
|
|
{
|
|
close( fd);
|
|
}
|
|
|
|
if( puiMajor)
|
|
{
|
|
*puiMajor = uiMajorVer;
|
|
}
|
|
|
|
if( puiMinor)
|
|
{
|
|
*puiMinor = uiMinorVer;
|
|
}
|
|
|
|
if( puiRevision)
|
|
{
|
|
*puiRevision = uiRevision;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/***************************************************************************
|
|
Desc:
|
|
***************************************************************************/
|
|
#ifdef FLM_LINUX
|
|
void f_setupLinuxKernelVersion( void)
|
|
{
|
|
f_getLinuxKernelVersion( &gv_uiLinuxMajorVer,
|
|
&gv_uiLinuxMinorVer, &gv_uiLinuxRevision);
|
|
}
|
|
#endif
|
|
|
|
/***************************************************************************
|
|
Desc: Determines if the linux system we are running on is 2.4 or greater.
|
|
***************************************************************************/
|
|
#ifdef FLM_LINUX
|
|
FLMUINT f_getLinuxMaxFileSize( void)
|
|
{
|
|
#ifdef FLM_32BIT
|
|
return( FLM_MAXIMUM_FILE_SIZE);
|
|
#else
|
|
FLMUINT uiMaxFileSize = 0x7FF00000;
|
|
|
|
f_assert( gv_uiLinuxMajorVer);
|
|
|
|
// Version 2.4 or greater?
|
|
|
|
if( gv_uiLinuxMajorVer > 2 ||
|
|
(gv_uiLinuxMajorVer == 2 && gv_uiLinuxMinorVer >= 4))
|
|
{
|
|
uiMaxFileSize = FLM_MAXIMUM_FILE_SIZE;
|
|
}
|
|
|
|
return( uiMaxFileSize);
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
Desc: This routine gets the block size for the file system a file belongs to.
|
|
****************************************************************************/
|
|
FLMUINT f_getFSBlockSize(
|
|
FLMBYTE * pszFileName)
|
|
{
|
|
FLMUINT uiFSBlkSize = 4096;
|
|
FLMBYTE * pszTmp = pszFileName + f_strlen( (const char *)pszFileName) - 1;
|
|
FLMBYTE * pszDir;
|
|
FLMBYTE ucRestoreByte = 0;
|
|
|
|
while( pszTmp != pszFileName && *pszTmp != '/')
|
|
{
|
|
pszTmp--;
|
|
}
|
|
|
|
if( *pszTmp == '/')
|
|
{
|
|
if (pszTmp == pszFileName)
|
|
{
|
|
pszTmp++;
|
|
}
|
|
ucRestoreByte = *pszTmp;
|
|
*pszTmp = 0;
|
|
pszDir = pszFileName;
|
|
}
|
|
else
|
|
{
|
|
pszDir = (FLMBYTE *)".";
|
|
}
|
|
|
|
#if defined( FLM_SOLARIS)
|
|
struct statvfs statfsbuf;
|
|
if (statvfs( (char *)pszDir, &statfsbuf) == 0)
|
|
{
|
|
uiFSBlkSize = (FLMUINT)statfsbuf.f_bsize;
|
|
}
|
|
#elif defined( FLM_LINUX) || defined( FLM_OSX)
|
|
struct statfs statfsbuf;
|
|
if (statfs( (char *)pszDir, &statfsbuf) == 0)
|
|
{
|
|
uiFSBlkSize = (FLMUINT)statfsbuf.f_bsize;
|
|
}
|
|
#endif
|
|
|
|
if( ucRestoreByte)
|
|
{
|
|
*pszTmp = ucRestoreByte;
|
|
}
|
|
|
|
return( uiFSBlkSize);
|
|
}
|
|
|
|
/****************************************************************************
|
|
Desc:
|
|
****************************************************************************/
|
|
#if defined( FLM_SOLARIS) && defined( FLM_SPARC) && !defined( FLM_GNUC)
|
|
static void sparc_asm_code( void)
|
|
{
|
|
asm( ".align 8");
|
|
asm( ".global sparc_atomic_add_32");
|
|
asm( ".type sparc_atomic_add_32, #function");
|
|
asm( "sparc_atomic_add_32:");
|
|
asm( " membar #LoadLoad | #LoadStore | #StoreStore | #StoreLoad");
|
|
asm( " ld [%o0], %l0");
|
|
asm( " add %l0, %o1, %l2");
|
|
asm( " cas [%o0], %l0, %l2");
|
|
asm( " cmp %l0, %l2");
|
|
asm( " bne sparc_atomic_add_32");
|
|
asm( " nop");
|
|
asm( " add %l2, %o1, %o0");
|
|
asm( " membar #LoadLoad | #LoadStore | #StoreStore | #StoreLoad");
|
|
asm( "retl");
|
|
asm( "nop");
|
|
|
|
asm( ".align 8");
|
|
asm( ".global sparc_atomic_xchg_32");
|
|
asm( ".type sparc_atomic_xchg_32, #function");
|
|
asm( "sparc_atomic_xchg_32:");
|
|
asm( " membar #LoadLoad | #LoadStore | #StoreStore | #StoreLoad");
|
|
asm( " ld [%o0], %l0");
|
|
asm( " mov %o1, %l1");
|
|
asm( " cas [%o0], %l0, %l1");
|
|
asm( " cmp %l0, %l1");
|
|
asm( " bne sparc_atomic_xchg_32");
|
|
asm( " nop");
|
|
asm( " mov %l0, %o0");
|
|
asm( " membar #LoadLoad | #LoadStore | #StoreStore | #StoreLoad");
|
|
asm( "retl");
|
|
asm( "nop");
|
|
}
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
Desc:
|
|
****************************************************************************/
|
|
void FLMAPI f_yieldCPU( void)
|
|
{
|
|
}
|
|
|
|
/**********************************************************************
|
|
Desc:
|
|
**********************************************************************/
|
|
FLMINT32 posix_atomic_add_32(
|
|
volatile FLMINT32 * piTarget,
|
|
FLMINT32 iDelta)
|
|
{
|
|
FLMINT32 i32RetVal;
|
|
|
|
pthread_mutex_lock( &gv_atomicMutex);
|
|
(*piTarget) += iDelta;
|
|
i32RetVal = *piTarget;
|
|
pthread_mutex_unlock( &gv_atomicMutex);
|
|
|
|
return( i32RetVal);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Desc:
|
|
**********************************************************************/
|
|
FLMINT32 posix_atomic_xchg_32(
|
|
volatile FLMINT32 * piTarget,
|
|
FLMINT32 iNewValue)
|
|
{
|
|
FLMINT32 i32RetVal;
|
|
|
|
pthread_mutex_lock( &gv_atomicMutex);
|
|
i32RetVal = *piTarget;
|
|
*piTarget = iNewValue;
|
|
pthread_mutex_unlock( &gv_atomicMutex);
|
|
|
|
return( i32RetVal);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Desc:
|
|
**********************************************************************/
|
|
RCODE FLMAPI f_chdir(
|
|
const char * pszDir)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
if( chdir( pszDir) != 0)
|
|
{
|
|
rc = f_mapPlatformError( errno, NE_FLM_IO_PATH_NOT_FOUND);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
/**********************************************************************
|
|
Desc:
|
|
**********************************************************************/
|
|
RCODE FLMAPI f_getcwd(
|
|
char * pszDir)
|
|
{
|
|
RCODE rc = NE_FLM_OK;
|
|
|
|
if( getcwd( pszDir, F_PATH_MAX_SIZE) == NULL)
|
|
{
|
|
*pszDir = 0;
|
|
rc = f_mapPlatformError( errno, NE_FLM_IO_PATH_NOT_FOUND);
|
|
goto Exit;
|
|
}
|
|
|
|
Exit:
|
|
|
|
return( rc);
|
|
}
|
|
|
|
#endif // FLM_UNIX
|
|
|
|
#if defined( FLM_WATCOM_NLM)
|
|
int fposixDummy(void)
|
|
{
|
|
return( 0);
|
|
}
|
|
#endif
|