From c659bf11f9f23fd65159c71949677bcb31018616 Mon Sep 17 00:00:00 2001 From: Darley Barreto <darleybarreto@gmail.com> Date: Tue, 9 May 2023 14:23:25 +0000 Subject: [PATCH] Implementing ungetwc --- src/header/wchar/mod.rs | 21 ++++++++++++++++++--- tests/Makefile | 1 + tests/wchar/ungetwc.c | 26 ++++++++++++++++++++++++++ tests/wchar/ungetwc.in | 1 + 4 files changed, 46 insertions(+), 3 deletions(-) create mode 100644 tests/wchar/ungetwc.c create mode 100644 tests/wchar/ungetwc.in diff --git a/src/header/wchar/mod.rs b/src/header/wchar/mod.rs index c947a4e4c..bf345e142 100644 --- a/src/header/wchar/mod.rs +++ b/src/header/wchar/mod.rs @@ -237,9 +237,24 @@ pub extern "C" fn swscanf(s: *const wchar_t, format: *const wchar_t, ap: va_list unimplemented!(); } -// #[no_mangle] -pub extern "C" fn ungetwc(wc: wint_t, stream: *mut FILE) -> wint_t { - unimplemented!(); +/// Push wide character `wc` back onto `stream` so it'll be read next +#[no_mangle] +pub unsafe extern "C" fn ungetwc(wc: wint_t, stream: &mut FILE) -> wint_t { + if wc == WEOF { + return wc; + } + static mut INTERNAL: mbstate_t = mbstate_t; + let mut bytes: [c_char; MB_CUR_MAX as usize] = [0; MB_CUR_MAX as usize]; + + let amount = wcrtomb(bytes.as_mut_ptr(), wc as wchar_t, &mut INTERNAL); + if amount == usize::MAX { + return WEOF; + } + for i in 0..amount { + ungetc(bytes[i] as c_int, &mut *stream); + } + + wc } // #[no_mangle] diff --git a/tests/Makefile b/tests/Makefile index 84d51f740..e3295ba65 100644 --- a/tests/Makefile +++ b/tests/Makefile @@ -43,6 +43,7 @@ EXPECT_NAMES=\ stdio/printf_space_pad \ stdio/ungetc_ftell \ stdio/ungetc_multiple \ + stdio/ungetwc \ stdio/fscanf_offby1 \ stdio/fscanf \ stdio/printf_neg_pad \ diff --git a/tests/wchar/ungetwc.c b/tests/wchar/ungetwc.c new file mode 100644 index 000000000..015ba174e --- /dev/null +++ b/tests/wchar/ungetwc.c @@ -0,0 +1,26 @@ +#include <wchar.h> +#include <wctype.h> +#include <stdio.h> +#include <stdlib.h> +#include <assert.h> + +int main() +{ + FILE *stream; + wint_t wc; + wint_t wc2; + + if (NULL == (stream = fopen("wchar/ungetwc.in", "r+"))) + return 1; + + while (WEOF != (wc = fgetwc(stream)) && iswdigit(wc)) {} + + if (WEOF != wc) + ungetwc(wc, stream); + + wc2 = fgetwc(stream); + assert(WEOF != wc2); + assert(wc == wc2); + + return 0; +} \ No newline at end of file diff --git a/tests/wchar/ungetwc.in b/tests/wchar/ungetwc.in new file mode 100644 index 000000000..5e6162a56 --- /dev/null +++ b/tests/wchar/ungetwc.in @@ -0,0 +1 @@ +123A \ No newline at end of file -- GitLab