1
Fork 0
mirror of https://github.com/RGBCube/serenity synced 2025-10-22 17:42:33 +00:00
serenity/Userland/Libraries/LibC/wstdio.cpp

275 lines
7.5 KiB
C++

/*
* Copyright (c) 2021, Ali Mohammad Pur <mpfard@serenityos.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <AK/Assertions.h>
#include <AK/BitCast.h>
#include <AK/PrintfImplementation.h>
#include <AK/StringBuilder.h>
#include <AK/Types.h>
#include <bits/stdio_file_implementation.h>
#include <errno.h>
#include <stdio.h>
#include <wchar.h>
static_assert(AssertSize<wchar_t, sizeof(u32)>());
extern "C" {
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fwide.html
int fwide(FILE*, int mode)
{
// Nope Nope Nope.
return mode;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fgetwc.html
wint_t fgetwc(FILE* stream)
{
VERIFY(stream);
Array<u8, 4> underlying;
auto underlying_bytes = underlying.span();
size_t encoded_length = 0;
size_t bytes_read = 0;
do {
size_t nread = fread(underlying_bytes.offset_pointer(bytes_read), 1, 1, stream);
if (nread != 1) {
errno = EILSEQ;
return WEOF;
}
++bytes_read;
if (bytes_read == 1) {
if (underlying[0] >> 7 == 0) {
encoded_length = 1;
} else if (underlying[0] >> 5 == 6) {
encoded_length = 2;
} else if (underlying[0] >> 4 == 0xe) {
encoded_length = 3;
} else if (underlying[0] >> 3 == 0x1e) {
encoded_length = 4;
} else {
errno = EILSEQ;
return WEOF;
}
}
} while (bytes_read < encoded_length);
wchar_t code_point;
auto read_bytes = mbrtowc(&code_point, bit_cast<char const*>(underlying.data()), encoded_length, nullptr);
VERIFY(read_bytes == encoded_length);
return code_point;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/getwc.html
wint_t getwc(FILE* stream)
{
return fgetwc(stream);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/getwchar.html
wint_t getwchar()
{
return getwc(stdin);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fputwc.html
wint_t fputwc(wchar_t wc, FILE* stream)
{
VERIFY(stream);
// Negative wide chars are weird
if constexpr (IsSigned<wchar_t>) {
if (wc < 0) {
errno = EILSEQ;
return WEOF;
}
}
StringBuilder sb;
sb.append_code_point(static_cast<u32>(wc));
auto bytes = sb.string_view().bytes();
ScopedFileLock lock(stream);
size_t nwritten = stream->write(bytes.data(), bytes.size());
if (nwritten < bytes.size())
return WEOF;
return wc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/putwc.html
wint_t putwc(wchar_t wc, FILE* stream)
{
return fputwc(wc, stream);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/putwchar.html
wint_t putwchar(wchar_t wc)
{
return fputwc(wc, stdout);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fgetws.html
wchar_t* fgetws(wchar_t* __restrict buffer, int size, FILE* __restrict stream)
{
VERIFY(stream);
ScopedFileLock lock(stream);
bool ok = stream->gets(bit_cast<u32*>(buffer), size);
return ok ? buffer : nullptr;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fputws.html
int fputws(wchar_t const* __restrict ws, FILE* __restrict stream)
{
VERIFY(stream);
ScopedFileLock lock(stream);
int size = 0;
for (auto const* p = ws; *p != 0; ++p, ++size) {
if (putwc(*p, stream) == WEOF)
return WEOF;
}
return size;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/ungetwc.html
wint_t ungetwc(wint_t wc, FILE* stream)
{
VERIFY(stream);
ScopedFileLock lock(stream);
StringBuilder sb;
sb.append_code_point(static_cast<u32>(wc));
auto bytes = sb.string_view().bytes();
size_t ok_bytes = 0;
for (auto byte : bytes) {
if (!stream->ungetc(byte)) {
// Discard the half-ungotten bytes.
stream->read(const_cast<u8*>(bytes.data()), ok_bytes);
return WEOF;
}
++ok_bytes;
}
return wc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/wprintf.html
int wprintf(wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vfwprintf(stdout, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fwprintf.html
int fwprintf(FILE* __restrict stream, wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vfwprintf(stream, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/swprintf.html
int swprintf(wchar_t* __restrict wcs, size_t max_length, wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vswprintf(wcs, max_length, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vwprintf.html
int vwprintf(wchar_t const* __restrict format, va_list args)
{
return vfwprintf(stdout, format, args);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vfwprintf.html
int vfwprintf(FILE* __restrict stream, wchar_t const* __restrict format, va_list args)
{
auto const* fmt = bit_cast<wchar_t const*>(format);
return printf_internal([stream](wchar_t*&, wchar_t wc) {
putwc(wc, stream);
},
nullptr, fmt, args);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vswprintf.html
int vswprintf(wchar_t* __restrict wcs, size_t max_length, wchar_t const* __restrict format, va_list args)
{
auto const* fmt = bit_cast<wchar_t const*>(format);
size_t length_so_far = 0;
printf_internal([max_length, &length_so_far](wchar_t*& buffer, wchar_t wc) {
if (length_so_far > max_length)
return;
*buffer++ = wc;
++length_so_far;
},
wcs, fmt, args);
if (length_so_far < max_length)
wcs[length_so_far] = L'\0';
else
wcs[max_length - 1] = L'\0';
return static_cast<int>(length_so_far);
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/fwscanf.html
int fwscanf(FILE* __restrict stream, wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vfwscanf(stream, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/swscanf.html
int swscanf(wchar_t const* __restrict ws, wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vswscanf(ws, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/wscanf.html
int wscanf(wchar_t const* __restrict format, ...)
{
va_list ap;
va_start(ap, format);
auto rc = vfwscanf(stdout, format, ap);
va_end(ap);
return rc;
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vfwscanf.html
int vfwscanf(FILE* __restrict stream, wchar_t const* __restrict format, va_list arg)
{
(void)stream;
(void)format;
(void)arg;
dbgln("FIXME: Implement vfwscanf()");
TODO();
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vswscanf.html
int vswscanf(wchar_t const* __restrict ws, wchar_t const* __restrict format, va_list arg)
{
(void)ws;
(void)format;
(void)arg;
dbgln("FIXME: Implement vswscanf()");
TODO();
}
// https://pubs.opengroup.org/onlinepubs/9699919799/functions/vwscanf.html
int vwscanf(wchar_t const* __restrict format, va_list arg)
{
(void)format;
(void)arg;
dbgln("FIXME: Implement vwscanf()");
TODO();
}
}