2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-10-29 11:53:47 +01:00
|
|
|
* 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-28 04:47:47 +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
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +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"
|
2003-11-11 13:19:20 +01:00
|
|
|
#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
|
|
|
|
|
2002-08-26 14:18:16 +02:00
|
|
|
#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
|
|
|
|
|
2004-05-06 20:09:24 +02:00
|
|
|
const int IO_RETRY = 20;
|
|
|
|
const int BUFSIZE = 32768;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-10-29 11:53:47 +01: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(
|
2003-12-05 11:35:47 +01:00
|
|
|
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
|
2003-12-05 11:35:47 +01:00
|
|
|
* the passed fill_char. If 'overwrite' is false and the file
|
2002-11-14 09:33:08 +01:00
|
|
|
* 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.
|
|
|
|
*
|
2002-11-14 09:33:08 +01:00
|
|
|
* 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;
|
|
|
|
|
2003-12-05 11:35:47 +01:00
|
|
|
SLONG fd;
|
|
|
|
if (LLIO_open(status_vector, filename, open_mode, true, &fd)) {
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01:00
|
|
|
SCHAR buffer[BUFSIZE];
|
2001-05-23 15:26:42 +02:00
|
|
|
memset(buffer, (int) fill_char, BUFSIZE);
|
2003-12-05 11:35:47 +01:00
|
|
|
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--) {
|
2003-12-05 11:35:47 +01:00
|
|
|
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,
|
2003-12-05 11:35:47 +01:00
|
|
|
reinterpret_cast<const UCHAR*>(buffer), length, 0))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
LLIO_close(0, fd);
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LLIO_close(0, fd);
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2003-12-05 11:35:47 +01: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 ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Close a file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return (CloseHandle((HANDLE) file_desc) ? FB_SUCCESS : FB_FAILURE);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_open(
|
2003-10-29 11:53:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
|
|
|
const TEXT* filename,
|
2003-12-05 11:35:47 +01:00
|
|
|
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.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-05 11:35:47 +01:00
|
|
|
DWORD create;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01: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;
|
|
|
|
}
|
|
|
|
|
2003-12-05 11:35:47 +01:00
|
|
|
*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));
|
2003-10-29 11:53:47 +01:00
|
|
|
if ((HANDLE) *file_desc == INVALID_HANDLE_VALUE
|
|
|
|
&& open_mode == LLIO_OPEN_WITH_TRUNC_RW)
|
|
|
|
{
|
|
|
|
*file_desc =
|
2003-12-05 11:35:47 +01:00
|
|
|
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));
|
2003-10-29 11:53:47 +01:00
|
|
|
}
|
|
|
|
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);
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_read(
|
2003-12-05 11:35:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG file_desc,
|
2003-12-05 11:35:47 +01:00
|
|
|
const TEXT* filename,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG offset,
|
|
|
|
USHORT whence,
|
2003-12-05 11:35:47 +01:00
|
|
|
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) ==
|
2003-12-05 11:35:47 +01:00
|
|
|
FB_FAILURE))
|
|
|
|
{
|
|
|
|
return FB_FAILURE;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01:00
|
|
|
DWORD len = 0;
|
|
|
|
if (buffer) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!ReadFile((HANDLE) file_desc, buffer, length, &len, NULL)) {
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "ReadFile", filename,
|
|
|
|
isc_io_read_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (length_read)
|
|
|
|
*length_read = len;
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_seek(
|
2003-12-05 11:35:47 +01:00
|
|
|
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;
|
|
|
|
}
|
2003-08-09 23:15:32 +02:00
|
|
|
if (SetFilePointer((HANDLE) file_desc, offset, NULL, method) == (DWORD) -1) {
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "SetFilePointer", filename,
|
|
|
|
isc_io_access_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-05 11:35:47 +01: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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return (FlushFileBuffers((HANDLE) file_desc) ? FB_SUCCESS : FB_FAILURE);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_write(
|
2003-12-05 11:35:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG file_desc,
|
2003-12-05 11:35:47 +01:00
|
|
|
const TEXT* filename,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG offset,
|
2003-12-05 11:35:47 +01:00
|
|
|
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) ==
|
2003-12-05 11:35:47 +01:00
|
|
|
FB_FAILURE))
|
|
|
|
{
|
|
|
|
return FB_FAILURE;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01:00
|
|
|
DWORD len = 0;
|
|
|
|
if (buffer) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!WriteFile((HANDLE) file_desc, buffer, length, &len, NULL)) {
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "WriteFile", filename,
|
|
|
|
isc_io_write_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (length_written)
|
|
|
|
*length_written = len;
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-10-29 11:53:47 +01: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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2004-05-06 20:09:24 +02:00
|
|
|
#else // WIN_NT
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01: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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return ((close((int) file_desc) != -1) ? FB_SUCCESS : FB_FAILURE);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_open(
|
2003-10-29 11:53:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
|
|
|
const TEXT* filename,
|
2003-12-05 11:35:47 +01:00
|
|
|
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;
|
|
|
|
}
|
2003-12-05 11:35:47 +01:00
|
|
|
|
|
|
|
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) {
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "open", filename, isc_io_open_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int LLIO_read(
|
2003-12-05 11:35:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG file_desc,
|
2003-12-05 11:35:47 +01:00
|
|
|
const TEXT* filename,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG offset,
|
|
|
|
USHORT whence,
|
2003-12-05 11:35:47 +01:00
|
|
|
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) ==
|
2003-12-05 11:35:47 +01:00
|
|
|
FB_FAILURE))
|
|
|
|
{
|
|
|
|
return FB_FAILURE;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01: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;
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "read", filename,
|
|
|
|
isc_io_read_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (!len)
|
|
|
|
break;
|
|
|
|
length -= len;
|
|
|
|
p += len;
|
|
|
|
}
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (length_read)
|
|
|
|
*length_read = p - buffer;
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_seek(
|
2003-12-05 11:35:47 +01:00
|
|
|
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) {
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "lseek", filename, isc_io_access_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-05 11:35:47 +01: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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return (fsync((int) file_desc) != -1) ? FB_SUCCESS : FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LLIO_write(
|
2003-12-05 11:35:47 +01:00
|
|
|
ISC_STATUS* status_vector,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG file_desc,
|
2003-12-05 11:35:47 +01:00
|
|
|
const TEXT* filename,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG offset,
|
2003-12-05 11:35:47 +01:00
|
|
|
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) ==
|
2003-12-05 11:35:47 +01:00
|
|
|
FB_FAILURE))
|
|
|
|
{
|
|
|
|
return FB_FAILURE;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-05 11:35:47 +01: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;
|
2003-12-05 11:35:47 +01:00
|
|
|
if (status_vector) {
|
2001-05-23 15:26:42 +02:00
|
|
|
io_error(status_vector, "write", filename,
|
|
|
|
isc_io_write_err);
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_FAILURE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (!len)
|
|
|
|
break;
|
|
|
|
length -= len;
|
|
|
|
p += len;
|
|
|
|
}
|
2003-12-05 11:35:47 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (length_written)
|
|
|
|
*length_written = p - buffer;
|
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void io_error(
|
2003-10-29 11:53:47 +01:00
|
|
|
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.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
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
|
|
|
}
|
2004-05-06 20:09:24 +02:00
|
|
|
#endif // WIN_NT
|
2003-10-29 11:53:47 +01:00
|
|
|
|