8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-24 09:23:03 +01:00
firebird-mirror/src/isql/InputDevices.cpp

303 lines
7.4 KiB
C++

/*
* The contents of this file are subject to the Initial
* Developer's 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.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* 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 Claudio Valderrama on 5-Oct-2007
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2007 Claudio Valderrama
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
*/
#include "firebird.h"
#if defined(DARWIN) && !defined(IOS)
#if defined(i386) || defined(__x86_64__)
#include <architecture/i386/io.h>
#else
#include <io.h>
#endif
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#include "../common/utils_proto.h"
#include "../common/os/os_utils.h"
#include "InputDevices.h"
using Firebird::PathName;
InputDevices::indev::indev()
: indev_fpointer(0), indev_line(0), indev_aux(0), indev_next(0),
indev_fn(*getDefaultMemoryPool()), indev_fn_display(*getDefaultMemoryPool())
{
makeFullFileName();
}
InputDevices::indev::indev(FILE* fp, const char* fn, const char* fn_display)
: indev_fpointer(fp), indev_line(0), indev_aux(0), indev_next(0),
indev_fn(*getDefaultMemoryPool()), indev_fn_display(*getDefaultMemoryPool())
{
indev_fn = fn;
indev_fn_display = fn_display;
makeFullFileName();
}
// Performs the same task that one of the constructors, but called manually.
// File handle and file name are assumed to be valid and the rest of the data
// members aren't copied but reset.
void InputDevices::indev::init(FILE* fp, const char* fn, const char* fn_display)
{
indev_fpointer = fp;
indev_line = 0;
indev_aux = 0;
indev_fn = fn;
indev_fn_display = fn_display;
indev_next = 0;
makeFullFileName();
}
// Copies only the file handle and file name from one indev to another.
// File handle and file name are assumed to be valid.
void InputDevices::indev::init(const indev& src)
{
indev_fpointer = src.indev_fpointer;
indev_line = 0;
indev_aux = 0;
indev_fn = src.indev_fn;
indev_fn_display = src.indev_fn_display;
indev_next = 0;
}
//InputDevices::indev::~indev()
//{
// Nothing for now. Let the owner or caller close the file pointer if needed.
//}
// Initializes one indev with another. Useful for method of the owning class
// that needs to copy from the input indev to the top of the stack and vice-versa.
void InputDevices::indev::copy_from(const indev* src)
{
fb_assert(src);
indev_fpointer = src->indev_fpointer;
indev_line = src->indev_line;
indev_aux = src->indev_aux;
indev_fn = src->indev_fn;
indev_fn_display = src->indev_fn_display;
// indev_next not copied.
}
// Drop a file associated with an indev.
void InputDevices::indev::drop()
{
fb_assert(indev_fpointer != stdin);
fb_assert(!indev_fn.isEmpty()); // Some name should exist.
fclose(indev_fpointer);
unlink(indev_fn.c_str());
}
// Save the reading position in the parameter.
void InputDevices::indev::getPos(fpos_t* out) const
{
fb_assert(out);
fb_assert(indev_fpointer);
os_utils::fgetpos(indev_fpointer, out);
}
// Restore a previously stored reading position held in the parameter.
void InputDevices::indev::setPos(const fpos_t* in)
{
fb_assert(in);
fb_assert(indev_fpointer);
#ifdef SFIO
// hack to fix bad sfio header
os_utils::fsetpos(indev_fpointer, const_cast<fpos_t*>(in));
#else
os_utils::fsetpos(indev_fpointer, in);
#endif
}
void InputDevices::indev::makeFullFileName()
{
if (!indev_fn.isEmpty() && PathUtils::isRelative(indev_fn))
{
PathName name = indev_fn;
PathName path;
fb_utils::getCwd(path);
PathUtils::concatPath(indev_fn, path, name);
}
}
// Clear the chain of indev, closing file handles.
void InputDevices::clear(FILE* fpointer)
{
while (m_head)
{
FILE* const p = m_head->indev_fpointer;
if (fpointer && p == fpointer)
break;
if (p != stdin && p != m_ofp.indev_fpointer)
fclose(p);
indev* const flist = m_head->indev_next;
delete m_head;
m_head = flist;
--m_count;
}
if (m_ifp.indev_fpointer && // In case we called clear(NULL) manually before.
m_ifp.indev_fpointer != stdin && m_ifp.indev_fpointer != m_ofp.indev_fpointer)
{
fclose(m_ifp.indev_fpointer);
m_ifp.indev_fpointer = 0;
}
fb_assert(m_count == 0 || fpointer != 0);
if (!fpointer)
{
m_head = 0;
m_count = 0;
}
}
// Insert an indev in the chain, always in LIFO way.
bool InputDevices::insert(FILE* fp, const char* name, const char* display)
{
if (!m_head)
{
fb_assert(m_count == 0);
m_head = FB_NEW indev(fp, name, display);
}
else
{
fb_assert(m_count > 0);
indev* p = m_head;
m_head = FB_NEW indev(fp, name, display);
m_head->indev_next = p;
}
++m_count;
return true;
}
// Shortcut for inserting the currently input file in the indev chain.
bool InputDevices::insertIfp()
{
if (insert(NULL, "", ""))
{
m_head->copy_from(&m_ifp);
return true;
}
return false;
}
// Remove the top (last inserted) indev in the chain.
bool InputDevices::remove()
{
if (m_head)
{
fb_assert(m_count > 0);
indev* p = m_head;
m_head = m_head->indev_next;
delete p;
--m_count;
return true;
}
return false;
}
// Shortcut for moving the top indev in the chain to the current input file.
void InputDevices::removeIntoIfp()
{
fb_assert(m_head && m_count > 0);
m_ifp.copy_from(m_head);
// When we come back from inout(), we continue inside the get_statement loop.
// If we are inside do_isql() due to Ctrl-C, it doesn't cause any harm.
m_ifp.indev_line = m_ifp.indev_aux;
remove();
}
// Shortcut for testing whether current input and output handles are the same.
// This may happen if we are reading from the command history file that's filled
// automatically with every interactive SQL command (isql commands aren't logged).
bool InputDevices::sameInputAndOutput() const
{
return m_ifp.indev_fpointer && (m_ifp.indev_fpointer == m_ofp.indev_fpointer);
}
// Save SQL command (not isql's own commands) to a history file only if we
// are in interactive mode.
void InputDevices::saveCommand(const char* statement, const char* term)
{
if (m_ifp.indev_fpointer == stdin)
{
FILE* f = m_ofp.indev_fpointer;
if (f)
{
fputs(statement, f);
fputs(term, f);
// Add newline to make the file more readable.
fputc('\n', f);
}
else
{
Command* command = FB_NEW Command(statement, term);
commands.add(command);
}
}
}
// Are we reading from stdin now or from a file?
bool InputDevices::readingStdin() const
{
fb_assert(m_ifp.indev_fpointer);
return m_ifp.indev_fpointer == stdin;
}
// Go to end of file if we are reading from a file (not stdin).
void InputDevices::gotoEof()
{
fb_assert(m_ifp.indev_fpointer);
if (m_ifp.indev_fpointer != stdin)
fseek(m_ifp.indev_fpointer, 0, SEEK_END);
}
InputDevices::Command::Command(const char* statement, const char* term)
: m_statement(getPool())
{
m_statement = statement;
m_statement += term;
}
void InputDevices::Command::toFile(FILE* f)
{
fputs(m_statement.c_str(), f);
// Add newline to make the file more readable.
fputc('\n', f);
}
void InputDevices::commandsToFile(FILE* f)
{
for (unsigned n = 0; n < commands.getCount(); ++n)
{
commands[n]->toFile(f);
delete commands[n];
}
commands.clear();
}