//===-- Implementation for mbrtowc function ---------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "src/__support/wchar/mbrtowc.h" #include "hdr/errno_macros.h" #include "hdr/types/size_t.h" #include "hdr/types/wchar_t.h" #include "src/__support/common.h" #include "src/__support/error_or.h" #include "src/__support/macros/config.h" #include "src/__support/wchar/character_converter.h" #include "src/__support/wchar/mbstate.h" namespace LIBC_NAMESPACE_DECL { namespace internal { ErrorOr mbrtowc(wchar_t *__restrict pwc, const char *__restrict s, size_t n, mbstate *__restrict ps) { CharacterConverter char_conv(ps); if (!char_conv.isValidState()) return Error(EINVAL); if (s == nullptr) return 0; size_t i = 0; // Reading in bytes until we have a complete wc or error for (; i < n && !char_conv.isFull(); ++i) { int err = char_conv.push(static_cast(s[i])); // Encoding error if (err == EILSEQ) return Error(err); } auto wc = char_conv.pop_utf32(); if (wc.has_value()) { if (pwc != nullptr) *pwc = wc.value(); // null terminator -> return 0 if (wc.value() == L'\0') return 0; return i; } // Incomplete but potentially valid return -2; } } // namespace internal } // namespace LIBC_NAMESPACE_DECL