serenityos support yayyy

This commit is contained in:
Mars 2025-05-03 22:23:11 -04:00
parent 735e3fb56f
commit e4cafe1960
Signed by: pupbrained
GPG key ID: 0FF5B8826803F895
6 changed files with 214 additions and 57 deletions

157
src/os/serenity.cpp Normal file
View file

@ -0,0 +1,157 @@
#ifdef __serenity__
// clang-format off
#include <cerrno> // For errno
#include <cstring> // For strerror
#include <filesystem>
#include <format>
#include <fstream>
#include <glaze/core/common.hpp> // glz::object
#include <glaze/core/context.hpp> // glz::error_ctx, glz::error_code
#include <glaze/core/meta.hpp> // glz::detail::Object
#include <glaze/json/read.hpp> // glz::read_json
#include <iostream>
#include <pwd.h> // For getpwuid(), struct passwd
#include <string>
#include <string_view>
#include <sys/statvfs.h>
#include <sys/types.h> // For uid_t
#include <sys/utsname.h>
#include <unistd.h>
#include "src/util/defs.hpp"
#include "src/util/error.hpp"
#include "src/util/helpers.hpp"
#include "src/util/logging.hpp"
#include "src/util/types.hpp"
#include "os.hpp"
// clang-format on
using namespace util::types;
using util::error::DracError, util::error::DracErrorCode;
using util::helpers::GetEnv;
namespace {
using glz::opts, glz::detail::Object, glz::object;
constexpr opts glaze_opts = { .error_on_unknown_keys = false };
struct MemStatData {
u64 physical_allocated = 0;
u64 physical_available = 0;
// NOLINTBEGIN(readability-identifier-naming)
struct glaze {
using T = MemStatData;
static constexpr Object value =
object("physical_allocated", &T::physical_allocated, "physical_available", &T::physical_available);
};
// NOLINTEND(readability-identifier-naming)
};
} // namespace
namespace os {
fn GetOSVersion() -> Result<String, DracError> {
utsname uts;
if (uname(&uts) == -1)
return Err(DracError::withErrno("uname call failed for OS Version"));
return uts.sysname;
}
fn GetMemInfo() -> Result<u64, DracError> {
CStr path = "/sys/kernel/memstat";
std::ifstream file(path);
if (!file)
return Err(DracError(DracErrorCode::NotFound, std::format("Could not open {}", path)));
String buffer((std::istreambuf_iterator<char>(file)), std::istreambuf_iterator<char>());
file.close();
if (buffer.empty())
return Err(DracError(DracErrorCode::IoError, std::format("File is empty: {}", path)));
MemStatData data;
glz::error_ctx error_context = glz::read<glaze_opts>(data, buffer);
if (error_context)
return Err(DracError(
DracErrorCode::ParseError,
std::format("Failed to parse JSON from {}: {}", path, glz::format_error(error_context, buffer))
));
if (data.physical_allocated > std::numeric_limits<u64>::max() - data.physical_available)
return Err(DracError(DracErrorCode::InternalError, "Memory size overflow during calculation"));
return (data.physical_allocated + data.physical_available) * PAGE_SIZE;
}
fn GetNowPlaying() -> Result<MediaInfo, DracError> {
return Err(DracError(DracErrorCode::NotSupported, "Now playing is not supported on SerenityOS"));
}
fn GetWindowManager() -> Result<String, DracError> { return "WindowManager"; }
fn GetDesktopEnvironment() -> Result<String, DracError> { return "SerenityOS Desktop"; }
fn GetShell() -> Result<String, DracError> {
uid_t userId = getuid();
errno = 0;
struct passwd* pw = getpwuid(userId);
if (pw == nullptr)
return Err(DracError(DracErrorCode::NotFound, std::format("User ID {} not found in /etc/passwd", userId)));
if (pw->pw_shell == nullptr || *(pw->pw_shell) == '\0')
return Err(DracError(
DracErrorCode::NotFound, std::format("User shell entry is empty in /etc/passwd for user ID {}", userId)
));
String shell = pw->pw_shell;
if (shell.starts_with("/bin/"))
shell = shell.substr(5);
return shell;
}
fn GetHost() -> Result<String, DracError> {
Array<char, HOST_NAME_MAX> hostname_buffer;
if (gethostname(hostname_buffer.data(), hostname_buffer.size()) != 0)
return Err(DracError::withErrno("gethostname() failed: {}"));
return String(hostname_buffer.data());
}
fn GetKernelVersion() -> Result<String, DracError> {
utsname uts;
if (uname(&uts) == -1)
return Err(DracError::withErrno("uname call failed for Kernel Version"));
return uts.release;
}
fn GetDiskUsage() -> Result<DiskSpace, DracError> {
struct statvfs stat;
if (statvfs("/", &stat) == -1)
return Err(DracError::withErrno("statvfs call failed for '/'"));
const u64 total_bytes = static_cast<u64>(stat.f_blocks) * stat.f_frsize;
const u64 free_bytes = static_cast<u64>(stat.f_bfree) * stat.f_frsize;
const u64 used_bytes = total_bytes - free_bytes;
return DiskSpace { .used_bytes = used_bytes, .total_bytes = total_bytes };
}
fn GetPackageCount() -> Result<u64, DracError> {
return Err(DracError(DracErrorCode::NotSupported, "Package count is not supported on SerenityOS"));
}
} // namespace os
#endif // __serenity__

View file

@ -1,3 +1,6 @@
#ifndef __serenity__
// clang-format off
#ifndef _WIN32
#include <SQLiteCpp/Database.h> // SQLite::{Database, OPEN_READONLY}
#include <SQLiteCpp/Exception.h> // SQLite::Exception
@ -22,6 +25,7 @@
#include "src/util/types.hpp"
#include "os.hpp"
// clang-format on
using util::error::DracError, util::error::DracErrorCode;
using util::types::u64, util::types::i64, util::types::String, util::types::StringView, util::types::Result,
@ -33,13 +37,13 @@ namespace {
using namespace std::chrono;
using namespace util::cache;
#ifndef _WIN32
#ifndef _WIN32
struct PackageManagerInfo {
String id;
fs::path dbPath;
String countQuery;
};
#endif
#endif
struct PkgCountCacheData {
u64 count {};
@ -55,7 +59,7 @@ namespace {
// NOLINTEND(readability-identifier-naming)
};
#ifndef _WIN32
#ifndef _WIN32
fn GetPackageCountInternalDb(const PackageManagerInfo& pmInfo) -> Result<u64, DracError> {
const auto& [pmId, dbPath, countQuery] = pmInfo;
@ -117,9 +121,9 @@ namespace {
return count;
}
#endif
#endif
#ifndef _WIN32
#ifndef _WIN32
fn GetNixPackageCount() -> Result<u64, DracError> {
debug_log("Attempting to get Nix package count.");
@ -140,7 +144,7 @@ namespace {
return GetPackageCountInternalDb(nixInfo);
}
#endif
#endif
fn GetCargoPackageCount() -> Result<u64, DracError> {
using util::helpers::GetEnv;
@ -171,12 +175,12 @@ namespace os::shared {
fn GetPackageCount() -> Result<u64, DracError> {
u64 count = 0;
#ifndef _WIN32
#ifndef _WIN32
if (const Result<u64, DracError> pkgCount = GetNixPackageCount())
count += *pkgCount;
else
debug_at(pkgCount.error());
#endif
#endif
if (const Result<u64, DracError> pkgCount = GetCargoPackageCount())
count += *pkgCount;
@ -186,3 +190,5 @@ namespace os::shared {
return count;
}
} // namespace os::shared
#endif // !__serenity__