/*------------------------------------------------------------------
 * test_vfwscanf_s
 * File 'wchar/vfwscanf_s.c'
 * Lines executed:85.71% of 21
 *
 *------------------------------------------------------------------
 */

#include "test_private.h"
#include "safe_str_lib.h"
#include "safe_lib.h"
#include <stdio.h>
#include <stdarg.h>
#include <unistd.h>

#undef USE_PIPE

#ifdef HAVE_VFWSCANF_S
#define HAVE_NATIVE 1
#else
#define HAVE_NATIVE 0
#endif
#include "test_msvcrt.h"

#define LEN (128)

static wchar_t wstr1[LEN];
static wchar_t wstr2[LEN];
static char str3[LEN];
#define TMP "tmpvfwscanf"
static FILE *stream = NULL;
static void win_stuff_stream(const wchar_t *dest);
static int vtwscanf_s(FILE *stream, const wchar_t *restrict fmt, ...);
static int test_vfwscanf_s(void);

static void win_stuff_stream(const wchar_t *dest) {
    if (!stream)
        stream = fopen(TMP, "w+");
    else
        rewind(stream);
    fwprintf(stream, L"%ls\n", dest);
    rewind(stream);
}

#ifndef USE_PIPE
#define stuff_stream(s)                                                        \
    wcscpy(wstr1, s);                                                          \
    win_stuff_stream(s);
#else
#define stuff_stream(s)                                                        \
    wcscpy(wstr1, s);                                                          \
    write(p[1], (s), sizeof(s) - 1);                                           \
    write(p[1], L"\n", sizeof(L"\n") - 1);
#endif

static int vtwscanf_s(FILE *restrict f, const wchar_t *restrict fmt, ...) {
    int rc;
    va_list ap;
    va_start(ap, fmt);
    rc = vfwscanf_s(f, fmt, ap);
    va_end(ap);
    return rc;
}

static int test_vfwscanf_s(void) {
    errno_t rc;
    int32_t ind;
    size_t len1;
    size_t len2;
    size_t len3;
    int num = 0;
    int errs = 0;
#ifdef USE_PIPE
    int p[2];
#endif

    /*--------------------------------------------------*/

    print_msvcrt(use_msvcrt);

/* This pipe does not work on windows */
#ifdef USE_PIPE
    pipe(p);
    stream = fdopen(p[0], "rb");
    if (!stream) {
        close(p[0]);
        close(p[1]);
        return 0;
    }
#endif
    stuff_stream(L"a");
    wstr2[0] = '\0';
    rc = vtwscanf_s(NULL, L"%ls", wstr2, LEN);
    init_msvcrt(errno == ESNULLP, &use_msvcrt);
    ERR(EOF);
    ERRNO_MSVC(ESNULLP, EINVAL);
    WEXPSTR(wstr2, L"");

    rc = vtwscanf_s(stream, NULL);
    ERR(EOF);
    ERRNO_MSVC(ESNULLP, EINVAL);

#if 0
    /* Illegal: */
    rc = vfwscanf_s(stream, NULL, NULL);
    ERR(EOF);
    ERRNO_MSVC(ESNULLP, EINVAL);

    /* SEGV: */
    rc = vfwscanf_s(stream, L"", NULL);
    ERR(EOF);
    ERRNO_MSVC(ESNULLP, EINVAL);
    rc = vfwscanf_s(stream, L"%ls", NULL);
    ERREOF(ESNULLP);
#endif

    /*--------------------------------------------------*/
    stuff_stream(L"      24");
    rc = vtwscanf_s(stream, L"%ls %n", wstr2, LEN, &ind);
    ERREOF(EINVAL);

    stuff_stream(L"      24");
    rc = vtwscanf_s(stream, L"%ls %%n", wstr2, LEN);
#ifdef BSD_LIKE
    if (rc != 0) { /* BSD's return -1 on %%n */
        printf("%s %u wrong vfwscanf(\"\",L\"%%n\"): %d\n", __FUNCTION__,
               __LINE__, (int)rc);
    } else
#endif
        ERR(1);
    ERRNO(0);

    stuff_stream(L"      24");
    rc = vtwscanf_s(stream, L"%ls %%n", wstr2, 6);
    ERR(1);
    ERRNO(0);

    stuff_stream(L"      24");
    rc = vtwscanf_s(stream, L"%s %%n", str3, 6);
    ERR(1);
    ERRNO(0);
#if !defined(_WIN32) || defined(HAVE_MINGW64)
    EXPSTR(str3, "24");
#else
    EXPSTR(str3, "2");
#endif

    stuff_stream(L"      24");
    rc = vtwscanf_s(stream, L" %d", &num);
    ERR(1);
    ERRNO(0);
    if (num != 24) {
        debug_printf("%s %u wrong arg: %d\n", __FUNCTION__, __LINE__, num);
        errs++;
    }

    /*--------------------------------------------------*/

    stuff_stream(L"aaaaaaaaaa");

    rc = vtwscanf_s(stream, L"%ls", wstr2, LEN);
    ERR(1)
    len2 = wcslen(wstr2);
    len3 = wcslen(wstr1);
    if (len3 != len2) {
#ifdef DEBUG
        len1 = wcslen(wstr1);
#endif
        debug_printf("%s %u lengths wrong: %d  %d  %d \n", __FUNCTION__,
                     __LINE__, (int)len1, (int)len2, (int)len3);
        errs++;
    }

    /*--------------------------------------------------*/

    stuff_stream(L"keep it simple");

    rc = vtwscanf_s(stream, L"%ls", wstr2, LEN);
    ERR(1);
    WEXPSTR(wstr1, L"keep it simple")

    /*--------------------------------------------------*/

    wcscpy(wstr2, L"keep it simple");
    stuff_stream(L"qqweqq");

    rc = vtwscanf_s(stream, L"%ls", wstr2);
    NOERR()
    WEXPSTR(wstr1, wstr2);

    /*--------------------------------------------------*/

    wstr1[0] = '\0';
    wstr2[0] = '\0';
    stuff_stream(L"");

    rc = vtwscanf_s(stream, L"%ls", wstr2, LEN);
    ERR(1)
    WEXPNULL(wstr1)

    /*--------------------------------------------------*/

    wstr1[0] = '\0';
    wcscpy(wstr2, L"keep it simple");
    stuff_stream(L"");

    rc = vtwscanf_s(stream, L"%ls", wstr2, LEN);
    ERR(1)
    WEXPSTR(wstr1, L"")

    /*--------------------------------------------------*/

    /* overlapping works fine on darwin, different on linux glibc */
    /*
    wcscpy(wstr1, L"12345678901234567890");

    rc = vtwscanf_s(stream, L"%ls", &wstr1[7]);
    ERR(1);
    WEXPSTR(wstr1, L"123456712345678901234567890");

    wcscpy(wstr1, L"123456789");

    rc = vtwscanf_s(stream, L"%ls", &wstr1[8]);
    ERR(1);
    WEXPSTR(wstr1, L"12345678123456789");
    */

    /*--------------------------------------------------*/

    /* TODO: we want to test a vfwscanf error propagated through vfwscanf_s.
       The only error is EOF(stream), but this is blocking.
       Reading from a closed stream is platform dependent.
     */
    fclose(stream);
#ifdef USE_PIPE
    close(p[1]);
#endif
#if 0
    rc = vtwscanf_s(stream, L"%ls", wstr2, LEN);

#if defined(__GLIBC__)
    if (rc < 0) {
        ERR(-1);
        ERRNO(EBADF);
    } else {
        ERR(14); /* older glibc upstream bug */
        ERRNO(0);
    }
#elif defined __CYGWIN__
    ERR(1);
#else
    ERR(-1);
#endif
    /*WEXPNULL(wstr2); TODO zero the output args */
#endif

    /*--------------------------------------------------*/

    unlink(TMP);

    return (errs);
}

#ifndef __KERNEL__
/* simple hack to get this to work for both userspace and Linux kernel,
   until a better solution can be created. */
int main(void) { return (test_vfwscanf_s()); }
#endif