mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-17 22:12:44 +01:00
581 lines
17 KiB
C
581 lines
17 KiB
C
/*
|
|
* Copyright (C) 2021 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @{
|
|
*
|
|
* @file
|
|
* @author Martine Lenders <m.lenders@fu-berlin.de>
|
|
*/
|
|
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "fmt.h"
|
|
#include "net/af.h"
|
|
#include "net/coap.h"
|
|
#include "net/credman.h"
|
|
#include "net/dns/msg.h"
|
|
#include "net/ipv4/addr.h"
|
|
#include "net/ipv6/addr.h"
|
|
#include "od.h"
|
|
#include "embUnit.h"
|
|
#include "shell.h"
|
|
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
#include "net/sock/dtls.h"
|
|
#endif
|
|
#include "net/gcoap/dns.h"
|
|
|
|
#define PSK_ID_LEN 32U
|
|
#define PSK_LEN 32U
|
|
|
|
#define TEST_TAG 2599U
|
|
#define TEST_PSK_ID "client_identity"
|
|
#define TEST_PSK "secretPSK"
|
|
|
|
static char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
static char _psk_id[PSK_ID_LEN];
|
|
static char _psk[PSK_LEN];
|
|
static credman_credential_t _credential = {
|
|
.type = CREDMAN_TYPE_PSK,
|
|
.params = {
|
|
.psk = {
|
|
.id = { .s = _psk_id, .len = 0U, },
|
|
.key = { .s = _psk, .len = 0U, },
|
|
}
|
|
},
|
|
};
|
|
static uint8_t _mock_response[CONFIG_DNS_MSG_LEN];
|
|
static size_t _mock_response_len = 0U;
|
|
static uint8_t _resp_code = COAP_CODE_EMPTY;
|
|
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
static_assert(CONFIG_GCOAP_DNS_CREDS_MAX == CONFIG_DTLS_CREDENTIALS_MAX,
|
|
"CONFIG_GCOAP_DNS_CREDS_MAX and CONFIG_DTLS_CREDENTIALS_MAX "
|
|
"must be equal for this test");
|
|
#endif
|
|
static_assert(!IS_USED(MODULE_GCOAP_DTLS) ||
|
|
(CONFIG_GCOAP_DNS_CREDS_MAX == CONFIG_CREDMAN_MAX_CREDENTIALS),
|
|
"CONFIG_GCOAP_DNS_CREDS_MAX and CONFIG_CREDMAN_MAX_CREDENTIALS "
|
|
"must be equal for this test");
|
|
|
|
#define INIT_TEST_PSK(t) \
|
|
_credential.type = CREDMAN_TYPE_PSK; \
|
|
_credential.tag = (t); \
|
|
_credential.params.psk.id.len = sizeof(TEST_PSK_ID); \
|
|
strcpy((char *)_credential.params.psk.id.s, TEST_PSK_ID); \
|
|
_credential.params.psk.key.len = sizeof(TEST_PSK); \
|
|
strcpy((char *)_credential.params.psk.key.s, TEST_PSK)
|
|
|
|
static void setup(void)
|
|
{
|
|
gcoap_dns_server_uri_set(NULL);
|
|
gcoap_dns_cred_reset();
|
|
_credential.tag = CREDMAN_TAG_EMPTY;
|
|
_credential.params.psk.id.len = 0U;
|
|
_credential.params.psk.key.len = 0U;
|
|
}
|
|
|
|
static void test_server_uri_set__success_1(void)
|
|
{
|
|
static const char uri[] = "coap://example.org/";
|
|
char res[sizeof(uri)];
|
|
|
|
TEST_ASSERT_EQUAL_INT(strlen(uri), gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT_EQUAL_INT(strlen(uri), gcoap_dns_server_uri_get(res, sizeof(res)));
|
|
TEST_ASSERT_EQUAL_STRING(uri, res);
|
|
TEST_ASSERT(gcoap_dns_server_uri_is_set());
|
|
TEST_ASSERT_EQUAL_INT(0, gcoap_dns_server_uri_set(NULL));
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
}
|
|
|
|
static void test_server_uri_set__success_2(void)
|
|
{
|
|
static const char uri[] = "coaps://example.org/";
|
|
|
|
if (IS_USED(MODULE_GCOAP_DTLS)) {
|
|
char res[sizeof(uri)];
|
|
TEST_ASSERT_EQUAL_INT(strlen(uri), gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT_EQUAL_INT(strlen(uri), gcoap_dns_server_uri_get(res, sizeof(res)));
|
|
TEST_ASSERT_EQUAL_STRING(uri, res);
|
|
TEST_ASSERT(gcoap_dns_server_uri_is_set());
|
|
}
|
|
else {
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
}
|
|
}
|
|
|
|
static void test_server_uri_set__uri_too_long(void)
|
|
{
|
|
static const char uri[] = "coap://a.very.long.host-name.org"
|
|
"/this/is/a/very/long/path/to/dns";
|
|
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
/* would not fit trailing \0 */
|
|
TEST_ASSERT_EQUAL_INT(CONFIG_GCOAP_DNS_SERVER_URI_LEN,
|
|
strlen(uri));
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
TEST_ASSERT_EQUAL_INT(-ENOBUFS, gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
}
|
|
|
|
static void test_server_uri_set__not_coap(void)
|
|
{
|
|
static const char uri[] = "https://example.org/";
|
|
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
}
|
|
|
|
static void test_server_uri_set__no_dns_var(void)
|
|
{
|
|
static const char uri[] = "https://example.org/";
|
|
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
TEST_ASSERT_EQUAL_INT(-EINVAL, gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT(!gcoap_dns_server_uri_is_set());
|
|
}
|
|
|
|
static void test_server_uri_get__buf_too_short(void)
|
|
{
|
|
static const char uri[] = "coap://example.org/";
|
|
char res[sizeof(uri) - 1];
|
|
|
|
TEST_ASSERT_EQUAL_INT(strlen(uri), gcoap_dns_server_uri_set(uri));
|
|
TEST_ASSERT_EQUAL_INT(-ENOBUFS, gcoap_dns_server_uri_get(res, sizeof(res)));
|
|
}
|
|
|
|
static void test_cred_add__success(void)
|
|
{
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
if (IS_USED(MODULE_GCOAP_DTLS)) {
|
|
TEST_ASSERT_EQUAL_INT(0, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
else {
|
|
TEST_ASSERT_EQUAL_INT(-ENOTSUP, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
}
|
|
|
|
static void test_cred_add__no_mem(void)
|
|
{
|
|
#if IS_USED(MODULE_GCOAP_DTLS)
|
|
for (unsigned i = 0; i < CONFIG_GCOAP_DNS_CREDS_MAX; i++) {
|
|
INIT_TEST_PSK(TEST_TAG + i);
|
|
TEST_ASSERT_EQUAL_INT(0, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
INIT_TEST_PSK(TEST_TAG + CONFIG_GCOAP_DNS_CREDS_MAX);
|
|
TEST_ASSERT_EQUAL_INT(-ENOMEM, gcoap_dns_cred_add(&_credential));
|
|
#else
|
|
TEST_ASSERT_EQUAL_INT(-ENOTSUP, gcoap_dns_cred_add(&_credential));
|
|
#endif
|
|
}
|
|
|
|
static void test_cred_add__credman_error(void)
|
|
{
|
|
if (IS_USED(MODULE_GCOAP_DTLS)) {
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
_credential.type = CREDMAN_TYPE_EMPTY;
|
|
TEST_ASSERT_EQUAL_INT(-EBADF, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
else {
|
|
TEST_ASSERT_EQUAL_INT(-ENOTSUP, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
}
|
|
|
|
static void test_cred_remove__success(void)
|
|
{
|
|
INIT_TEST_PSK(TEST_TAG);
|
|
if (IS_USED(MODULE_GCOAP_DTLS)) {
|
|
TEST_ASSERT_EQUAL_INT(CREDMAN_NOT_FOUND, credman_get(&_credential,
|
|
_credential.tag, _credential.type));
|
|
TEST_ASSERT_EQUAL_INT(0, gcoap_dns_cred_add(&_credential));
|
|
TEST_ASSERT_EQUAL_INT(CREDMAN_OK, credman_get(&_credential,
|
|
_credential.tag, _credential.type));
|
|
gcoap_dns_cred_remove(_credential.tag, _credential.type);
|
|
TEST_ASSERT_EQUAL_INT(CREDMAN_NOT_FOUND, credman_get(&_credential,
|
|
_credential.tag, _credential.type));
|
|
}
|
|
else {
|
|
TEST_ASSERT_EQUAL_INT(-ENOTSUP, gcoap_dns_cred_add(&_credential));
|
|
}
|
|
}
|
|
|
|
static int _unittests(int argc, char **argv)
|
|
{
|
|
(void)argc;
|
|
(void)argv;
|
|
|
|
EMB_UNIT_TESTFIXTURES(fixtures) {
|
|
new_TestFixture(test_server_uri_set__success_1),
|
|
new_TestFixture(test_server_uri_set__success_2),
|
|
new_TestFixture(test_server_uri_set__uri_too_long),
|
|
new_TestFixture(test_server_uri_set__not_coap),
|
|
new_TestFixture(test_server_uri_set__no_dns_var),
|
|
new_TestFixture(test_server_uri_get__buf_too_short),
|
|
new_TestFixture(test_cred_add__success),
|
|
new_TestFixture(test_cred_add__no_mem),
|
|
new_TestFixture(test_cred_add__credman_error),
|
|
new_TestFixture(test_cred_remove__success),
|
|
};
|
|
|
|
EMB_UNIT_TESTCALLER(gcoap_dns_tests, setup, NULL, fixtures);
|
|
TESTS_START();
|
|
TESTS_RUN((Test *)&gcoap_dns_tests);
|
|
TESTS_END();
|
|
puts("Be aware that URI and credentials are now reset.");
|
|
return 0;
|
|
}
|
|
|
|
static ssize_t _mock_dns_server(coap_pkt_t *pdu, uint8_t *buf, size_t len, coap_request_ctx_t *ctx)
|
|
{
|
|
(void)ctx;
|
|
if ((_resp_code >> 5) == COAP_CLASS_SUCCESS) {
|
|
gcoap_resp_init(pdu, buf, len, _resp_code);
|
|
coap_opt_add_format(pdu, COAP_FORMAT_DNS_MESSAGE);
|
|
size_t resp_len = coap_opt_finish(pdu, COAP_OPT_FINISH_PAYLOAD);
|
|
/* write the RIOT board name in the response buffer */
|
|
if (pdu->payload_len >= _mock_response_len) {
|
|
memcpy(pdu->payload, _mock_response, _mock_response_len);
|
|
return resp_len + _mock_response_len;
|
|
}
|
|
else {
|
|
puts("gcoap_cli: msg buffer too small");
|
|
return gcoap_response(pdu, buf, len, COAP_CODE_INTERNAL_SERVER_ERROR);
|
|
}
|
|
}
|
|
else {
|
|
return gcoap_response(pdu, buf, len, _resp_code);
|
|
}
|
|
}
|
|
|
|
static void _uri_usage(const char *cmd)
|
|
{
|
|
printf("usage: %s -d\n", cmd);
|
|
printf(" %s <uri>\n", cmd);
|
|
}
|
|
|
|
static int _set_uri(int argc, char **argv)
|
|
{
|
|
int res;
|
|
|
|
if ((argc > 1) && (strcmp(argv[1], "-d") == 0)) {
|
|
gcoap_dns_server_uri_set(NULL);
|
|
puts("Successfully reset URI\n");
|
|
return 0;
|
|
}
|
|
else if (argc == 1) {
|
|
static char uri[CONFIG_GCOAP_DNS_SERVER_URI_LEN];
|
|
ssize_t res;
|
|
|
|
if ((res = gcoap_dns_server_uri_get(uri, sizeof(uri))) < 0) {
|
|
errno = -res;
|
|
perror("Unable to get URI");
|
|
return 1;
|
|
}
|
|
else {
|
|
if (res > 0) {
|
|
puts(uri);
|
|
return 0;
|
|
}
|
|
else {
|
|
_uri_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
/* argc > 1 can be assumed since argc == 1 returns above */
|
|
res = gcoap_dns_server_uri_set(argv[1]);
|
|
if (res < 0) {
|
|
errno = -res;
|
|
perror("Unable to set URI");
|
|
return errno;
|
|
}
|
|
printf("Successfully added URI %s\n", argv[1]);
|
|
return 0;
|
|
}
|
|
|
|
static void _creds_usage(const char *cmd)
|
|
{
|
|
printf("usage: %s -d <cred_tag>\n", cmd);
|
|
printf(" %s <cred_tag> <psk_id> <psk>\n", cmd);
|
|
}
|
|
|
|
static int _creds(int argc, char **argv)
|
|
{
|
|
if ((argc > 2) && (strcmp(argv[1], "-d") == 0)) {
|
|
credman_tag_t tag = atoi(argv[2]);
|
|
|
|
if (tag == 0) {
|
|
_creds_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
gcoap_dns_cred_remove(tag, CREDMAN_TYPE_PSK);
|
|
printf("Successfully removed credentials with tag %u\n", tag);
|
|
return 0;
|
|
}
|
|
if (argc < 4) {
|
|
_creds_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
if ((_credential.tag = atoi(argv[1])) == 0) {
|
|
_creds_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
if ((_credential.params.psk.id.len = strlen(argv[2])) > PSK_ID_LEN) {
|
|
printf("PSK ID too long (max. %u bytes allowed)\n", PSK_ID_LEN);
|
|
return 1;
|
|
}
|
|
if ((_credential.params.psk.key.len = strlen(argv[3])) > PSK_LEN) {
|
|
printf("PSK too long (max. %u bytes allowed)\n", PSK_LEN);
|
|
return 1;
|
|
}
|
|
_credential.type = CREDMAN_TYPE_PSK;
|
|
strcpy((char *)_credential.params.psk.id.s, argv[2]);
|
|
strcpy((char *)_credential.params.psk.key.s, argv[3]);
|
|
|
|
int res = gcoap_dns_cred_add(&_credential);
|
|
|
|
if (res < 0) {
|
|
errno = -res;
|
|
perror("Unable to add credential");
|
|
return errno;
|
|
}
|
|
printf("Successfully added creds: %u, %s, %s\n", _credential.tag, argv[2], argv[3]);
|
|
return 0;
|
|
}
|
|
|
|
static int _proxy(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
static char proxy[CONFIG_GCOAP_DNS_SERVER_URI_LEN];
|
|
int res;
|
|
|
|
if ((res = gcoap_dns_server_proxy_get(proxy, sizeof(proxy))) < 0) {
|
|
errno = -res;
|
|
perror("Unable to get URI");
|
|
return 1;
|
|
}
|
|
else {
|
|
if (res > 0) {
|
|
puts(proxy);
|
|
return 0;
|
|
}
|
|
else {
|
|
printf("usage: %s [<proxy URI>|-]\n", argv[0]);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
else if (argv[1][0] == '-') {
|
|
gcoap_dns_server_proxy_reset();
|
|
puts("Successfully reset proxy URI\n");
|
|
}
|
|
else {
|
|
int res = gcoap_dns_server_proxy_set(argv[1]);
|
|
|
|
if (res < 0) {
|
|
errno = -res;
|
|
perror("Unable to set proxy URI");
|
|
return errno;
|
|
}
|
|
printf("Successfully added proxy URI %s\n", argv[1]);
|
|
}
|
|
return 0;
|
|
}
|
|
static void _query_usage(const char *cmd)
|
|
{
|
|
printf("usage: %s <hostname> [inet|inet6]\n", cmd);
|
|
}
|
|
|
|
int _parse_af(const char *family_name)
|
|
{
|
|
if (strcmp("inet6", family_name) == 0) {
|
|
return AF_INET6;
|
|
}
|
|
else if (strcmp("inet", family_name) == 0) {
|
|
return AF_INET;
|
|
}
|
|
else {
|
|
printf("Unexpected family %s\n", family_name);
|
|
return AF_UNSPEC;
|
|
}
|
|
}
|
|
|
|
int _print_addr(const char *hostname, const uint8_t *addr, int addr_len)
|
|
{
|
|
char addr_str[IPV6_ADDR_MAX_STR_LEN];
|
|
|
|
switch (addr_len) {
|
|
case sizeof(ipv4_addr_t):
|
|
printf("Hostname %s resolves to %s (IPv4)\n",
|
|
hostname, ipv4_addr_to_str(addr_str, (ipv4_addr_t *)addr,
|
|
sizeof(addr_str)));
|
|
break;
|
|
case sizeof(ipv6_addr_t):
|
|
printf("Hostname %s resolves to %s (IPv6)\n",
|
|
hostname, ipv6_addr_to_str(addr_str, (ipv6_addr_t *)addr,
|
|
sizeof(addr_str)));
|
|
break;
|
|
default:
|
|
printf("Unexpected address format resolved for hostname %s\n",
|
|
hostname);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int _query(int argc, char **argv)
|
|
{
|
|
uint8_t addr_out[sizeof(ipv6_addr_t)];
|
|
const char *hostname;
|
|
int family = AF_INET6;
|
|
int res;
|
|
int offset = 0;
|
|
|
|
if (argc < 2) {
|
|
_query_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
if (argc < (offset + 2)) {
|
|
_query_usage(argv[0]);
|
|
return 1;
|
|
}
|
|
if (argc == (offset + 3)) {
|
|
if ((family = _parse_af(argv[2 + offset])) == AF_UNSPEC) {
|
|
return 1;
|
|
}
|
|
}
|
|
hostname = argv[offset + 1];
|
|
res = gcoap_dns_query(hostname, addr_out, family);
|
|
if (res < 0) {
|
|
errno = -res;
|
|
perror("Unable to resolve query");
|
|
return errno;
|
|
}
|
|
return _print_addr(hostname, addr_out, res);
|
|
}
|
|
|
|
static ssize_t _copy_mock_response(const char *str)
|
|
{
|
|
bool msn = true; /* most significant nibble */
|
|
ssize_t start = _mock_response_len;
|
|
|
|
for (unsigned i = 0; i < strlen(str); i++) {
|
|
uint8_t nibble = scn_u32_hex(&str[i], 1);
|
|
|
|
if (_mock_response_len == CONFIG_DNS_MSG_LEN) {
|
|
break;
|
|
}
|
|
if (msn) {
|
|
_mock_response[_mock_response_len] = nibble << 4;
|
|
}
|
|
else {
|
|
_mock_response[_mock_response_len++] |= nibble;
|
|
}
|
|
msn = !msn;
|
|
}
|
|
if (!msn) {
|
|
_mock_response_len++;
|
|
}
|
|
if (_mock_response_len >= CONFIG_DNS_MSG_LEN) {
|
|
printf("Too many bytes added to response");
|
|
return -ENOBUFS;
|
|
}
|
|
return _mock_response_len - start;
|
|
}
|
|
|
|
static int _resp(int argc, char **argv)
|
|
{
|
|
if (argc < 2) {
|
|
od_hex_dump(_mock_response, _mock_response_len, OD_WIDTH_DEFAULT);
|
|
return 0;
|
|
}
|
|
if (argc > 2 && (strcmp(argv[1], "-c") == 0)) {
|
|
ssize_t size = _copy_mock_response(argv[2]);
|
|
if (size >= 0) {
|
|
printf("Successfully continued response\n");
|
|
od_hex_dump(&_mock_response[_mock_response_len - size], size,
|
|
OD_WIDTH_DEFAULT);
|
|
return 0;
|
|
}
|
|
else {
|
|
return 1;
|
|
}
|
|
}
|
|
_resp_code = atoi(argv[1]);
|
|
if ((argc > 2) && ((_resp_code >> 5) == 2)) {
|
|
_mock_response_len = 0;
|
|
if (_copy_mock_response(argv[2]) < 0) {
|
|
return 1;
|
|
}
|
|
printf("Successfully set response with code %u.%02u\n", _resp_code >> 5, _resp_code & 0x1f);
|
|
od_hex_dump(_mock_response, _mock_response_len, OD_WIDTH_DEFAULT);
|
|
}
|
|
else {
|
|
printf("Successfully set response code %u.%02u\n", _resp_code >> 5, _resp_code & 0x1f);
|
|
_mock_response_len = 0;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int _has_dns_cache(int argc, char **argv)
|
|
{
|
|
(void)argc;
|
|
(void)argv;
|
|
if (IS_USED(MODULE_DNS_CACHE)) {
|
|
puts("DNS cache exists");
|
|
}
|
|
else {
|
|
puts("DNS cache does not exist");
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static const coap_resource_t _resources[] = {
|
|
{ "/", COAP_FETCH, _mock_dns_server, NULL },
|
|
};
|
|
|
|
static gcoap_listener_t _listener = {
|
|
&_resources[0],
|
|
ARRAY_SIZE(_resources),
|
|
GCOAP_SOCKET_TYPE_UNDEF,
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
};
|
|
|
|
static const shell_command_t _shell_commands[] = {
|
|
{ "unittests", "Runs unittests", _unittests},
|
|
{ "uri", "Sets URI to DoC server", _set_uri},
|
|
{ "creds", "Adds/removes credentials for DoC server", _creds},
|
|
{ "proxy", "Sets proxy URI for DoC queries", _proxy},
|
|
{ "query", "Sends DoC query for a hostname", _query},
|
|
{ "resp", "Set static response for mock DoC server", _resp},
|
|
{ "has_dns_cache", "Check if DNS cache is activated", _has_dns_cache},
|
|
{ NULL, NULL, NULL }
|
|
};
|
|
|
|
int main(void)
|
|
{
|
|
gcoap_register_listener(&_listener);
|
|
shell_run(_shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
|
return 0;
|
|
}
|
|
|
|
/** @} */
|