Browse Source

Merge pull request #2133 from MikuAuahDark/nativefile-sdl

Replace NativeFile to use SDL IO functions.
Sasha Szpakowski 6 months ago
parent
commit
176a3c17d5

+ 1 - 1
CMakeLists.txt

@@ -1962,7 +1962,7 @@ love_group_projects(NAME "liblove" NESTED TARGETS ${LIBLOVE_DEPENDENCIES})
 love_group_projects(NAME "liblove/libraries" NESTED TARGETS ${LIBLOVE_LIBRARIES})
 love_group_projects(NAME "liblove/libraries" NESTED TARGETS ${LIBLOVE_LIBRARIES})
 love_group_projects(NAME "liblove" TARGETS liblove ${LOVE_EXTRA_DEPENDECIES})
 love_group_projects(NAME "liblove" TARGETS liblove ${LOVE_EXTRA_DEPENDECIES})
 
 
-love_group_projects(NAME "lovedep" TARGETS lovedep::SDL3 lovedep::Freetype lovedep::Harfbuzz lovedep::OpenAL lovedep::Modplug lovedep::Theora lovedep::Vorbis lovedep::Ogg lovedep::Zlib lovedep::Lua)
+love_group_projects(NAME "lovedep" TARGETS lovedep::SDL lovedep::Freetype lovedep::Harfbuzz lovedep::OpenAL lovedep::Modplug lovedep::Theora lovedep::Vorbis lovedep::Ogg lovedep::Zlib lovedep::Lua)
 love_group_projects(NAME "lovedep" TARGETS lua51 alcommon al-excommon harfbuzz-subset zlib)
 love_group_projects(NAME "lovedep" TARGETS lua51 alcommon al-excommon harfbuzz-subset zlib)
 
 
 
 

+ 6 - 0
extra/cmake/LoveMacros.cmake

@@ -16,6 +16,12 @@ function(love_group_projects)
 				foreach(TARGET_LIB ${TARGET_LIBS})
 				foreach(TARGET_LIB ${TARGET_LIBS})
 					# Is this a target? (Could also be a .lib file)
 					# Is this a target? (Could also be a .lib file)
 					if (TARGET ${TARGET_LIB})
 					if (TARGET ${TARGET_LIB})
+						# Resolve aliased target
+						get_target_property(ORIGINAL_TARGET ${TARGET_LIB} ALIASED_TARGET)
+						if (ORIGINAL_TARGET)
+							set(TARGET_LIB ${ORIGINAL_TARGET})
+						endif()
+
 						# Do we want to nest per-project?
 						# Do we want to nest per-project?
 						if (LOVE_GROUP_NESTED)
 						if (LOVE_GROUP_NESTED)
 							set_target_properties(${TARGET_LIB} PROPERTIES FOLDER "${LOVE_GROUP_NAME}/${TARGET_NAME}")
 							set_target_properties(${TARGET_LIB} PROPERTIES FOLDER "${LOVE_GROUP_NAME}/${TARGET_NAME}")

+ 265 - 134
src/modules/filesystem/NativeFile.cpp

@@ -20,21 +20,12 @@
 
 
 // LOVE
 // LOVE
 #include "NativeFile.h"
 #include "NativeFile.h"
-#include "common/utf8.h"
 
 
-#ifdef LOVE_ANDROID
-#include "common/android.h"
-#endif
+// C
+#include <cstring>
 
 
-// Assume POSIX or Visual Studio.
-#include <sys/types.h>
-#include <sys/stat.h>
-
-#ifdef LOVE_WINDOWS
-#include <wchar.h>
-#else
-#include <unistd.h> // POSIX.
-#endif
+// SDL
+#include <SDL3/SDL_iostream.h>
 
 
 namespace love
 namespace love
 {
 {
@@ -42,22 +33,26 @@ namespace filesystem
 {
 {
 
 
 NativeFile::NativeFile(const std::string &filename, Mode mode)
 NativeFile::NativeFile(const std::string &filename, Mode mode)
-	: filename(filename)
-	, file(nullptr)
-	, mode(MODE_CLOSED)
-	, bufferMode(BUFFER_NONE)
-	, bufferSize(0)
+: filename(filename)
+, file(nullptr)
+, mode(MODE_CLOSED)
+, buffer(nullptr)
+, bufferMode(BUFFER_NONE)
+, bufferSize(0)
+, bufferUsed(0)
 {
 {
 	if (!open(mode))
 	if (!open(mode))
 		throw love::Exception("Could not open file at path %s", filename.c_str());
 		throw love::Exception("Could not open file at path %s", filename.c_str());
 }
 }
 
 
 NativeFile::NativeFile(const NativeFile &other)
 NativeFile::NativeFile(const NativeFile &other)
-	: filename(other.filename)
-	, file(nullptr)
-	, mode(MODE_CLOSED)
-	, bufferMode(other.bufferMode)
-	, bufferSize(other.bufferSize)
+: filename(other.filename)
+, file(nullptr)
+, mode(MODE_CLOSED)
+, buffer(nullptr)
+, bufferMode(other.bufferMode)
+, bufferSize(other.bufferSize)
+, bufferUsed(0)
 {
 {
 	if (!open(other.mode))
 	if (!open(other.mode))
 		throw love::Exception("Could not open file at path %s", filename.c_str());
 		throw love::Exception("Could not open file at path %s", filename.c_str());
@@ -86,41 +81,18 @@ bool NativeFile::open(Mode newmode)
 	if (file != nullptr)
 	if (file != nullptr)
 		return false;
 		return false;
 
 
-#if defined(LOVE_ANDROID)
-	// Try to handle content:// URI
-	int fd = love::android::getFDFromContentProtocol(filename.c_str());
-	if (fd != -1)
-	{
-		if (newmode != MODE_READ)
-		{
-			::close(fd);
-			throw love::Exception("%s is read-only.", filename.c_str());
-		}
-
-		file = fdopen(fd, "rb");
-	}
-	else
-		file = fopen(filename.c_str(), getModeString(newmode));
-#elif defined(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());
+	file = SDL_IOFromFile(filename.c_str(), getModeString(newmode));
+	if (file == nullptr)
+		throw love::Exception("Could not open file %s: %s", filename.c_str(), SDL_GetError());
 
 
 	mode = newmode;
 	mode = newmode;
 
 
-	if (file != nullptr && !setBuffer(bufferMode, bufferSize))
+	if (!setupBuffering(bufferMode, bufferSize))
 	{
 	{
-		// Revert to buffer defaults if we don't successfully set the buffer.
-		bufferMode = BUFFER_NONE;
-		bufferSize = 0;
+		SDL_CloseIO(file);
+		file = nullptr;
+		mode = MODE_CLOSED;
+		throw love::Exception("Could not open file %s: cannot setup buffering", filename.c_str());
 	}
 	}
 
 
 	return file != nullptr;
 	return file != nullptr;
@@ -128,13 +100,18 @@ bool NativeFile::open(Mode newmode)
 
 
 bool NativeFile::close()
 bool NativeFile::close()
 {
 {
-	if (file == nullptr || fclose(file) != 0)
+	if (file == nullptr)
 		return false;
 		return false;
 
 
+	bool success = flush();
+	success = SDL_CloseIO(file) && success;
+	// Regardless whetever SDL_CloseIO succeeded or failed, the `file`
+	// pointer is no longer valid.
 	mode = MODE_CLOSED;
 	mode = MODE_CLOSED;
 	file = nullptr;
 	file = nullptr;
+	setupBuffering(BUFFER_NONE, 0);
 
 
-	return true;
+	return success;
 }
 }
 
 
 bool NativeFile::isOpen() const
 bool NativeFile::isOpen() const
@@ -144,44 +121,17 @@ bool NativeFile::isOpen() const
 
 
 int64 NativeFile::getSize()
 int64 NativeFile::getSize()
 {
 {
-	int fd = file ? fileno(file) : -1;
-
-#ifdef LOVE_WINDOWS
-	
-	struct _stat64 buf;
-
-	if (fd != -1)
+	int64 size = -1;
+	if (!file)
 	{
 	{
-		if (_fstat64(fd, &buf) != 0)
-			return -1;
+		open(MODE_READ);
+		size = SDL_GetIOSize(file);
+		close();
 	}
 	}
 	else
 	else
-	{
-		// make sure non-ASCII filenames work.
-		std::wstring wfilename = to_widestr(filename);
-
-		if (_wstat64(wfilename.c_str(), &buf) != 0)
-			return -1;
-	}
-
-	return (int64) buf.st_size;
-
-#else
-
-	// Assume POSIX support...
-	struct stat buf;
+		size = SDL_GetIOSize(file);
 
 
-	if (fd != -1)
-	{
-		if (fstat(fd, &buf) != 0)
-			return -1;
-	}
-	else if (stat(filename.c_str(), &buf) != 0)
-		return -1;
-
-	return (int64) buf.st_size;
-
-#endif
+	return std::max<int64>(size, -1);
 }
 }
 
 
 int64 NativeFile::read(void *dst, int64 size)
 int64 NativeFile::read(void *dst, int64 size)
@@ -192,8 +142,12 @@ int64 NativeFile::read(void *dst, int64 size)
 	if (size < 0)
 	if (size < 0)
 		throw love::Exception("Invalid read size.");
 		throw love::Exception("Invalid read size.");
 
 
-	size_t read = fread(dst, 1, (size_t) size, file);
+	// Are we using buffers?
+	if (buffer)
+		return bufferedRead(dst, size);
 
 
+	// No buffering.
+	size_t read = SDL_ReadIO(file, dst, (size_t) size);
 	return (int64) read;
 	return (int64) read;
 }
 }
 
 
@@ -205,17 +159,60 @@ bool NativeFile::write(const void *data, int64 size)
 	if (size < 0)
 	if (size < 0)
 		throw love::Exception("Invalid write size.");
 		throw love::Exception("Invalid write size.");
 
 
-	int64 written = (int64) fwrite(data, 1, (size_t) size, file);
+	if (buffer)
+	{
+		if (!bufferedWrite(data, size))
+			return false;
+
+		// There's newline? force flush
+		if (bufferMode == BUFFER_LINE && memchr(data, '\n', size) != nullptr)
+		{
+			if (!flush())
+				return false;
+		}
+	}
+	else
+		return SDL_WriteIO(file, data, (size_t) size) == (size_t) size;
 
 
-	return written == size;
+	return true;
 }
 }
 
 
 bool NativeFile::flush()
 bool NativeFile::flush()
 {
 {
-	if (!file || (mode != MODE_WRITE && mode != MODE_APPEND))
-		throw love::Exception("File is not opened for writing.");
+	switch (mode)
+	{
+		case MODE_READ:
+		{
+			if (buffer)
+			{
+				// Seek to already consumed buffer
+				if (SDL_SeekIO(file, (size_t) (bufferUsed - bufferSize), SDL_IO_SEEK_CUR) < 0)
+					return false;
+
+				// Mark as depleted
+				bufferUsed = bufferSize;
+			}
+			
+			return true;
+		}
+		case MODE_WRITE:
+		case MODE_APPEND:
+		{
+			if (buffer && bufferUsed > 0)
+			{
+				size_t written = SDL_WriteIO(file, buffer, (size_t) bufferUsed);
+				memmove(buffer, buffer + written, (size_t) bufferSize - written);
+				bufferUsed = std::max<int64>(bufferUsed - (int64) written, 0);
+			}
+
+			return SDL_FlushIO(file);
+		}
+		default:
+			throw love::Exception("Invalid file mode.");
+	}
 
 
-	return fflush(file) == 0;
+	// Make sure compiler doesn't emit warnings.
+	return true;
 }
 }
 
 
 bool NativeFile::isEOF()
 bool NativeFile::isEOF()
@@ -228,11 +225,23 @@ int64 NativeFile::tell()
 	if (file == nullptr)
 	if (file == nullptr)
 		return -1;
 		return -1;
 
 
-#ifdef LOVE_WINDOWS
-	return (int64) _ftelli64(file);
-#else
-	return (int64) ftello(file);
-#endif
+	int64 offset = 0;
+	if (buffer)
+	{
+		switch (mode)
+		{
+			case MODE_READ:
+				// Note: We want offset be negative for reading
+				offset = bufferUsed - bufferSize;
+				break;
+			case MODE_WRITE:
+			case MODE_APPEND:
+				offset = bufferUsed;
+				break;
+		}
+	}
+
+	return SDL_TellIO(file) + offset;
 }
 }
 
 
 bool NativeFile::seek(int64 pos, SeekOrigin origin)
 bool NativeFile::seek(int64 pos, SeekOrigin origin)
@@ -240,55 +249,68 @@ bool NativeFile::seek(int64 pos, SeekOrigin origin)
 	if (file == nullptr)
 	if (file == nullptr)
 		return false;
 		return false;
 
 
-	int forigin = SEEK_SET;
+	if (mode == MODE_APPEND)
+		// FIXME: PhysFS "append" allows the user to
+		// seek the write pointer, but it's not possible
+		// to do so with standard fopen-style modes.
+		return false;
+
+	SDL_IOWhence whence = SDL_IO_SEEK_SET;
 	if (origin == SEEKORIGIN_CURRENT)
 	if (origin == SEEKORIGIN_CURRENT)
-		forigin = SEEK_CUR;
+		whence = SDL_IO_SEEK_CUR;
 	else if (origin == SEEKORIGIN_END)
 	else if (origin == SEEKORIGIN_END)
-		forigin = SEEK_END;
-
-	// TODO
-#ifdef LOVE_WINDOWS
-	return _fseeki64(file, pos, forigin) == 0;
-#else
-	return fseeko(file, (off_t) pos, forigin) == 0;
-#endif
+		whence = SDL_IO_SEEK_END;
+
+	if (mode == MODE_READ && whence == SDL_IO_SEEK_SET && buffer)
+	{
+		// Retain the buffer if it's forward.
+		// TODO: Handle SDL_IO_SEEK_CUR.
+		int64 offset = pos - tell();
+		if (offset >= 0 && (offset + bufferUsed) < bufferSize)
+		{
+			// Seek success
+			bufferUsed += offset;
+			return true;
+		}
+
+		// Note: We don't handle backward seek because the
+		// contents past `bufferUsed` is not necessarily valid.
+	}
+
+	// If the read is buffered, the flush() will ensure
+	// the read pointer is in correct place before doing seek.
+	return flush() && (SDL_SeekIO(file, pos, whence) >= 0);
 }
 }
 
 
 bool NativeFile::setBuffer(BufferMode bufmode, int64 size)
 bool NativeFile::setBuffer(BufferMode bufmode, int64 size)
 {
 {
+	// BUFSIZ in Windows is too low on 512 bytes.
+	// Make sure the default buffer size is at least 4KiB.
+	constexpr int64 DEFAULT_BUFFER_SIZE = std::max<int64>(BUFSIZ, 4096);
+
 	if (size < 0)
 	if (size < 0)
 		return false;
 		return false;
+	else if (sizeof(uintptr_t) == 4 && size > 0x80000000LL)
+		// Safeguards against 32-bit integer truncation?
+		return false;
 
 
 	if (bufmode == BUFFER_NONE)
 	if (bufmode == BUFFER_NONE)
 		size = 0;
 		size = 0;
+	else if (size == 0)
+		size = DEFAULT_BUFFER_SIZE;
 
 
-	// If the file isn't open, we'll make sure the buffer values are set in
-	// NativeFile::open.
-	if (!isOpen())
+	// If there's no file handle, we'll setup the buffering later in open()
+	if (file)
 	{
 	{
-		bufferMode = bufmode;
-		bufferSize = size;
-		return true;
-	}
+		// Ideally we don't want to flush if user request larger buffer size
+		// but the added complexity is not worth it for now.
+		if (!flush())
+			return false;
 
 
-	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 (!setupBuffering(bufmode, size))
+			return false;
 	}
 	}
 
 
-	if (setvbuf(file, nullptr, vbufmode, (size_t) size) != 0)
-		return false;
-
 	bufferMode = bufmode;
 	bufferMode = bufmode;
 	bufferSize = size;
 	bufferSize = size;
 
 
@@ -311,6 +333,115 @@ File::Mode NativeFile::getMode() const
 	return mode;
 	return mode;
 }
 }
 
 
+bool NativeFile::setupBuffering(BufferMode mode, int64 bufferSize)
+{
+	int8 *newbuf = nullptr;
+	if (mode != BUFFER_NONE)
+	{
+		newbuf = new (std::nothrow) int8[(size_t) bufferSize];
+		if (newbuf == nullptr)
+			return false;
+	}
+
+	delete[] buffer;
+	buffer = newbuf;
+	bufferUsed = this->mode == MODE_READ ? bufferSize : 0;
+	return true;
+}
+
+int64 NativeFile::bufferedRead(void *dst, int64 size)
+{
+	int8 *ptr = (int8 *) dst;
+	int64 readed = 0;
+
+	while (size > 0)
+	{
+		int64 available = bufferSize - bufferUsed;
+
+		if (available > 0)
+		{
+			// There's leftover buffers.
+			size_t copy = (size_t) std::min(size, available);
+			memcpy(ptr, buffer + (size_t) bufferUsed, copy);
+
+			ptr += copy;
+			size -= (int64) copy;
+			bufferUsed += (int64) copy;
+			readed += copy;
+		}
+		else
+		{
+			// Buffer is empty. Fill it.
+			size_t ureaded = SDL_ReadIO(file, buffer, (size_t) bufferSize);
+			bufferUsed = bufferSize - (int64) ureaded;
+
+			if (ureaded == 0)
+				break;
+
+			if (bufferUsed > 0)
+				// Shift the buffer so code above can properly index it
+				memmove(buffer + bufferUsed, buffer, ureaded);
+		}
+	}
+
+	return readed;
+}
+
+bool NativeFile::bufferedWrite(const void *data, int64 size)
+{
+	const int8 *ptr = (const int8 *) data;
+
+	int64 inBuffer = std::min<int64>(size, bufferSize - bufferUsed);
+	if (inBuffer > 0)
+	{
+		// Put the data into buffer
+		memcpy(buffer + bufferUsed, ptr, (size_t) inBuffer);
+		bufferUsed += inBuffer;
+		size -= inBuffer;
+		ptr += (size_t) inBuffer;
+	}
+
+	if (size > 0)
+	{
+		// This means the buffer is full. Soft-flush the buffers.
+		size_t bufferWritten = SDL_WriteIO(file, buffer, bufferSize);
+		bufferUsed -= bufferWritten;
+
+		if (bufferWritten < (size_t) bufferSize)
+		{
+			memmove(buffer, buffer + bufferWritten, (size_t) bufferSize - bufferWritten);
+			return false;
+		}
+
+		int64 directWriteCount = size / bufferSize;
+		if (directWriteCount > 0)
+		{
+			// Batch write from the source pointer directly, bypassing
+			// our buffer.
+			size_t targetDirectBuffer = (size_t) (directWriteCount * bufferSize);
+			if (SDL_WriteIO(file, ptr, targetDirectBuffer) < targetDirectBuffer)
+				return false;
+
+			ptr += targetDirectBuffer;
+			size -= targetDirectBuffer;
+		}
+
+		if (size > 0)
+		{
+			// Store the rest in our buffer.
+			// Note that bufferUsed will always be 0 here.
+			memcpy(buffer, ptr, size);
+			bufferUsed = size;
+		}
+
+		return SDL_FlushIO(file);
+	}
+
+	// If above codeblock is not taken, it means the
+	// whole data goes into our buffer. Report success.
+	return true;
+}
+
 const char *NativeFile::getModeString(Mode mode)
 const char *NativeFile::getModeString(Mode mode)
 {
 {
 	switch (mode)
 	switch (mode)

+ 8 - 1
src/modules/filesystem/NativeFile.h

@@ -25,6 +25,8 @@
 // C
 // C
 #include <cstdio>
 #include <cstdio>
 
 
+typedef struct SDL_IOStream SDL_IOStream;
+
 namespace love
 namespace love
 {
 {
 namespace filesystem
 namespace filesystem
@@ -64,17 +66,22 @@ public:
 private:
 private:
 
 
 	NativeFile(const NativeFile &other);
 	NativeFile(const NativeFile &other);
+	bool setupBuffering(BufferMode mode, int64 bufferSize);
+	int64 bufferedRead(void* dst, int64 size);
+	bool bufferedWrite(const void* data, int64 size);
 
 
 	static const char *getModeString(Mode mode);
 	static const char *getModeString(Mode mode);
 
 
 	std::string filename;
 	std::string filename;
 
 
-	FILE *file;
+	SDL_IOStream *file;
 
 
 	Mode mode;
 	Mode mode;
 
 
+	int8 *buffer;
 	BufferMode bufferMode;
 	BufferMode bufferMode;
 	int64 bufferSize;
 	int64 bufferSize;
+	int64 bufferUsed;
 
 
 }; // NativeFile
 }; // NativeFile