/*
* Copyright (C) 2019 Red Hat, Inc.
*
* Author: Daiki Ueno
*
* This file is part of GnuTLS.
*
* GnuTLS is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* GnuTLS is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this program. If not, see
*/
#ifdef HAVE_CONFIG_H
#include
#endif
#include "gnutls_int.h"
#include "../lib/iov.h"
#include "utils.h"
struct exp_st {
ssize_t ret;
size_t iov_index;
size_t iov_offset;
};
struct test_st {
const char *name;
const giovec_t *iov;
size_t iovcnt;
size_t block_size;
const struct exp_st *exp;
size_t expcnt;
};
static const giovec_t iov16[] = { { (void *)"0123456789012345", 16 },
{ (void *)"0123456789012345", 16 },
{ (void *)"0123456789012345", 16 },
{ (void *)"0123456789012345", 16 } };
static const struct exp_st exp16_64[] = { { 64, 4, 0 }, { 0, 0, 0 } };
static const struct exp_st exp16_32[] = { { 32, 2, 0 },
{ 32, 4, 0 },
{ 0, 0, 0 } };
static const struct exp_st exp16_16[] = { { 16, 1, 0 },
{ 16, 2, 0 },
{ 16, 3, 0 },
{ 16, 4, 0 },
{ 0, 0, 0 } };
static const struct exp_st exp16_4[] = { { 16, 1, 0 },
{ 16, 2, 0 },
{ 16, 3, 0 },
{ 16, 4, 0 },
{ 0, 0, 0 } };
static const struct exp_st exp16_3[] = { { 15, 0, 15 }, { 3, 1, 2 },
{ 12, 1, 14 }, { 3, 2, 1 },
{ 15, 3, 0 }, { 15, 3, 15 },
{ 1, 4, 0 }, { 0, 0, 0 } };
static const giovec_t iov8[] = { { (void *)"01234567", 8 },
{ (void *)"01234567", 8 },
{ (void *)"01234567", 8 },
{ (void *)"01234567", 8 } };
static const struct exp_st exp8_64[] = { { 32, 4, 0 }, { 0, 0, 0 } };
static const giovec_t iov_odd[] = { { (void *)"0", 1 },
{ (void *)"012", 3 },
{ (void *)"01234", 5 },
{ (void *)"0123456", 7 },
{ (void *)"012345678", 9 },
{ (void *)"01234567890", 11 },
{ (void *)"0123456789012", 13 },
{ (void *)"012345678901234", 15 } };
static const struct exp_st exp_odd_16[] = { { 16, 4, 0 },
{ 16, 5, 7 },
{ 16, 6, 12 },
{ 16, 8, 0 },
{ 0, 0, 0 } };
static const giovec_t iov_skip[] = { { (void *)"0123456789012345", 16 },
{ (void *)"01234567", 8 },
{ (void *)"", 0 },
{ (void *)"", 0 },
{ (void *)"0123456789012345", 16 } };
static const struct exp_st exp_skip_16[] = { { 16, 1, 0 },
{ 16, 4, 8 },
{ 8, 5, 0 },
{ 0, 0, 0 } };
static const giovec_t iov_empty[] = { { (void *)"", 0 },
{ (void *)"", 0 },
{ (void *)"", 0 },
{ (void *)"", 0 } };
static const struct exp_st exp_empty_16[] = { { 0, 0, 0 } };
static const struct test_st tests[] = {
{ "16/64", iov16, sizeof(iov16) / sizeof(iov16[0]), 64, exp16_64,
sizeof(exp16_64) / sizeof(exp16_64[0]) },
{ "16/32", iov16, sizeof(iov16) / sizeof(iov16[0]), 32, exp16_32,
sizeof(exp16_32) / sizeof(exp16_32[0]) },
{ "16/16", iov16, sizeof(iov16) / sizeof(iov16[0]), 16, exp16_16,
sizeof(exp16_16) / sizeof(exp16_16[0]) },
{ "16/4", iov16, sizeof(iov16) / sizeof(iov16[0]), 4, exp16_4,
sizeof(exp16_4) / sizeof(exp16_4[0]) },
{ "16/3", iov16, sizeof(iov16) / sizeof(iov16[0]), 3, exp16_3,
sizeof(exp16_3) / sizeof(exp16_3[0]) },
{ "8/64", iov8, sizeof(iov8) / sizeof(iov8[0]), 64, exp8_64,
sizeof(exp8_64) / sizeof(exp8_64[0]) },
{ "odd/16", iov_odd, sizeof(iov_odd) / sizeof(iov_odd[0]), 16,
exp_odd_16, sizeof(exp_odd_16) / sizeof(exp_odd_16[0]) },
{ "skip/16", iov_skip, sizeof(iov_skip) / sizeof(iov_skip[0]), 16,
exp_skip_16, sizeof(exp_skip_16) / sizeof(exp_skip_16[0]) },
{ "empty/16", iov_empty, sizeof(iov_empty) / sizeof(iov_empty[0]), 16,
exp_empty_16, sizeof(exp_empty_16) / sizeof(exp_empty_16[0]) },
};
static void copy(giovec_t *dst, uint8_t *buffer, const giovec_t *src,
size_t iovcnt)
{
uint8_t *p = buffer;
size_t i;
for (i = 0; i < iovcnt; i++) {
dst[i].iov_base = p;
dst[i].iov_len = src[i].iov_len;
memcpy(dst[i].iov_base, src[i].iov_base, src[i].iov_len);
p += src[i].iov_len;
}
}
static void translate(uint8_t *data, size_t len)
{
for (; len > 0; len--) {
uint8_t *p = &data[len - 1];
if (*p >= '0' && *p <= '9')
*p = 'A' + *p - '0';
else if (*p >= 'A' && *p <= 'Z')
*p = '0' + *p - 'A';
}
}
#define MAX_BUF 1024
#define MAX_IOV 16
void doit(void)
{
uint8_t buffer[MAX_BUF];
size_t i;
for (i = 0; i < sizeof(tests) / sizeof(tests[0]); i++) {
giovec_t iov[MAX_IOV];
struct iov_iter_st iter;
const struct exp_st *exp = tests[i].exp;
uint8_t *data;
size_t j;
copy(iov, buffer, tests[i].iov, tests[i].iovcnt);
success("%s\n", tests[i].name);
assert(_gnutls_iov_iter_init(&iter, iov, tests[i].iovcnt,
tests[i].block_size) == 0);
for (j = 0; j < tests[i].expcnt; j++) {
ssize_t ret;
ret = _gnutls_iov_iter_next(&iter, &data);
if (ret != exp[j].ret)
fail("iov_iter_next: %d != %d\n", (int)ret,
(int)exp[j].ret);
else if (debug)
success("iov_iter_next: %d == %d\n", (int)ret,
(int)exp[j].ret);
if (ret == 0)
break;
if (ret > 0) {
if (iter.iov_index != exp[j].iov_index)
fail("iter.iov_index: %u != %u\n",
(unsigned)iter.iov_index,
(unsigned)exp[j].iov_index);
else if (debug)
success("iter.iov_index: %u == %u\n",
(unsigned)iter.iov_index,
(unsigned)exp[j].iov_index);
if (iter.iov_offset != exp[j].iov_offset)
fail("iter.iov_offset: %u != %u\n",
(unsigned)iter.iov_offset,
(unsigned)exp[j].iov_offset);
else if (debug)
success("iter.iov_offset: %u == %u\n",
(unsigned)iter.iov_offset,
(unsigned)exp[j].iov_offset);
if (iter.block_offset != 0)
fail("iter.block_offset: %u != 0\n",
(unsigned)iter.block_offset);
else if (debug)
success("iter.block_offset: %u == 0\n",
(unsigned)iter.block_offset);
translate(data, ret);
ret = _gnutls_iov_iter_sync(&iter, data, ret);
if (ret < 0)
fail("sync failed\n");
}
}
for (j = 0; j < tests[i].iovcnt; j++) {
translate(iov[j].iov_base, iov[j].iov_len);
if (memcmp(iov[j].iov_base, tests[i].iov[j].iov_base,
iov[j].iov_len) != 0)
fail("iov doesn't match: %*s != %*s\n",
(int)iov[j].iov_len,
(char *)iov[j].iov_base,
(int)tests[i].iov[j].iov_len,
(char *)tests[i].iov[j].iov_len);
}
}
}