2014-01-10 02:01:34 +08:00
|
|
|
/**********************************************************************
|
|
|
|
* File: fileio.cpp
|
|
|
|
* Description: File I/O utilities.
|
|
|
|
* Author: Samuel Charron
|
|
|
|
*
|
|
|
|
* (C) Copyright 2013, Google Inc.
|
|
|
|
* Licensed under the Apache License, Version 2.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.apache.org/licenses/LICENSE-2.0 Unless required
|
|
|
|
* by applicable law or agreed to in writing, software distributed under the
|
|
|
|
* License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS
|
|
|
|
* OF ANY KIND, either express or implied. See the License for the specific
|
|
|
|
* language governing permissions and limitations under the License.
|
|
|
|
*
|
|
|
|
**********************************************************************/
|
2019-04-01 00:58:56 +08:00
|
|
|
|
2014-01-10 02:01:34 +08:00
|
|
|
#ifdef _WIN32
|
2014-01-27 06:39:20 +08:00
|
|
|
#ifndef unlink
|
|
|
|
#include <io.h>
|
|
|
|
#endif
|
2014-01-10 02:01:34 +08:00
|
|
|
#else
|
|
|
|
#include <glob.h>
|
2014-01-27 06:39:20 +08:00
|
|
|
#include <unistd.h>
|
2014-01-10 02:01:34 +08:00
|
|
|
#endif
|
|
|
|
|
2018-05-20 05:52:04 +08:00
|
|
|
#include <cstdlib>
|
2014-01-27 06:39:20 +08:00
|
|
|
#include <cstdio>
|
|
|
|
#include <string>
|
|
|
|
|
2018-06-21 04:32:17 +08:00
|
|
|
#include "errcode.h"
|
2014-01-27 06:39:20 +08:00
|
|
|
#include "fileio.h"
|
2019-04-23 03:22:00 +08:00
|
|
|
#include "host.h" // includes windows.h for BOOL, ...
|
2013-09-23 23:26:50 +08:00
|
|
|
#include "tprintf.h"
|
|
|
|
|
|
|
|
namespace tesseract {
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// File::
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
2018-03-03 21:36:28 +08:00
|
|
|
FILE* File::Open(const std::string& filename, const std::string& mode) {
|
2013-09-23 23:26:50 +08:00
|
|
|
return fopen(filename.c_str(), mode.c_str());
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
FILE* File::OpenOrDie(const std::string& filename,
|
|
|
|
const std::string& mode) {
|
2013-09-23 23:26:50 +08:00
|
|
|
FILE* stream = fopen(filename.c_str(), mode.c_str());
|
2016-12-13 15:08:01 +08:00
|
|
|
if (stream == nullptr) {
|
2014-01-10 02:01:34 +08:00
|
|
|
tprintf("Unable to open '%s' in mode '%s'\n", filename.c_str(),
|
|
|
|
mode.c_str());
|
2013-09-23 23:26:50 +08:00
|
|
|
}
|
|
|
|
return stream;
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
void File::WriteStringToFileOrDie(const std::string& str,
|
|
|
|
const std::string& filename) {
|
2013-09-23 23:26:50 +08:00
|
|
|
FILE* stream = fopen(filename.c_str(), "wb");
|
2016-12-13 15:08:01 +08:00
|
|
|
if (stream == nullptr) {
|
2014-01-10 02:01:34 +08:00
|
|
|
tprintf("Unable to open '%s' for writing\n", filename.c_str());
|
2013-09-23 23:26:50 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
fputs(str.c_str(), stream);
|
|
|
|
ASSERT_HOST(fclose(stream) == 0);
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
bool File::Readable(const std::string& filename) {
|
2013-09-23 23:26:50 +08:00
|
|
|
FILE* stream = fopen(filename.c_str(), "rb");
|
2016-12-13 15:08:01 +08:00
|
|
|
if (stream == nullptr) {
|
2013-09-23 23:26:50 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
fclose(stream);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
bool File::ReadFileToString(const std::string& filename, std::string* out) {
|
2013-09-23 23:26:50 +08:00
|
|
|
FILE* stream = File::Open(filename.c_str(), "rb");
|
2017-01-26 08:20:19 +08:00
|
|
|
if (stream == nullptr) return false;
|
2013-09-23 23:26:50 +08:00
|
|
|
InputBuffer in(stream);
|
|
|
|
*out = "";
|
2015-05-02 01:33:56 +08:00
|
|
|
in.Read(out);
|
2013-09-23 23:26:50 +08:00
|
|
|
return in.CloseFile();
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
std::string File::JoinPath(const std::string& prefix, const std::string& suffix) {
|
2016-11-08 02:46:33 +08:00
|
|
|
return (prefix.empty() || prefix[prefix.size() - 1] == '/')
|
|
|
|
? prefix + suffix
|
|
|
|
: prefix + "/" + suffix;
|
2014-01-10 02:01:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool File::Delete(const char* pathname) {
|
2018-11-09 05:55:04 +08:00
|
|
|
#if !defined(_WIN32) || defined(__MINGW32__)
|
2018-11-09 06:10:14 +08:00
|
|
|
const int status = unlink(pathname);
|
2018-11-09 05:55:04 +08:00
|
|
|
#else
|
|
|
|
const int status = _unlink(pathname);
|
|
|
|
#endif
|
2014-01-10 02:01:34 +08:00
|
|
|
if (status != 0) {
|
|
|
|
tprintf("ERROR: Unable to delete file %s\n", pathname);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
bool File::DeleteMatchingFiles(const char* pattern) {
|
|
|
|
WIN32_FIND_DATA data;
|
|
|
|
BOOL result = TRUE;
|
|
|
|
HANDLE handle = FindFirstFile(pattern, &data);
|
|
|
|
bool all_deleted = true;
|
|
|
|
if (handle != INVALID_HANDLE_VALUE) {
|
|
|
|
for (; result; result = FindNextFile(handle, &data)) {
|
|
|
|
all_deleted &= File::Delete(data.cFileName);
|
|
|
|
}
|
|
|
|
FindClose(handle);
|
|
|
|
}
|
|
|
|
return all_deleted;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
bool File::DeleteMatchingFiles(const char* pattern) {
|
|
|
|
glob_t pglob;
|
|
|
|
char **paths;
|
|
|
|
bool all_deleted = true;
|
2016-12-13 15:08:01 +08:00
|
|
|
if (glob(pattern, 0, nullptr, &pglob) == 0) {
|
|
|
|
for (paths = pglob.gl_pathv; *paths != nullptr; paths++) {
|
2014-01-10 02:01:34 +08:00
|
|
|
all_deleted &= File::Delete(*paths);
|
|
|
|
}
|
|
|
|
globfree(&pglob);
|
|
|
|
}
|
|
|
|
return all_deleted;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-09-23 23:26:50 +08:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// InputBuffer::
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
InputBuffer::InputBuffer(FILE* stream)
|
|
|
|
: stream_(stream) {
|
|
|
|
fseek(stream_, 0, SEEK_END);
|
|
|
|
filesize_ = ftell(stream_);
|
|
|
|
fseek(stream_, 0, SEEK_SET);
|
|
|
|
}
|
|
|
|
|
|
|
|
InputBuffer::InputBuffer(FILE* stream, size_t)
|
|
|
|
: stream_(stream) {
|
|
|
|
fseek(stream_, 0, SEEK_END);
|
|
|
|
filesize_ = ftell(stream_);
|
|
|
|
fseek(stream_, 0, SEEK_SET);
|
|
|
|
}
|
|
|
|
|
|
|
|
InputBuffer::~InputBuffer() {
|
2016-12-13 15:08:01 +08:00
|
|
|
if (stream_ != nullptr) {
|
2013-09-23 23:26:50 +08:00
|
|
|
fclose(stream_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
bool InputBuffer::Read(std::string* out) {
|
2015-07-10 05:28:20 +08:00
|
|
|
char buf[BUFSIZ + 1];
|
2015-05-02 01:33:56 +08:00
|
|
|
int l;
|
2015-07-10 05:28:20 +08:00
|
|
|
while ((l = fread(buf, 1, BUFSIZ, stream_)) > 0) {
|
|
|
|
if (ferror(stream_)) {
|
2015-05-02 01:33:56 +08:00
|
|
|
clearerr(stream_);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
buf[l] = 0;
|
|
|
|
out->append(buf);
|
2014-01-10 02:01:34 +08:00
|
|
|
}
|
2013-09-23 23:26:50 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool InputBuffer::CloseFile() {
|
|
|
|
int ret = fclose(stream_);
|
2016-12-13 15:08:01 +08:00
|
|
|
stream_ = nullptr;
|
2013-09-23 23:26:50 +08:00
|
|
|
return ret == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// OutputBuffer::
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
OutputBuffer::OutputBuffer(FILE* stream)
|
|
|
|
: stream_(stream) {
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputBuffer::OutputBuffer(FILE* stream, size_t)
|
|
|
|
: stream_(stream) {
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputBuffer::~OutputBuffer() {
|
2016-12-13 15:08:01 +08:00
|
|
|
if (stream_ != nullptr) {
|
2013-09-23 23:26:50 +08:00
|
|
|
fclose(stream_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-03 21:36:28 +08:00
|
|
|
void OutputBuffer::WriteString(const std::string& str) {
|
2013-09-23 23:26:50 +08:00
|
|
|
fputs(str.c_str(), stream_);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool OutputBuffer::CloseFile() {
|
|
|
|
int ret = fclose(stream_);
|
2016-12-13 15:08:01 +08:00
|
|
|
stream_ = nullptr;
|
2013-09-23 23:26:50 +08:00
|
|
|
return ret == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace tesseract
|