mirror of https://github.com/axmolengine/axmol.git
819 lines
39 KiB
C++
819 lines
39 KiB
C++
//////////////////////////////////////////////////////////////////////////////////////////
|
|
// A multi-platform support c++11 library with focus on asynchronous socket I/O for any
|
|
// client application.
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
/*
|
|
The MIT License (MIT)
|
|
Copyright (c) 2012-2023 HALX99
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
*/
|
|
#if !defined(LUA_LIB)
|
|
# define LUA_LIB
|
|
#endif
|
|
|
|
#include "yasio/yasio.hpp"
|
|
#include "yasio/bindings/lyasio.hpp"
|
|
using namespace yasio;
|
|
|
|
namespace lyasio
|
|
{
|
|
enum
|
|
{
|
|
BUFFER_DEFAULT,
|
|
BUFFER_RAW,
|
|
BUFFER_FAST,
|
|
};
|
|
|
|
template <typename _Stream>
|
|
static void obstream_write_v(_Stream* obs, cxx17::string_view val, int length_field_bits)
|
|
{
|
|
// default: Use variant length of length field, just like .net BinaryWriter.Write(String),
|
|
// see:
|
|
// https://github.com/mono/mono/blob/master/mcs/class/referencesource/mscorlib/system/io/binarywriter.cs
|
|
switch (length_field_bits)
|
|
{
|
|
case -1:
|
|
default:
|
|
return obs->write_v(val);
|
|
case 32:
|
|
return obs->write_v32(val);
|
|
case 16:
|
|
return obs->write_v16(val);
|
|
case 8:
|
|
return obs->write_v8(val);
|
|
}
|
|
};
|
|
template <typename _Stream>
|
|
static cxx17::string_view ibstream_read_v(_Stream* ibs, int length_field_bits)
|
|
{
|
|
// default: Use variant length of length field, just like .net BinaryReader.ReadString,
|
|
// see:
|
|
// https://github.com/mono/mono/blob/master/mcs/class/referencesource/mscorlib/system/io/binaryreader.cs
|
|
switch (length_field_bits)
|
|
{
|
|
case -1:
|
|
default:
|
|
return ibs->read_v();
|
|
case 32:
|
|
return ibs->read_v32();
|
|
case 16:
|
|
return ibs->read_v16();
|
|
case 8:
|
|
return ibs->read_v8();
|
|
}
|
|
};
|
|
} // namespace lyasio
|
|
|
|
#if YASIO__HAS_CXX14
|
|
|
|
# include "yasio/bindings/yasio_sol.hpp"
|
|
|
|
namespace lyasio
|
|
{
|
|
template <typename _Stream>
|
|
static void register_obstream(sol::table& lib, const char* usertype)
|
|
{
|
|
lib.new_usertype<_Stream>(
|
|
usertype,
|
|
# if defined(YASIO_OBS_BUILTIN_STACK)
|
|
"push32", &_Stream::push32, "pop32",
|
|
sol::overload(static_cast<void (_Stream ::*)()>(&_Stream::pop32), static_cast<void (_Stream ::*)(uint32_t)>(&_Stream::pop32)), "push16", &_Stream::push16,
|
|
"pop16", sol::overload(static_cast<void (_Stream ::*)()>(&_Stream::pop16), static_cast<void (_Stream ::*)(uint16_t)>(&_Stream::pop16)), "push8",
|
|
&_Stream::push8, "pop8", sol::overload(static_cast<void (_Stream ::*)()>(&_Stream::pop8), static_cast<void (_Stream ::*)(uint8_t)>(&_Stream::pop8)),
|
|
# else
|
|
"push32", &_Stream::template push<uint32_t>, "pop32",
|
|
sol::overload(static_cast<void (_Stream ::*)(size_t)>(&_Stream::template pop<uint32_t>),
|
|
static_cast<void (_Stream ::*)(size_t, uint32_t)>(&_Stream::template pop<uint32_t>)),
|
|
"push16", &_Stream::template push<uint16_t>, "pop16",
|
|
sol::overload(static_cast<void (_Stream ::*)(size_t)>(&_Stream::template pop<uint16_t>),
|
|
static_cast<void (_Stream ::*)(size_t, uint16_t)>(&_Stream::template pop<uint16_t>)),
|
|
"push8", &_Stream::template push<uint8_t>, "pop8",
|
|
sol::overload(static_cast<void (_Stream ::*)(size_t)>(&_Stream::template pop<uint8_t>),
|
|
static_cast<void (_Stream ::*)(size_t, uint8_t)>(&_Stream::template pop<uint8_t>)),
|
|
# endif
|
|
"write_ix", &_Stream::template write_ix<int64_t>, "write_bool", &_Stream::template write<bool>, "write_i8", &_Stream::template write<int8_t>, "write_i16",
|
|
&_Stream::template write<int16_t>, "write_i32", &_Stream::template write<int32_t>, "write_i64", &_Stream::template write<int64_t>, "write_u8",
|
|
&_Stream::template write<uint8_t>, "write_u16", &_Stream::template write<uint16_t>, "write_u32", &_Stream::template write<uint32_t>, "write_u64",
|
|
&_Stream::template write<uint64_t>,
|
|
# if defined(YASIO_ENABLE_HALF_FLOAT)
|
|
"write_f16", &_Stream::template write<fp16_t>,
|
|
# endif
|
|
"write_f", &_Stream::template write<float>, "write_lf", &_Stream::template write<double>, "write_v",
|
|
[](_Stream* obs, cxx17::string_view sv, sol::variadic_args args) {
|
|
int lfl = -1;
|
|
if (args.size() > 0)
|
|
lfl = static_cast<int>(args[0]);
|
|
return obstream_write_v<_Stream>(obs, sv, lfl);
|
|
},
|
|
"write_bytes", static_cast<void (_Stream::*)(cxx17::string_view)>(&_Stream::write_bytes), "length", &_Stream::length, "to_string",
|
|
[](_Stream* obs) { return cxx17::string_view(obs->data(), obs->length()); }, "save", &_Stream::save);
|
|
}
|
|
|
|
template <typename _Stream, typename _StreamView, typename _OStream>
|
|
static void register_ibstream(sol::table& lib, const char* usertype)
|
|
{
|
|
lib.new_usertype<_Stream>(
|
|
usertype, sol::constructors<_Stream(), _Stream(yasio::sbyte_buffer), _Stream(const _OStream*)>(), "load", &_Stream::load, "read_ix",
|
|
&_Stream::template read_ix<int64_t>, "read_bool", &_Stream::template read<bool>, "read_i8", &_Stream::template read<int8_t>, "read_i16",
|
|
&_Stream::template read<int16_t>, "read_i32", &_Stream::template read<int32_t>, "read_i64", &_Stream::template read<int64_t>, "read_u8",
|
|
&_Stream::template read<uint8_t>, "read_u16", &_Stream::template read<uint16_t>, "read_u32", &_Stream::template read<uint32_t>, "read_u64",
|
|
&_Stream::template read<uint64_t>,
|
|
# if defined(YASIO_ENABLE_HALF_FLOAT)
|
|
"read_f16", &_Stream::template read<fp16_t>,
|
|
# endif
|
|
"read_f", &_Stream::template read<float>, "read_lf", &_Stream::template read<double>, "read_v",
|
|
[](_Stream* ibs, sol::variadic_args args) {
|
|
int lfl = -1;
|
|
if (args.size() > 0)
|
|
lfl = static_cast<int>(args[0]);
|
|
return ibstream_read_v<_Stream>(ibs, lfl);
|
|
},
|
|
"read_bytes", static_cast<cxx17::string_view (_Stream::*)(int)>(&_Stream::read_bytes), "seek", &_StreamView::seek, "length", &_StreamView::length,
|
|
"to_string", [](_Stream* ibs) { return cxx17::string_view(ibs->data(), ibs->length()); });
|
|
}
|
|
} // namespace lyasio
|
|
|
|
extern "C" {
|
|
|
|
YASIO_LUA_API int luaopen_yasio(lua_State* L)
|
|
{
|
|
sol::state_view state_view(L);
|
|
|
|
# if !YASIO_LUA_ENABLE_GLOBAL
|
|
auto yasio_lib = state_view.create_table();
|
|
# else
|
|
auto yasio_lib = state_view.create_named_table("yasio");
|
|
# endif
|
|
yasio_lib.new_usertype<io_event>(
|
|
"io_event", "kind", &io_event::kind, "status", &io_event::status, "passive", [](io_event* e) { return !!e->passive(); }, "packet",
|
|
[](io_event* ev, sol::variadic_args args, sol::this_state s) {
|
|
sol::state_view L(s);
|
|
auto& pkt = ev->packet();
|
|
if (is_packet_empty(pkt))
|
|
return sol::make_object(L, sol::lua_nil);
|
|
|
|
int buffer_type = (args.size() >= 1) ? static_cast<int>(args[0]) : static_cast<int>(lyasio::BUFFER_DEFAULT);
|
|
switch (buffer_type)
|
|
{
|
|
case lyasio::BUFFER_RAW:
|
|
return sol::make_object(L, cxx17::string_view{packet_data(pkt), packet_len(pkt)});
|
|
case lyasio::BUFFER_FAST:
|
|
return sol::make_object(L, cxx14::make_unique<yasio::fast_ibstream>(forward_packet((packet_t &&) pkt)));
|
|
default:
|
|
return sol::make_object(L, cxx14::make_unique<yasio::ibstream>(forward_packet((packet_t &&) pkt)));
|
|
}
|
|
},
|
|
"cindex", &io_event::cindex, "transport", &io_event::transport
|
|
# if !defined(YASIO_MINIFY_EVENT)
|
|
,
|
|
"timestamp", &io_event::timestamp
|
|
# endif
|
|
);
|
|
|
|
yasio_lib.new_usertype<io_service>(
|
|
"io_service", "new",
|
|
sol::initializers([](io_service& uninitialized_memory) { return new (&uninitialized_memory) io_service(); },
|
|
[](io_service& uninitialized_memory, int n) { return new (&uninitialized_memory) io_service(n); },
|
|
[](io_service& uninitialized_memory, sol::table channel_eps) {
|
|
std::vector<io_hostent> hosts;
|
|
auto host = channel_eps["host"];
|
|
if (host.valid())
|
|
hosts.push_back(io_hostent(host.get<cxx17::string_view>(), channel_eps["port"]));
|
|
else
|
|
{
|
|
for (auto item : channel_eps)
|
|
{
|
|
auto ep = item.second.as<sol::table>();
|
|
hosts.push_back(io_hostent(ep["host"].get<cxx17::string_view>(), ep["port"]));
|
|
}
|
|
}
|
|
return new (&uninitialized_memory)
|
|
io_service(!hosts.empty() ? &hosts.front() : nullptr, (std::max)(static_cast<int>(hosts.size()), 1));
|
|
}),
|
|
sol::meta_function::garbage_collect, sol::destructor([](io_service& memory_from_lua) { memory_from_lua.~io_service(); }), "start",
|
|
[](io_service* service, sol::function cb) {
|
|
service->set_option(YOPT_S_NO_DISPATCH, 1); // script doesn't support handle event at network thread
|
|
service->start([=](event_ptr ev) { cb(std::move(ev)); });
|
|
},
|
|
"stop", &io_service::stop, "set_option",
|
|
[](io_service* service, int opt, sol::variadic_args args) {
|
|
switch (opt)
|
|
{
|
|
case YOPT_C_LOCAL_HOST:
|
|
case YOPT_C_REMOTE_HOST:
|
|
service->set_option(opt, static_cast<int>(args[0]), args[1].as<const char*>());
|
|
break;
|
|
case YOPT_C_UNPACK_STRIP:
|
|
case YOPT_C_LOCAL_PORT:
|
|
case YOPT_C_REMOTE_PORT:
|
|
case YOPT_C_KCP_CONV:
|
|
case YOPT_C_UNPACK_NO_BSWAP:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]));
|
|
break;
|
|
case YOPT_C_ENABLE_MCAST:
|
|
case YOPT_C_LOCAL_ENDPOINT:
|
|
case YOPT_C_REMOTE_ENDPOINT:
|
|
service->set_option(opt, static_cast<int>(args[0]), args[1].as<const char*>(), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_C_MOD_FLAGS:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_S_TCP_KEEPALIVE:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_C_UNPACK_PARAMS:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]), static_cast<int>(args[3]),
|
|
static_cast<int>(args[4]));
|
|
break;
|
|
case YOPT_S_EVENT_CB:
|
|
(void)0;
|
|
{
|
|
sol::function fn = args[0];
|
|
io_event_cb_t fnwrap = [=](event_ptr e) mutable -> void { fn(std::move(e)); };
|
|
service->set_option(opt, std::addressof(fnwrap));
|
|
}
|
|
break;
|
|
default:
|
|
service->set_option(opt, static_cast<int>(args[0]));
|
|
}
|
|
},
|
|
"dispatch", &io_service::dispatch, "open", &io_service::open, "is_open",
|
|
sol::overload(static_cast<bool (io_service::*)(int) const>(&io_service::is_open),
|
|
static_cast<bool (io_service::*)(transport_handle_t) const>(&io_service::is_open)),
|
|
"close",
|
|
sol::overload(static_cast<void (io_service::*)(transport_handle_t)>(&io_service::close), static_cast<void (io_service::*)(int)>(&io_service::close)),
|
|
"write",
|
|
sol::overload(
|
|
[](io_service* service, transport_handle_t transport, cxx17::string_view s) {
|
|
return service->write(transport, yasio::sbyte_buffer{s.data(), s.data() + s.length()});
|
|
},
|
|
[](io_service* service, transport_handle_t transport, yasio::obstream* obs) { return service->write(transport, std::move(obs->buffer())); }),
|
|
"write_to",
|
|
sol::overload(
|
|
[](io_service* service, transport_handle_t transport, cxx17::string_view s, cxx17::string_view ip, u_short port) {
|
|
return service->write_to(transport, yasio::sbyte_buffer{s.data(), s.data() + s.length()}, ip::endpoint{ip.data(), port});
|
|
},
|
|
[](io_service* service, transport_handle_t transport, yasio::obstream* obs, cxx17::string_view ip, u_short port) {
|
|
return service->write_to(transport, std::move(obs->buffer()), ip::endpoint{ip.data(), port});
|
|
}),
|
|
"native_ptr", [](io_service* service) { return (void*)service; });
|
|
|
|
// ##-- obstream
|
|
lyasio::register_obstream<obstream>(yasio_lib, "obstream");
|
|
lyasio::register_obstream<fast_obstream>(yasio_lib, "fast_obstream");
|
|
|
|
// ##-- yasio::ibstream
|
|
lyasio::register_ibstream<ibstream, ibstream_view, obstream>(yasio_lib, "ibstream");
|
|
lyasio::register_ibstream<fast_ibstream, fast_ibstream_view, fast_obstream>(yasio_lib, "fast_ibstream");
|
|
|
|
yasio_lib["highp_clock"] = &highp_clock<yasio::steady_clock_t>;
|
|
yasio_lib["highp_time"] = &highp_clock<yasio::system_clock_t>;
|
|
|
|
yasio_lib["unwrap_ptr"] = [](lua_State* L) -> int {
|
|
auto& pkt = *(packet_t*)lua_touserdata(L, 1);
|
|
int offset = lua_isinteger(L, 2) ? static_cast<int>(lua_tointeger(L, 2)) : 0;
|
|
lua_pushlightuserdata(L, packet_data(pkt) + offset);
|
|
return 1;
|
|
};
|
|
|
|
yasio_lib["unwrap_len"] = [](lua_State* L) -> int {
|
|
auto& pkt = *(packet_t*)lua_touserdata(L, 1);
|
|
int offset = lua_isinteger(L, 2) ? static_cast<int>(lua_tointeger(L, 2)) : 0;
|
|
lua_pushinteger(L, packet_len(pkt) - offset);
|
|
return 1;
|
|
};
|
|
|
|
// ##-- yasio enums
|
|
# define YASIO_EXPORT_ANY(v) yasio_lib[#v] = v
|
|
YASIO_EXPORT_ANY(YCK_TCP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_TCP_SERVER);
|
|
YASIO_EXPORT_ANY(YCK_UDP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_UDP_SERVER);
|
|
# if defined(YASIO_ENABLE_KCP)
|
|
YASIO_EXPORT_ANY(YCK_KCP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_KCP_SERVER);
|
|
# endif
|
|
# if defined(YASIO_SSL_BACKEND)
|
|
YASIO_EXPORT_ANY(YCK_SSL_CLIENT);
|
|
# endif
|
|
|
|
YASIO_EXPORT_ANY(YOPT_S_CONNECT_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_DNS_CACHE_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_DNS_QUERIES_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_TCP_KEEPALIVE);
|
|
YASIO_EXPORT_ANY(YOPT_S_EVENT_CB);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_PARAMS);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_STRIP);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_NO_BSWAP);
|
|
YASIO_EXPORT_ANY(YOPT_C_LFBFD_PARAMS); // alias for YOPT_C_UNPACK_PARAMS
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_HOST);
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_PORT);
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_ENDPOINT);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_HOST);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_PORT);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_ENDPOINT);
|
|
YASIO_EXPORT_ANY(YOPT_C_ENABLE_MCAST);
|
|
YASIO_EXPORT_ANY(YOPT_C_DISABLE_MCAST);
|
|
# if defined(YASIO_ENABLE_KCP)
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_CONV);
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_NODELAY);
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_WINDOW_SIZE);
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_MTU);
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_RTO_MIN);
|
|
# endif
|
|
YASIO_EXPORT_ANY(YOPT_C_MOD_FLAGS);
|
|
|
|
YASIO_EXPORT_ANY(YCF_REUSEADDR);
|
|
YASIO_EXPORT_ANY(YCF_EXCLUSIVEADDRUSE);
|
|
|
|
YASIO_EXPORT_ANY(YEK_ON_OPEN);
|
|
YASIO_EXPORT_ANY(YEK_ON_CLOSE);
|
|
YASIO_EXPORT_ANY(YEK_ON_PACKET);
|
|
YASIO_EXPORT_ANY(YEK_CONNECT_RESPONSE);
|
|
YASIO_EXPORT_ANY(YEK_CONNECTION_LOST);
|
|
YASIO_EXPORT_ANY(YEK_PACKET);
|
|
|
|
YASIO_EXPORT_ANY(SEEK_CUR);
|
|
YASIO_EXPORT_ANY(SEEK_SET);
|
|
YASIO_EXPORT_ANY(SEEK_END);
|
|
|
|
using namespace lyasio;
|
|
YASIO_EXPORT_ANY(BUFFER_DEFAULT);
|
|
YASIO_EXPORT_ANY(BUFFER_RAW);
|
|
YASIO_EXPORT_ANY(BUFFER_FAST);
|
|
|
|
char version[32];
|
|
snprintf(version, sizeof(version), "%x.%x.%x", (YASIO_VERSION_NUM >> 16) & 0xff, (YASIO_VERSION_NUM >> 8) & 0xff, YASIO_VERSION_NUM & 0xff);
|
|
YASIO_EXPORT_ANY(version);
|
|
|
|
return yasio_lib.push(); /* return 'yasio' table */
|
|
}
|
|
|
|
} /* extern "C" */
|
|
|
|
#else
|
|
|
|
# include "kaguya/kaguya.hpp"
|
|
/// customize the type conversion from/to lua
|
|
namespace kaguya
|
|
{
|
|
// cxx17::string_view
|
|
template <>
|
|
struct lua_type_traits<cxx17::string_view> {
|
|
typedef cxx17::string_view get_type;
|
|
typedef cxx17::string_view push_type;
|
|
|
|
static bool strictCheckType(lua_State* l, int index) { return lua_type(l, index) == LUA_TSTRING; }
|
|
static bool checkType(lua_State* l, int index) { return lua_isstring(l, index) != 0; }
|
|
static get_type get(lua_State* l, int index)
|
|
{
|
|
size_t size = 0;
|
|
const char* buffer = lua_tolstring(l, index, &size);
|
|
return cxx17::string_view(buffer, size);
|
|
}
|
|
static int push(lua_State* l, push_type s)
|
|
{
|
|
lua_pushlstring(l, s.data(), s.size());
|
|
return 1;
|
|
}
|
|
};
|
|
// yasio::sbyte_buffer
|
|
template <>
|
|
struct lua_type_traits<yasio::sbyte_buffer> {
|
|
typedef yasio::sbyte_buffer get_type;
|
|
typedef const yasio::sbyte_buffer& push_type;
|
|
|
|
static bool strictCheckType(lua_State* l, int index) { return lua_type(l, index) == LUA_TSTRING; }
|
|
static bool checkType(lua_State* l, int index) { return lua_isstring(l, index) != 0; }
|
|
static get_type get(lua_State* l, int index)
|
|
{
|
|
size_t size = 0;
|
|
const char* buffer = lua_tolstring(l, index, &size);
|
|
return yasio::sbyte_buffer(buffer, buffer + size);
|
|
}
|
|
static int push(lua_State* l, push_type s)
|
|
{
|
|
lua_pushlstring(l, s.data(), s.size());
|
|
return 1;
|
|
}
|
|
};
|
|
// transport_handle_t
|
|
template <>
|
|
struct lua_type_traits<yasio::inet::transport_handle_t> {
|
|
typedef yasio::inet::transport_handle_t get_type;
|
|
typedef yasio::inet::transport_handle_t push_type;
|
|
|
|
static bool strictCheckType(lua_State* l, int index) { return lua_type(l, index) == LUA_TLIGHTUSERDATA; }
|
|
static bool checkType(lua_State* l, int index) { return lua_islightuserdata(l, index) != 0; }
|
|
static get_type get(lua_State* l, int index) { return reinterpret_cast<get_type>(lua_touserdata(l, index)); }
|
|
static int push(lua_State* l, push_type s)
|
|
{
|
|
lua_pushlightuserdata(l, s);
|
|
return 1;
|
|
}
|
|
};
|
|
|
|
template <>
|
|
struct lua_type_traits<std::vector<yasio::inet::io_hostent>> {
|
|
typedef std::vector<yasio::inet::io_hostent> get_type;
|
|
typedef std::vector<yasio::inet::io_hostent> push_type;
|
|
|
|
static bool strictCheckType(lua_State* l, int index) { return lua_type(l, index) == LUA_TTABLE; }
|
|
static bool checkType(lua_State* l, int index) { return lua_istable(l, index) != 0; }
|
|
static get_type get(lua_State* l, int index)
|
|
{
|
|
lua_pushvalue(l, index);
|
|
|
|
kaguya::LuaTable channel_eps(l, kaguya::StackTop{});
|
|
std::vector<yasio::inet::io_hostent> hosts;
|
|
auto host = channel_eps["host"];
|
|
if (host)
|
|
hosts.push_back(yasio::inet::io_hostent(host, channel_eps["port"]));
|
|
else
|
|
{
|
|
channel_eps.foreach_table<int, kaguya::LuaTable>([&](int, kaguya::LuaTable ep) { hosts.push_back(yasio::inet::io_hostent(ep["host"], ep["port"])); });
|
|
}
|
|
|
|
lua_pop(l, 1);
|
|
|
|
return hosts;
|
|
}
|
|
static int push(lua_State* l, push_type s)
|
|
{ // don't need push yasio::inet::io_hostent to lua
|
|
assert(false);
|
|
return 1;
|
|
}
|
|
};
|
|
# if defined(YASIO_ENABLE_HALF_FLOAT)
|
|
template <>
|
|
struct lua_type_traits<fp16_t> {
|
|
typedef fp16_t get_type;
|
|
typedef fp16_t push_type;
|
|
|
|
static bool strictCheckType(lua_State* l, int index) { return lua_type(l, index) == LUA_TNUMBER; }
|
|
static bool checkType(lua_State* l, int index) { return lua_isnumber(l, index) != 0; }
|
|
static get_type get(lua_State* l, int index) { return fp16_t{static_cast<float>(lua_tonumber(l, index))}; }
|
|
static int push(lua_State* l, push_type value)
|
|
{
|
|
lua_pushnumber(l, static_cast<float>(value));
|
|
return 1;
|
|
}
|
|
};
|
|
# endif
|
|
}; // namespace kaguya
|
|
|
|
namespace lyasio
|
|
{
|
|
# define kaguya_obstream_class(_Stream, _BaseStream) kaguya::UserdataMetatable<_Stream, _BaseStream>().setConstructors<_Stream(), _Stream(size_t)>()
|
|
# define kaguya_obstream_base_class(_BaseStream) kaguya::UserdataMetatable<_BaseStream>().setConstructors<_BaseStream(_BaseStream::buffer_type*)>()
|
|
# define kaguya_ibstream_view_class(_StreamView, _OStream) \
|
|
kaguya::UserdataMetatable<_StreamView>().setConstructors<_StreamView(), _StreamView(const void*, size_t), _StreamView(const _OStream*)>()
|
|
# define kaguya_ibstream_class(_Stream, _StreamView, _OStream) \
|
|
kaguya::UserdataMetatable<_Stream, _StreamView>().setConstructors<_Stream(yasio::sbyte_buffer), _Stream(const _OStream*)>()
|
|
|
|
template <typename _Stream, typename _BaseStream>
|
|
static void register_obstream(kaguya::LuaTable& lib, const char* usertype, const char* basetype, kaguya::UserdataMetatable<_Stream, _BaseStream>& userclass,
|
|
kaguya::UserdataMetatable<_BaseStream>& baseclass)
|
|
{
|
|
lib[basetype].setClass(
|
|
# if defined(YASIO_OBS_BUILTIN_STACK)
|
|
userclass.addFunction("push32", &_BaseStream::push32)
|
|
.addOverloadedFunctions("pop32", static_cast<void (_BaseStream ::*)()>(&_BaseStream::pop32),
|
|
static_cast<void (_BaseStream ::*)(uint32_t)>(&_Stream::pop32))
|
|
.addFunction("push16", &_BaseStream::push16)
|
|
.addOverloadedFunctions("pop16", static_cast<void (_BaseStream ::*)()>(&_BaseStream::pop16),
|
|
static_cast<void (_BaseStream ::*)(uint16_t)>(&_BaseStream::pop16))
|
|
.addFunction("push8", &_BaseStream::push8)
|
|
.addOverloadedFunctions("pop8", static_cast<void (_BaseStream ::*)()>(&_BaseStream::pop8),
|
|
static_cast<void (_BaseStream ::*)(uint8_t)>(&_BaseStream::pop8))
|
|
# else
|
|
baseclass.addFunction("push32", &_BaseStream::template push<uint32_t>)
|
|
.addOverloadedFunctions("pop32", static_cast<void (_BaseStream ::*)(size_t)>(&_BaseStream::template pop<uint32_t>),
|
|
static_cast<void (_BaseStream ::*)(size_t, uint32_t)>(&_BaseStream::template pop<uint32_t>))
|
|
.addFunction("push16", &_BaseStream::template push<uint16_t>)
|
|
.addOverloadedFunctions("pop16", static_cast<void (_BaseStream ::*)(size_t)>(&_BaseStream::template pop<uint16_t>),
|
|
static_cast<void (_BaseStream ::*)(size_t, uint16_t)>(&_BaseStream::template pop<uint16_t>))
|
|
.addFunction("push8", &_BaseStream::template push<uint8_t>)
|
|
.addOverloadedFunctions("pop8", static_cast<void (_BaseStream ::*)(size_t)>(&_BaseStream::template pop<uint8_t>),
|
|
static_cast<void (_BaseStream ::*)(size_t, uint8_t)>(&_BaseStream::template pop<uint8_t>))
|
|
# endif
|
|
.addFunction("write_ix", &_BaseStream::template write_ix<int64_t>)
|
|
.addFunction("write_bool", &_BaseStream::template write<bool>)
|
|
.addFunction("write_i8", &_BaseStream::template write<int8_t>)
|
|
.addFunction("write_i16", &_BaseStream::template write<int16_t>)
|
|
.addFunction("write_i32", &_BaseStream::template write<int32_t>)
|
|
.addFunction("write_i64", &_BaseStream::template write<int64_t>)
|
|
.addFunction("write_u8", &_BaseStream::template write<uint8_t>)
|
|
.addFunction("write_u16", &_BaseStream::template write<uint16_t>)
|
|
.addFunction("write_u32", &_BaseStream::template write<uint32_t>)
|
|
.addFunction("write_u64", &_BaseStream::template write<uint64_t>)
|
|
# if defined(YASIO_ENABLE_HALF_FLOAT)
|
|
.addFunction("write_f16", &_BaseStream::template write<fp16_t>)
|
|
# endif
|
|
.addFunction("write_f", &_BaseStream::template write<float>)
|
|
.addFunction("write_lf", &_BaseStream::template write<double>)
|
|
.addStaticFunction("write_v",
|
|
[](_BaseStream* obs, cxx17::string_view sv, kaguya::VariadicArgType args) {
|
|
int lfl = -1;
|
|
if (args.size() > 0)
|
|
lfl = static_cast<int>(args[0]);
|
|
return lyasio::obstream_write_v(obs, sv, lfl);
|
|
})
|
|
.addFunction("write_bytes", static_cast<void (_BaseStream::*)(cxx17::string_view)>(&_BaseStream::write_bytes))
|
|
.addFunction("length", &_BaseStream::length)
|
|
.addStaticFunction("to_string", [](_BaseStream* obs) { return cxx17::string_view(obs->data(), obs->length()); }));
|
|
// ##-- obstream
|
|
lib[usertype].setClass(userclass);
|
|
}
|
|
template <typename _Stream, typename _StreamView, typename _OStream>
|
|
static void register_ibstream(kaguya::LuaTable& lib, const char* usertype, const char* basetype, kaguya::UserdataMetatable<_Stream, _StreamView>& userclass,
|
|
kaguya::UserdataMetatable<_StreamView>& baseclass)
|
|
{
|
|
lib[basetype].setClass(baseclass.addFunction("read_ix", &_StreamView::template read_ix<int64_t>)
|
|
.addFunction("read_bool", &_StreamView::template read<bool>)
|
|
.addFunction("read_i8", &_StreamView::template read<int8_t>)
|
|
.addFunction("read_i16", &_StreamView::template read<int16_t>)
|
|
.addFunction("read_i32", &_StreamView::template read<int32_t>)
|
|
.addFunction("read_i64", &_StreamView::template read<int64_t>)
|
|
.addFunction("read_u8", &_StreamView::template read<uint8_t>)
|
|
.addFunction("read_u16", &_StreamView::template read<uint16_t>)
|
|
.addFunction("read_u32", &_StreamView::template read<uint32_t>)
|
|
.addFunction("read_u64", &_StreamView::template read<uint64_t>)
|
|
# if defined(YASIO_ENABLE_HALF_FLOAT)
|
|
.addFunction("read_f16", &_Stream::template read<fp16_t>)
|
|
# endif
|
|
.addFunction("read_f", &_StreamView::template read<float>)
|
|
.addFunction("read_lf", &_StreamView::template read<double>)
|
|
.addStaticFunction("read_v",
|
|
[](_StreamView* ibs, kaguya::VariadicArgType args) {
|
|
int length_field_bits = -1;
|
|
if (args.size() > 0)
|
|
length_field_bits = static_cast<int>(args[0]);
|
|
return lyasio::ibstream_read_v(ibs, length_field_bits);
|
|
})
|
|
.addFunction("read_bytes", static_cast<cxx17::string_view (_StreamView::*)(int)>(&_StreamView::read_bytes))
|
|
.addFunction("seek", &_StreamView::seek)
|
|
.addFunction("length", &_StreamView::length)
|
|
.addStaticFunction("to_string", [](_StreamView* ibs) { return cxx17::string_view(ibs->data(), ibs->length()); }));
|
|
|
|
// ##-- ibstream
|
|
lib[usertype].setClass(userclass);
|
|
}
|
|
} // namespace lyasio
|
|
|
|
extern "C" {
|
|
|
|
YASIO_LUA_API int luaopen_yasio(lua_State* L)
|
|
{
|
|
kaguya::State state(L);
|
|
|
|
auto yasio_lib = state.newTable();
|
|
# if YASIO_LUA_ENABLE_GLOBAL
|
|
state["yasio"] = yasio_lib;
|
|
# endif
|
|
|
|
yasio_lib["io_event"].setClass(kaguya::UserdataMetatable<io_event>()
|
|
|
|
.addFunction("kind", &io_event::kind)
|
|
.addFunction("status", &io_event::status)
|
|
.addStaticFunction("passive", [](io_event* ev) { return !!ev->passive(); })
|
|
.addStaticFunction("default_packet",
|
|
[](io_event* ev) {
|
|
auto& pkt = ev->packet();
|
|
if (is_packet_empty(pkt))
|
|
return std::unique_ptr<yasio::ibstream>{};
|
|
return cxx14::make_unique<yasio::ibstream>(forward_packet((packet_t &&) pkt));
|
|
})
|
|
.addStaticFunction("raw_packet",
|
|
[](io_event* ev) {
|
|
auto& pkt = ev->packet();
|
|
if (is_packet_empty(pkt))
|
|
return cxx17::string_view{""};
|
|
return cxx17::string_view{packet_data(pkt), packet_len(pkt)};
|
|
})
|
|
.addStaticFunction("fast_packet",
|
|
[](io_event* ev) {
|
|
auto& pkt = ev->packet();
|
|
if (is_packet_empty(pkt))
|
|
return std::unique_ptr<yasio::fast_ibstream>{};
|
|
return cxx14::make_unique<yasio::fast_ibstream>(forward_packet((packet_t &&) pkt));
|
|
})
|
|
.addFunction("cindex", &io_event::cindex)
|
|
.addFunction("transport", &io_event::transport)
|
|
.addFunction("timestamp", &io_event::timestamp));
|
|
# if !YASIO_LUA_ENABLE_GLOBAL
|
|
state["yasio"] = yasio_lib;
|
|
# endif
|
|
bool succeed = state.dostring(R"(
|
|
local yasio = yasio;
|
|
yasio.io_event.packet = function(self, buffer_type)
|
|
if buffer_type == yasio.BUFFER_RAW then
|
|
return yasio.io_event.raw_packet(self)
|
|
elseif buffer_type == yasio.BUFFER_FAST then
|
|
return yasio.io_event.fast_packet(self)
|
|
else
|
|
return yasio.io_event.default_packet(self)
|
|
end
|
|
end
|
|
)");
|
|
assert(succeed);
|
|
# if !YASIO_LUA_ENABLE_GLOBAL
|
|
state["yasio"] = kaguya::NilValue{};
|
|
# endif
|
|
yasio_lib["io_service"].setClass(
|
|
kaguya::UserdataMetatable<io_service>()
|
|
.setConstructors<io_service(), io_service(int), io_service(const std::vector<io_hostent>&)>()
|
|
.addStaticFunction("start",
|
|
[](io_service* service, kaguya::LuaFunction cb) {
|
|
io_event_cb_t fnwrap = [=](event_ptr e) mutable -> void { cb(e.get()); };
|
|
service->set_option(YOPT_S_NO_DISPATCH, 1); // script doesn't support handle event at network thread
|
|
service->start(std::move(fnwrap));
|
|
})
|
|
.addFunction("stop", &io_service::stop)
|
|
.addFunction("dispatch", &io_service::dispatch)
|
|
.addFunction("open", &io_service::open)
|
|
.addOverloadedFunctions("is_open", static_cast<bool (io_service::*)(int) const>(&io_service::is_open),
|
|
static_cast<bool (io_service::*)(transport_handle_t) const>(&io_service::is_open))
|
|
.addOverloadedFunctions("close", static_cast<void (io_service::*)(transport_handle_t)>(&io_service::close),
|
|
static_cast<void (io_service::*)(int)>(&io_service::close))
|
|
.addOverloadedFunctions(
|
|
"write",
|
|
[](io_service* service, transport_handle_t transport, cxx17::string_view s) {
|
|
return service->write(transport, yasio::sbyte_buffer(s.data(), s.data() + s.length()));
|
|
},
|
|
[](io_service* service, transport_handle_t transport, yasio::obstream* obs) { return service->write(transport, std::move(obs->buffer())); })
|
|
.addOverloadedFunctions(
|
|
"write_to",
|
|
[](io_service* service, transport_handle_t transport, cxx17::string_view s, cxx17::string_view ip, u_short port) {
|
|
return service->write_to(transport, yasio::sbyte_buffer(s.data(), s.data() + s.length()), ip::endpoint{ip.data(), port});
|
|
},
|
|
[](io_service* service, transport_handle_t transport, yasio::obstream* obs, cxx17::string_view ip, u_short port) {
|
|
return service->write_to(transport, std::move(obs->buffer()), ip::endpoint{ip.data(), port});
|
|
})
|
|
.addStaticFunction("set_option",
|
|
[](io_service* service, int opt, kaguya::VariadicArgType args) {
|
|
switch (opt)
|
|
{
|
|
case YOPT_C_LOCAL_HOST:
|
|
case YOPT_C_REMOTE_HOST:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<const char*>(args[1]));
|
|
break;
|
|
|
|
case YOPT_C_UNPACK_STRIP:
|
|
case YOPT_C_LOCAL_PORT:
|
|
case YOPT_C_REMOTE_PORT:
|
|
case YOPT_C_KCP_CONV:
|
|
case YOPT_C_UNPACK_NO_BSWAP:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]));
|
|
break;
|
|
case YOPT_C_ENABLE_MCAST:
|
|
case YOPT_C_LOCAL_ENDPOINT:
|
|
case YOPT_C_REMOTE_ENDPOINT:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<const char*>(args[1]), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_C_MOD_FLAGS:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_S_TCP_KEEPALIVE:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]));
|
|
break;
|
|
case YOPT_C_UNPACK_PARAMS:
|
|
service->set_option(opt, static_cast<int>(args[0]), static_cast<int>(args[1]), static_cast<int>(args[2]),
|
|
static_cast<int>(args[3]), static_cast<int>(args[4]));
|
|
break;
|
|
case YOPT_S_EVENT_CB:
|
|
(void)0;
|
|
{
|
|
kaguya::LuaFunction fn = args[0];
|
|
io_event_cb_t fnwrap = [=](event_ptr e) mutable -> void { fn(e.get()); };
|
|
service->set_option(opt, std::addressof(fnwrap));
|
|
}
|
|
break;
|
|
default:
|
|
service->set_option(opt, static_cast<int>(args[0]));
|
|
}
|
|
})
|
|
.addStaticFunction("native_ptr", [](io_service* service) { return (void*)service; }));
|
|
|
|
// ##-- obstream
|
|
lyasio::register_obstream<obstream, obstream::super_type>(yasio_lib, "obstream", "obstream_base", kaguya_obstream_class(obstream, obstream::super_type),
|
|
kaguya_obstream_base_class(obstream::super_type));
|
|
lyasio::register_obstream<fast_obstream, fast_obstream::super_type>(yasio_lib, "fast_obstream", "fast_obstream_base",
|
|
kaguya_obstream_class(fast_obstream, fast_obstream::super_type),
|
|
kaguya_obstream_base_class(fast_obstream::super_type));
|
|
|
|
// ##-- yasio::ibstream
|
|
lyasio::register_ibstream<ibstream, ibstream_view, obstream>(yasio_lib, "ibstream", "ibstream_view", kaguya_ibstream_class(ibstream, ibstream_view, obstream),
|
|
kaguya_ibstream_view_class(ibstream_view, obstream));
|
|
lyasio::register_ibstream<fast_ibstream, fast_ibstream_view, fast_obstream>(yasio_lib, "fast_ibstream", "fast_ibstream_view",
|
|
kaguya_ibstream_class(fast_ibstream, fast_ibstream_view, fast_obstream),
|
|
kaguya_ibstream_view_class(fast_ibstream_view, fast_obstream));
|
|
|
|
yasio_lib.setField("highp_clock", &highp_clock<yasio::steady_clock_t>);
|
|
yasio_lib.setField("highp_time", &highp_clock<yasio::system_clock_t>);
|
|
|
|
yasio_lib.setField("unwrap_ptr", [](lua_State* L) -> int {
|
|
auto& pkt = *(packet_t*)lua_touserdata(L, 1);
|
|
int offset = lua_isinteger(L, 2) ? static_cast<int>(lua_tointeger(L, 2)) : 0;
|
|
lua_pushlightuserdata(L, packet_data(pkt) + offset);
|
|
return 1;
|
|
});
|
|
|
|
yasio_lib.setField("unwrap_len", [](lua_State* L) -> int {
|
|
auto& pkt = *(packet_t*)lua_touserdata(L, 1);
|
|
int offset = lua_isinteger(L, 2) ? static_cast<int>(lua_tointeger(L, 2)) : 0;
|
|
lua_pushinteger(L, packet_len(pkt) - offset);
|
|
return 1;
|
|
});
|
|
|
|
// ##-- yasio enums
|
|
# define YASIO_EXPORT_ANY(v) yasio_lib[#v] = v
|
|
YASIO_EXPORT_ANY(YCK_TCP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_TCP_SERVER);
|
|
YASIO_EXPORT_ANY(YCK_UDP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_UDP_SERVER);
|
|
# if defined(YASIO_ENABLE_KCP)
|
|
YASIO_EXPORT_ANY(YCK_KCP_CLIENT);
|
|
YASIO_EXPORT_ANY(YCK_KCP_SERVER);
|
|
# endif
|
|
# if YASIO_SSL_BACKEND
|
|
YASIO_EXPORT_ANY(YCK_SSL_CLIENT);
|
|
# endif
|
|
|
|
YASIO_EXPORT_ANY(YOPT_S_CONNECT_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_DNS_CACHE_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_DNS_QUERIES_TIMEOUT);
|
|
YASIO_EXPORT_ANY(YOPT_S_TCP_KEEPALIVE);
|
|
YASIO_EXPORT_ANY(YOPT_S_EVENT_CB);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_PARAMS);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_STRIP);
|
|
YASIO_EXPORT_ANY(YOPT_C_UNPACK_NO_BSWAP);
|
|
YASIO_EXPORT_ANY(YOPT_C_LFBFD_PARAMS); // alias for YOPT_C_UNPACK_PARAMS
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_HOST);
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_PORT);
|
|
YASIO_EXPORT_ANY(YOPT_C_LOCAL_ENDPOINT);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_HOST);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_PORT);
|
|
YASIO_EXPORT_ANY(YOPT_C_REMOTE_ENDPOINT);
|
|
YASIO_EXPORT_ANY(YOPT_C_ENABLE_MCAST);
|
|
YASIO_EXPORT_ANY(YOPT_C_DISABLE_MCAST);
|
|
YASIO_EXPORT_ANY(YOPT_C_KCP_CONV);
|
|
YASIO_EXPORT_ANY(YOPT_C_MOD_FLAGS);
|
|
|
|
YASIO_EXPORT_ANY(YCF_REUSEADDR);
|
|
YASIO_EXPORT_ANY(YCF_EXCLUSIVEADDRUSE);
|
|
|
|
YASIO_EXPORT_ANY(YEK_ON_OPEN);
|
|
YASIO_EXPORT_ANY(YEK_ON_CLOSE);
|
|
YASIO_EXPORT_ANY(YEK_ON_PACKET);
|
|
YASIO_EXPORT_ANY(YEK_CONNECT_RESPONSE);
|
|
YASIO_EXPORT_ANY(YEK_CONNECTION_LOST);
|
|
YASIO_EXPORT_ANY(YEK_PACKET);
|
|
|
|
YASIO_EXPORT_ANY(SEEK_CUR);
|
|
YASIO_EXPORT_ANY(SEEK_SET);
|
|
YASIO_EXPORT_ANY(SEEK_END);
|
|
|
|
using namespace lyasio;
|
|
YASIO_EXPORT_ANY(BUFFER_DEFAULT);
|
|
YASIO_EXPORT_ANY(BUFFER_RAW);
|
|
YASIO_EXPORT_ANY(BUFFER_FAST);
|
|
|
|
char version[32];
|
|
snprintf(version, sizeof(version), "%x.%x.%x", (YASIO_VERSION_NUM >> 16) & 0xff, (YASIO_VERSION_NUM >> 8) & 0xff, YASIO_VERSION_NUM & 0xff);
|
|
YASIO_EXPORT_ANY(version);
|
|
|
|
return yasio_lib.push(); /* return 'yasio' table */
|
|
}
|
|
|
|
} /* extern "C" */
|
|
|
|
#endif /* YASIO__HAS_CXX17 */
|
|
|
|
extern "C" {
|
|
YASIO_LUA_API void luaregister_yasio(lua_State* L)
|
|
{
|
|
lua_getglobal(L, "package");
|
|
lua_getfield(L, -1, "preload");
|
|
|
|
lua_pushcfunction(L, luaopen_yasio);
|
|
lua_setfield(L, -2, "yasio");
|
|
|
|
lua_pop(L, 2);
|
|
}
|
|
}
|