8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 20:43:03 +01:00
firebird-mirror/src/jrd/unix.cpp
2003-02-20 06:15:46 +00:00

1392 lines
33 KiB
C++
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* PROGRAM: JRD Access Method
* MODULE: unix.c
* DESCRIPTION: UNIX (BSD) specific physical IO
*
* 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): ______________________________________.
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
* conditionals, as the engine now fully supports
* readonly databases.
*
* 2002.10.27 Sean Leyne - Completed removal of "DELTA" port
*
*/
#ifdef SHLIB_DEFS
#define LOCAL_SHLIB_DEFS
#endif
#include "firebird.h"
#include "../jrd/common.h"
#include "../jrd/ib_stdio.h"
#include <fcntl.h>
#include <errno.h>
#include <sys/types.h>
#ifdef HAVE_PWD_H
#include <pwd.h>
#endif
#include <sys/file.h>
#include <sys/stat.h>
#include <string.h>
#include <errno.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifdef HAVE_AIO_H
#include <aio.h>
#endif
#include "../jrd/jrd.h"
#include "../jrd/pio.h"
#include "../jrd/ods.h"
#include "../jrd/lck.h"
#include "../jrd/cch.h"
#include "../jrd/y_ref.h"
#include "../jrd/ibase.h"
#include "gen/codes.h"
#include "../jrd/all_proto.h"
#include "../jrd/cch_proto.h"
#include "../jrd/err_proto.h"
#include "../jrd/gds_proto.h"
#include "../jrd/isc_proto.h"
#include "../jrd/isc_f_proto.h"
#include "../jrd/isc_i_proto.h"
#include "../jrd/lck_proto.h"
#include "../jrd/mov_proto.h"
#include "../jrd/pio_proto.h"
#include "../jrd/thd_proto.h"
/* SUPERSERVER uses a mutex to allow atomic seek/read(write) sequences.
SUPERSERVER_V2 uses "positioned" read (write) calls to avoid a seek
and allow multiple threads to overlap database I/O. */
#if defined SUPERSERVER
#if (defined PREAD && defined PWRITE) || defined HAVE_AIO_H
#define PREAD_PWRITE
#endif
#endif
extern "C" {
#ifndef SUPERSERVER
#undef THD_MUTEX_INIT
#undef THD_MUTEX_LOCK
#undef THD_MUTEX_UNLOCK
#undef THD_MUTEX_DESTROY
#define THD_MUTEX_INIT(mutx)
#define THD_MUTEX_LOCK(mutx)
#define THD_MUTEX_UNLOCK(mutx)
#define THD_MUTEX_DESTROY(mutx)
#endif
#define IO_RETRY 20
#ifdef O_SYNC
#define SYNC O_SYNC
#endif
/* Changed to not redfine SYNC if O_SYNC already exists
they seem to be the same values anyway. MOD 13-07-2001 */
#if (!(defined SYNC) && (defined O_FSYNC))
#define SYNC O_FSYNC
#endif
#ifdef O_DSYNC
#undef SYNC
#define SYNC O_DSYNC
#endif
#ifndef SYNC
#define SYNC 0
#endif
#ifndef O_BINARY
#define O_BINARY 0
#endif
#ifdef SUPERSERVER
#define MASK 0600
#else
#define MASK 0666
#endif
static void close_marker_file(TEXT *);
static FIL seek_file(FIL, BDB, UINT64 *, STATUS *);
static FIL setup_file(DBB, TEXT *, USHORT, int);
static BOOLEAN unix_error(TEXT *, FIL, STATUS, STATUS *);
#if defined PREAD_PWRITE && !(defined HAVE_PREAD && defined HAVE_PWRITE)
static SLONG pread(int, SCHAR *, SLONG, SLONG);
static SLONG pwrite(int, SCHAR *, SLONG, SLONG);
#endif
#ifdef SUPPORT_RAW_DEVICES
static BOOLEAN raw_devices_check_file (TEXT *);
static BOOLEAN raw_devices_validate_database (int, TEXT *, USHORT);
static int raw_devices_unlink_database (TEXT *);
#endif
#ifdef hpux
union fcntlun {
int val;
struct flock *lockdes;
};
#endif
#ifdef SHLIB_DEFS
#define strlen (*_libgds_strlen)
#define close (*_libgds_close)
#define open (*_libgds_open)
#define errno (*_libgds_errno)
#define lseek (*_libgds_lseek)
#define read (*_libgds_read)
#define fstat (*_libgds_fstat)
#define write (*_libgds_write)
#define fcntl (*_libgds_fcntl)
#define ib_fprintf (*_libgds_fprintf)
#define ib_fflush (*_libgds_fflush)
#define sscanf (*_libgds_sscanf)
#define access (*_libgds_access)
#define ib_fopen (*_libgds_fopen)
#define ib_fclose (*_libgds_fclose)
#define ib_fputs (*_libgds_fputs)
#define ib_fgets (*_libgds_fgets)
#define strcat (*_libgds_strcat)
#define strcpy (*_libgds_strcpy)
#define fsync (*_libgds_fsync)
#define _iob (*_libgds__iob)
extern int strlen();
extern int close();
extern int open();
extern int errno;
extern off_t lseek();
extern int read();
extern int fstat();
extern int write();
extern int fcntl();
extern int ib_fprintf();
extern int ib_fflush();
extern int sscanf();
extern int access();
extern IB_FILE *ib_fopen();
extern int ib_fclose();
extern int ib_fputs();
extern SCHAR *ib_fgets();
extern SCHAR *strcat();
extern SCHAR *strcpy();
extern int fsync();
extern IB_FILE _iob[];
#endif
int PIO_add_file(DBB dbb, FIL main_file, TEXT * file_name, SLONG start)
{
/**************************************
*
* P I O _ a d d _ f i l e
*
**************************************
*
* Functional description
* Add a file to an existing database. Return the sequence
* number of the new file. If anything goes wrong, return a
* sequence of 0.
* NOTE: This routine does not lock any mutexes on
* its own behalf. It is assumed that mutexes will
* have been locked before entry.
*
**************************************/
USHORT sequence;
FIL file, new_file;
if (!(new_file = PIO_create(dbb, file_name, strlen(file_name), FALSE)))
return 0;
new_file->fil_min_page = start;
sequence = 1;
for (file = main_file; file->fil_next; file = file->fil_next)
++sequence;
file->fil_max_page = start - 1;
file->fil_next = new_file;
return sequence;
}
void PIO_close(FIL main_file)
{
/**************************************
*
* P I O _ c l o s e
*
**************************************
*
* Functional description
* NOTE: This routine does not lock any mutexes on
* its own behalf. It is assumed that mutexes will
* have been locked before entry.
*
**************************************/
FIL file;
if (main_file) {
TEXT marker_filename[MAXPATHLEN];
/* If a marker file is found for single user access, close it. */
strcpy(marker_filename, main_file->fil_string);
strcat(marker_filename, "_m");
if (!access(marker_filename, F_OK)) /* Marker file exists. */
close_marker_file(marker_filename);
}
for (file = main_file; file; file = file->fil_next) {
if (file->fil_desc == -1)
continue; /* This really should be an error */
if (file->fil_desc) {
close(file->fil_desc);
file->fil_desc = -1;
#ifndef PREAD_PWRITE
THD_MUTEX_DESTROY(file->fil_mutex);
#endif
}
}
}
int PIO_connection(TEXT * file_name, USHORT * file_length)
{
/**************************************
*
* P I O _ c o n n e c t i o n
*
**************************************
*
* Functional description
* Analyze a file specification and determine whether a page/lock
* server is required and available. If so, return a "connection"
* block. If not, return NULL.
*
* Note: The file name must have been expanded prior to this call.
*
**************************************/
return 0;
}
FIL PIO_create(DBB dbb, TEXT * string, SSHORT length, BOOLEAN overwrite)
{
/**************************************
*
* P I O _ c r e a t e
*
**************************************
*
* Functional description
* Create a new database file.
* NOTE: This routine does not lock any mutexes on
* its own behalf. It is assumed that mutexes will
* have been locked before entry.
*
**************************************/
int desc, flag;
FIL file;
TEXT expanded_name[256], temp[256], *file_name;
file_name = string;
if (length) {
MOVE_FAST(file_name, temp, length);
temp[length] = 0;
file_name = temp;
}
#ifdef SUPERSERVER_V2
flag =
SYNC | O_RDWR | O_CREAT | (overwrite ? O_TRUNC : O_EXCL) | O_BINARY;
#else
#ifdef SUPPORT_RAW_DEVICES
flag = O_RDWR |
(raw_devices_check_file(file_name) ? 0 : O_CREAT) |
(overwrite ? O_TRUNC : O_EXCL) |
O_BINARY;
#else
flag = O_RDWR | O_CREAT | (overwrite ? O_TRUNC : O_EXCL) | O_BINARY;
#endif
#endif
if ((desc = open(file_name, flag, MASK)) == -1)
ERR_post(isc_io_error,
gds_arg_string, "open O_CREAT",
gds_arg_cstring, length, ERR_string(string, length),
isc_arg_gds, isc_io_create_err, gds_arg_unix, errno, 0);
/* File open succeeded. Now expand the file name. */
length = PIO_expand(string, length, expanded_name);
file = setup_file(dbb, expanded_name, length, desc);
return file;
}
int PIO_expand(TEXT * file_name, USHORT file_length, TEXT * expanded_name)
{
/**************************************
*
* P I O _ e x p a n d
*
**************************************
*
* Functional description
* Fully expand a file name. If the file doesn't exist, do something
* intelligent.
*
**************************************/
return ISC_expand_filename(file_name, file_length, expanded_name);
}
void PIO_flush(FIL main_file)
{
/**************************************
*
* P I O _ f l u s h
*
**************************************
*
* Functional description
* Flush the operating system cache back to good, solid oxide.
*
**************************************/
FIL file;
/* Since all SUPERSERVER_V2 database and shadow I/O is synchronous, this
is a no-op. */
#ifndef SUPERSERVER_V2
for (file = main_file; file; file = file->fil_next) {
if (file->fil_desc != -1) { /* This really should be an error */
THD_MUTEX_LOCK(file->fil_mutex);
fsync(file->fil_desc);
THD_MUTEX_UNLOCK(file->fil_mutex);
}
}
#endif
}
void PIO_force_write(FIL file, USHORT flag)
{
/**************************************
*
* P I O _ f o r c e _ w r i t e ( G E N E R I C )
*
**************************************
*
* Functional description
* Set (or clear) force write, if possible, for the database.
*
**************************************/
#ifdef hpux
union fcntlun control;
#else
int control;
#endif
/* Since all SUPERSERVER_V2 database and shadow I/O is synchronous, this
is a no-op. */
#ifndef SUPERSERVER_V2
#ifdef hpux
control.val = (flag) ? SYNC : NULL;
#else
control = (flag) ? SYNC : 0;
#endif
if (fcntl(file->fil_desc, F_SETFL, control) == -1)
{
ERR_post(isc_io_error,
gds_arg_string, "fcntl SYNC",
gds_arg_cstring, file->fil_length,
ERR_string(file->fil_string, file->fil_length), isc_arg_gds,
isc_io_access_err, gds_arg_unix, errno, 0);
}
else
{
if (flag) {
file->fil_flags |= (FIL_force_write | FIL_force_write_init);
} else {
file->fil_flags &= ~FIL_force_write;
}
}
#endif
}
void PIO_header(DBB dbb, SCHAR * address, int length)
{
/**************************************
*
* P I O _ h e a d e r
*
**************************************
*
* Functional description
* Read the page header. This assumes that the file has not been
* repositioned since the file was originally mapped.
*
**************************************/
FIL file;
SSHORT i;
UINT64 bytes;
file = dbb->dbb_file;
ISC_inhibit();
if (file->fil_desc == -1)
unix_error("PIO_header", file, isc_io_read_err, 0);
for (i = 0; i < IO_RETRY; i++) {
#ifndef PREAD_PWRITE
THD_MUTEX_LOCK(file->fil_mutex);
if ((lseek(file->fil_desc, LSEEK_OFFSET_CAST 0, 0)) == -1) {
THD_MUTEX_UNLOCK(file->fil_mutex);
unix_error("lseek", file, isc_io_read_err, 0);
}
#endif
#ifdef ISC_DATABASE_ENCRYPTION
if (dbb->dbb_encrypt_key) {
SLONG spare_buffer[MAX_PAGE_SIZE / sizeof(SLONG)];
#ifdef PREAD_PWRITE
if ((bytes = pread(file->fil_desc, spare_buffer, length, 0)) ==
-1) {
#else
if ((bytes = read(file->fil_desc, spare_buffer, length)) == -1) {
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (SYSCALL_INTERRUPTED(errno))
continue;
unix_error("read", file, isc_io_read_err, 0);
}
(*dbb->dbb_decrypt) (dbb->dbb_encrypt_key->str_data,
spare_buffer, length, address);
}
else
#endif /* ISC_DATABASE_ENCRYPTION */
#ifdef PREAD_PWRITE
if ((bytes = pread(file->fil_desc, address, length, 0)) == -1) {
#else
if ((bytes = read(file->fil_desc, address, length)) == -1) {
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (SYSCALL_INTERRUPTED(errno))
continue;
unix_error("read", file, isc_io_read_err, 0);
}
else
break;
}
if (i == IO_RETRY) {
if (bytes == 0) {
#ifdef DEV_BUILD
ib_fprintf(ib_stderr, "PIO_header: an empty page read!\n");
ib_fflush(ib_stderr);
#endif
}
else {
#ifdef DEV_BUILD
ib_fprintf(ib_stderr, "PIO_header: retry count exceeded\n");
ib_fflush(ib_stderr);
#endif
unix_error("read_retry", file, isc_io_read_err, 0);
}
}
#ifndef PREAD_PWRITE
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
ISC_enable();
}
SLONG PIO_max_alloc(DBB dbb)
{
/**************************************
*
* P I O _ m a x _ a l l o c
*
**************************************
*
* Functional description
* Compute last physically allocated page of database.
*
**************************************/
struct stat statistics;
UINT64 length;
FIL file;
for (file = dbb->dbb_file; file->fil_next; file = file->fil_next);
if (file->fil_desc == -1) {
ISC_inhibit();
unix_error("fstat", file, isc_io_access_err, 0);
return (0);
}
if (fstat(file->fil_desc, &statistics)) {
ISC_inhibit();
unix_error("fstat", file, isc_io_access_err, 0);
}
length = statistics.st_size;
/****
#ifndef sun
length = statistics.st_size;
#else
length = statistics.st_blocks * statistics.st_blksize;
#endif
****/
return file->fil_min_page - file->fil_fudge +
(length + dbb->dbb_page_size - 1) / dbb->dbb_page_size;
}
SLONG PIO_act_alloc(DBB dbb)
{
/**************************************
*
* P I O _ a c t _ a l l o c
*
**************************************
*
* Functional description
* Compute actual number of physically allocated pages of database.
*
**************************************/
struct stat statistics;
UINT64 length;
FIL file;
ULONG tot_pages = 0;
/**
** Traverse the linked list of files and add up the number of pages
** in each file
**/
for (file = dbb->dbb_file; file != NULL; file = file->fil_next) {
if (file->fil_desc == -1) {
ISC_inhibit();
unix_error("fstat", file, isc_io_access_err, 0);
return (0);
}
if (fstat(file->fil_desc, &statistics)) {
ISC_inhibit();
unix_error("fstat", file, isc_io_access_err, 0);
}
length = statistics.st_size;
tot_pages += (length + dbb->dbb_page_size - 1) / dbb->dbb_page_size;
}
return tot_pages;
}
FIL PIO_open(DBB dbb,
TEXT * string,
SSHORT length,
SSHORT trace_flag,
BLK connection, TEXT * file_name, USHORT file_length)
{
/**************************************
*
* P I O _ o p e n
*
**************************************
*
* Functional description
* Open a database file. If a "connection" block is provided, use
* the connection to communication with a page/lock server.
*
**************************************/
TEXT *ptr, temp[256];
int desc, i, flag;
if (string) {
ptr = string;
if (length) {
MOVE_FAST(string, temp, length);
temp[length] = 0;
ptr = temp;
}
}
else {
ptr = file_name;
if (file_length) {
MOVE_FAST(file_name, temp, file_length);
temp[file_length] = 0;
ptr = temp;
}
}
#ifdef SUPERSERVER_V2
flag = SYNC | O_RDWR | O_BINARY;
#else
flag = O_RDWR | O_BINARY;
#endif
for (i = 0; i < IO_RETRY; i++) {
if ((desc = open(ptr, flag)) != -1)
break;
if (!SYSCALL_INTERRUPTED(errno))
break;
}
if (desc == -1) {
/* Try opening the database file in ReadOnly mode. The database file could
* be on a RO medium (CD-ROM etc.). If this fileopen fails, return error.
*/
flag &= ~O_RDWR;
flag |= O_RDONLY;
if ((desc = open(ptr, flag)) == -1) {
ERR_post(isc_io_error,
gds_arg_string, "open",
gds_arg_cstring, file_length, ERR_string(file_name,
file_length),
isc_arg_gds, isc_io_open_err, gds_arg_unix, errno, 0);
}
else {
/* If this is the primary file, set DBB flag to indicate that it is
* being opened ReadOnly. This flag will be used later to compare with
* the Header Page flag setting to make sure that the database is set
* ReadOnly.
*/
if (!dbb->dbb_file)
dbb->dbb_flags |= DBB_being_opened_read_only;
}
}
#ifdef SUPPORT_RAW_DEVICES
/* At this point the file has successfully been opened in either RW or RO
* mode. Check if it is a special file (i.e. raw block device) and if a
* valid database is on it. If not, return an error.
*/
if (raw_devices_check_file(file_name)
&& !raw_devices_validate_database(desc, file_name, file_length))
{
ERR_post (isc_io_error,
gds_arg_string, "open",
gds_arg_cstring, file_length,
ERR_string (file_name, file_length),
isc_arg_gds, isc_io_open_err,
gds_arg_unix, ENOENT, 0);
}
#endif /* SUPPORT_RAW_DEVICES */
return setup_file(dbb, string, length, desc);
}
int PIO_read(FIL file, BDB bdb, PAG page, STATUS * status_vector)
{
/**************************************
*
* P I O _ r e a d
*
**************************************
*
* Functional description
* Read a data page. Oh wow.
*
**************************************/
DBB dbb;
SSHORT i;
UINT64 bytes, size, offset;
ISC_inhibit();
if (file->fil_desc == -1)
return unix_error("read", file, isc_io_read_err, status_vector);
dbb = bdb->bdb_dbb;
size = dbb->dbb_page_size;
#ifdef ISC_DATABASE_ENCRYPTION
if (dbb->dbb_encrypt_key) {
SLONG spare_buffer[MAX_PAGE_SIZE / sizeof(SLONG)];
for (i = 0; i < IO_RETRY; i++) {
if (!(file = seek_file(file, bdb, &offset, status_vector)))
return FALSE;
#ifdef PREAD_PWRITE
if ((bytes = pread (file->fil_desc, spare_buffer, size, LSEEK_OFFSET_CAST offset)) == size)
#else
if ((bytes = read(file->fil_desc, spare_buffer, size)) == size)
#endif
{
(*dbb->dbb_decrypt) (dbb->dbb_encrypt_key->str_data,
spare_buffer, size, page);
break;
}
#ifndef PREAD_PWRITE
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (bytes == -1U && !SYSCALL_INTERRUPTED(errno))
return unix_error("read", file, isc_io_read_err,
status_vector);
}
}
else
#endif /* ISC_DATABASE_ENCRYPTION */
{
for (i = 0; i < IO_RETRY; i++) {
if (!(file = seek_file(file, bdb, &offset, status_vector)))
return FALSE;
#ifdef PREAD_PWRITE
if ((bytes = pread(file->fil_desc, page, size, LSEEK_OFFSET_CAST offset)) == size)
break;
#else
if ((bytes = read(file->fil_desc, page, size)) == size)
break;
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (bytes == -1U && !SYSCALL_INTERRUPTED(errno))
return unix_error("read", file, isc_io_read_err,
status_vector);
}
}
#ifndef PREAD_PWRITE
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (i == IO_RETRY) {
if (bytes == 0) {
#ifdef DEV_BUILD
ib_fprintf(ib_stderr, "PIO_read: an empty page read!\n");
ib_fflush(ib_stderr);
#endif
}
else {
#ifdef DEV_BUILD
ib_fprintf(ib_stderr, "PIO_read: retry count exceeded\n");
ib_fflush(ib_stderr);
#endif
unix_error("read_retry", file, isc_io_read_err, 0);
}
}
ISC_enable();
return TRUE;
}
int PIO_write(FIL file, BDB bdb, PAG page, STATUS * status_vector)
{
/**************************************
*
* P I O _ w r i t e
*
**************************************
*
* Functional description
* Write a data page. Oh wow.
*
**************************************/
DBB dbb;
SSHORT i;
SLONG bytes, size;
UINT64 offset;
ISC_inhibit();
if (file->fil_desc == -1)
return unix_error("write", file, isc_io_write_err, status_vector);
dbb = bdb->bdb_dbb;
size = dbb->dbb_page_size;
#ifdef ISC_DATABASE_ENCRYPTION
if (dbb->dbb_encrypt_key) {
SLONG spare_buffer[MAX_PAGE_SIZE / sizeof(SLONG)];
(*dbb->dbb_encrypt) (dbb->dbb_encrypt_key->str_data,
page, size, spare_buffer);
for (i = 0; i < IO_RETRY; i++) {
if (!(file = seek_file(file, bdb, &offset, status_vector)))
return FALSE;
#ifdef PREAD_PWRITE
if ((bytes = pwrite(file->fil_desc, spare_buffer, size, LSEEK_OFFSET_CAST offset)) == size)
break;
#else
if ((bytes = write(file->fil_desc, spare_buffer, size)) == size)
break;
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (bytes == -1U && !SYSCALL_INTERRUPTED(errno))
return unix_error("write", file, isc_io_write_err,
status_vector);
}
}
else
#endif /* ISC_DATABASE_ENCRYPTION */
{
for (i = 0; i < IO_RETRY; i++) {
if (!(file = seek_file(file, bdb, &offset, status_vector)))
return FALSE;
#ifdef PREAD_PWRITE
if ((bytes = pwrite(file->fil_desc, page, size, LSEEK_OFFSET_CAST offset)) == size)
break;
#else
if ((bytes = write(file->fil_desc, page, size)) == size)
break;
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
if (bytes == -1U && !SYSCALL_INTERRUPTED(errno))
return unix_error("write", file, isc_io_write_err,
status_vector);
}
}
#ifndef PREAD_PWRITE
THD_MUTEX_UNLOCK(file->fil_mutex);
#endif
ISC_enable();
return TRUE;
}
#ifdef UNIX
static void close_marker_file(TEXT * marker_filename)
{
/***************************************
*
* c l o s e _ m a r k e r _ f i l e
*
***************************************
*
* Functional description
* The caller has already found a marker
* file exists, so open the file, read
* the absolute path to the database,
* then start reading fildes and closing
* the open marker files.
*
***************************************/
TEXT fildes_str[5], marker_file_path[MAXPATHLEN];
IB_FILE *fp;
int fd;
/* Open the marker file and save the marker_file_path for later. */
if ((fp = ib_fopen(marker_filename, "r")) == NULL)
return;
if (ib_fgets(marker_file_path, sizeof(marker_file_path), fp) == NULL)
return;
/* Read the fildes string, convert it into a file descriptor and close. */
while (ib_fgets(fildes_str, sizeof(fildes_str), fp) != NULL) {
sscanf(fildes_str, "%d", &fd);
close(fd);
}
/* Remove all fildes from marker file by writing just the path to the file. */
if ((fp = ib_fopen(marker_filename, "w")) == NULL)
return;
ib_fputs(marker_file_path, fp);
ib_fclose(fp);
}
#endif
static FIL seek_file(FIL file, BDB bdb, UINT64 * offset, STATUS * status_vector)
{
/**************************************
*
* s e e k _ f i l e
*
**************************************
*
* Functional description
* Given a buffer descriptor block, find the appropropriate
* file block and seek to the proper page in that file.
*
**************************************/
ULONG page;
DBB dbb;
UINT64 lseek_offset;
dbb = bdb->bdb_dbb;
page = bdb->bdb_page;
for (;; file = file->fil_next)
if (!file) {
ISC_enable();
CORRUPT(158); /* msg 158 cannot sort on a field that does not exist */
}
else if (page >= file->fil_min_page && page <= file->fil_max_page)
break;
if (file->fil_desc == -1)
return (FIL) unix_error("lseek", file, isc_io_access_err,
status_vector);
page -= file->fil_min_page - file->fil_fudge;
lseek_offset = page;
lseek_offset *= dbb->dbb_page_size;
#if _FILE_OFFSET_BITS != 64
if (lseek_offset > MAX_SLONG) {
return (FIL) unix_error ("lseek", file, isc_io_32bit_exceeded_err, status_vector);
}
#endif
#ifdef PREAD_PWRITE
*offset = lseek_offset;
#else
THD_MUTEX_LOCK(file->fil_mutex);
if ((lseek(file->fil_desc, LSEEK_OFFSET_CAST lseek_offset, 0)) == (off_t)-1) {
THD_MUTEX_UNLOCK(file->fil_mutex);
return (FIL) unix_error("lseek", file, isc_io_access_err,
status_vector);
}
#endif
return file;
}
static FIL setup_file(
DBB dbb, TEXT * file_name, USHORT file_length, int desc)
{
/**************************************
*
* s e t u p _ f i l e
*
**************************************
*
* Functional description
* Set up file and lock blocks for a file.
*
**************************************/
FIL file;
LCK lock;
UCHAR *p, *q, lock_string[32];
USHORT l;
struct stat statistics;
/* Allocate file block and copy file name string */
file = FB_NEW_RPT(*dbb->dbb_permanent, file_length + 1) fil();
file->fil_desc = desc;
file->fil_length = file_length;
file->fil_max_page = -1;
MOVE_FAST(file_name, file->fil_string, file_length);
file->fil_string[file_length] = '\0';
#ifndef PREAD_PWRITE
THD_MUTEX_INIT(file->fil_mutex);
#endif
/* If this isn't the primary file, we're done */
if (dbb->dbb_file)
return file;
/* Build unique lock string for file and construct lock block */
fstat(desc, &statistics);
p = lock_string;
q = (UCHAR *) & statistics.st_dev;
l = sizeof(statistics.st_dev);
do
*p++ = *q++;
while (--l);
q = (UCHAR *) & statistics.st_ino;
l = sizeof(statistics.st_ino);
do
*p++ = *q++;
while (--l);
l = p - lock_string;
dbb->dbb_lock = lock = FB_NEW_RPT(*dbb->dbb_permanent, l) lck();
lock->lck_type = LCK_database;
lock->lck_owner_handle = LCK_get_owner_handle(NULL_TDBB, lock->lck_type);
lock->lck_object = reinterpret_cast<blk*>(dbb);
lock->lck_length = l;
lock->lck_dbb = dbb;
lock->lck_ast = reinterpret_cast<lck_ast_t>(CCH_down_grade_dbb);
MOVE_FAST(lock_string, lock->lck_key.lck_string, l);
/* Try to get an exclusive lock on database. If this fails, insist
on at least a shared lock */
dbb->dbb_flags |= DBB_exclusive;
if (!LCK_lock(NULL_TDBB, lock, LCK_EX, LCK_NO_WAIT)) {
dbb->dbb_flags &= ~DBB_exclusive;
LCK_lock(NULL_TDBB, lock, LCK_SW, LCK_WAIT);
}
return file;
}
static BOOLEAN unix_error(
TEXT * string,
FIL file, STATUS operation, STATUS * status_vector)
{
/**************************************
*
* u n i x _ e r r o r
*
**************************************
*
* Functional description
* Somebody has noticed a file system error and expects error
* to do something about it. Harumph!
*
**************************************/
STATUS *status;
ISC_enable();
if (status = status_vector) {
*status++ = isc_arg_gds;
*status++ = isc_io_error;
*status++ = gds_arg_string;
*status++ = (STATUS) string;
*status++ = gds_arg_string;
*status++ = (STATUS) ERR_string(file->fil_string, file->fil_length);
*status++ = isc_arg_gds;
*status++ = operation;
*status++ = gds_arg_unix;
*status++ = errno;
*status++ = gds_arg_end;
gds__log_status(0, status_vector);
return FALSE;
}
else
ERR_post(isc_io_error,
gds_arg_string, string,
gds_arg_string, ERR_string(file->fil_string,
file->fil_length), isc_arg_gds,
operation, gds_arg_unix, errno, 0);
// Added a FALSE for final return - which seems to be the answer,
// but is better than what it was which was nothing ie random
// Most usages within here want it to return a failure.
// MOD 01-July-2002
return FALSE;
}
#if defined PREAD_PWRITE && !(defined HAVE_PREAD && defined HAVE_PWRITE)
/* pread() and pwrite() behave like read() and write() except that they
take an additional 'offset' argument. The I/O takes place at the specified
'offset' from the beginning of the file and does not affect the offset
associated with the file descriptor.
This is done in order to allow more than one thread to operate on
individual records within the same file simultaneously. This is
called Positioned I/O. Since positioned I/O is not currently directly
available through the POSIX interfaces, it has been implemented
using the POSIX asynchronous I/O calls.
NOTE: pread() and pwrite() are defined in UNIX International system
interface and are a part of POSIX systems.
*/
static SLONG pread(int fd, SCHAR * buf, SLONG nbytes, SLONG offset)
/**************************************
*
* p r e a d
*
**************************************
*
* Functional description
*
* This function uses Asynchronous I/O calls to implement
* positioned read from a given offset
**************************************/
{
struct aiocb io;
struct aiocb *list[1];
int err;
io.aio_fildes = fd;
io.aio_offset = offset;
io.aio_buf = buf;
io.aio_nbytes = nbytes;
io.aio_reqprio = 0;
io.aio_sigevent.sigev_notify = SIGEV_NONE;
err = aio_read(&io); /* atomically reads at offset */
if (err != 0)
return (err); /* errno is set */
list[0] = &io;
err = aio_suspend(list, 1, NULL); /* wait for I/O to complete */
if (err != 0)
return (err); /* errno is set */
return (aio_return(&io)); /* return I/O status */
}
static SLONG pwrite(int fd, SCHAR * buf, SLONG nbytes, SLONG offset)
/**************************************
*
* p w r i t e
*
**************************************
*
* Functional description
*
* This function uses Asynchronous I/O calls to implement
* positioned write from a given offset
**************************************/
{
struct aiocb io;
struct aiocb *list[1];
int err;
io.aio_fildes = fd;
io.aio_offset = offset;
io.aio_buf = buf;
io.aio_nbytes = nbytes;
io.aio_reqprio = 0;
io.aio_sigevent.sigev_notify = SIGEV_NONE;
err = aio_write(&io); /* atomically reads at offset */
if (err != 0)
return (err); /* errno is set */
list[0] = &io;
err = aio_suspend(list, 1, NULL); /* wait for I/O to complete */
if (err != 0)
return (err); /* errno is set */
return (aio_return(&io)); /* return I/O status */
}
#endif /* PREAD_PWRITE && !(HAVE_PREAD && HAVE_PWRITE)*/
int PIO_unlink (
TEXT *file_name)
{
/**************************************
*
* P I O _ u n l i n k
*
**************************************
*
* Functional description
* Delete a database file.
*
**************************************/
#ifdef SUPPORT_RAW_DEVICES
if (raw_devices_check_file(file_name))
return raw_devices_unlink_database(file_name);
else
#endif
return unlink(file_name);
}
#ifdef SUPPORT_RAW_DEVICES
static BOOLEAN
raw_devices_check_file (
TEXT *file_name)
{
/**************************************
*
* raw_devices_check_file
*
**************************************
*
* Functional description
* Checks if the supplied file name is a special file
*
**************************************/
struct stat s;
return (stat(file_name, &s) == 0
&& (S_ISCHR(s.st_mode) || S_ISBLK(s.st_mode)));
}
static BOOLEAN
raw_devices_validate_database (
int desc,
TEXT *file_name,
USHORT file_length)
{
/**************************************
*
* raw_devices_validate_database
*
**************************************
*
* Functional description
* Checks if the special file contains a valid database
*
**************************************/
char header[MIN_PAGE_SIZE];
HDR hp = (HDR)header;
ssize_t bytes;
BOOLEAN retval = FALSE;
int i;
/* Read in database header. Code lifted from PIO_header. */
if (desc == -1)
ERR_post (isc_io_error,
gds_arg_string, "raw_devices_validate_database",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_read_err,
gds_arg_unix, errno, 0);
for (i = 0; i < IO_RETRY; i++)
{
if (lseek (desc, LSEEK_OFFSET_CAST 0, 0) == (off_t)-1)
ERR_post (isc_io_error,
gds_arg_string, "lseek",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_read_err,
gds_arg_unix, errno, 0);
if ((bytes = read (desc, header, sizeof(header))) == sizeof(header))
goto read_finished;
if (bytes == -1 && !SYSCALL_INTERRUPTED(errno))
ERR_post (isc_io_error,
gds_arg_string, "read",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_read_err,
gds_arg_unix, errno, 0);
}
ERR_post (isc_io_error,
gds_arg_string, "read_retry",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_read_err,
gds_arg_unix, errno, 0);
read_finished:
/* Rewind file pointer */
if (lseek (desc, LSEEK_OFFSET_CAST 0, 0) == (off_t)-1)
ERR_post (isc_io_error,
gds_arg_string, "lseek",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_read_err,
gds_arg_unix, errno, 0);
/* Validate database header. Code lifted from PAG_header. */
if (hp->hdr_header.pag_type != pag_header || hp->hdr_sequence)
goto quit;
#ifdef ODS_8_TO_CURRENT
/* This Server understands ODS greater than 8 *ONLY* upto current major
* ODS_VERSION defined in ods.h, Refuse connections to older or newer ODS's.
*/
if ((hp->hdr_ods_version < ODS_VERSION8)
|| (hp->hdr_ods_version > ODS_VERSION))
#else
if (hp->hdr_ods_version != ODS_VERSION)
#endif
goto quit;
if (hp->hdr_page_size < MIN_PAGE_SIZE || hp->hdr_page_size > MAX_PAGE_SIZE)
goto quit;
/* At this point we think we have identified a database on the device.
* PAG_header will validate the entire structure later.
*/
retval = TRUE;
quit:
#ifdef DEV_BUILD
gds__log ("raw_devices_validate_database: %s -> %s\n",
file_name, retval ? "true" : "false");
#endif
return retval;
}
static int
raw_devices_unlink_database (
TEXT *file_name)
{
char header[MIN_PAGE_SIZE];
size_t file_length = strlen(file_name);
ssize_t bytes;
int desc, i;
for (i = 0; i < IO_RETRY; i++)
{
if ((desc = open (file_name, O_RDWR | O_BINARY)) != -1)
break;
if (!SYSCALL_INTERRUPTED(errno))
ERR_post (isc_io_error,
gds_arg_string, "open",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_open_err,
gds_arg_unix, errno, 0);
}
memset(header, 0xa5, sizeof(header));
for (i = 0; i < IO_RETRY; i++)
{
if ((bytes = write (desc, header, sizeof(header))) == sizeof(header))
break;
if (bytes == -1 && SYSCALL_INTERRUPTED(errno))
continue;
ERR_post (isc_io_error,
gds_arg_string, "write",
gds_arg_string, ERR_string (file_name, file_length),
isc_arg_gds, isc_io_write_err,
gds_arg_unix, errno, 0);
}
(void)close(desc);
#if DEV_BUILD
gds__log ("raw_devices_unlink_database: %s -> %s\n",
file_name, i < IO_RETRY ? "true" : "false");
#endif
return 0;
}
#endif /* SUPPORT_RAW_DEVICES */
} // extern "C"