Files
hak/lib/x-proto.c
2025-09-05 10:52:02 +09:00

195 lines
5.2 KiB
C

/*
Copyright (c) 2016-2018 Chung, Hyung-Hwan. All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions
are met:
1. Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR
IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include <hak-x.h>
#include "hak-prv.h"
enum hak_xproto_rcv_state_t
{
HAK_XPROTO_RCV_HDR,
HAK_XPROTO_RCV_PLD
};
typedef enum hak_xproto_rcv_state_t hak_xproto_rcv_state_t;
struct hak_xproto_t
{
hak_oow_t _instsize;
hak_mmgr_t* _mmgr;
hak_xproto_cb_t _cb;
struct
{
hak_xproto_rcv_state_t state;
hak_oow_t len_needed;
unsigned int eof: 1;
hak_oow_t len;
hak_uint8_t buf[HAK_XPKT_MAX_PLD_LEN];
/* normalize header of hak_xpkt_hdr_t with combined bits into separate placeholders */
struct
{
hak_uint8_t id;
hak_uint8_t type;
hak_uint16_t len; /* this is wider than the len field of hak_xpkt_hdr_t */
} hdr;
} rcv;
struct
{
} snd;
};
hak_xproto_t* hak_xproto_open (hak_mmgr_t* mmgr, hak_xproto_cb_t* cb, hak_oow_t xtnsize)
{
hak_xproto_t* proto;
proto = (hak_xproto_t*)HAK_MMGR_ALLOC(mmgr, HAK_SIZEOF(*proto) + xtnsize);
if (HAK_UNLIKELY(!proto)) return HAK_NULL;
HAK_MEMSET(proto, 0, HAK_SIZEOF(*proto));
proto->_instsize = HAK_SIZEOF(*proto);
proto->_mmgr = mmgr;
proto->_cb = *cb;
proto->rcv.state = HAK_XPROTO_RCV_HDR;
proto->rcv.len_needed = HAK_XPKT_HDR_LEN;
proto->rcv.eof = 0;
return proto;
}
void hak_xproto_close (hak_xproto_t* proto)
{
HAK_MMGR_FREE (proto->_mmgr, proto);
}
void* hak_xproto_getxtn (hak_xproto_t* proto)
{
return (proto + 1);
}
hak_uint8_t* hak_xproto_getbuf (hak_xproto_t* proto, hak_oow_t* capa)
{
*capa = HAK_COUNTOF(proto->rcv.buf) - proto->rcv.len;
return &proto->rcv.buf[proto->rcv.len];
}
int hak_xproto_geteof (hak_xproto_t* proto)
{
return proto->rcv.eof;
}
void hak_xproto_seteof (hak_xproto_t* proto, int v)
{
proto->rcv.eof = v;
}
void hak_xproto_advbuf (hak_xproto_t* proto, hak_oow_t inc)
{
proto->rcv.len += inc;
}
int hak_xproto_ready (hak_xproto_t* proto)
{
/* has it received suffient data for processing? */
return proto->rcv.len >= proto->rcv.len_needed;
}
int hak_xproto_process (hak_xproto_t* proto)
{
int n;
hak_xpkt_hdr_t* hdr;
switch (proto->rcv.state)
{
case HAK_XPROTO_RCV_HDR:
if (proto->rcv.len < HAK_XPKT_HDR_LEN) goto carry_on; /* need more data */
hdr = (hak_xpkt_hdr_t*)proto->rcv.buf;
proto->rcv.hdr.id = hdr->id;
proto->rcv.hdr.type = hdr->type & 0x0F;
proto->rcv.hdr.len = (hak_uint16_t)hdr->len | ((hak_uint16_t)(hdr->type >> 4) << 8);
/* consume the header */
HAK_MEMMOVE(proto->rcv.buf, &proto->rcv.buf[HAK_XPKT_HDR_LEN], proto->rcv.len - HAK_XPKT_HDR_LEN);
proto->rcv.len -= HAK_XPKT_HDR_LEN;
/* switch to the payload mode */
if (proto->rcv.hdr.len > 0)
{
proto->rcv.state = HAK_XPROTO_RCV_PLD;
proto->rcv.len_needed = proto->rcv.hdr.len;
}
else
{
/* take shortcut */
/* TODO: convert handle_packet as call back */
n = proto->_cb.on_packet(proto, proto->rcv.hdr.type, proto->rcv.buf, proto->rcv.hdr.len);
if (n <= -1) goto fail_with_errmsg;
if (n == 0) return 0;
}
break;
case HAK_XPROTO_RCV_PLD:
if (proto->rcv.len < proto->rcv.hdr.len) goto carry_on; /* need more payload data */
n = proto->_cb.on_packet(proto, proto->rcv.hdr.type, proto->rcv.buf, proto->rcv.hdr.len);
/* switch to the header mode */
if (proto->rcv.hdr.len > 0)
{
/* TODO: minimize the use of HAK_MEMOVE... use the buffer */
HAK_MEMMOVE(proto->rcv.buf, &proto->rcv.buf[proto->rcv.hdr.len], proto->rcv.len - proto->rcv.hdr.len);
proto->rcv.len -= proto->rcv.hdr.len;
}
proto->rcv.state = HAK_XPROTO_RCV_HDR;
proto->rcv.len_needed = HAK_XPKT_HDR_LEN;
if (n <= -1) goto fail_with_errmsg;
if (n == 0) return 0;
break;
default:
/*
hak_seterrbfmt(hak, HAK_EINTERN, "invalid request state %d", (int)proto->rcv.state);
*/
/* TODO: call back */
goto fail_with_errmsg;
}
carry_on:
return 1;
fail_with_errmsg:
// TODO: proper error handling
//send_proto_hak_error (proto);
//HAK_LOG1 (hak, SERVER_LOGMASK_ERROR, "Unable to compile .SCRIPT contents - %js\n", hak_geterrmsg(worker->hak));
return -1;
}