AuroraRuntime/Source/IO/AuIOHandle.Unix.cpp

281 lines
6.6 KiB
C++
Raw Normal View History

/***
Copyright (C) 2023 J Reece Wilson (a/k/a "Reece"). All rights reserved.
File: AuIOHandle.Unix.cpp
Date: 2023-8-11
Author: Reece
***/
#include <RuntimeInternal.hpp>
#include "AuIOHandle.hpp"
#include "AuIOHandle.Unix.hpp"
#include "FS/FS.hpp"
#include "FS/FileAdvisory.Unix.hpp"
#include "FS/FileStream.Unix.hpp"
#include <fcntl.h>
#include <unistd.h>
#include <sys/stat.h>
2023-08-27 11:41:51 +00:00
#if defined(AURORA_COMPILER_CLANG)
// warning: enumeration values 'kEnumCount' and 'kEnumInvalid' not handled in switch [-Wswitch
#pragma clang diagnostic ignored "-Wswitch"
// Yea, I don't give a shit.
#endif
namespace Aurora::IO
{
AuUInt64 AFileHandle::DupHandle(AuUInt64 uOSHandle, bool bWriteAccess)
{
int fd = dup(uOSHandle);
if (fd < 0)
{
return 0;
}
return AuUInt64(fd);
}
void AFileHandle::CloseHandle(AuUInt64 uOSHandle)
{
int fd = (int)uOSHandle;
if (fd < 0)
{
return;
}
::close(fd);
}
struct UnixIOHandle final : AFileHandle
{
2023-08-27 11:41:51 +00:00
bool InitFromPath(HandleCreate create) override;
2023-08-27 11:41:51 +00:00
bool IsFile() override;
bool IsTTY() override;
bool IsPipe() override;
2023-08-27 11:41:51 +00:00
AuOptionalEx<bool> optIsFile {};
AuOptionalEx<bool> optIsPipe {};
AuOptionalEx<bool> optIsTTY {};
};
2023-08-27 11:41:51 +00:00
bool UnixIOHandle::InitFromPath(HandleCreate create)
{
int iFileDescriptor { -1 };
2023-08-27 11:41:51 +00:00
if (create.path.empty())
{
SysPushErrorArg("Cannot open an IO handle to the provided empty path");
return false;
}
2023-08-27 11:41:51 +00:00
if (!FS::EFileOpenModeIsValid(create.eMode))
{
SysPushErrorParam("Invalid open mode");
return false;
}
2023-08-27 11:41:51 +00:00
if (!FS::EFileAdvisoryLockLevelIsValid(create.eAdvisoryLevel))
{
SysPushErrorParam("Invalid lock mode");
return false;
}
2023-08-27 11:41:51 +00:00
auto pathex = FS::NormalizePathRet(create.path);
if (pathex.empty())
{
SysPushErrorMemory();
return false;
}
2023-08-27 11:41:51 +00:00
if (create.bAsyncHandle ||
create.bDirectIOMode)
{
this->bDirectIO = true;
}
2023-08-27 11:41:51 +00:00
switch (create.eMode)
{
case FS::EFileOpenMode::eRead:
{
2023-08-27 11:41:51 +00:00
break;
}
case FS::EFileOpenMode::eReadWrite:
case FS::EFileOpenMode::eWrite:
{
if (create.bAlwaysCreateDirTree)
{
2023-08-27 11:41:51 +00:00
FS::CreateDirectories(pathex, true);
}
if (create.bFailIfNonEmptyFile)
{
2023-08-27 11:41:51 +00:00
if (AuFS::FileExists(pathex.c_str()))
{
SysPushErrorResourceExists("File {} already exists", create.path);
return false;
}
}
2023-08-27 11:41:51 +00:00
break;
}
};
2023-08-27 11:41:51 +00:00
iFileDescriptor = ::open(pathex.c_str(),
(create.eMode == FS::EFileOpenMode::eRead ? O_RDONLY : (O_RDWR | O_CREAT)) | O_CLOEXEC | (this->bDirectIO ? O_DIRECT : 0),
0664);
2023-08-27 11:41:51 +00:00
if (iFileDescriptor < 0)
{
SysPushErrorIO("Couldn't open file: {} ({}) {}", path, pathex, errno);
return false;
}
2023-08-27 11:41:51 +00:00
if (!FS::ApplyDumbAdvisoryLock(iFileDescriptor, create.eAdvisoryLevel))
{
SysPushErrorIO("Couldn't open file: {}. File node (not section) is locked.", path);
::close(iFileDescriptor);
return false;
}
2023-08-27 11:41:51 +00:00
if (create.bFailIfNonEmptyFile)
{
if (FS::PosixGetLength(iFileDescriptor))
{
SysPushErrorResourceExists("File {} already exists", create.path);
::close(iFileDescriptor);
return false;
}
}
switch (create.eMode)
{
case FS::EFileOpenMode::eRead:
{
this->uOSReadHandle = AuUInt64(iFileDescriptor);
break;
}
case FS::EFileOpenMode::eReadWrite:
{
this->uOSWriteHandle = AuUInt64(iFileDescriptor);
this->uOSReadHandle = AuUInt64(iFileDescriptor);
break;
}
case FS::EFileOpenMode::eWrite:
{
this->uOSWriteHandle = AuUInt64(iFileDescriptor);
break;
}
};
2023-08-27 11:41:51 +00:00
this->bIsAsync = create.bAsyncHandle;
this->path = create.path;
return this->IsValid();
}
bool UnixIOHandle::IsFile()
{
bool bIsFile {};
struct stat st;
int iFileDescriptor {};
if (auto file = this->optIsFile)
{
return file.value();
}
if (auto optHandle = this->GetOSHandleSafe())
{
iFileDescriptor = (int)optHandle.value();
}
else
{
SysPushErrorUninitialized();
return false;
}
if (::fstat(iFileDescriptor, &st) != 0)
{
SysPushErrorIO("fstat failed");
return false;
}
bIsFile = S_ISREG(st.st_mode);
this->optIsFile = bIsFile;
return bIsFile;
}
bool UnixIOHandle::IsTTY()
{
bool bIsTTY {};
int iFileDescriptor {};
if (auto file = this->optIsTTY)
{
return file.value();
}
if (auto optHandle = this->GetOSHandleSafe())
{
iFileDescriptor = (int)optHandle.value();
}
else
{
SysPushErrorUninitialized();
return false;
}
bIsTTY = ::isatty(iFileDescriptor);
this->optIsTTY = bIsTTY;
return bIsTTY;
}
bool UnixIOHandle::IsPipe()
{
bool bIsPipe {};
struct stat st;
int iFileDescriptor {};
if (auto file = this->optIsPipe)
{
return file.value();
}
if (auto optHandle = this->GetOSHandleSafe())
{
iFileDescriptor = (int)optHandle.value();
}
else
{
SysPushErrorUninitialized();
return false;
}
if (::fstat(iFileDescriptor, &st) != 0)
{
SysPushErrorIO("fstat failed");
return false;
}
bIsPipe = S_ISFIFO(st.st_mode);
this->optIsPipe = bIsPipe;
return bIsPipe;
}
AUKN_SYM IIOHandle *IOHandleNew()
{
return _new UnixIOHandle();
}
AUKN_SYM void IOHandleRelease(IIOHandle *pIOHandle)
{
AuSafeDelete<UnixIOHandle *>(pIOHandle);
}
AUROXTL_INTERFACE_SOO_SRC_EX(AURORA_SYMBOL_EXPORT, IOHandle, UnixIOHandle)
}