Merge remote branch 'gonzoleeman/cunit' into cunit
This commit is contained in:
18
Makefile.am
18
Makefile.am
@@ -58,6 +58,7 @@ dist_noinst_HEADERS += test-tool/iscsi-test.h
|
||||
bin_iscsi_test_CPPFLAGS = $(AM_CPPFLAGS) -I$(srcdir)/test-tool
|
||||
bin_iscsi_test_LDFLAGS = -ldl
|
||||
bin_iscsi_test_SOURCES = test-tool/iscsi-test.c \
|
||||
test-tool/iscsi-support.c \
|
||||
test-tool/0000_testunitready_simple.c \
|
||||
test-tool/0100_read10_simple.c test-tool/0101_read10_beyond_eol.c \
|
||||
test-tool/0102_read10_0blocks.c test-tool/0103_read10_rdprotect.c \
|
||||
@@ -162,6 +163,23 @@ bin_iscsi_test_SOURCES = test-tool/iscsi-test.c \
|
||||
test-tool/1144_persistent_reserve_access_check_eaar.c \
|
||||
test-tool/1145_persistent_reserve_access_check_wear.c
|
||||
|
||||
# libiscsi test tool using cunit
|
||||
|
||||
noinst_PROGRAMS += bin/iscsi-test-cu
|
||||
# dist_noinst_HEADERS += test-tool/iscsi-test-cu.h
|
||||
bin_iscsi_test_cu_CPPFLAGS = $(AM_CPPFLAGS) -I$(srcdir)/test-tool \
|
||||
-I/usr/local/include
|
||||
bin_iscsi_test_cu_LDFLAGS = -ldl -lcunit
|
||||
bin_iscsi_test_cu_SOURCES = test-tool/iscsi-test-cu.c \
|
||||
test-tool/iscsi-support.c \
|
||||
test-tool/test_testunitready_simple.c \
|
||||
test-tool/test_read10_simple.c \
|
||||
test-tool/test_read10_beyond_eol.c \
|
||||
test-tool/test_read10_0blocks.c \
|
||||
test-tool/test_read10_rdprotect.c \
|
||||
test-tool/test_read10_flags.c \
|
||||
test-tool/test_read10_invalid.c
|
||||
|
||||
endif
|
||||
|
||||
# LD_PRELOAD library.
|
||||
|
||||
1382
test-tool/iscsi-support.c
Normal file
1382
test-tool/iscsi-support.c
Normal file
File diff suppressed because it is too large
Load Diff
130
test-tool/iscsi-support.h
Normal file
130
test-tool/iscsi-support.h
Normal file
@@ -0,0 +1,130 @@
|
||||
/*
|
||||
iscsi-test tool support
|
||||
|
||||
Copyright (C) 2012 by Lee Duncan <leeman.duncan@gmail.com>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef _ISCSI_SUPPORT_H_
|
||||
#define _ISCSI_SUPPORT_H_
|
||||
|
||||
#include <time.h>
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
extern const char *initiatorname1;
|
||||
extern const char *initiatorname2;
|
||||
extern const char *tgt_url;
|
||||
|
||||
extern uint32_t block_size;
|
||||
extern uint64_t num_blocks;
|
||||
extern int lbpme;
|
||||
extern int lbppb;
|
||||
extern int lbpme;
|
||||
extern int data_loss;
|
||||
extern int removable;
|
||||
extern enum scsi_inquiry_peripheral_device_type device_type;
|
||||
extern int sccs;
|
||||
extern int encserv;
|
||||
|
||||
|
||||
struct iscsi_context *iscsi_context_login(const char *initiatorname, const char *url, int *lun);
|
||||
|
||||
struct iscsi_async_state {
|
||||
struct scsi_task *task;
|
||||
int status;
|
||||
int finished;
|
||||
};
|
||||
void wait_until_test_finished(struct iscsi_context *iscsi, struct iscsi_async_state *test_state);
|
||||
|
||||
struct iscsi_pdu;
|
||||
int (*local_iscsi_queue_pdu)(struct iscsi_context *iscsi, struct iscsi_pdu *pdu);
|
||||
|
||||
|
||||
|
||||
/*
|
||||
* PGR support
|
||||
*/
|
||||
|
||||
static inline long rand_key(void)
|
||||
{
|
||||
time_t t;
|
||||
pid_t p;
|
||||
unsigned int s;
|
||||
long l;
|
||||
|
||||
(void)time(&t);
|
||||
p = getpid();
|
||||
s = ((int)p * (t & 0xffff));
|
||||
srandom(s);
|
||||
l = random();
|
||||
return l;
|
||||
}
|
||||
|
||||
static inline int pr_type_is_all_registrants(
|
||||
enum scsi_persistent_out_type pr_type)
|
||||
{
|
||||
switch (pr_type) {
|
||||
case SCSI_PERSISTENT_RESERVE_TYPE_WRITE_EXCLUSIVE_ALL_REGISTRANTS:
|
||||
case SCSI_PERSISTENT_RESERVE_TYPE_EXCLUSIVE_ACCESS_ALL_REGISTRANTS:
|
||||
return 1;
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
int register_and_ignore(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key);
|
||||
int register_key(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long sark, unsigned long long rk);
|
||||
int verify_key_presence(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, int present);
|
||||
int reregister_key_fails(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long sark);
|
||||
int reserve(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int release(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int verify_reserved_as(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int verify_read_works(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_write_works(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_read_fails(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_write_fails(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int testunitready(struct iscsi_context *iscsi, int lun);
|
||||
int testunitready_nomedium(struct iscsi_context *iscsi, int lun);
|
||||
int testunitready_conflict(struct iscsi_context *iscsi, int lun);
|
||||
int prefetch10(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch10_lbaoutofrange(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch10_nomedium(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16_lbaoutofrange(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16_nomedium(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int verify10(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int inquiry(struct iscsi_context *iscsi, int lun, int evpd, int page_code, int maxsize);
|
||||
|
||||
#endif /* _ISCSI_SUPPORT_H_ */
|
||||
482
test-tool/iscsi-test-cu.c
Normal file
482
test-tool/iscsi-test-cu.c
Normal file
@@ -0,0 +1,482 @@
|
||||
/*
|
||||
iscsi-test tool
|
||||
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#define _GNU_SOURCE
|
||||
#include <sys/syscall.h>
|
||||
#include <dlfcn.h>
|
||||
#include <sys/types.h>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdint.h>
|
||||
#include <unistd.h>
|
||||
#include <string.h>
|
||||
#include <getopt.h>
|
||||
#include <fnmatch.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
#include <CUnit/Basic.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-private.h"
|
||||
|
||||
#include "iscsi-support.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
|
||||
|
||||
#ifndef discard_const
|
||||
#define discard_const(ptr) ((void *)((intptr_t)(ptr)))
|
||||
#endif
|
||||
|
||||
#define PROG "iscsi-test-cu"
|
||||
|
||||
|
||||
/* XXX what is this for? */
|
||||
int (*real_iscsi_queue_pdu)(struct iscsi_context *iscsi, struct iscsi_pdu *pdu);
|
||||
|
||||
/*****************************************************************
|
||||
*
|
||||
* list of tests and test suites
|
||||
*
|
||||
*****************************************************************/
|
||||
static CU_TestInfo tests_testunitready[] = {
|
||||
{ (char *)"testTurSimple", test_testunitready_simple },
|
||||
CU_TEST_INFO_NULL
|
||||
};
|
||||
|
||||
static CU_TestInfo tests_read10[] = {
|
||||
{ (char *)"testRead10Simple", test_read10_simple },
|
||||
{ (char *)"testRead10BeyondEol", test_read10_beyond_eol },
|
||||
{ (char *)"testRead10ZeroBlocks", test_read10_0blocks },
|
||||
{ (char *)"testRead10ReadProtect", test_read10_rdprotect },
|
||||
{ (char *)"testRead10Flags", test_read10_flags },
|
||||
{ (char *)"testRead10Invalid", test_read10_invalid },
|
||||
CU_TEST_INFO_NULL
|
||||
};
|
||||
|
||||
static CU_SuiteInfo suites[] = {
|
||||
{ (char *)"TestTestUnitReady", test_setup, test_teardown,
|
||||
tests_testunitready },
|
||||
{ (char *)"TestRead10", test_setup, test_teardown,
|
||||
tests_read10 },
|
||||
CU_SUITE_INFO_NULL
|
||||
};
|
||||
|
||||
/*
|
||||
* globals for test setup and teardown
|
||||
*/
|
||||
int tgt_lun;
|
||||
struct iscsi_context *iscsic;
|
||||
struct scsi_task *task;
|
||||
|
||||
|
||||
static void
|
||||
print_usage(void)
|
||||
{
|
||||
fprintf(stderr,
|
||||
"Usage: %s [-?|--help] print this message and exit\n",
|
||||
PROG);
|
||||
fprintf(stderr,
|
||||
"or %s [OPTIONS] <iscsi-url>\n", PROG);
|
||||
fprintf(stderr,
|
||||
"Where OPTIONS are from:\n");
|
||||
fprintf(stderr,
|
||||
" -i|--initiator-name=iqn-name Initiatorname to use [%s]\n",
|
||||
initiatorname1);
|
||||
fprintf(stderr,
|
||||
" -I|--initiator-name-2=iqn-name 2nd Initiatorname to use [%s]\n",
|
||||
initiatorname2);
|
||||
fprintf(stderr,
|
||||
" -t|--test=test-name-reg-exp Test(s) to run [ALL] <NOT YET IMPLEMENTED>\n");
|
||||
fprintf(stderr,
|
||||
" -l|--list List all tests and exit\n");
|
||||
fprintf(stderr,
|
||||
" -X|--dataloss Allow destructive tests\n");
|
||||
fprintf(stderr,
|
||||
" -g|--ignore Error Action: Ignore test errors [DEFAULT]\n");
|
||||
fprintf(stderr,
|
||||
" -f|--fail Error Action: FAIL if any tests fail\n");
|
||||
fprintf(stderr,
|
||||
" -A|--abort Error Action: ABORT if any tests fail\n");
|
||||
fprintf(stderr,
|
||||
" -s|--silent Test Mode: Silent\n");
|
||||
fprintf(stderr,
|
||||
" -n|--normal Test Mode: Normal\n");
|
||||
fprintf(stderr,
|
||||
" -v|--verbose Test Mode: Verbose [DEFAULT]\n");
|
||||
fprintf(stderr, "\n");
|
||||
fprintf(stderr,
|
||||
"Where <iscsi-url> iSCSI URL format is: %s\n", ISCSI_URL_SYNTAX);
|
||||
fprintf(stderr, "\n");
|
||||
fprintf(stderr,
|
||||
"<host> is one of:\n");
|
||||
fprintf(stderr,
|
||||
" \"hostname\" e.g. iscsi.example\n");
|
||||
fprintf(stderr,
|
||||
" \"ipv4-address\" e.g. 10.1.1.27\n");
|
||||
fprintf(stderr,
|
||||
" \"ipv6-address\" e.g. [fce0::1]\n");
|
||||
fprintf(stderr, "\n");
|
||||
fprintf(stderr,
|
||||
"and <test-name-reg-exp> is of the form: SUITENAME_RE[.[SUBTESTNAME_RE]]\n");
|
||||
fprintf(stderr, "\n");
|
||||
}
|
||||
|
||||
|
||||
int
|
||||
test_setup(void)
|
||||
{
|
||||
iscsic = iscsi_context_login(initiatorname1, tgt_url, &tgt_lun);
|
||||
if (iscsic == NULL) {
|
||||
fprintf(stderr,
|
||||
"error: Failed to login to target for test set-up\n");
|
||||
return 1;
|
||||
}
|
||||
task = NULL;
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
int
|
||||
test_teardown(void)
|
||||
{
|
||||
if (task)
|
||||
scsi_free_scsi_task(task);
|
||||
iscsi_logout_sync(iscsic);
|
||||
iscsi_destroy_context(iscsic);
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static void
|
||||
list_all_tests(void)
|
||||
{
|
||||
CU_SuiteInfo *sp;
|
||||
CU_TestInfo *tp;
|
||||
|
||||
for (sp = suites; sp->pName != NULL; sp++)
|
||||
for (tp = sp->pTests; tp->pName != NULL; tp++)
|
||||
printf("%s.%s\n", sp->pName, tp->pName);
|
||||
}
|
||||
|
||||
|
||||
static CU_ErrorCode
|
||||
add_tests(const char *testname_re)
|
||||
{
|
||||
char *suitename_re = NULL;
|
||||
char *subtestname_re = NULL;
|
||||
char *cp;
|
||||
CU_SuiteInfo *sp;
|
||||
CU_TestInfo *tp;
|
||||
|
||||
|
||||
/* if not testname(s) register all tests and return */
|
||||
if (!testname_re)
|
||||
return CU_register_suites(suites);
|
||||
|
||||
/*
|
||||
* break testname_re into suitename_re and subtestname_re
|
||||
*
|
||||
* syntax is: SUITE_RE[.[SUBTEST_RE]]
|
||||
*/
|
||||
|
||||
/* is there a subtest name? */
|
||||
if ((cp = strchr(testname_re, '.')) == NULL) {
|
||||
suitename_re = strdup(testname_re);
|
||||
} else {
|
||||
size_t suitename_sz;
|
||||
size_t subtestname_sz;
|
||||
|
||||
suitename_sz = cp - testname_re;
|
||||
suitename_re = malloc(suitename_sz+1);
|
||||
memset(suitename_re, 0, suitename_sz+1);
|
||||
strncpy(suitename_re, testname_re, suitename_sz);
|
||||
|
||||
subtestname_sz = strlen(testname_re) - (suitename_sz+1);
|
||||
if (subtestname_sz) {
|
||||
subtestname_re = malloc(subtestname_sz+1);
|
||||
memset(subtestname_re, 0, subtestname_sz+1);
|
||||
strncpy(subtestname_re, cp+1, subtestname_sz);
|
||||
}
|
||||
}
|
||||
if (!suitename_re) {
|
||||
fprintf(stderr, "error: can't parse testsuite name: %s\n",
|
||||
testname_re);
|
||||
return CUE_NOTEST;
|
||||
}
|
||||
|
||||
/*
|
||||
* cycle through the test suites and tests, adding
|
||||
* ones that match
|
||||
*/
|
||||
for (sp = suites; sp->pName != NULL; sp++) {
|
||||
int suite_added = 0;
|
||||
CU_pSuite pSuite = NULL;
|
||||
|
||||
if (fnmatch(suitename_re, sp->pName, 0) != 0)
|
||||
continue;
|
||||
|
||||
for (tp = sp->pTests; tp->pName != NULL; tp++) {
|
||||
if (subtestname_re)
|
||||
if (fnmatch(subtestname_re, tp->pName, 0) != 0)
|
||||
continue;
|
||||
if (!suite_added) {
|
||||
suite_added++;
|
||||
pSuite = CU_add_suite(sp->pName,
|
||||
sp->pInitFunc, sp->pCleanupFunc);
|
||||
}
|
||||
CU_add_test(pSuite, tp->pName, tp->pTestFunc);
|
||||
}
|
||||
}
|
||||
|
||||
/* all done -- clean up */
|
||||
if (suitename_re)
|
||||
free(suitename_re);
|
||||
if (subtestname_re)
|
||||
free(subtestname_re);
|
||||
|
||||
return CUE_SUCCESS;
|
||||
}
|
||||
|
||||
|
||||
int
|
||||
main(int argc, char *argv[])
|
||||
{
|
||||
char *testname_re = NULL;
|
||||
int lun;
|
||||
CU_BasicRunMode mode = CU_BRM_VERBOSE;
|
||||
CU_ErrorAction error_action = CUEA_IGNORE;
|
||||
int res;
|
||||
struct scsi_readcapacity10 *rc10;
|
||||
struct scsi_readcapacity16 *rc16;
|
||||
struct scsi_inquiry_standard *inq;
|
||||
int full_size;
|
||||
static struct option long_opts[] = {
|
||||
{ "help", no_argument, 0, '?' },
|
||||
{ "list", no_argument, 0, 'l' },
|
||||
{ "initiator-name", required_argument, 0, 'i' },
|
||||
{ "initiator-name-2", required_argument, 0, 'I' },
|
||||
{ "test", required_argument, 0, 't' },
|
||||
{ "dataloss", no_argument, 0, 'd' },
|
||||
{ "ignore", no_argument, 0, 'g' },
|
||||
{ "fail", no_argument, 0, 'f' },
|
||||
{ "abort", no_argument, 0, 'A' },
|
||||
{ "silent", no_argument, 0, 's' },
|
||||
{ "normal", no_argument, 0, 'n' },
|
||||
{ "verbose", no_argument, 0, 'v' },
|
||||
{ NULL, 0, 0, 0 }
|
||||
};
|
||||
int c;
|
||||
int opt_idx = 0;
|
||||
|
||||
while ((c = getopt_long(argc, argv, "?hli:I:t:sdgfAsnv", long_opts,
|
||||
&opt_idx)) > 0) {
|
||||
switch (c) {
|
||||
case 'h':
|
||||
case '?':
|
||||
print_usage();
|
||||
return 0;
|
||||
case 'l':
|
||||
list_all_tests();
|
||||
return 0;
|
||||
case 'i':
|
||||
initiatorname1 = strdup(optarg);
|
||||
break;
|
||||
case 'I':
|
||||
initiatorname2 = strdup(optarg);
|
||||
break;
|
||||
case 't':
|
||||
testname_re = strdup(optarg);
|
||||
break;
|
||||
case 'd':
|
||||
data_loss++;
|
||||
break;
|
||||
case 'g':
|
||||
error_action = CUEA_IGNORE; /* default */
|
||||
break;
|
||||
case 'f':
|
||||
error_action = CUEA_FAIL;
|
||||
break;
|
||||
case 'A':
|
||||
error_action = CUEA_ABORT;
|
||||
break;
|
||||
case 's':
|
||||
mode = CU_BRM_SILENT;
|
||||
break;
|
||||
case 'n':
|
||||
mode = CU_BRM_NORMAL;
|
||||
break;
|
||||
case 'v':
|
||||
mode = CU_BRM_VERBOSE; /* default */
|
||||
break;
|
||||
default:
|
||||
fprintf(stderr,
|
||||
"error: unknown option return: %c (option %s)\n",
|
||||
c, argv[optind]);
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
|
||||
if (optind < argc) {
|
||||
tgt_url = strdup(argv[optind++]);
|
||||
}
|
||||
if (optind < argc) {
|
||||
fprintf(stderr, "error: too many arguments\n");
|
||||
print_usage();
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* XXX why is this done? */
|
||||
real_iscsi_queue_pdu = dlsym(RTLD_NEXT, "iscsi_queue_pdu");
|
||||
|
||||
if (tgt_url == NULL) {
|
||||
fprintf(stderr, "You must specify the URL\n");
|
||||
print_usage();
|
||||
if (testname_re)
|
||||
free(testname_re);
|
||||
return 10;
|
||||
}
|
||||
|
||||
iscsic = iscsi_context_login(initiatorname1, tgt_url, &lun);
|
||||
if (iscsic == NULL) {
|
||||
printf("Failed to login to target\n");
|
||||
return -1;
|
||||
}
|
||||
|
||||
/*
|
||||
* find the size of the LUN
|
||||
* All devices support readcapacity10 but only some support
|
||||
* readcapacity16
|
||||
*/
|
||||
task = iscsi_readcapacity10_sync(iscsic, lun, 0, 0);
|
||||
if (task == NULL) {
|
||||
printf("Failed to send READCAPACITY10 command: %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
iscsi_destroy_context(iscsic);
|
||||
return -1;
|
||||
}
|
||||
if (task->status != SCSI_STATUS_GOOD) {
|
||||
printf("READCAPACITY10 command: failed with sense. %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
scsi_free_scsi_task(task);
|
||||
iscsi_destroy_context(iscsic);
|
||||
return -1;
|
||||
}
|
||||
rc10 = scsi_datain_unmarshall(task);
|
||||
if (rc10 == NULL) {
|
||||
printf("failed to unmarshall READCAPACITY10 data. %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
scsi_free_scsi_task(task);
|
||||
iscsi_destroy_context(iscsic);
|
||||
return -1;
|
||||
}
|
||||
block_size = rc10->block_size;
|
||||
num_blocks = rc10->lba;
|
||||
scsi_free_scsi_task(task);
|
||||
|
||||
task = iscsi_readcapacity16_sync(iscsic, lun);
|
||||
if (task == NULL) {
|
||||
printf("Failed to send READCAPACITY16 command: %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
iscsi_destroy_context(iscsic);
|
||||
return -1;
|
||||
}
|
||||
if (task->status == SCSI_STATUS_GOOD) {
|
||||
rc16 = scsi_datain_unmarshall(task);
|
||||
if (rc16 == NULL) {
|
||||
printf("failed to unmarshall READCAPACITY16 data. %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
scsi_free_scsi_task(task);
|
||||
iscsi_destroy_context(iscsic);
|
||||
return -1;
|
||||
}
|
||||
block_size = rc16->block_length;
|
||||
num_blocks = rc16->returned_lba;
|
||||
lbpme = rc16->lbpme;
|
||||
lbppb = 1 << rc16->lbppbe;
|
||||
lbpme = rc16->lbpme;
|
||||
|
||||
scsi_free_scsi_task(task);
|
||||
}
|
||||
|
||||
task = iscsi_inquiry_sync(iscsic, lun, 0, 0, 64);
|
||||
if (task == NULL || task->status != SCSI_STATUS_GOOD) {
|
||||
printf("Inquiry command failed : %s\n", iscsi_get_error(iscsic));
|
||||
return -1;
|
||||
}
|
||||
full_size = scsi_datain_getfullsize(task);
|
||||
if (full_size > task->datain.size) {
|
||||
scsi_free_scsi_task(task);
|
||||
|
||||
/* we need more data for the full list */
|
||||
task = iscsi_inquiry_sync(iscsic, lun, 0, 0, full_size);
|
||||
if (task == NULL) {
|
||||
printf("Inquiry command failed : %s\n",
|
||||
iscsi_get_error(iscsic));
|
||||
return -1;
|
||||
}
|
||||
}
|
||||
inq = scsi_datain_unmarshall(task);
|
||||
if (inq == NULL) {
|
||||
printf("failed to unmarshall inquiry datain blob\n");
|
||||
scsi_free_scsi_task(task);
|
||||
return -1;
|
||||
}
|
||||
removable = inq->rmb;
|
||||
device_type = inq->device_type;
|
||||
sccs = inq->sccs;
|
||||
encserv = inq->encserv;
|
||||
scsi_free_scsi_task(task);
|
||||
|
||||
iscsi_logout_sync(iscsic);
|
||||
iscsi_destroy_context(iscsic);
|
||||
|
||||
if (CU_initialize_registry() != 0) {
|
||||
fprintf(stderr, "error: unable to initialize test registry\n");
|
||||
return 1;
|
||||
}
|
||||
if (CU_is_test_running()) {
|
||||
fprintf(stderr, "error: test suite(s) already running!?\n");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (add_tests(testname_re) != CUE_SUCCESS) {
|
||||
fprintf(stderr, "error: suite registration failed: %s\n",
|
||||
CU_get_error_msg());
|
||||
exit(1);
|
||||
}
|
||||
CU_basic_set_mode(mode);
|
||||
CU_set_error_action(error_action);
|
||||
printf("\n");
|
||||
|
||||
/*
|
||||
* this actually runs the tests ...
|
||||
*/
|
||||
res = CU_basic_run_tests();
|
||||
|
||||
printf("Tests completed with return value: %d\n", res);
|
||||
|
||||
CU_cleanup_registry();
|
||||
if (testname_re)
|
||||
free(testname_re);
|
||||
free(discard_const(tgt_url));
|
||||
|
||||
return 0;
|
||||
}
|
||||
48
test-tool/iscsi-test-cu.h
Normal file
48
test-tool/iscsi-test-cu.h
Normal file
@@ -0,0 +1,48 @@
|
||||
/*
|
||||
iscsi-test tool
|
||||
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef _ISCSI_TEST_CU_H_
|
||||
#define _ISCSI_TEST_CU_H_
|
||||
|
||||
#include <time.h>
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include "iscsi-support.h"
|
||||
|
||||
/* globals between setup, tests, and teardown */
|
||||
extern struct iscsi_context *iscsic;
|
||||
extern int tgt_lun;
|
||||
extern struct scsi_task *task;
|
||||
|
||||
int test_setup(void);
|
||||
int test_teardown(void);
|
||||
|
||||
void test_testunitready_simple(void);
|
||||
|
||||
void test_read10_simple(void);
|
||||
void test_read10_beyond_eol(void);
|
||||
void test_read10_0blocks(void);
|
||||
void test_read10_rdprotect(void);
|
||||
void test_read10_flags(void);
|
||||
void test_read10_invalid(void);
|
||||
|
||||
|
||||
#endif /* _ISCSI_TEST_CU_H_ */
|
||||
File diff suppressed because it is too large
Load Diff
@@ -25,32 +25,11 @@
|
||||
#include <unistd.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
extern const char *initiatorname1;
|
||||
extern const char *initiatorname2;
|
||||
#include "iscsi-support.h"
|
||||
|
||||
|
||||
extern uint32_t block_size;
|
||||
extern uint64_t num_blocks;
|
||||
extern int lbpme;
|
||||
extern int lbppb;
|
||||
extern int lbpme;
|
||||
extern int data_loss;
|
||||
extern int show_info;
|
||||
extern int removable;
|
||||
extern enum scsi_inquiry_peripheral_device_type device_type;
|
||||
extern int sccs;
|
||||
extern int encserv;
|
||||
|
||||
struct iscsi_context *iscsi_context_login(const char *initiatorname, const char *url, int *lun);
|
||||
|
||||
struct iscsi_async_state {
|
||||
struct scsi_task *task;
|
||||
int status;
|
||||
int finished;
|
||||
};
|
||||
void wait_until_test_finished(struct iscsi_context *iscsi, struct iscsi_async_state *test_state);
|
||||
|
||||
struct iscsi_pdu;
|
||||
int (*local_iscsi_queue_pdu)(struct iscsi_context *iscsi, struct iscsi_pdu *pdu);
|
||||
|
||||
int T0000_testunitready_simple(const char *initiator, const char *url);
|
||||
|
||||
@@ -216,77 +195,4 @@ int T1143_persistent_reserve_access_check_wero(const char *initiator, const char
|
||||
int T1144_persistent_reserve_access_check_eaar(const char *initiator, const char *url);
|
||||
int T1145_persistent_reserve_access_check_wear(const char *initiator, const char *url);
|
||||
|
||||
|
||||
/*
|
||||
* PGR support
|
||||
*/
|
||||
|
||||
static inline long rand_key(void)
|
||||
{
|
||||
time_t t;
|
||||
pid_t p;
|
||||
unsigned int s;
|
||||
long l;
|
||||
|
||||
(void)time(&t);
|
||||
p = getpid();
|
||||
s = ((int)p * (t & 0xffff));
|
||||
srandom(s);
|
||||
l = random();
|
||||
return l;
|
||||
}
|
||||
|
||||
static inline int pr_type_is_all_registrants(
|
||||
enum scsi_persistent_out_type pr_type)
|
||||
{
|
||||
switch (pr_type) {
|
||||
case SCSI_PERSISTENT_RESERVE_TYPE_WRITE_EXCLUSIVE_ALL_REGISTRANTS:
|
||||
case SCSI_PERSISTENT_RESERVE_TYPE_EXCLUSIVE_ACCESS_ALL_REGISTRANTS:
|
||||
return 1;
|
||||
default:
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
int register_and_ignore(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key);
|
||||
int register_key(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long sark, unsigned long long rk);
|
||||
int verify_key_presence(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, int present);
|
||||
int reregister_key_fails(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long sark);
|
||||
int reserve(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int release(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int verify_reserved_as(struct iscsi_context *iscsi, int lun,
|
||||
unsigned long long key, enum scsi_persistent_out_type pr_type);
|
||||
int verify_read_works(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_write_works(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_read_fails(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int verify_write_fails(struct iscsi_context *iscsi, int lun, unsigned char *buf);
|
||||
int testunitready(struct iscsi_context *iscsi, int lun);
|
||||
int testunitready_nomedium(struct iscsi_context *iscsi, int lun);
|
||||
int testunitready_conflict(struct iscsi_context *iscsi, int lun);
|
||||
int prefetch10(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch10_lbaoutofrange(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch10_nomedium(struct iscsi_context *iscsi, int lun, uint32_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16_lbaoutofrange(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int prefetch16_nomedium(struct iscsi_context *iscsi, int lun, uint64_t lba, int num_blocks, int immed, int group);
|
||||
int verify10(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify10_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify12_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint32_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_nomedium(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_miscompare(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int verify16_lbaoutofrange(struct iscsi_context *iscsi, int lun, unsigned char *data, uint32_t datalen, uint64_t lba, int vprotect, int dpo, int bytchk, int blocksize);
|
||||
int inquiry(struct iscsi_context *iscsi, int lun, int evpd, int page_code, int maxsize);
|
||||
|
||||
#endif /* _ISCSI_TEST_H_ */
|
||||
|
||||
73
test-tool/test_read10_0blocks.c
Normal file
73
test-tool/test_read10_0blocks.c
Normal file
@@ -0,0 +1,73 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
void
|
||||
test_read10_0blocks(void)
|
||||
{
|
||||
/* read zero blocks at LBA 0 ... */
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, 0, 0, block_size,
|
||||
0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* READ10 0blocks at one block beyond <end-of-LUN> ... */
|
||||
if (num_blocks > 0x80000000) {
|
||||
CU_PASS("[SKIPPED] LUN is too big");
|
||||
return;
|
||||
}
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, num_blocks + 1, 0,
|
||||
block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* READ10 0blocks at LBA 2^31 ... */
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, 0x80000000, 0, block_size,
|
||||
0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* READ10 0blocks at LBA -1 ... */
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, -1, 0, block_size,
|
||||
0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
88
test-tool/test_read10_beyond_eol.c
Normal file
88
test-tool/test_read10_beyond_eol.c
Normal file
@@ -0,0 +1,88 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
|
||||
void
|
||||
test_read10_beyond_eol(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
if (num_blocks >= 0x80000000) {
|
||||
CU_PASS("LUN is too big for read-beyond-eol tests with READ10. Skipping test.\n");
|
||||
return;
|
||||
}
|
||||
|
||||
/* read 1-256 blocks, one block beyond the end-of-lun */
|
||||
for (i = 1; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, num_blocks + 2 - i,
|
||||
i * block_size, block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
|
||||
/* Reading 1-256 blocks at LBA 2^31 */
|
||||
for (i = 1; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, 0x80000000,
|
||||
i * block_size, block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
|
||||
/* read 1 - 256 blocks at LBA -1 */
|
||||
for (i = 1; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, -1, i * block_size,
|
||||
block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
|
||||
/* read 2-256 blocks, all but one block beyond the eol */
|
||||
for (i = 2; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, num_blocks,
|
||||
i * block_size, block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_LBA_OUT_OF_RANGE);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
}
|
||||
131
test-tool/test_read10_flags.c
Normal file
131
test-tool/test_read10_flags.c
Normal file
@@ -0,0 +1,131 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
void
|
||||
test_read10_flags(void)
|
||||
{
|
||||
struct scsi_task *task_ret;
|
||||
|
||||
|
||||
fprintf(stderr, "DEBUG: %s: entering\n", __FUNCTION__);
|
||||
|
||||
/* This test is only valid for SBC devices */
|
||||
if (device_type != SCSI_INQUIRY_PERIPHERAL_DEVICE_TYPE_DIRECT_ACCESS) {
|
||||
CU_PASS("[SKIPPED] LUN is not SBC device. Skipping test");
|
||||
return;
|
||||
}
|
||||
|
||||
/* Try out READ10 with DPO : 1 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = 0x10;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try out READ10 with FUA : 1 FUA_NV : 0 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = 0x08;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try out READ10 with FUA : 1 FUA_NV : 1 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = 0x0a;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try out READ10 with FUA : 0 FUA_NV : 1 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = 0x02;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try out READM10 with DPO : 1 FUA : 1 FUA_NV : 1 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = 0x18;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
145
test-tool/test_read10_invalid.c
Normal file
145
test-tool/test_read10_invalid.c
Normal file
@@ -0,0 +1,145 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "iscsi-private.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
|
||||
void
|
||||
test_read10_invalid(void)
|
||||
{
|
||||
struct iscsi_data data;
|
||||
char buf[4096];
|
||||
struct scsi_task *task_ret;
|
||||
|
||||
|
||||
fprintf(stderr, "DEBUG: %s: entering\n", __FUNCTION__);
|
||||
|
||||
/* Try a read10 of 1 block but xferlength == 0 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = 0;
|
||||
|
||||
/*
|
||||
* we dont want autoreconnect since some targets will drop the session
|
||||
* on this condition.
|
||||
*/
|
||||
iscsi_set_noautoreconnect(iscsic, 1);
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_NOT_EQUAL(task->status, SCSI_STATUS_CANCELLED); /* XXX redundant? */
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->residual_status, SCSI_RESIDUAL_OVERFLOW);
|
||||
CU_ASSERT_EQUAL(task->residual, (int64_t)block_size);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* in case the previous test failed the session */
|
||||
iscsi_set_noautoreconnect(iscsic, 0);
|
||||
|
||||
/* Try a read10 of 1 block but xferlength == 1024 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = 1024;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->residual_status, SCSI_RESIDUAL_OVERFLOW);
|
||||
CU_ASSERT_EQUAL(task->residual, (int64_t)block_size);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try a read10 of 1 block but xferlength == 200 */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = 200;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->residual_status, SCSI_RESIDUAL_OVERFLOW);
|
||||
CU_ASSERT_EQUAL(task->residual, (int64_t)block_size - 200);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try a read10 of 2 blocks but xferlength == 'block_size' */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[8] = 2;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
CU_ASSERT_EQUAL(task->residual_status, SCSI_RESIDUAL_OVERFLOW);
|
||||
CU_ASSERT_EQUAL(task->residual, (int64_t)block_size);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
|
||||
/* Try a read10 of 1 block but make it a data-out write on the iscsi layer */
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_WRITE;
|
||||
task->expxferlen = sizeof(buf);
|
||||
|
||||
data.size = sizeof(buf);
|
||||
data.data = (unsigned char *)&buf[0];
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, &data);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
68
test-tool/test_read10_rdprotect.c
Normal file
68
test-tool/test_read10_rdprotect.c
Normal file
@@ -0,0 +1,68 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <string.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
void
|
||||
test_read10_rdprotect(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
|
||||
if (device_type != SCSI_INQUIRY_PERIPHERAL_DEVICE_TYPE_DIRECT_ACCESS) {
|
||||
CU_PASS("[SKIPPED] LUN is not SBC device. Skipping test");
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* Try out Different non-zero values for RDPROTECT.
|
||||
* They should all fail.
|
||||
*/
|
||||
|
||||
/* Read10 with non-zero RDPROTECT ... */
|
||||
for (i = 1; i < 8; i++) {
|
||||
struct scsi_task *task_ret;
|
||||
|
||||
task = malloc(sizeof(struct scsi_task));
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
|
||||
memset(task, 0, sizeof(struct scsi_task));
|
||||
task->cdb[0] = SCSI_OPCODE_READ10;
|
||||
task->cdb[1] = (i<<5)&0xe0;
|
||||
task->cdb[8] = 1;
|
||||
task->cdb_size = 10;
|
||||
task->xfer_dir = SCSI_XFER_READ;
|
||||
task->expxferlen = block_size;
|
||||
|
||||
task_ret = iscsi_scsi_command_sync(iscsic, tgt_lun, task, NULL);
|
||||
CU_ASSERT_PTR_NOT_NULL(task_ret);
|
||||
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_CHECK_CONDITION);
|
||||
CU_ASSERT_EQUAL(task->sense.key, SCSI_SENSE_ILLEGAL_REQUEST);
|
||||
CU_ASSERT_EQUAL(task->sense.ascq, SCSI_SENSE_ASCQ_INVALID_FIELD_IN_CDB);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
}
|
||||
53
test-tool/test_read10_simple.c
Normal file
53
test-tool/test_read10_simple.c
Normal file
@@ -0,0 +1,53 @@
|
||||
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
|
||||
void
|
||||
test_read10_simple(void)
|
||||
{
|
||||
int i;
|
||||
|
||||
|
||||
/* read the first 1 - 256 blocks at the start of the LUN */
|
||||
for (i = 1; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, 0, i * block_size,
|
||||
block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
|
||||
/* read the last 1 - 256 blocks at the end of the LUN */
|
||||
for (i = 1; i <= 256; i++) {
|
||||
task = iscsi_read10_sync(iscsic, tgt_lun, num_blocks +1 - i,
|
||||
i * block_size, block_size, 0, 0, 0, 0, 0);
|
||||
CU_ASSERT_PTR_NOT_NULL(task);
|
||||
CU_ASSERT_EQUAL(task->status, SCSI_STATUS_GOOD);
|
||||
scsi_free_scsi_task(task);
|
||||
task = NULL;
|
||||
}
|
||||
}
|
||||
33
test-tool/test_testunitready_simple.c
Normal file
33
test-tool/test_testunitready_simple.c
Normal file
@@ -0,0 +1,33 @@
|
||||
/*
|
||||
Copyright (C) 2012 by Lee Duncan <lee@gonzoleeman.net>
|
||||
|
||||
This program 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 2 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program 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 General Public License
|
||||
along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <stdio.h>
|
||||
|
||||
#include <CUnit/CUnit.h>
|
||||
|
||||
#include "iscsi.h"
|
||||
#include "scsi-lowlevel.h"
|
||||
#include "iscsi-test-cu.h"
|
||||
|
||||
void
|
||||
test_testunitready_simple(void)
|
||||
{
|
||||
int ret;
|
||||
|
||||
ret = testunitready(iscsic, tgt_lun);
|
||||
CU_ASSERT_EQUAL(ret, 0);
|
||||
}
|
||||
Reference in New Issue
Block a user