/** * Copyright (c) 2006-2016 LOVE Development Team * * This software is provided 'as-is', without any express or implied * warranty. In no event will the authors be held liable for any damages * arising from the use of this software. * * Permission is granted to anyone to use this software for any purpose, * including commercial applications, and to alter it and redistribute it * freely, subject to the following restrictions: * * 1. The origin of this software must not be misrepresented; you must not * claim that you wrote the original software. If you use this software * in a product, an acknowledgment in the product documentation would be * appreciated but is not required. * 2. Altered source versions must be plainly marked as such, and must not be * misrepresented as being the original software. * 3. This notice may not be removed or altered from any source distribution. **/ // LOVE #include "DroppedFile.h" #include "common/utf8.h" // Assume POSIX or Visual Studio. #include #include #ifdef LOVE_WINDOWS #include #else #include // POSIX. #endif namespace love { namespace filesystem { love::Type DroppedFile::type(&File::type); DroppedFile::DroppedFile(const std::string &filename) : filename(filename) , file(nullptr) , mode(MODE_CLOSED) , bufferMode(BUFFER_NONE) , bufferSize(0) { } DroppedFile::~DroppedFile() { if (mode != MODE_CLOSED) close(); } bool DroppedFile::open(Mode newmode) { if (newmode == MODE_CLOSED) return true; // File already open? if (file != nullptr) return false; #ifdef LOVE_WINDOWS // make sure non-ASCII filenames work. std::wstring modestr = to_widestr(getModeString(newmode)); std::wstring wfilename = to_widestr(filename); file = _wfopen(wfilename.c_str(), modestr.c_str()); #else file = fopen(filename.c_str(), getModeString(newmode)); #endif if (newmode == MODE_READ && file == nullptr) throw love::Exception("Could not open file %s. Does not exist.", filename.c_str()); mode = newmode; if (file != nullptr && !setBuffer(bufferMode, bufferSize)) { // Revert to buffer defaults if we don't successfully set the buffer. bufferMode = BUFFER_NONE; bufferSize = 0; } return file != nullptr; } bool DroppedFile::close() { if (file == nullptr || fclose(file) != 0) return false; mode = MODE_CLOSED; file = nullptr; return true; } bool DroppedFile::isOpen() const { return mode != MODE_CLOSED && file != nullptr; } int64 DroppedFile::getSize() { #ifdef LOVE_WINDOWS // make sure non-ASCII filenames work. std::wstring wfilename = to_widestr(filename); struct _stat buf; if (_wstat(wfilename.c_str(), &buf) != 0) return -1; return (int64) buf.st_size; #else // Assume POSIX support... struct stat buf; if (stat(filename.c_str(), &buf) != 0) return -1; return (int64) buf.st_size; #endif } int64 DroppedFile::read(void *dst, int64 size) { if (!file || mode != MODE_READ) throw love::Exception("File is not opened for reading."); if (size < 0) throw love::Exception("Invalid read size."); size_t read = fread(dst, 1, (size_t) size, file); return (int64) read; } bool DroppedFile::write(const void *data, int64 size) { if (!file || (mode != MODE_WRITE && mode != MODE_APPEND)) throw love::Exception("File is not opened for writing."); if (size < 0) throw love::Exception("Invalid write size."); int64 written = (int64) fwrite(data, 1, (size_t) size, file); return written == size; } bool DroppedFile::flush() { if (!file || (mode != MODE_WRITE && mode != MODE_APPEND)) throw love::Exception("File is not opened for writing."); return fflush(file) == 0; } bool DroppedFile::isEOF() { return file == nullptr || feof(file) != 0; } int64 DroppedFile::tell() { if (file == nullptr) return -1; return (int64) ftell(file); } bool DroppedFile::seek(uint64 pos) { return file != nullptr && fseek(file, (long) pos, SEEK_SET) == 0; } bool DroppedFile::setBuffer(BufferMode bufmode, int64 size) { if (size < 0) return false; if (bufmode == BUFFER_NONE) size = 0; // If the file isn't open, we'll make sure the buffer values are set in // DroppedFile::open. if (!isOpen()) { bufferMode = bufmode; bufferSize = size; return true; } int vbufmode; switch (bufmode) { case File::BUFFER_NONE: default: vbufmode = _IONBF; break; case File::BUFFER_LINE: vbufmode = _IOLBF; break; case File::BUFFER_FULL: vbufmode = _IOFBF; break; } if (setvbuf(file, nullptr, vbufmode, (size_t) size) != 0) return false; bufferMode = bufmode; bufferSize = size; return true; } File::BufferMode DroppedFile::getBuffer(int64 &size) const { size = bufferSize; return bufferMode; } const std::string &DroppedFile::getFilename() const { return filename; } File::Mode DroppedFile::getMode() const { return mode; } const char *DroppedFile::getModeString(Mode mode) { switch (mode) { case File::MODE_CLOSED: default: return "c"; case File::MODE_READ: return "rb"; case File::MODE_WRITE: return "wb"; case File::MODE_APPEND: return "ab"; } } } // filesystem } // love