8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-26 07:23:08 +01:00
firebird-mirror/src/jrd/llio.cpp

662 lines
15 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
* MODULE: llio.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Low-level I/O routines
*
* The contents of this file are subject to the Interbase Public
* License Version 1.0 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy
* of the License at http://www.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
* or implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code was created by Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
2002-02-16 03:21:35 +01:00
*
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "EPSON" define
2002-02-16 04:05:21 +01:00
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "DELTA" port
2002-02-16 04:27:33 +01:00
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "IMP" port
2002-02-16 03:21:35 +01:00
*
* 2002.10.27 Sean Leyne - Completed removal of "DELTA" port
*
2002-10-30 07:40:58 +01:00
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
2001-05-23 15:26:42 +02:00
*/
#include "firebird.h"
2004-04-29 00:43:34 +02:00
#include <stdio.h>
2001-05-23 15:26:42 +02:00
#include <errno.h>
#include <string.h>
#include "../jrd/common.h"
#include "../jrd/llio.h"
#include "gen/iberror.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/iberr_proto.h"
#include "../jrd/llio_proto.h"
2001-07-12 07:46:06 +02:00
#ifdef HAVE_SYS_TYPES_H
#include <sys/types.h>
#endif
#ifdef HAVE_SYS_STAT_H
#include <sys/stat.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
2001-05-23 15:26:42 +02:00
# ifndef WIN_NT
# ifndef VMS
# include <fcntl.h>
2002-02-16 04:27:33 +01:00
# include <sys/file.h>
2001-05-23 15:26:42 +02:00
# else
# include <file.h>
# endif
# endif
#ifdef HAVE_UNISTD_H
2001-05-23 15:26:42 +02:00
#include <unistd.h>
#endif
#ifdef WIN_NT
#include <windows.h>
#endif
#ifndef O_SYNC
#define O_SYNC 0
#endif
const int IO_RETRY = 20;
const int BUFSIZE = 32768;
2001-05-23 15:26:42 +02:00
static void io_error(ISC_STATUS*, const TEXT*, const TEXT*, ISC_STATUS);
2001-05-23 15:26:42 +02:00
int LLIO_allocate_file_space(
ISC_STATUS* status_vector,
const TEXT* filename,
SLONG size, UCHAR fill_char, bool overwrite)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ a l l o c a t e _ f i l e _ s p a c e
*
**************************************
*
* Functional description
* Open (create if necessary) the given file and write 'size'
* number of bytes to it. Each byte is initialized to
* the passed fill_char. If 'overwrite' is false and the file
* already exists, return FB_FAILURE.
2001-05-23 15:26:42 +02:00
*
* This routine may be used to make sure that the file already
* has enough space allocated to it.
*
* Other side benefit of using this routine could be that once the
* space is allocated, file system would not update the inode every
* time something is written to the file unless the file size grows
* beyond the given 'size'. So 'write' performance should improve.
*
* If there is any error, return FB_FAILURE else return FB_SUCCESS.
2001-05-23 15:26:42 +02:00
* In case of an error, status_vector would be updated.
*
**************************************/
USHORT open_mode;
if (overwrite)
open_mode = LLIO_OPEN_RW;
else
open_mode = LLIO_OPEN_NEW_RW;
SLONG fd;
if (LLIO_open(status_vector, filename, open_mode, true, &fd)) {
return FB_FAILURE;
}
2001-05-23 15:26:42 +02:00
SCHAR buffer[BUFSIZE];
2001-05-23 15:26:42 +02:00
memset(buffer, (int) fill_char, BUFSIZE);
SLONG times = size / BUFSIZE + 1; /* For the while loop below */
const SLONG last_size = size % BUFSIZE;
2001-05-23 15:26:42 +02:00
while (times--) {
const SLONG length = (times != 0) ? BUFSIZE : last_size;
2001-05-23 15:26:42 +02:00
if (LLIO_write(status_vector, fd, filename, 0L, LLIO_SEEK_NONE,
reinterpret_cast<const UCHAR*>(buffer), length, 0))
{
2001-05-23 15:26:42 +02:00
LLIO_close(0, fd);
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
}
LLIO_close(0, fd);
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
#ifdef WIN_NT
int LLIO_close(ISC_STATUS* status_vector, SLONG file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ c l o s e ( W I N _ N T )
*
**************************************
*
* Functional description
* Close a file.
*
**************************************/
return (CloseHandle((HANDLE) file_desc) ? FB_SUCCESS : FB_FAILURE);
2001-05-23 15:26:42 +02:00
}
int LLIO_open(
ISC_STATUS* status_vector,
const TEXT* filename,
USHORT open_mode,
bool, //share_flag, // unused, must keep for the non-NT version
SLONG* file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ o p e n ( W I N _ N T )
*
**************************************
*
* Functional description
* Open a file.
*
**************************************/
DWORD create;
2001-05-23 15:26:42 +02:00
DWORD access = GENERIC_READ | GENERIC_WRITE;
DWORD attributes = 0;
2001-05-23 15:26:42 +02:00
switch (open_mode) {
case LLIO_OPEN_R:
access = GENERIC_READ;
create = OPEN_EXISTING;
break;
case LLIO_OPEN_RW:
create = OPEN_ALWAYS;
break;
case LLIO_OPEN_WITH_SYNC_RW:
create = OPEN_ALWAYS;
attributes = FILE_FLAG_WRITE_THROUGH;
break;
case LLIO_OPEN_WITH_TRUNC_RW:
create = TRUNCATE_EXISTING;
break;
case LLIO_OPEN_EXISTING_RW:
create = OPEN_EXISTING;
break;
case LLIO_OPEN_WITH_SYNC_W:
access = GENERIC_WRITE;
create = OPEN_ALWAYS;
attributes = FILE_FLAG_WRITE_THROUGH;
break;
case LLIO_OPEN_NEW_RW:
create = CREATE_NEW;
break;
}
*file_desc = reinterpret_cast<SLONG> (CreateFile(filename,
2001-05-23 15:26:42 +02:00
access,
FILE_SHARE_READ |
FILE_SHARE_WRITE,
NULL, create,
FILE_ATTRIBUTE_NORMAL
|
FILE_FLAG_RANDOM_ACCESS
| attributes, 0));
if ((HANDLE) *file_desc == INVALID_HANDLE_VALUE
&& open_mode == LLIO_OPEN_WITH_TRUNC_RW)
{
*file_desc =
reinterpret_cast<SLONG>
2001-05-23 15:26:42 +02:00
(CreateFile
(filename, access, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL,
CREATE_NEW,
FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS | attributes,
0));
}
if ((HANDLE) *file_desc == INVALID_HANDLE_VALUE) {
2001-05-23 15:26:42 +02:00
if (status_vector)
io_error(status_vector, "CreateFile", filename, isc_io_open_err);
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_read(
ISC_STATUS* status_vector,
2001-05-23 15:26:42 +02:00
SLONG file_desc,
const TEXT* filename,
2001-05-23 15:26:42 +02:00
SLONG offset,
USHORT whence,
UCHAR* buffer, SLONG length, SLONG* length_read)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ r e a d ( W I N _ N T )
*
**************************************
*
* Functional description
* Read a record from a file.
*
**************************************/
if ((whence != LLIO_SEEK_NONE) &&
(LLIO_seek(status_vector, file_desc, filename, offset, whence) ==
FB_FAILURE))
{
return FB_FAILURE;
}
2001-05-23 15:26:42 +02:00
DWORD len = 0;
if (buffer) {
2001-05-23 15:26:42 +02:00
if (!ReadFile((HANDLE) file_desc, buffer, length, &len, NULL)) {
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "ReadFile", filename,
isc_io_read_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
}
2001-05-23 15:26:42 +02:00
if (length_read)
*length_read = len;
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_seek(
ISC_STATUS* status_vector,
SLONG file_desc, const TEXT* filename, SLONG offset, USHORT whence)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ s e e k ( W I N _ N T )
*
**************************************
*
* Functional description
* Seek to the given offset.
*
**************************************/
DWORD method;
if (whence != LLIO_SEEK_NONE) {
switch (whence) {
case LLIO_SEEK_BEGIN:
method = FILE_BEGIN;
break;
case LLIO_SEEK_CURRENT:
method = FILE_CURRENT;
break;
case LLIO_SEEK_END:
method = FILE_END;
break;
}
if (SetFilePointer((HANDLE) file_desc, offset, NULL, method) == (DWORD) -1) {
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "SetFilePointer", filename,
isc_io_access_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
}
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_sync(ISC_STATUS* status_vector, SLONG file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ s y n c ( W I N _ N T )
*
**************************************
*
* Functional description
* Flush the buffers in a file.
*
**************************************/
return (FlushFileBuffers((HANDLE) file_desc) ? FB_SUCCESS : FB_FAILURE);
2001-05-23 15:26:42 +02:00
}
int LLIO_write(
ISC_STATUS* status_vector,
2001-05-23 15:26:42 +02:00
SLONG file_desc,
const TEXT* filename,
2001-05-23 15:26:42 +02:00
SLONG offset,
USHORT whence, const UCHAR* buffer, SLONG length, SLONG* length_written)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ w r i t e ( W I N _ N T )
*
**************************************
*
* Functional description
* Write a record to a file.
*
**************************************/
if ((whence != LLIO_SEEK_NONE) &&
(LLIO_seek(status_vector, file_desc, filename, offset, whence) ==
FB_FAILURE))
{
return FB_FAILURE;
}
2001-05-23 15:26:42 +02:00
DWORD len = 0;
if (buffer) {
2001-05-23 15:26:42 +02:00
if (!WriteFile((HANDLE) file_desc, buffer, length, &len, NULL)) {
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "WriteFile", filename,
isc_io_write_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
}
2001-05-23 15:26:42 +02:00
if (length_written)
*length_written = len;
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
static void io_error(ISC_STATUS* status_vector,
const TEXT* op, const TEXT* filename, ISC_STATUS operation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* i o _ e r r o r ( W I N _ N T )
*
**************************************
*
* Functional description
* Fill up the status_vector with error info.
*
**************************************/
IBERR_build_status(status_vector, isc_io_error, isc_arg_string, op,
isc_arg_string, filename, isc_arg_gds, operation,
isc_arg_win32, GetLastError(), 0);
2001-05-23 15:26:42 +02:00
}
#else // WIN_NT
2001-05-23 15:26:42 +02:00
int LLIO_close(ISC_STATUS* status_vector, SLONG file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ c l o s e ( g e n e r i c )
*
**************************************
*
* Functional description
* Close a file.
*
**************************************/
return ((close((int) file_desc) != -1) ? FB_SUCCESS : FB_FAILURE);
2001-05-23 15:26:42 +02:00
}
int LLIO_open(
ISC_STATUS* status_vector,
const TEXT* filename,
USHORT open_mode, bool share_flag, SLONG* file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ o p e n ( g e n e r i c )
*
**************************************
*
* Functional description
* Open a file.
*
**************************************/
int mode;
switch (open_mode) {
case LLIO_OPEN_R:
mode = O_RDONLY;
break;
case LLIO_OPEN_RW:
mode = O_RDWR | O_CREAT;
break;
case LLIO_OPEN_WITH_SYNC_RW:
mode = O_RDWR | O_CREAT | O_SYNC;
break;
case LLIO_OPEN_WITH_TRUNC_RW:
mode = O_RDWR | O_CREAT | O_TRUNC;
break;
case LLIO_OPEN_EXISTING_RW:
mode = O_RDWR;
break;
case LLIO_OPEN_WITH_SYNC_W:
mode = O_WRONLY | O_CREAT | O_SYNC;
break;
case LLIO_OPEN_NEW_RW:
mode = O_CREAT | O_EXCL | O_RDWR;
break;
}
const int oldmask = share_flag ? umask(0) : 0;
2001-05-23 15:26:42 +02:00
*file_desc = open(filename, mode, 0666);
if (share_flag)
umask(oldmask);
if (*file_desc == -1) {
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "open", filename, isc_io_open_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_read(
ISC_STATUS* status_vector,
2001-05-23 15:26:42 +02:00
SLONG file_desc,
const TEXT* filename,
2001-05-23 15:26:42 +02:00
SLONG offset,
USHORT whence,
UCHAR* buffer, SLONG length, SLONG* length_read)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ r e a d ( g e n e r i c )
*
**************************************
*
* Functional description
* Read a record from a file.
*
**************************************/
if ((whence != LLIO_SEEK_NONE) &&
(LLIO_seek(status_vector, file_desc, filename, offset, whence) ==
FB_FAILURE))
{
return FB_FAILURE;
}
2001-05-23 15:26:42 +02:00
UCHAR* p = buffer;
if (p) {
for (int i = 0; length && i++ < IO_RETRY;) {
const int len = read((int) file_desc, p, (int) length);
if (len == -1) {
2001-05-23 15:26:42 +02:00
if (SYSCALL_INTERRUPTED(errno) && i < IO_RETRY)
continue;
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "read", filename,
isc_io_read_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
if (!len)
break;
length -= len;
p += len;
}
}
2001-05-23 15:26:42 +02:00
if (length_read)
*length_read = p - buffer;
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_seek(
ISC_STATUS* status_vector,
SLONG file_desc, const TEXT* filename, SLONG offset, USHORT whence)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ s e e k ( g e n e r i c )
*
**************************************
*
* Functional description
* Seek to the given offset.
*
**************************************/
if (whence != LLIO_SEEK_NONE) {
switch (whence) {
case LLIO_SEEK_BEGIN:
whence = SEEK_SET;
break;
case LLIO_SEEK_CURRENT:
whence = SEEK_CUR;
break;
case LLIO_SEEK_END:
whence = SEEK_END;
break;
}
2001-07-12 07:46:06 +02:00
/* Darwin - Add the offset cast because the types are different
* between the implicit declaration and the actual declaration,
* causing problems with the stack frame, etc. Bad.
*/
if (lseek((int) file_desc, LSEEK_OFFSET_CAST offset, (int) whence) == -1) {
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "lseek", filename, isc_io_access_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
}
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
int LLIO_sync(ISC_STATUS* status_vector, SLONG file_desc)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ s y n c ( g e n e r i c )
*
**************************************
*
* Functional description
* Flush the buffers in a file.
*
**************************************/
return (fsync((int) file_desc) != -1) ? FB_SUCCESS : FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
int LLIO_write(
ISC_STATUS* status_vector,
2001-05-23 15:26:42 +02:00
SLONG file_desc,
const TEXT* filename,
2001-05-23 15:26:42 +02:00
SLONG offset,
USHORT whence, const UCHAR* buffer, SLONG length, SLONG* length_written)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* L L I O _ w r i t e ( g e n e r i c )
*
**************************************
*
* Functional description
* Write a record to a file.
*
**************************************/
if ((whence != LLIO_SEEK_NONE) &&
(LLIO_seek(status_vector, file_desc, filename, offset, whence) ==
FB_FAILURE))
{
return FB_FAILURE;
}
2001-05-23 15:26:42 +02:00
// CVC: it may be necessary to write here instead
// UCHAR* p = const_cast<UCHAR*>(buffer);
// to make your platform's C headers happy. Still, it's logically const.
const UCHAR* p = buffer;
if (p) {
for (int i = 0; length && i++ < IO_RETRY;) {
const int len = write((int) file_desc, p, (int) length);
if (len == -1) {
2001-05-23 15:26:42 +02:00
if (SYSCALL_INTERRUPTED(errno) && i < IO_RETRY)
continue;
if (status_vector) {
2001-05-23 15:26:42 +02:00
io_error(status_vector, "write", filename,
isc_io_write_err);
}
return FB_FAILURE;
2001-05-23 15:26:42 +02:00
}
if (!len)
break;
length -= len;
p += len;
}
}
2001-05-23 15:26:42 +02:00
if (length_written)
*length_written = p - buffer;
return FB_SUCCESS;
2001-05-23 15:26:42 +02:00
}
static void io_error(
ISC_STATUS* status_vector,
const TEXT* op, const TEXT* filename, ISC_STATUS operation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* i o _ e r r o r ( g e n e r i c )
*
**************************************
*
* Functional description
* Fill up the status_vector with error info.
*
**************************************/
IBERR_build_status(status_vector, isc_io_error, isc_arg_string, op,
isc_arg_string, filename, isc_arg_gds, operation,
isc_arg_unix, errno, 0);
2001-05-23 15:26:42 +02:00
}
#endif // WIN_NT