byedpi/extend.c

638 lines
16 KiB
C
Raw Permalink Normal View History

2024-08-07 11:25:26 +00:00
#include "extend.h"
2024-04-23 05:47:27 +00:00
#ifdef _WIN32
#include <ws2tcpip.h>
#ifndef TCP_MAXRT
#define TCP_MAXRT 5
#endif
#else
#include <arpa/inet.h>
#include <netinet/tcp.h>
2024-04-27 23:50:46 +00:00
#include <sys/un.h>
#include <sys/time.h>
2024-04-23 05:47:27 +00:00
#endif
#include <string.h>
2024-05-08 17:15:57 +00:00
#include <assert.h>
2024-04-23 05:47:27 +00:00
2024-05-02 16:36:29 +00:00
#include "proxy.h"
#include "error.h"
#include "params.h"
2024-04-23 05:47:27 +00:00
2024-05-02 18:23:14 +00:00
#include "desync.h"
#include "packets.h"
2024-04-23 05:47:27 +00:00
#define KEY_SIZE sizeof(struct sockaddr_ina)
2024-04-23 05:47:27 +00:00
2024-10-20 15:50:08 +00:00
static int set_timeout(int fd, unsigned int s)
2024-04-23 05:47:27 +00:00
{
#ifdef __linux__
if (setsockopt(fd, IPPROTO_TCP,
TCP_USER_TIMEOUT, (char *)&s, sizeof(s))) {
uniperror("setsockopt TCP_USER_TIMEOUT");
return -1;
}
#else
#ifdef _WIN32
if (setsockopt(fd, IPPROTO_TCP,
TCP_MAXRT, (char *)&s, sizeof(s))) {
uniperror("setsockopt TCP_MAXRT");
return -1;
}
#endif
#endif
return 0;
}
static ssize_t serialize_addr(const struct sockaddr_ina *dst,
uint8_t *const out, const size_t out_len)
{
#define serialize(raw, field, len, counter){ \
const size_t size = sizeof(field); \
if ((counter + size) <= len) { \
memcpy(raw + counter, &(field), size); \
counter += size; \
} else return 0; \
}
size_t c = 0;
serialize(out, dst->in.sin_port, out_len, c);
serialize(out, dst->sa.sa_family, out_len, c);
if (dst->sa.sa_family == AF_INET) {
serialize(out, dst->in.sin_addr, out_len, c);
} else {
serialize(out, dst->in6.sin6_addr, out_len, c);
}
#undef serialize
return c;
}
2024-11-09 15:07:27 +00:00
static int cache_get(const struct sockaddr_ina *dst)
2024-04-23 05:47:27 +00:00
{
2024-10-20 15:50:08 +00:00
uint8_t key[KEY_SIZE] = { 0 };
int len = serialize_addr(dst, key, sizeof(key));
2024-09-18 12:29:53 +00:00
2024-12-16 16:32:07 +00:00
struct elem_i *val = (struct elem_i *)mem_get(params.mempool, (char *)key, len);
2024-10-20 15:50:08 +00:00
if (!val) {
return -1;
}
time_t t = time(0);
if (t > val->time + params.cache_ttl) {
LOG(LOG_S, "time=%jd, now=%jd, ignore\n", (intmax_t)val->time, (intmax_t)t);
return 0;
}
return val->m;
}
2024-11-09 15:07:27 +00:00
static int cache_add(const struct sockaddr_ina *dst, int m)
2024-10-20 15:50:08 +00:00
{
assert(m >= 0 && m < params.dp_count);
2024-09-18 12:29:53 +00:00
uint8_t key[KEY_SIZE] = { 0 };
int len = serialize_addr(dst, key, sizeof(key));
INIT_ADDR_STR((*dst));
2024-04-23 05:47:27 +00:00
if (m == 0) {
LOG(LOG_S, "delete ip: %s\n", ADDR_STR);
mem_delete(params.mempool, (char *)key, len);
2024-04-23 05:47:27 +00:00
return 0;
}
2024-10-20 15:50:08 +00:00
LOG(LOG_S, "save ip: %s, m=%d\n", ADDR_STR, m);
time_t t = time(0);
2024-12-20 01:15:13 +00:00
char *key_d = malloc(len);
if (!key_d) {
return -1;
}
memcpy(key_d, key, len);
struct elem_i *val = (struct elem_i *)mem_add(params.mempool, key_d, len, sizeof(struct elem_i));
2024-10-20 15:50:08 +00:00
if (!val) {
uniperror("mem_add");
2024-12-20 01:15:13 +00:00
free(key_d);
2024-10-20 15:50:08 +00:00
return -1;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
val->m = m;
val->time = t;
return 0;
2024-04-23 05:47:27 +00:00
}
2024-12-17 02:27:34 +00:00
static bool check_l34(struct desync_params *dp, int st, const struct sockaddr_in6 *dst);
2024-04-23 05:47:27 +00:00
int connect_hook(struct poolhd *pool, struct eval *val,
2024-11-09 15:07:27 +00:00
const struct sockaddr_ina *dst, int next)
2024-04-23 05:47:27 +00:00
{
2024-10-20 15:50:08 +00:00
int m = cache_get(dst);
2024-04-23 05:47:27 +00:00
val->cache = (m == 0);
val->attempt = m < 0 ? 0 : m;
2024-04-23 05:47:27 +00:00
return create_conn(pool, val, dst, next);
2024-04-23 05:47:27 +00:00
}
2024-11-09 15:07:27 +00:00
int socket_mod(int fd)
2024-08-14 12:20:35 +00:00
{
if (params.custom_ttl) {
2024-10-15 18:52:18 +00:00
if (setttl(fd, params.def_ttl) < 0) {
2024-08-14 12:20:35 +00:00
return -1;
}
}
if (params.protect_path) {
return protect(fd, params.protect_path);
}
return 0;
}
2024-10-20 15:50:08 +00:00
static int reconnect(struct poolhd *pool, struct eval *val, int m)
2024-04-23 05:47:27 +00:00
{
2024-10-20 15:50:08 +00:00
assert(val->flag == FLAG_CONN);
2024-04-23 05:47:27 +00:00
struct eval *client = val->pair;
if (create_conn(pool, client,
2024-10-20 15:50:08 +00:00
(struct sockaddr_ina *)&val->in6, EV_FIRST_TUNNEL)) {
2024-04-23 05:47:27 +00:00
return -1;
}
val->pair = 0;
del_event(pool, val);
2024-10-20 15:50:08 +00:00
//client->type = EV_IGNORE;
2024-04-23 05:47:27 +00:00
client->attempt = m;
client->cache = 1;
2024-05-11 14:23:34 +00:00
client->buff.offset = 0;
2024-10-20 15:50:08 +00:00
client->round_sent = 0;
2024-04-23 05:47:27 +00:00
return 0;
}
2024-11-09 15:07:27 +00:00
static bool check_host(
struct mphdr *hosts, const char *buffer, ssize_t n)
2024-04-23 05:47:27 +00:00
{
2024-04-29 20:35:40 +00:00
char *host = 0;
2024-04-23 05:47:27 +00:00
int len;
2024-10-20 15:50:08 +00:00
if (!(len = parse_tls(buffer, n, &host))) {
len = parse_http(buffer, n, &host, 0);
2024-04-23 05:47:27 +00:00
}
2024-05-08 17:15:57 +00:00
assert(len == 0 || host != 0);
2024-07-18 20:59:44 +00:00
if (len <= 0) {
return 0;
}
2024-12-20 01:15:13 +00:00
struct elem *v = mem_get(hosts, host, len);
return v && v->len <= len;
2024-04-23 05:47:27 +00:00
}
2024-12-17 02:27:34 +00:00
static bool check_ip(
struct mphdr *ipset, const struct sockaddr_in6 *addr)
{
const struct sockaddr_ina *dst = (const struct sockaddr_ina *)addr;
int len = sizeof(dst->in.sin_addr);
char *data = (char *)&dst->in.sin_addr;
2024-11-09 23:39:21 +00:00
2024-12-17 02:27:34 +00:00
if (dst->sa.sa_family == AF_INET6) {
len = sizeof(dst->in6.sin6_addr);
data = (char *)&dst->in6.sin6_addr;
}
if (mem_get(ipset, data, len * 8)) {
return 1;
}
return 0;
}
2024-11-09 15:07:27 +00:00
static bool check_proto_tcp(int proto, const char *buffer, ssize_t n)
2024-04-23 05:47:27 +00:00
{
2024-11-09 23:39:21 +00:00
if (!(proto & ~IS_IPV4)) {
2024-07-29 09:26:42 +00:00
return 1;
}
else if ((proto & IS_HTTP) &&
2024-10-20 15:50:08 +00:00
is_http(buffer, n)) {
2024-04-23 05:47:27 +00:00
return 1;
}
2024-07-29 09:26:42 +00:00
else if ((proto & IS_HTTPS) &&
2024-10-20 15:50:08 +00:00
is_tls_chello(buffer, n)) {
2024-04-23 05:47:27 +00:00
return 1;
}
return 0;
}
2024-12-17 02:27:34 +00:00
static bool check_l34(struct desync_params *dp, int st, const struct sockaddr_in6 *dst)
2024-11-09 23:39:21 +00:00
{
2024-12-17 02:27:34 +00:00
if ((dp->proto & IS_UDP) && (st != SOCK_DGRAM)) {
2024-11-09 23:39:21 +00:00
return 0;
}
2024-12-17 02:27:34 +00:00
if (dp->proto & IS_IPV4) {
2024-11-09 23:39:21 +00:00
static const char *pat = "\0\0\0\0\0\0\0\0\0\0\xff\xff";
if (dst->sin6_family != AF_INET
&& memcmp(&dst->sin6_addr, pat, 12)) {
return 0;
}
}
2024-12-17 02:27:34 +00:00
if (dp->pf[0] &&
(dst->sin6_port < dp->pf[0] || dst->sin6_port > dp->pf[1])) {
return 0;
}
if (dp->ipset && !check_ip(dp->ipset, dst)) {
2024-11-09 23:39:21 +00:00
return 0;
}
return 1;
}
2024-11-09 15:07:27 +00:00
static bool check_round(const int *nr, int r)
2024-10-20 15:50:08 +00:00
{
return (!nr[1] && r <= 1) || (r >= nr[0] && r <= nr[1]);
}
static int on_trigger(int type, struct poolhd *pool, struct eval *val)
2024-04-23 05:47:27 +00:00
{
int m = val->pair->attempt + 1;
bool can_reconn = (
val->pair->buff.data && !val->recv_count
2024-10-20 15:50:08 +00:00
&& params.auto_level > AUTO_NOBUFF
);
2024-10-20 15:50:08 +00:00
if (!can_reconn && params.auto_level <= AUTO_NOSAVE) {
return -1;
}
for (; m < params.dp_count; m++) {
struct desync_params *dp = &params.dp[m];
if (!dp->detect) {
break;
2024-09-24 01:59:19 +00:00
}
2024-10-20 15:50:08 +00:00
if (!(dp->detect & type)) {
continue;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
if (can_reconn) {
return reconnect(pool, val, m);
2024-09-24 01:59:19 +00:00
}
2024-10-20 15:50:08 +00:00
cache_add(
2024-09-24 01:59:19 +00:00
(struct sockaddr_ina *)&val->in6, m);
2024-10-20 15:50:08 +00:00
break;
}
if (m >= params.dp_count && m > 1) {
cache_add(
(struct sockaddr_ina *)&val->in6, 0);
}
return -1;
}
static int on_torst(struct poolhd *pool, struct eval *val)
{
if (on_trigger(DETECT_TORST, pool, val) == 0) {
return 0;
}
struct linger l = { .l_onoff = 1 };
if (setsockopt(val->pair->fd, SOL_SOCKET,
SO_LINGER, (char *)&l, sizeof(l)) < 0) {
uniperror("setsockopt SO_LINGER");
2024-04-23 05:47:27 +00:00
}
return -1;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
static int on_fin(struct poolhd *pool, struct eval *val)
2024-08-19 23:56:10 +00:00
{
2024-10-20 15:50:08 +00:00
if (!(val->pair->mark && val->round_count <= 1)) {
return -1;
}
2024-10-20 15:50:08 +00:00
if (on_trigger(DETECT_TLS_ERR, pool, val) == 0) {
return 0;
2024-08-19 23:56:10 +00:00
}
return -1;
}
2024-10-20 15:50:08 +00:00
static int on_response(struct poolhd *pool, struct eval *val,
2024-11-09 15:07:27 +00:00
const char *resp, ssize_t sn)
2024-04-23 05:47:27 +00:00
{
int m = val->pair->attempt + 1;
char *req = val->pair->buff.data;
ssize_t qn = val->pair->buff.size;
for (; m < params.dp_count; m++) {
struct desync_params *dp = &params.dp[m];
if (!dp->detect) {
return -1;
2024-04-23 05:47:27 +00:00
}
if ((dp->detect & DETECT_HTTP_LOCAT)
&& is_http_redirect(req, qn, resp, sn)) {
2024-04-23 05:47:27 +00:00
break;
}
2024-08-13 14:00:03 +00:00
else if ((dp->detect & DETECT_TLS_ERR)
&& ((is_tls_chello(req, qn) && !is_tls_shello(resp, sn))
|| neq_tls_sid(req, qn, resp, sn))) {
break;
}
2024-04-23 05:47:27 +00:00
}
if (m < params.dp_count) {
return reconnect(pool, val, m);
}
return -1;
}
2024-10-20 15:50:08 +00:00
static inline void free_first_req(struct eval *client)
2024-05-11 14:23:34 +00:00
{
client->type = EV_TUNNEL;
2024-10-20 15:50:08 +00:00
client->pair->type = EV_TUNNEL;
2024-05-11 14:23:34 +00:00
free(client->buff.data);
2024-10-20 15:50:08 +00:00
memset(&client->buff, 0, sizeof(client->buff));
2024-05-11 14:23:34 +00:00
}
2024-11-09 15:07:27 +00:00
static int setup_conn(struct eval *client, const char *buffer, ssize_t n)
2024-04-23 05:47:27 +00:00
{
2024-10-20 15:50:08 +00:00
int m = client->attempt;
if (!m) for (; m < params.dp_count; m++) {
struct desync_params *dp = &params.dp[m];
2024-11-09 23:39:21 +00:00
if (!dp->detect
2024-12-17 02:27:34 +00:00
&& check_l34(dp, SOCK_STREAM, &client->pair->in6)
&& check_proto_tcp(dp->proto, buffer, n)
2024-11-09 23:39:21 +00:00
&& (!dp->hosts || check_host(dp->hosts, buffer, n))) {
2024-10-20 15:50:08 +00:00
break;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
}
if (m >= params.dp_count) {
LOG(LOG_E, "drop connection (m=%d)\n", m);
return -1;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
if (params.auto_level > AUTO_NOBUFF && params.dp_count > 1) {
client->mark = is_tls_chello(buffer, n);
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
client->attempt = m;
2024-04-23 05:47:27 +00:00
2024-10-20 15:50:08 +00:00
if (params.timeout
&& set_timeout(client->pair->fd, params.timeout)) {
2024-04-23 05:47:27 +00:00
return -1;
}
2024-10-20 15:50:08 +00:00
return 0;
}
static int cancel_setup(struct eval *remote)
{
if (params.timeout && params.auto_level <= AUTO_NOSAVE &&
set_timeout(remote->fd, 0)) {
return -1;
}
2024-10-20 15:50:08 +00:00
if (post_desync(remote->fd, remote->pair->attempt)) {
2024-04-23 05:47:27 +00:00
return -1;
}
2024-10-20 15:50:08 +00:00
return 0;
}
2024-11-09 15:07:27 +00:00
static int send_saved_req(struct poolhd *pool,
2024-10-20 15:50:08 +00:00
struct eval *client, char *buffer, ssize_t bfsize)
{
ssize_t offset = client->buff.offset;
ssize_t n = client->buff.size - offset;
assert(bfsize >= n);
memcpy(buffer, client->buff.data + offset, n);
2024-04-23 05:47:27 +00:00
2024-10-20 15:50:08 +00:00
ssize_t sn = tcp_send_hook(client->pair, buffer, bfsize, n);
if (sn < 0) {
2024-08-09 19:49:44 +00:00
return -1;
}
2024-10-20 15:50:08 +00:00
client->buff.offset += sn;
if (sn < n) {
if (mod_etype(pool, client->pair, POLLOUT) ||
mod_etype(pool, client, 0)) {
uniperror("mod_etype");
return -1;
}
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
return 0;
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
int on_first_tunnel(struct poolhd *pool,
struct eval *val, char *buffer, ssize_t bfsize, int etype)
2024-04-23 05:47:27 +00:00
{
2024-10-20 15:50:08 +00:00
if ((etype & POLLOUT) && val->flag == FLAG_CONN) {
if (mod_etype(pool, val, POLLIN) ||
mod_etype(pool, val->pair, POLLIN)) {
2024-04-23 05:47:27 +00:00
uniperror("mod_etype");
return -1;
}
val->pair->type = EV_FIRST_TUNNEL;
2024-10-20 15:50:08 +00:00
return send_saved_req(pool, val->pair, buffer, bfsize);
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
ssize_t n = tcp_recv_hook(pool, val, buffer, bfsize);
if (n < 1) {
return n;
}
if (val->flag != FLAG_CONN) {
val->buff.size += n;
if (val->buff.size >= bfsize) {
free_first_req(val);
}
else {
val->buff.data = realloc(val->buff.data, val->buff.size);
if (val->buff.data == 0) {
uniperror("realloc");
return -1;
}
memcpy(val->buff.data + val->buff.size - n, buffer, n);
return send_saved_req(pool, val, buffer, bfsize);
}
2024-04-23 05:47:27 +00:00
}
2024-10-20 15:50:08 +00:00
else {
if (on_response(pool, val, buffer, n) == 0) {
return 0;
}
free_first_req(val->pair);
int m = val->pair->attempt;
if (val->pair->cache &&
cache_add((struct sockaddr_ina *)&val->in6, m) < 0) {
2024-04-23 05:47:27 +00:00
return -1;
}
}
2024-10-20 15:50:08 +00:00
if (tcp_send_hook(val->pair, buffer, bfsize, n) < n) {
return -1;
}
2024-04-23 05:47:27 +00:00
return 0;
}
2024-04-27 23:50:46 +00:00
2024-05-11 14:23:34 +00:00
2024-10-20 15:50:08 +00:00
ssize_t tcp_send_hook(struct eval *remote,
char *buffer, size_t bfsize, ssize_t n)
2024-05-11 14:23:34 +00:00
{
2024-10-20 15:50:08 +00:00
ssize_t sn = -1;
int skip = remote->flag != FLAG_CONN;
if (!skip) {
struct eval *client = remote->pair;
if (client->recv_count == n
&& setup_conn(client, buffer, n) < 0) {
return -1;
}
int m = client->attempt, r = client->round_count;
if (!check_round(params.dp[m].rounds, r)) {
skip = 1;
}
else {
2024-11-09 13:20:13 +00:00
LOG(LOG_S, "desync TCP: group=%d, round=%d, fd=%d\n", m, r, remote->fd);
2024-10-20 15:50:08 +00:00
ssize_t offset = remote->pair->round_sent;
if (!offset && remote->round_count) offset = -1;
sn = desync(remote->fd, buffer, bfsize, n,
offset, (struct sockaddr *)&remote->in6, m);
}
2024-05-11 14:23:34 +00:00
}
2024-10-20 15:50:08 +00:00
if (skip) {
sn = send(remote->fd, buffer, n, 0);
if (sn < 0 && get_e() == EAGAIN) {
return 0;
}
2024-05-11 14:23:34 +00:00
}
2024-10-20 15:50:08 +00:00
remote->pair->round_sent += sn;
return sn;
}
ssize_t tcp_recv_hook(struct poolhd *pool, struct eval *val,
char *buffer, size_t bfsize)
{
ssize_t n = recv(val->fd, buffer, bfsize, 0);
if (n < 1) {
if (!n) {
if (val->flag != FLAG_CONN) {
val = val->pair;
}
return on_fin(pool, val);
}
if (get_e() == EAGAIN) {
return 0;
}
uniperror("recv");
switch (get_e()) {
case ECONNRESET:
case ECONNREFUSED:
case ETIMEDOUT:
if (val->flag == FLAG_CONN)
return on_torst(pool, val);
else
return on_fin(pool, val->pair);
}
2024-05-11 14:23:34 +00:00
return -1;
}
val->recv_count += n;
2024-10-20 15:50:08 +00:00
if (val->round_sent == 0) {
val->round_count++;
val->pair->round_sent = 0;
}
if (val->flag == FLAG_CONN && !val->round_sent) {
int *nr = params.dp[val->pair->attempt].rounds;
if (check_round(nr, val->round_count)
&& !check_round(nr, val->round_count + 1)
&& cancel_setup(val)) {
return -1;
}
2024-05-11 14:23:34 +00:00
}
2024-10-20 15:50:08 +00:00
return n;
2024-05-11 14:23:34 +00:00
}
2024-07-29 09:26:42 +00:00
ssize_t udp_hook(struct eval *val,
char *buffer, size_t bfsize, ssize_t n, struct sockaddr_ina *dst)
{
2024-10-20 15:50:08 +00:00
struct eval *pair = val->pair->pair;
int m = pair->attempt, r = pair->round_count;
if (!m) {
for (; m < params.dp_count; m++) {
struct desync_params *dp = &params.dp[m];
2024-11-09 23:39:21 +00:00
if (!dp->detect
2024-12-17 02:27:34 +00:00
&& check_l34(dp, SOCK_DGRAM, &dst->in6)) {
2024-10-20 15:50:08 +00:00
break;
}
2024-07-29 09:26:42 +00:00
}
2024-10-20 15:50:08 +00:00
if (m >= params.dp_count) {
return -1;
}
pair->attempt = m;
2024-07-29 09:26:42 +00:00
}
2024-10-20 15:50:08 +00:00
if (!check_round(params.dp[m].rounds, r)) {
return send(val->fd, buffer, n, 0);
}
2024-11-09 13:20:13 +00:00
LOG(LOG_S, "desync UDP: group=%d, round=%d, fd=%d\n", m, r, val->fd);
2024-07-29 13:08:35 +00:00
return desync_udp(val->fd, buffer, bfsize, n, &dst->sa, m);
2024-07-29 09:26:42 +00:00
}
2024-04-27 23:50:46 +00:00
#ifdef __linux__
2024-11-09 15:07:27 +00:00
static int protect(int conn_fd, const char *path)
2024-04-27 23:50:46 +00:00
{
struct sockaddr_un sa;
sa.sun_family = AF_UNIX;
strcpy(sa.sun_path, path);
int fd = socket(AF_UNIX, SOCK_STREAM, 0);
if (fd < 0) {
uniperror("socket");
return -1;
}
struct timeval tv = { .tv_sec = 1 };
setsockopt(fd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv));
setsockopt(fd, SOL_SOCKET, SO_SNDTIMEO, &tv, sizeof(tv));
int err = connect(fd, (struct sockaddr *)&sa, sizeof(sa));
if (err) {
uniperror("connect");
close(fd);
return -1;
}
char buf[CMSG_SPACE(sizeof(fd))] = {};
struct iovec io = { .iov_base = "1", .iov_len = 1 };
struct msghdr msg = { .msg_iov = &io };
msg.msg_iovlen = 1;
msg.msg_control = buf;
msg.msg_controllen = sizeof(buf);
struct cmsghdr *cmsg = CMSG_FIRSTHDR(&msg);
cmsg->cmsg_level = SOL_SOCKET;
cmsg->cmsg_type = SCM_RIGHTS;
cmsg->cmsg_len = CMSG_LEN(sizeof(conn_fd));
*((int *)CMSG_DATA(cmsg)) = conn_fd;
msg.msg_controllen = CMSG_SPACE(sizeof(conn_fd));
if (sendmsg(fd, &msg, 0) < 0) {
uniperror("sendmsg");
close(fd);
return -1;
}
if (recv(fd, buf, 1, 0) < 1) {
uniperror("recv");
close(fd);
return -1;
}
close(fd);
return 0;
}
#endif