2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-11-07 09:06:35 +01:00
|
|
|
* MODULE: ext.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: External file access
|
|
|
|
*
|
|
|
|
* 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-08-08 04:01:26 +02:00
|
|
|
*
|
2008-12-05 02:20:14 +01:00
|
|
|
* 26-Sept-2001 Paul Beach - Windows External File Directory Config. Parameter
|
2002-06-30 11:58:20 +02:00
|
|
|
*
|
2001-07-10 19:35:13 +02:00
|
|
|
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
|
|
* conditionals, as the engine now fully supports
|
|
|
|
* readonly databases.
|
2001-08-08 04:01:26 +02:00
|
|
|
*
|
|
|
|
* 2001.08.07 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
|
|
* conditionals, second attempt
|
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:36:29 +02:00
|
|
|
#include <stdio.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
2010-05-22 04:19:01 +02:00
|
|
|
#include <sys/stat.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/val.h"
|
|
|
|
#include "../jrd/exe.h"
|
|
|
|
#include "../jrd/ext.h"
|
|
|
|
#include "../jrd/tra.h"
|
2010-10-24 02:26:00 +02:00
|
|
|
#include "../dsql/ExprNodes.h"
|
2021-09-15 13:37:17 +02:00
|
|
|
#include "iberror.h"
|
2006-07-18 08:08:15 +02:00
|
|
|
#include "../jrd/cmp_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/ext_proto.h"
|
2010-10-12 10:02:57 +02:00
|
|
|
#include "../yvalve/gds_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#include "../jrd/mov_proto.h"
|
|
|
|
#include "../jrd/vio_proto.h"
|
2003-03-15 21:20:41 +01:00
|
|
|
#include "../common/config/config.h"
|
|
|
|
#include "../common/config/dir_list.h"
|
2010-10-12 10:02:57 +02:00
|
|
|
#include "../common/os/path_utils.h"
|
2004-03-14 14:40:14 +01:00
|
|
|
#include "../common/classes/init.h"
|
2014-04-22 15:37:14 +02:00
|
|
|
#include "../common/isc_f_proto.h"
|
2014-12-17 15:31:02 +01:00
|
|
|
#include "../common/os/os_utils.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2009-08-30 21:00:46 +02:00
|
|
|
#if defined _MSC_VER && _MSC_VER < 1400
|
|
|
|
// NS: in VS2003 these only work with static CRT
|
|
|
|
extern "C" {
|
2009-09-02 06:23:02 +02:00
|
|
|
int __cdecl _fseeki64(FILE*, __int64, int);
|
|
|
|
__int64 __cdecl _ftelli64(FILE*);
|
2009-08-30 21:00:46 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-06-03 11:56:29 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
#define FTELL64 _ftelli64
|
|
|
|
#define FSEEK64 _fseeki64
|
2016-05-25 14:14:01 +02:00
|
|
|
#elif defined(LSB_BUILD)
|
|
|
|
#define FTELL64 ftello64
|
|
|
|
#define FSEEK64 fseeko64
|
2009-06-03 11:56:29 +02:00
|
|
|
#else
|
|
|
|
#define FTELL64 ftello
|
|
|
|
#define FSEEK64 fseeko
|
|
|
|
#endif
|
|
|
|
|
2008-08-27 14:20:47 +02:00
|
|
|
using namespace Firebird;
|
2004-03-20 15:57:40 +01:00
|
|
|
|
2010-03-01 03:14:36 +01:00
|
|
|
namespace Jrd
|
|
|
|
{
|
|
|
|
class ExternalFileDirectoryList : public DirectoryList
|
2004-03-14 14:40:14 +01:00
|
|
|
{
|
|
|
|
private:
|
2010-03-01 03:14:36 +01:00
|
|
|
const PathName getConfigString() const
|
2008-05-10 05:44:57 +02:00
|
|
|
{
|
2010-03-01 03:14:36 +01:00
|
|
|
return PathName(config->getExternalFileAccess());
|
2004-03-14 14:40:14 +01:00
|
|
|
}
|
2010-03-01 03:14:36 +01:00
|
|
|
|
2004-03-14 14:40:14 +01:00
|
|
|
public:
|
2010-03-07 05:30:35 +01:00
|
|
|
explicit ExternalFileDirectoryList(const Database* dbb)
|
2010-02-28 19:00:51 +01:00
|
|
|
: DirectoryList(*dbb->dbb_permanent), config(dbb->dbb_config)
|
2004-03-14 14:40:14 +01:00
|
|
|
{
|
|
|
|
initialize();
|
|
|
|
}
|
2010-03-01 03:14:36 +01:00
|
|
|
|
2010-02-28 19:00:51 +01:00
|
|
|
static void create(Database* dbb)
|
|
|
|
{
|
|
|
|
if (!dbb->dbb_external_file_directory_list)
|
|
|
|
{
|
2010-03-01 03:14:36 +01:00
|
|
|
dbb->dbb_external_file_directory_list =
|
2015-10-12 16:26:00 +02:00
|
|
|
FB_NEW_POOL(*dbb->dbb_permanent) ExternalFileDirectoryList(dbb);
|
2010-02-28 19:00:51 +01:00
|
|
|
}
|
|
|
|
}
|
2010-03-01 03:14:36 +01:00
|
|
|
|
2010-02-28 19:00:51 +01:00
|
|
|
private:
|
2017-01-31 15:38:53 +01:00
|
|
|
RefPtr<const Config> config;
|
2004-03-14 14:40:14 +01:00
|
|
|
};
|
2010-02-28 19:00:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
using namespace Jrd;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
static const char* const FOPEN_TYPE = "a+b";
|
|
|
|
#else
|
|
|
|
static const char* const FOPEN_TYPE = "a+";
|
|
|
|
#endif
|
|
|
|
static const char* const FOPEN_READ_ONLY = "rb";
|
2004-03-14 14:40:14 +01:00
|
|
|
|
2015-10-13 04:22:57 +02:00
|
|
|
FILE* ext_fopen(Database* dbb, ExternalFile* ext_file)
|
2007-03-19 22:34:52 +01:00
|
|
|
{
|
2008-04-22 13:27:42 +02:00
|
|
|
const char* file_name = ext_file->ext_filename;
|
2007-03-19 22:34:52 +01:00
|
|
|
|
2010-02-28 19:00:51 +01:00
|
|
|
ExternalFileDirectoryList::create(dbb);
|
|
|
|
if (!dbb->dbb_external_file_directory_list->isPathInList(file_name))
|
2008-05-10 05:44:57 +02:00
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
ERR_post(Arg::Gds(isc_conf_access_denied) << Arg::Str("external file") <<
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Str(file_name));
|
2008-05-10 05:44:57 +02:00
|
|
|
}
|
2004-03-14 14:40:14 +01:00
|
|
|
|
2007-03-19 22:34:52 +01:00
|
|
|
// If the database is updateable, then try opening the external files in
|
|
|
|
// RW mode. If the DB is ReadOnly, then open the external files only in
|
|
|
|
// ReadOnly mode, thus being consistent.
|
2012-04-12 11:02:13 +02:00
|
|
|
if (!dbb->readOnly())
|
2014-12-17 15:31:02 +01:00
|
|
|
ext_file->ext_ifi = os_utils::fopen(file_name, FOPEN_TYPE);
|
2007-03-19 22:34:52 +01:00
|
|
|
|
|
|
|
if (!ext_file->ext_ifi)
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// could not open the file as read write attempt as read only
|
2014-12-17 15:31:02 +01:00
|
|
|
if (!(ext_file->ext_ifi = os_utils::fopen(file_name, FOPEN_READ_ONLY)))
|
2007-03-19 22:34:52 +01:00
|
|
|
{
|
2008-12-20 09:12:19 +01:00
|
|
|
ERR_post(Arg::Gds(isc_io_error) << Arg::Str("fopen") << Arg::Str(file_name) <<
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Gds(isc_io_open_err) << SYS_ERR(errno));
|
2007-03-19 22:34:52 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
ext_file->ext_flags |= EXT_readonly;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ext_file->ext_ifi;
|
2003-03-23 17:50:54 +01:00
|
|
|
}
|
2004-06-14 01:45:02 +02:00
|
|
|
} // namespace
|
2003-03-23 17:50:54 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2010-05-21 08:14:25 +02:00
|
|
|
double EXT_cardinality(thread_db* tdbb, jrd_rel* relation)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ c a r d i n a l i t y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return cardinality for the external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
ExternalFile* const file = relation->rel_file;
|
|
|
|
fb_assert(file);
|
|
|
|
|
2013-08-06 20:49:49 +02:00
|
|
|
try
|
2010-05-21 08:14:25 +02:00
|
|
|
{
|
2013-08-06 20:49:49 +02:00
|
|
|
bool must_close = false;
|
|
|
|
if (!file->ext_ifi)
|
|
|
|
{
|
|
|
|
ext_fopen(tdbb->getDatabase(), file);
|
|
|
|
must_close = true;
|
|
|
|
}
|
2010-05-21 08:14:25 +02:00
|
|
|
|
2013-08-06 20:49:49 +02:00
|
|
|
FB_UINT64 file_size = 0;
|
2010-05-21 08:14:25 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2013-08-06 20:49:49 +02:00
|
|
|
struct __stat64 statistics;
|
|
|
|
if (!_fstat64(_fileno(file->ext_ifi), &statistics))
|
2010-05-21 08:14:25 +02:00
|
|
|
#else
|
2016-05-25 14:14:01 +02:00
|
|
|
struct STAT statistics;
|
2016-05-30 16:50:02 +02:00
|
|
|
if (!os_utils::fstat(fileno(file->ext_ifi), &statistics))
|
2010-05-21 08:14:25 +02:00
|
|
|
#endif
|
2013-08-06 20:49:49 +02:00
|
|
|
{
|
|
|
|
file_size = statistics.st_size;
|
|
|
|
}
|
2010-05-21 08:14:25 +02:00
|
|
|
|
2013-08-06 20:49:49 +02:00
|
|
|
if (must_close)
|
|
|
|
{
|
|
|
|
fclose(file->ext_ifi);
|
|
|
|
file->ext_ifi = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Format* const format = MET_current(tdbb, relation);
|
|
|
|
fb_assert(format && format->fmt_length);
|
|
|
|
const USHORT offset = (USHORT)(IPTR) format->fmt_desc[0].dsc_address;
|
|
|
|
const ULONG record_length = format->fmt_length - offset;
|
|
|
|
|
|
|
|
return (double) file_size / record_length;
|
|
|
|
}
|
|
|
|
catch (const Exception&)
|
2010-05-21 08:14:25 +02:00
|
|
|
{
|
2013-08-06 20:49:49 +02:00
|
|
|
fb_utils::init_status(tdbb->tdbb_status_vector);
|
2010-05-21 08:14:25 +02:00
|
|
|
}
|
|
|
|
|
2013-08-06 20:49:49 +02:00
|
|
|
return 10000; // just a wild guess
|
2010-05-21 08:14:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-28 15:08:04 +02:00
|
|
|
void EXT_erase(record_param*, jrd_tra*)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ e r a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Update an external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post(Arg::Gds(isc_ext_file_delete));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
// Third param is unused.
|
2009-04-28 15:08:04 +02:00
|
|
|
ExternalFile* EXT_file(jrd_rel* relation, const TEXT* file_name) //, bid* description)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ f i l e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create a file block for external file access.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-23 05:18:10 +02:00
|
|
|
Database* dbb = GET_DBB();
|
2001-05-23 15:26:42 +02:00
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
2009-08-22 14:16:47 +02:00
|
|
|
// if we already have a external file associated with this relation just
|
|
|
|
// return the file structure
|
2001-05-23 15:26:42 +02:00
|
|
|
if (relation->rel_file) {
|
2007-03-19 22:34:52 +01:00
|
|
|
EXT_fini(relation, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-06-30 11:58:20 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2009-08-22 14:16:47 +02:00
|
|
|
// Default number of file handles stdio.h on Windows is 512, use this
|
|
|
|
// call to increase and set to the maximum
|
2002-06-30 11:58:20 +02:00
|
|
|
_setmaxstdio(2048);
|
|
|
|
#endif
|
|
|
|
|
2014-04-22 15:37:14 +02:00
|
|
|
// If file_name is relative expand it in ExternalFilesPath.
|
|
|
|
PathName newName, name(file_name);
|
|
|
|
if (PathUtils::isRelative(name))
|
2009-08-23 09:31:48 +02:00
|
|
|
{
|
2010-02-28 19:00:51 +01:00
|
|
|
ExternalFileDirectoryList::create(dbb);
|
2014-04-22 15:37:14 +02:00
|
|
|
if (!(dbb->dbb_external_file_directory_list->expandFileName(newName, name)))
|
2005-01-26 19:02:30 +01:00
|
|
|
{
|
2014-04-22 15:37:14 +02:00
|
|
|
if (!dbb->dbb_external_file_directory_list->defaultName(newName, name))
|
|
|
|
{
|
|
|
|
ISC_expand_filename(newName, false);
|
|
|
|
}
|
2005-01-26 19:02:30 +01:00
|
|
|
}
|
2014-04-22 15:37:14 +02:00
|
|
|
file_name = newName.c_str();
|
|
|
|
name = newName;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create missing path components
|
|
|
|
ObjectsArray<PathName> paths;
|
2014-05-09 01:26:40 +02:00
|
|
|
|
|
|
|
for (;;)
|
2014-04-22 15:37:14 +02:00
|
|
|
{
|
|
|
|
PathName path, file;
|
|
|
|
PathUtils::splitLastComponent(path, file, name);
|
|
|
|
if (path.isEmpty())
|
|
|
|
break;
|
|
|
|
|
|
|
|
int rc = PathUtils::makeDir(path.c_str());
|
|
|
|
if (rc == 0 || rc == EEXIST)
|
|
|
|
break;
|
|
|
|
paths.push(path);
|
|
|
|
name = path;
|
|
|
|
}
|
2014-05-09 01:26:40 +02:00
|
|
|
|
|
|
|
while (paths.hasData())
|
2014-04-22 15:37:14 +02:00
|
|
|
{
|
|
|
|
PathName path(paths.pop());
|
|
|
|
if (PathUtils::makeDir(path.c_str()) != 0)
|
|
|
|
break;
|
2003-03-23 17:50:54 +01:00
|
|
|
}
|
2014-05-09 01:26:40 +02:00
|
|
|
|
2014-04-22 15:37:14 +02:00
|
|
|
paths.clear();
|
2003-03-23 17:50:54 +01:00
|
|
|
|
2011-05-09 12:15:19 +02:00
|
|
|
ExternalFile* file = FB_NEW_RPT(*relation->rel_pool, (strlen(file_name) + 1)) ExternalFile();
|
2003-12-31 06:36:12 +01:00
|
|
|
relation->rel_file = file;
|
2008-04-22 13:27:42 +02:00
|
|
|
strcpy(file->ext_filename, file_name);
|
2001-05-23 15:26:42 +02:00
|
|
|
file->ext_flags = 0;
|
2003-08-28 15:16:03 +02:00
|
|
|
file->ext_ifi = NULL;
|
2002-06-30 11:58:20 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-19 22:34:52 +01:00
|
|
|
void EXT_fini(jrd_rel* relation, bool close_only)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ f i n i
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Close the file associated with a relation.
|
|
|
|
*
|
|
|
|
**************************************/
|
2009-08-23 09:31:48 +02:00
|
|
|
if (relation->rel_file)
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
ExternalFile* file = relation->rel_file;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (file->ext_ifi)
|
2007-03-19 22:34:52 +01:00
|
|
|
{
|
2005-03-26 06:37:59 +01:00
|
|
|
fclose(file->ext_ifi);
|
2007-03-19 22:34:52 +01:00
|
|
|
file->ext_ifi = NULL;
|
|
|
|
}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// before zeroing out the rel_file we need to deallocate the memory
|
2007-03-19 22:34:52 +01:00
|
|
|
if (!close_only)
|
|
|
|
{
|
|
|
|
delete file;
|
2007-03-20 15:26:10 +01:00
|
|
|
relation->rel_file = NULL;
|
2007-03-19 22:34:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-11 15:59:55 +01:00
|
|
|
bool EXT_get(thread_db* tdbb, record_param* rpb, FB_UINT64& position)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ g e t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get a record from an external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
2009-12-09 19:45:44 +01:00
|
|
|
jrd_rel* const relation = rpb->rpb_relation;
|
|
|
|
ExternalFile* const file = relation->rel_file;
|
2007-03-20 15:26:10 +01:00
|
|
|
fb_assert(file->ext_ifi);
|
2007-03-19 22:34:52 +01:00
|
|
|
|
2009-12-09 19:45:44 +01:00
|
|
|
Record* const record = rpb->rpb_record;
|
2015-02-19 15:15:00 +01:00
|
|
|
const Format* const format = record->getFormat();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2013-08-21 09:19:43 +02:00
|
|
|
const USHORT offset = (USHORT) (IPTR) format->fmt_desc[0].dsc_address;
|
2015-02-19 15:15:00 +01:00
|
|
|
UCHAR* p = record->getData() + offset;
|
|
|
|
const ULONG l = record->getLength() - offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2011-01-21 17:35:24 +01:00
|
|
|
if (file->ext_ifi == NULL)
|
|
|
|
{
|
|
|
|
ERR_post(Arg::Gds(isc_io_error) << "fseek" << Arg::Str(file->ext_filename) <<
|
|
|
|
Arg::Gds(isc_io_open_err) << Arg::Unix(EBADF) <<
|
|
|
|
Arg::Gds(isc_random) << "File not opened");
|
|
|
|
}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// hvlad: fseek will flush file buffer and degrade performance, so don't
|
|
|
|
// call it if it is not necessary. Note that we must flush file buffer if we
|
2006-12-05 23:01:58 +01:00
|
|
|
// do read after write
|
2011-01-21 17:35:24 +01:00
|
|
|
|
|
|
|
bool doSeek = false;
|
|
|
|
if (!(file->ext_flags & EXT_last_read))
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2011-01-21 17:35:24 +01:00
|
|
|
doSeek = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
SINT64 offset = FTELL64(file->ext_ifi);
|
|
|
|
if (offset < 0)
|
|
|
|
{
|
|
|
|
ERR_post(Arg::Gds(isc_io_error) << STRINGIZE(FTELL64) << Arg::Str(file->ext_filename) <<
|
|
|
|
Arg::Gds(isc_io_read_err) << SYS_ERR(errno));
|
|
|
|
}
|
|
|
|
doSeek = (static_cast<FB_UINT64>(offset) != position);
|
|
|
|
}
|
|
|
|
|
|
|
|
// reset both flags cause we are going to move the file pointer
|
|
|
|
file->ext_flags &= ~(EXT_last_write | EXT_last_read);
|
|
|
|
|
|
|
|
if (doSeek)
|
|
|
|
{
|
|
|
|
if (FSEEK64(file->ext_ifi, position, SEEK_SET) != 0)
|
|
|
|
{
|
|
|
|
ERR_post(Arg::Gds(isc_io_error) << STRINGIZE(FSEEK64) << Arg::Str(file->ext_filename) <<
|
|
|
|
Arg::Gds(isc_io_open_err) << SYS_ERR(errno));
|
|
|
|
}
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-11-02 10:35:00 +01:00
|
|
|
if (!fread(p, l, 1, file->ext_ifi))
|
2011-01-21 17:35:24 +01:00
|
|
|
{
|
2006-11-02 10:35:00 +01:00
|
|
|
return false;
|
2011-01-21 17:35:24 +01:00
|
|
|
}
|
2006-11-02 10:35:00 +01:00
|
|
|
|
2009-12-09 19:45:44 +01:00
|
|
|
position += l;
|
2006-12-05 23:01:58 +01:00
|
|
|
file->ext_flags |= EXT_last_read;
|
|
|
|
|
2009-08-22 14:16:47 +02:00
|
|
|
// Loop thru fields setting missing fields to either blanks/zeros or the missing value
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
dsc desc;
|
2004-03-30 06:10:52 +02:00
|
|
|
Format::fmt_desc_const_iterator desc_ptr = format->fmt_desc.begin();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
SSHORT i = 0;
|
2005-12-02 08:35:34 +01:00
|
|
|
for (vec<jrd_fld*>::iterator itr = relation->rel_fields->begin();
|
2008-12-20 09:12:19 +01:00
|
|
|
i < format->fmt_count; ++i, ++itr, ++desc_ptr)
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2005-12-02 08:35:34 +01:00
|
|
|
const jrd_fld* field = *itr;
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2013-03-17 18:35:53 +01:00
|
|
|
record->setNull(i);
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
if (!desc_ptr->dsc_length || !field)
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2017-06-09 19:09:36 +02:00
|
|
|
const LiteralNode* literal = nodeAs<LiteralNode>(field->fld_missing_value);
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2009-08-23 09:31:48 +02:00
|
|
|
if (literal)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
desc = *desc_ptr;
|
2015-02-19 15:15:00 +01:00
|
|
|
desc.dsc_address = record->getData() + (IPTR) desc.dsc_address;
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2016-11-11 15:59:55 +01:00
|
|
|
if (!MOV_compare(tdbb, &literal->litDesc, &desc))
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2013-03-17 18:35:53 +01:00
|
|
|
record->clearNull(i);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-28 15:08:04 +02:00
|
|
|
void EXT_modify(record_param* /*old_rpb*/, record_param* /*new_rpb*/, jrd_tra* /*transaction*/)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ m o d i f y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Update an external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post(Arg::Gds(isc_ext_file_modify));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-12-09 19:45:44 +01:00
|
|
|
void EXT_open(Database* dbb, ExternalFile* file)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ o p e n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Open a record stream for an external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
2007-03-19 22:34:52 +01:00
|
|
|
if (!file->ext_ifi) {
|
2009-12-09 19:45:44 +01:00
|
|
|
ext_fopen(dbb, file);
|
2007-03-19 22:34:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-28 15:08:04 +02:00
|
|
|
void EXT_store(thread_db* tdbb, record_param* rpb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ s t o r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Update an external file.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-31 06:36:12 +01:00
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
2004-03-18 06:56:06 +01:00
|
|
|
ExternalFile* file = relation->rel_file;
|
|
|
|
Record* record = rpb->rpb_record;
|
2015-02-19 15:15:00 +01:00
|
|
|
const Format* const format = record->getFormat();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2007-03-19 22:34:52 +01:00
|
|
|
if (!file->ext_ifi) {
|
2007-12-03 16:46:39 +01:00
|
|
|
ext_fopen(tdbb->getDatabase(), file);
|
2007-03-19 22:34:52 +01:00
|
|
|
}
|
|
|
|
|
2009-08-22 14:16:47 +02:00
|
|
|
// Loop thru fields setting missing fields to either blanks/zeros or the missing value
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2009-08-22 14:16:47 +02:00
|
|
|
// check if file is read only if read only then post error we cannot write to this file
|
2009-08-23 09:31:48 +02:00
|
|
|
if (file->ext_flags & EXT_readonly)
|
|
|
|
{
|
2007-12-03 16:46:39 +01:00
|
|
|
Database* dbb = tdbb->getDatabase();
|
2001-05-23 15:26:42 +02:00
|
|
|
CHECK_DBB(dbb);
|
2009-08-22 14:16:47 +02:00
|
|
|
// Distinguish error message for a ReadOnly database
|
2012-04-12 11:02:13 +02:00
|
|
|
if (dbb->readOnly())
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post(Arg::Gds(isc_read_only_database));
|
2009-08-23 09:31:48 +02:00
|
|
|
else
|
|
|
|
{
|
2008-12-20 09:12:19 +01:00
|
|
|
ERR_post(Arg::Gds(isc_io_error) << Arg::Str("insert") << Arg::Str(file->ext_filename) <<
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Gds(isc_io_write_err) <<
|
|
|
|
Arg::Gds(isc_ext_readonly_err));
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-31 06:36:12 +01:00
|
|
|
dsc desc;
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<jrd_fld*>::iterator field_ptr = relation->rel_fields->begin();
|
2004-03-30 06:10:52 +02:00
|
|
|
Format::fmt_desc_const_iterator desc_ptr = format->fmt_desc.begin();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-02-10 04:28:43 +01:00
|
|
|
for (USHORT i = 0; i < format->fmt_count; ++i, ++field_ptr, ++desc_ptr)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2005-12-02 08:35:34 +01:00
|
|
|
const jrd_fld* field = *field_ptr;
|
2013-03-17 18:35:53 +01:00
|
|
|
if (field && !field->fld_computation && desc_ptr->dsc_length && record->isNull(i))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2015-02-19 15:15:00 +01:00
|
|
|
UCHAR* p = record->getData() + (IPTR) desc_ptr->dsc_address;
|
2017-06-09 19:09:36 +02:00
|
|
|
LiteralNode* literal = nodeAs<LiteralNode>(field->fld_missing_value);
|
2010-10-24 02:26:00 +02:00
|
|
|
|
2009-08-23 09:31:48 +02:00
|
|
|
if (literal)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
desc = *desc_ptr;
|
|
|
|
desc.dsc_address = p;
|
2010-10-24 02:26:00 +02:00
|
|
|
MOV_move(tdbb, &literal->litDesc, &desc);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2009-08-23 09:31:48 +02:00
|
|
|
else
|
|
|
|
{
|
2003-12-31 06:36:12 +01:00
|
|
|
const UCHAR pad = (desc_ptr->dsc_dtype == dtype_text) ? ' ' : 0;
|
2006-02-23 06:08:26 +01:00
|
|
|
memset(p, pad, desc_ptr->dsc_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-21 08:18:30 +01:00
|
|
|
const USHORT offset = (USHORT) (IPTR) format->fmt_desc[0].dsc_address;
|
2015-02-19 15:15:00 +01:00
|
|
|
const UCHAR* p = record->getData() + offset;
|
|
|
|
const ULONG l = record->getLength() - offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// hvlad: fseek will flush file buffer and degrade performance, so don't
|
|
|
|
// call it if it is not necessary. Note that we must flush file buffer if we
|
2006-12-05 23:01:58 +01:00
|
|
|
// do write after read
|
2011-01-21 17:35:24 +01:00
|
|
|
file->ext_flags &= ~EXT_last_read;
|
2008-12-05 02:20:14 +01:00
|
|
|
if (file->ext_ifi == NULL ||
|
2009-06-06 20:28:29 +02:00
|
|
|
(!(file->ext_flags & EXT_last_write) && FSEEK64(file->ext_ifi, (SINT64) 0, SEEK_END) != 0) )
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2011-01-21 17:35:24 +01:00
|
|
|
file->ext_flags &= ~EXT_last_write;
|
2008-12-20 09:12:19 +01:00
|
|
|
ERR_post(Arg::Gds(isc_io_error) << Arg::Str("fseek") << Arg::Str(file->ext_filename) <<
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Gds(isc_io_open_err) << SYS_ERR(errno));
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2006-11-02 10:35:00 +01:00
|
|
|
|
|
|
|
if (!fwrite(p, l, 1, file->ext_ifi))
|
|
|
|
{
|
2011-01-21 17:35:24 +01:00
|
|
|
file->ext_flags &= ~EXT_last_write;
|
2008-12-20 09:12:19 +01:00
|
|
|
ERR_post(Arg::Gds(isc_io_error) << Arg::Str("fwrite") << Arg::Str(file->ext_filename) <<
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Gds(isc_io_open_err) << SYS_ERR(errno));
|
2006-11-02 10:35:00 +01:00
|
|
|
}
|
|
|
|
|
2006-12-05 23:01:58 +01:00
|
|
|
// fflush(file->ext_ifi);
|
|
|
|
file->ext_flags |= EXT_last_write;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-03 11:59:45 +02:00
|
|
|
void EXT_tra_attach(ExternalFile* file, jrd_tra*)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ t r a _ a t t a c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Transaction going to use external table.
|
|
|
|
* Increment transactions use count.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
file->ext_tra_cnt++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EXT_tra_detach(ExternalFile* file, jrd_tra*)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E X T _ t r a _ d e t a c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2009-06-06 20:13:57 +02:00
|
|
|
* Transaction used external table is finished.
|
2009-06-03 11:59:45 +02:00
|
|
|
* Decrement transactions use count and close
|
|
|
|
* external file if count is zero.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
file->ext_tra_cnt--;
|
|
|
|
if (!file->ext_tra_cnt && file->ext_ifi)
|
|
|
|
{
|
|
|
|
fclose(file->ext_ifi);
|
|
|
|
file->ext_ifi = NULL;
|
|
|
|
}
|
|
|
|
}
|