#ifndef TOOLS_FILE_IO_H_
#define TOOLS_FILE_IO_H_
+#include <errno.h>
+#include <limits.h>
#include <stdint.h>
+#include <stdio.h>
+#include <string.h>
+#include <sys/stat.h>
+#include <list>
+#include <string>
#include <vector>
+#include "lib/jxl/base/compiler_specific.h"
+
namespace jpegxl {
namespace tools {
-bool ReadFile(const char* filename, std::vector<uint8_t>* out);
+namespace {
+
+// RAII, ensures files are closed even when returning early.
+class FileWrapper {
+ public:
+ FileWrapper(const FileWrapper& other) = delete;
+ FileWrapper& operator=(const FileWrapper& other) = delete;
+
+ explicit FileWrapper(const std::string& pathname, const char* mode)
+ : file_(pathname == "-" ? (mode[0] == 'r' ? stdin : stdout)
+ : fopen(pathname.c_str(), mode)),
+ close_on_delete_(pathname != "-") {
+#ifdef _WIN32
+ struct __stat64 s = {};
+ const int err = _stat64(pathname.c_str(), &s);
+ const bool is_file = (s.st_mode & S_IFREG) != 0;
+#else
+ struct stat s = {};
+ const int err = stat(pathname.c_str(), &s);
+ const bool is_file = S_ISREG(s.st_mode);
+#endif
+ if (err == 0 && is_file) {
+ size_ = s.st_size;
+ }
+ }
+
+ ~FileWrapper() {
+ if (file_ != nullptr && close_on_delete_) {
+ const int err = fclose(file_);
+ if (err) {
+ fprintf(stderr,
+ "Could not close file\n"
+ "Error: %s",
+ strerror(errno));
+ }
+ }
+ }
+
+ // We intend to use FileWrapper as a replacement of FILE.
+ // NOLINTNEXTLINE(google-explicit-constructor)
+ operator FILE*() const { return file_; }
+
+ int64_t size() { return size_; }
+
+ private:
+ FILE* const file_;
+ bool close_on_delete_ = true;
+ int64_t size_ = -1;
+};
+
+} // namespace
+
+template <typename ContainerType>
+static inline bool ReadFile(FileWrapper& f, ContainerType* JXL_RESTRICT bytes) {
+ if (!f) return false;
+
+ // Get size of file in bytes
+ const int64_t size = f.size();
+ if (size < 0) {
+ // Size is unknown, loop reading chunks until EOF.
+ bytes->clear();
+ std::list<std::vector<uint8_t>> chunks;
+
+ size_t total_size = 0;
+ while (true) {
+ std::vector<uint8_t> chunk(16 * 1024);
+ const size_t bytes_read = fread(chunk.data(), 1, chunk.size(), f);
+ if (ferror(f) || bytes_read > chunk.size()) {
+ return false;
+ }
+
+ chunk.resize(bytes_read);
+ total_size += bytes_read;
+ if (bytes_read != 0) {
+ chunks.emplace_back(std::move(chunk));
+ }
+ if (feof(f)) {
+ break;
+ }
+ }
+ bytes->resize(total_size);
+ size_t pos = 0;
+ for (const auto& chunk : chunks) {
+ memcpy(bytes->data() + pos, chunk.data(), chunk.size());
+ pos += chunk.size();
+ }
+ } else {
+ // Size is known, read the file directly.
+ bytes->resize(static_cast<size_t>(size));
+
+ const size_t bytes_read = fread(bytes->data(), 1, bytes->size(), f);
+ if (bytes_read != static_cast<size_t>(size)) return false;
+ }
+
+ return true;
+}
+
+template <typename ContainerType>
+static inline bool ReadFile(const std::string& filename,
+ ContainerType* JXL_RESTRICT bytes) {
+ FileWrapper f(filename, "rb");
+ return ReadFile(f, bytes);
+}
-bool WriteFile(const char* filename, const std::vector<uint8_t>& bytes);
+template <typename ContainerType>
+static inline bool WriteFile(const std::string& filename,
+ const ContainerType& bytes) {
+ FileWrapper file(filename, "wb");
+ if (!file) {
+ fprintf(stderr,
+ "Could not open %s for writing\n"
+ "Error: %s",
+ filename.c_str(), strerror(errno));
+ return false;
+ }
+ if (fwrite(bytes.data(), 1, bytes.size(), file) != bytes.size()) {
+ fprintf(stderr,
+ "Could not write to file\n"
+ "Error: %s",
+ strerror(errno));
+ return false;
+ }
+ return true;
+}
} // namespace tools
} // namespace jpegxl