/*
 * Copyright (C) Alexander Borisov
 * Copyright (C) NGINX, Inc.
 */

#include <perl/nxt_perl_psgi_layer.h>


typedef struct {
    struct _PerlIO  base;

    SV              *var;
} nxt_perl_psgi_layer_stream_t;


static IV nxt_perl_psgi_layer_stream_pushed(pTHX_ PerlIO *f, const char *mode,
    SV *arg, PerlIO_funcs *tab);
static IV nxt_perl_psgi_layer_stream_popped(pTHX_ PerlIO *f);

static PerlIO *nxt_perl_psgi_layer_stream_open(pTHX_ PerlIO_funcs *self,
    PerlIO_list_t *layers, IV n,
    const char *mode, int fd, int imode, int perm,
    PerlIO *f, int narg, SV **args);

static IV nxt_perl_psgi_layer_stream_close(pTHX_ PerlIO *f);

static SSize_t nxt_perl_psgi_layer_stream_read(pTHX_ PerlIO *f,
    void *vbuf, Size_t count);
static SSize_t nxt_perl_psgi_layer_stream_write(pTHX_ PerlIO *f,
    const void *vbuf, Size_t count);

static IV nxt_perl_psgi_layer_stream_fileno(pTHX_ PerlIO *f);
static IV nxt_perl_psgi_layer_stream_seek(pTHX_ PerlIO *f,
    Off_t offset, int whence);
static Off_t nxt_perl_psgi_layer_stream_tell(pTHX_ PerlIO *f);
static IV nxt_perl_psgi_layer_stream_fill(pTHX_ PerlIO *f);
static IV nxt_perl_psgi_layer_stream_flush(pTHX_ PerlIO *f);

static SV *nxt_perl_psgi_layer_stream_arg(pTHX_ PerlIO *f,
    CLONE_PARAMS *param, int flags);

static PerlIO *nxt_perl_psgi_layer_stream_dup(pTHX_ PerlIO *f, PerlIO *o,
    CLONE_PARAMS *param, int flags);
static IV nxt_perl_psgi_layer_stream_eof(pTHX_ PerlIO *f);

static STDCHAR *nxt_perl_psgi_layer_stream_get_base(pTHX_ PerlIO *f);
static STDCHAR *nxt_perl_psgi_layer_stream_get_ptr(pTHX_ PerlIO *f);
static SSize_t nxt_perl_psgi_layer_stream_get_cnt(pTHX_ PerlIO *f);
static Size_t nxt_perl_psgi_layer_stream_buffersize(pTHX_ PerlIO *f);
static void nxt_perl_psgi_layer_stream_set_ptrcnt(pTHX_ PerlIO *f,
    STDCHAR *ptr, SSize_t cnt);


static PERLIO_FUNCS_DECL(PerlIO_NGINX_Unit) = {
    sizeof(PerlIO_funcs),
    "NGINX_Unit_PSGI_Layer_Stream",
    sizeof(nxt_perl_psgi_layer_stream_t),
    PERLIO_K_BUFFERED | PERLIO_K_RAW,
    nxt_perl_psgi_layer_stream_pushed,
    nxt_perl_psgi_layer_stream_popped,
    nxt_perl_psgi_layer_stream_open,
    PerlIOBase_binmode,
    nxt_perl_psgi_layer_stream_arg,
    nxt_perl_psgi_layer_stream_fileno,
    nxt_perl_psgi_layer_stream_dup,
    nxt_perl_psgi_layer_stream_read,
    NULL,
    nxt_perl_psgi_layer_stream_write,
    nxt_perl_psgi_layer_stream_seek,
    nxt_perl_psgi_layer_stream_tell,
    nxt_perl_psgi_layer_stream_close,
    nxt_perl_psgi_layer_stream_flush,
    nxt_perl_psgi_layer_stream_fill,
    nxt_perl_psgi_layer_stream_eof,
    PerlIOBase_error,
    PerlIOBase_clearerr,
    PerlIOBase_setlinebuf,
    nxt_perl_psgi_layer_stream_get_base,
    nxt_perl_psgi_layer_stream_buffersize,
    nxt_perl_psgi_layer_stream_get_ptr,
    nxt_perl_psgi_layer_stream_get_cnt,
    nxt_perl_psgi_layer_stream_set_ptrcnt,
};


static IV
nxt_perl_psgi_layer_stream_pushed(pTHX_ PerlIO *f, const char *mode, SV *arg,
    PerlIO_funcs *tab)
{
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);

    if (arg != NULL && SvOK(arg)) {
        unit_stream->var = arg;
    }

    SvSETMAGIC(unit_stream->var);

    return PerlIOBase_pushed(aTHX_ f, mode, Nullsv, tab);
}


static IV
nxt_perl_psgi_layer_stream_popped(pTHX_ PerlIO *f)
{
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);

    if (unit_stream->var != NULL) {
        SvREFCNT_dec(unit_stream->var);
        unit_stream->var = Nullsv;
    }

    return 0;
}


static PerlIO *
nxt_perl_psgi_layer_stream_open(pTHX_ PerlIO_funcs *self,
    PerlIO_list_t *layers, IV n,
    const char *mode, int fd, int imode, int perm,
    PerlIO *f, int narg, SV **args)
{
    SV  *arg;

    arg = (narg > 0) ? *args : PerlIOArg;

    PERL_UNUSED_ARG(fd);
    PERL_UNUSED_ARG(imode);
    PERL_UNUSED_ARG(perm);

    if (SvROK(arg) || SvPOK(arg)) {

        if (f == NULL) {
            f = PerlIO_allocate(aTHX);
        }

        f = PerlIO_push(aTHX_ f, self, mode, arg);

        if (f != NULL) {
            PerlIOBase(f)->flags |= PERLIO_F_OPEN;
        }

        return f;
    }

    return NULL;
}


static IV
nxt_perl_psgi_layer_stream_close(pTHX_ PerlIO *f)
{
    IV  code;

    code = PerlIOBase_close(aTHX_ f);
    PerlIOBase(f)->flags &= ~(PERLIO_F_RDBUF | PERLIO_F_WRBUF);

    return code;
}


static IV
nxt_perl_psgi_layer_stream_fileno(pTHX_ PerlIO *f)
{
    PERL_UNUSED_ARG(f);
    return -1;
}


static SSize_t
nxt_perl_psgi_layer_stream_read(pTHX_ PerlIO *f, void *vbuf, Size_t count)
{
    nxt_perl_psgi_io_arg_t        *arg;
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    if (f == NULL) {
        return 0;
    }

    unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);
    arg = (nxt_perl_psgi_io_arg_t *) (intptr_t) SvIV(SvRV(unit_stream->var));

    if ((PerlIOBase(f)->flags & PERLIO_F_CANREAD) == 0) {
        PerlIOBase(f)->flags |= PERLIO_F_ERROR;

        SETERRNO(EBADF, SS_IVCHAN);

        return 0;
    }

    return (SSize_t) arg->read(PERL_GET_CONTEXT, arg, vbuf, count);
}


static SSize_t
nxt_perl_psgi_layer_stream_write(pTHX_ PerlIO *f,
    const void *vbuf, Size_t count)
{
    nxt_perl_psgi_io_arg_t        *arg;
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    if (PerlIOBase(f)->flags & PERLIO_F_CANWRITE) {

        unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);

        arg = (nxt_perl_psgi_io_arg_t *)
            (intptr_t) SvIV(SvRV(unit_stream->var));

        return (SSize_t) arg->write(PERL_GET_CONTEXT, arg, vbuf, count);
    }

    return 0;
}


static IV
nxt_perl_psgi_layer_stream_seek(pTHX_ PerlIO *f, Off_t offset, int whence)
{
    PERL_UNUSED_ARG(f);
    return 0;
}


static Off_t
nxt_perl_psgi_layer_stream_tell(pTHX_ PerlIO *f)
{
    PERL_UNUSED_ARG(f);
    return 0;
}


static IV
nxt_perl_psgi_layer_stream_fill(pTHX_ PerlIO *f)
{
    PERL_UNUSED_ARG(f);
    return -1;
}


static IV
nxt_perl_psgi_layer_stream_flush(pTHX_ PerlIO *f)
{
    nxt_perl_psgi_io_arg_t        *arg;
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);
    arg = (nxt_perl_psgi_io_arg_t *) (intptr_t) SvIV(SvRV(unit_stream->var));

    return (IV) arg->flush(PERL_GET_CONTEXT, arg);
}


static SV *
nxt_perl_psgi_layer_stream_arg(pTHX_ PerlIO * f,
    CLONE_PARAMS *param, int flags)
{
    SV                            *var;
    nxt_perl_psgi_layer_stream_t  *unit_stream;

    unit_stream = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);
    var = unit_stream->var;

    if (flags & PERLIO_DUP_CLONE) {
        var = PerlIO_sv_dup(aTHX_ var, param);

    } else if (flags & PERLIO_DUP_FD) {
        var = newSVsv(var);

    } else {
        var = SvREFCNT_inc(var);
    }

    return var;
}


static PerlIO *
nxt_perl_psgi_layer_stream_dup(pTHX_ PerlIO *f, PerlIO *o,
    CLONE_PARAMS *param, int flags)
{
    nxt_perl_psgi_layer_stream_t  *fs;

    f = PerlIOBase_dup(aTHX_ f, o, param, flags);

    if (f != NULL) {
        fs = PerlIOSelf(f, nxt_perl_psgi_layer_stream_t);
        fs->var = nxt_perl_psgi_layer_stream_arg(aTHX_ o, param, flags);
    }

    return f;
}


static IV
nxt_perl_psgi_layer_stream_eof(pTHX_ PerlIO *f)
{
    return 1;
}


static STDCHAR *
nxt_perl_psgi_layer_stream_get_base(pTHX_ PerlIO *f)
{
    return (STDCHAR *) NULL;
}


static STDCHAR *
nxt_perl_psgi_layer_stream_get_ptr(pTHX_ PerlIO *f)
{
    return (STDCHAR *) NULL;
}


static SSize_t
nxt_perl_psgi_layer_stream_get_cnt(pTHX_ PerlIO *f)
{
    return 0;
}


static Size_t
nxt_perl_psgi_layer_stream_buffersize(pTHX_ PerlIO *f)
{
    return 0;
}


static void
nxt_perl_psgi_layer_stream_set_ptrcnt(pTHX_ PerlIO *f,
    STDCHAR *ptr, SSize_t cnt)
{
    /* Need some code. */
}


void
nxt_perl_psgi_layer_stream_init(pTHX)
{
    PerlIO_define_layer(aTHX_ PERLIO_FUNCS_CAST(&PerlIO_NGINX_Unit));
}


PerlIO *
nxt_perl_psgi_layer_stream_fp_create(pTHX_ nxt_perl_psgi_io_arg_t *arg,
    const char *mode)
{
    SV      *arg_rv;
    PerlIO  *fp;

    arg_rv = newSV_type(SVt_RV);

    if (arg_rv == NULL) {
        return NULL;
    }

    sv_setptrref(arg_rv, arg);

    fp = PerlIO_openn(aTHX_ "NGINX_Unit_PSGI_Layer_Stream",
                      mode, 0, 0, 0, NULL, 1, &arg_rv);

    if (fp == NULL) {
        SvREFCNT_dec(arg_rv);
        return NULL;
    }

    return fp;
}


void
nxt_perl_psgi_layer_stream_fp_destroy(pTHX_ PerlIO *io)
{
    PerlIO_close(io);
}


SV *
nxt_perl_psgi_layer_stream_io_create(pTHX_ PerlIO *fp)
{
    SV  *rvio;
    IO  *thatio;

    thatio = newIO();

    if (thatio == NULL) {
        return NULL;
    }

    IoOFP(thatio) = fp;
    IoIFP(thatio) = fp;

    rvio = newRV_noinc((SV *) thatio);

    if (rvio == NULL) {
        SvREFCNT_dec(thatio);
        return NULL;
    }

    return rvio;
}


void
nxt_perl_psgi_layer_stream_io_destroy(pTHX_ SV *rvio)
{
    SvREFCNT_dec(rvio);
}