sfftools/sfftobmp3/src/common.cpp

288 lines
7.6 KiB
C++

// Utility classes
//
// This file is part of sfftobmp, a program to convert
// structured fax files (sff) to windows bitmap files (bmp),
// portable bitmap graphics (pbm), tagged image file format (tiff)
// or JPEG (jpg).
//
// Copyright (C) 1998-2012 Peter Schaefer-Hutter and contributors ("THE AUTHORS")
//
// Permission to use, copy, modify, distribute, and sell this software and
// its documentation for any purpose is hereby granted without fee.
//
// THE SOFTWARE IS PROVIDED "AS-IS" AND WITHOUT WARRANTY OF ANY KIND,
// EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY
// WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.
//
// IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL,
// INCIDENTAL, INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY
// DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
// WHETHER OR NOT ADVISED OF THE POSSIBILITY OF DAMAGE, AND ON ANY
// THEORY OF LIABILITY, ARISING OUT OF OR IN CONNECTION WITH THE USE
// OR PERFORMANCE OF THIS SOFTWARE.
//
// Contributor(s):
// Ulf Zibis <ulf.zibis@gmx.de> (Prepared for inheritance,
// destructor handling, preserving filetime)
//
// You can contact the original author by email at peter.schaefer@gmx.de.
//
// I'm always pleased to hear that somebody is actually using my software.
// If you can manage it, e-mail me a quick notice. Thanks!
//
/*-RCS-Info----------------------------------------------------
$Id: common.cpp,v 1.5 2009/04/28 19:02:08 pschaefer Exp $
---RCS-Info--------------------------------------------------*/
#include "sfftypes.h"
#include "common.h"
#include <time.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#ifndef WIN32
#include <utime.h>
#else
#include <sys/utime.h>
#endif
#include <cassert>
#include <cstdio>
#include <iostream>
#ifdef WIN32
#include <io.h> // _mktemp
#endif
#ifdef _MSC_VER
#define fileno _fileno
#endif
using namespace std;
//-----------------------------------------------------------------
const std::string CSimpleException::m_aReasons[err_count+1] =
{
// err_invalidfile
"Not a valid sff file.",
// err_corruptfile,
"File seems corrupt. Reading abandoned.",
// err_lastpageread,
"Last page read.",
// err_notsupported,
"Operation not supported.",
// err_openfile,
"Can't open file.",
// err_closedfile,
"Operation on closed file.",
// err_findPath,
"Path not found.",
// err_nowhitestart,
"Line doesn't begin with white code.",
// err_noblackcode,
"White code not followed by black code.",
// err_noblackterm,
"Black MUC not followed by black TERM.",
// err_nowhiteterm,
"White MUC not followed by white TERM.",
// err_invalidversion,
"OOps. Don't know how to handle this Fileversion.",
// err_unknowncoding,
"Oh my dear. Don't know how to handle this encoding.",
// err_toomuchformats,
"Please specify only one output format.",
// err_cmdline,
"Error in commandline.",
// err_noformat,
"No output format specified.",
// err_outfileexists,
"Output file already exists, use -f to force overwrite.",
// err_outfileisdir,
"Given output file is directory.",
// err_outdir,
"Cannot create output directory.",
// err_stdoutnotallowed
"Output on stdout is not available with this output format.",
// err_stdoutonlyonefile
"Output on stdout is not available with more than one input file.",
//
// ------------------- INSERT HERE
//
// err_count
"Unknown error."
};
const string& CSimpleException::what() const
{
if (m_nError < err_count) {
return m_aReasons[m_nError];
}
return m_aReasons[err_count];
}
//-----------------------------------------------------------------
CFile::CFile(const std::string& strPath) :
m_hFile(NULL)
{
Open( strPath, "rb");
}
CFile::~CFile()
{
Close();
}
void CFile::Open(const std::string& strPath, const char *pszMode)
{
m_strPath = strPath;
if (( m_hFile ) || !( m_hFile = fopen(strPath.c_str(), pszMode))) {
throw CSimpleException(CSimpleException::err_openfile);
}
m_nFileNo = fileno( m_hFile);
}
void CFile::OpenTemp()
{
char tmp_name[] = "sfftobmp-tmpXXXXXX";
#ifndef WIN32
if (( m_nFileNo ) || !(( m_nFileNo = mkstemp(tmp_name)) > 0)) {
m_nFileNo = 0;
cerr << "temp file couldn't be created: error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
unlink(tmp_name);
#else
_mktemp(tmp_name);
if (( m_nFileNo ) ||
(( m_nFileNo = (int)::CreateFile(tmp_name,
GENERIC_READ|GENERIC_WRITE, 0, NULL,
CREATE_NEW,FILE_ATTRIBUTE_TEMPORARY|
FILE_FLAG_DELETE_ON_CLOSE, NULL)) == 0))
{
cerr << "temp file couldn't be created: error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
#endif
m_strPath = tmp_name;
}
void CFile::Close()
{
if (m_hFile) {
fclose(m_hFile);
} else if (m_nFileNo) {
#ifndef WIN32
close(m_nFileNo);
#else
::CloseHandle((HANDLE)m_nFileNo);
#endif
}
m_nFileNo = 0;
m_hFile = NULL;
}
time_t CFile::GetModificationTime()
{
struct stat buf;
if (::fstat( m_nFileNo, &buf ) == 0)
return buf.st_mtime ;
else
throw CSimpleException(CSimpleException::err_notsupported);
}
void CFile::SetModificationTime(const time_t &modtime)
{
time_t ltime;
time( &ltime);
utimbuf filetime;
filetime.actime = ltime;
filetime.modtime = modtime;
if (::utime( m_strPath.string().c_str(), &filetime) != 0) {
cerr << "CFile::SetModificationTime(): ErrorNr.: " << errno << endl;
}
}
sff_byte CFile::GetC()
{
return ::fgetc(m_hFile);
}
void CFile::Read(void *pTarget, int nLen)
{
size_t rc = ::fread(pTarget, 1, nLen, m_hFile);
}
void CFile::Write(void *pSource, int nLen)
{
::fwrite(pSource, nLen, 1, m_hFile);
}
sff_dword CFile::Tell()
{
return ::ftell(m_hFile);
}
void CFile::Seek(int pos, CFile::seek_offset dir)
{
int whence;
if (dir == sk_from_start) {
whence = SEEK_SET;
} else if (dir == sk_current) {
whence = SEEK_CUR;
} else {
return;
}
::fseek(m_hFile, pos, whence);
}
bool CFile::Eof()
{
return (feof(m_hFile) != 0);
}
void CFile::DumpToStdOut()
{
char buf[1024];
sff_dword count;
#ifndef WIN32
int result=lseek(m_nFileNo,0,SEEK_SET);
if (result!=0) {
cerr << "seeking to temp file start failed " << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
while ( (count = read(m_nFileNo, buf, 1024)) > 0) {
result=write(STDOUT_FILENO, buf, count);
if (result<0) {
cerr << "writing to stdout failed with error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
}
if (count<0) {
cerr << "reading from temp file failed with error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
#else
SetFilePointer((HANDLE)m_nFileNo,0,0,FILE_BEGIN);
HANDLE hStdOut = ::GetStdHandle(STD_OUTPUT_HANDLE);
sff_dword written;
for (;;) {
::ReadFile((HANDLE)m_nFileNo, buf, 1024, (DWORD*)&count, NULL);
if (count <= 0) break;
if (!::WriteFile(hStdOut, buf, count, (DWORD *)&written, NULL)) {
cerr << "writing to stdout failed with error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
}
if (count<0) {
cerr << "reading from temp file failed with error " << errno << endl;
throw CSimpleException(CSimpleException::err_openfile);
}
#endif
}