+static int jt_getattr(int argc, char **argv)
+{
+ struct obd_ioctl_data data;
+ int rc;
+
+ if (argc != 2) {
+ fprintf(stderr, "usage: %s id\n", cmdname(argv[0]));
+ return -1;
+ }
+
+ IOCINIT(data);
+ data.ioc_obdo1.o_id = strtoul(argv[1], NULL, 0);
+ /* to help obd filter */
+ data.ioc_obdo1.o_mode = 0100644;
+ data.ioc_obdo1.o_valid = 0xffffffff;
+ printf("%s: object id %Ld\n", cmdname(argv[0]), data.ioc_obdo1.o_id);
+
+ rc = ioctl(fd, OBD_IOC_GETATTR , &data);
+ if (rc) {
+ fprintf(stderr, "error: %s: %s\n", cmdname(argv[0]),
+ strerror(rc=errno));
+ } else {
+ printf("%s: object id %Ld, mode %o\n", cmdname(argv[0]),
+ data.ioc_obdo1.o_id, data.ioc_obdo1.o_mode);
+ }
+ return rc;
+}
+
+static int jt_test_getattr(int argc, char **argv)
+{
+ struct obd_ioctl_data data;
+ struct timeval start, next_time;
+ int i, count, next_count;
+ int verbose;
+ int rc;
+
+ if (argc != 2 && argc != 3) {
+ fprintf(stderr, "usage: %s count [verbose]\n",cmdname(argv[0]));
+ return -1;
+ }
+
+ IOCINIT(data);
+ count = strtoul(argv[1], NULL, 0);
+
+ verbose = get_verbose(argv[2]);
+
+ data.ioc_obdo1.o_valid = 0xffffffff;
+ data.ioc_obdo1.o_id = 2;
+ gettimeofday(&start, NULL);
+ next_time.tv_sec = start.tv_sec - verbose;
+ next_time.tv_usec = start.tv_usec;
+ printf("%s: getting %d attrs (testing only): %s", cmdname(argv[0]),
+ count, ctime(&start.tv_sec));
+
+ for (i = 1, next_count = verbose; i <= count; i++) {
+ rc = ioctl(fd, OBD_IOC_GETATTR , &data);
+ if (rc < 0) {
+ fprintf(stderr, "error: %s: #%d - %s\n",
+ cmdname(argv[0]), i, strerror(rc = errno));
+ break;
+ } else if (be_verbose(verbose, &next_time, i,&next_count,count))
+ printf("%s: got attr #%d\n", cmdname(argv[0]), i);
+ }
+
+ if (!rc) {
+ struct timeval end;
+ double diff;
+
+ gettimeofday(&end, NULL);
+
+ diff = difftime(&end, &start);
+
+ --i;
+ printf("%s: %d attrs in %.4gs (%.4g attr/s): %s",
+ cmdname(argv[0]), i, diff, (double)i / diff,
+ ctime(&end.tv_sec));
+ }
+ return rc;
+}
+
+static int jt_test_brw(int argc, char **argv)
+{
+ struct obd_ioctl_data data;
+ struct timeval start, next_time;
+ char *bulk, *b;
+ int pages = 1, obdos = 1, count, next_count;
+ int verbose = 1, write = 0, rw;
+ int i, o, p;
+ int len;
+ int rc;
+
+ if (argc < 2 || argc > 6) {
+ fprintf(stderr,
+ "usage: %s count [write [verbose [pages [obdos]]]]\n",
+ cmdname(argv[0]));
+ return -1;
+ }
+
+ count = strtoul(argv[1], NULL, 0);
+
+ if (argc >= 3) {
+ if (argv[2][0] == 'w' || argv[2][0] == '1')
+ write = 1;
+ else if (argv[2][0] == 'r' || argv[2][0] == '0')
+ write = 0;
+
+ verbose = get_verbose(argv[3]);
+ }
+
+ if (argc >= 5)
+ pages = strtoul(argv[4], NULL, 0);
+ if (argc >= 6)
+ obdos = strtoul(argv[5], NULL, 0);
+
+ if (obdos != 1 && obdos != 2) {
+ fprintf(stderr, "error: %s: only 1 or 2 obdos supported\n",
+ cmdname(argv[0]));
+ return -2;
+ }
+
+ len = pages * PAGE_SIZE;
+
+ bulk = calloc(obdos, len);
+ if (!bulk) {
+ fprintf(stderr, "error: %s: no memory allocating %dx%d pages\n",
+ cmdname(argv[0]), obdos, pages);
+ return -2;
+ }
+ IOCINIT(data);
+ data.ioc_conn2 = connid;
+ data.ioc_obdo1.o_id = 2;
+ data.ioc_count = len;
+ data.ioc_offset = 0;
+ data.ioc_plen1 = len;
+ data.ioc_pbuf1 = bulk;
+ if (obdos > 1) {
+ data.ioc_obdo2.o_id = 2;
+ data.ioc_plen2 = len;
+ data.ioc_pbuf2 = bulk + len;
+ }
+
+ gettimeofday(&start, NULL);
+ next_time.tv_sec = start.tv_sec - verbose;
+ next_time.tv_usec = start.tv_usec;
+
+ printf("%s: %s %d (%dx%d pages) (testing only): %s",
+ cmdname(argv[0]), write ? "writing" : "reading",
+ count, obdos, pages, ctime(&start.tv_sec));
+
+ /*
+ * We will put in the start time (and loop count inside the loop)
+ * at the beginning of each page so that we will be able to validate
+ * (at some later time) whether the data actually made it or not.
+ */
+ for (o = 0, b = bulk; o < obdos; o++)
+ for (p = 0; p < pages; p++, b += PAGE_SIZE)
+ memcpy(b, &start, sizeof(start));
+
+ rw = write ? OBD_IOC_BRW_WRITE : OBD_IOC_BRW_READ;
+ for (i = 1, next_count = verbose; i <= count; i++) {
+ if (write) {
+ b = bulk + sizeof(struct timeval);
+ for (o = 0; o < obdos; o++)
+ for (p = 0; p < pages; p++, b += PAGE_SIZE)
+ memcpy(b, &count, sizeof(count));
+ }
+
+ rc = ioctl(fd, rw, &data);
+ if (rc) {
+ fprintf(stderr, "error: %s: #%d - %s on %s\n",
+ cmdname(argv[0]), i, strerror(rc = errno),
+ write ? "write" : "read");
+ break;
+ } else if (be_verbose(verbose, &next_time, i,&next_count,count))
+ printf("%s: %s number %d\n", cmdname(argv[0]),
+ write ? "write" : "read", i);
+ }
+
+ free(bulk);
+
+ if (!rc) {
+ struct timeval end;
+ double diff;
+
+ gettimeofday(&end, NULL);
+
+ diff = difftime(&end, &start);
+
+ --i;
+ printf("%s: %s %dx%dx%d pages in %.4gs (%.4g pg/s): %s",
+ cmdname(argv[0]), write ? "wrote" : "read", obdos,
+ pages, i, diff, (double)obdos * i * pages / diff,
+ ctime(&end.tv_sec));
+ }
+ return rc;
+}
+
+static int jt_test_ldlm(int argc, char **argv)
+{
+ struct obd_ioctl_data data;
+ int rc;
+
+ IOCINIT(data);
+ if (argc != 1) {
+ fprintf(stderr, "usage: %s\n", cmdname(argv[0]));
+ return 1;
+ }
+
+ rc = ioctl(fd, IOC_LDLM_TEST, &data);
+ if (rc)
+ fprintf(stderr, "error: %s: test failed: %s\n",
+ cmdname(argv[0]), strerror(rc = errno));
+ return rc;
+}
+
+command_t cmdlist[] = {
+ /* Metacommands */
+ {"--device", jt__device, 0, "--device <devno> <command [args ...]>"},
+ {"--threads", jt__threads, 0,
+ "--threads <threads> <devno> <command [args ...]>"},
+
+ /* Device configuration commands */
+ {"device", jt_device, 0, "set current device (args device no)"},
+ {"attach", jt_attach, 0, "name the type of device (args: type data"},
+ {"setup", jt_setup, 0, "setup device (args: <blkdev> [data]"},
+ {"detach", jt_detach, 0, "detach the current device (arg: )"},
+ {"cleanup", jt_cleanup, 0, "cleanup the current device (arg: )"},
+
+ /* Session commands */
+ {"connect", jt_connect, 0, "connect - get a connection to device"},
+ {"disconnect", jt_disconnect, 0,
+ "disconnect - break connection to device"},
+
+ /* Session operations */
+ {"create", jt_create, 0, "create [count [mode [verbose]]]"},
+ {"destroy", jt_destroy, 0, "destroy <id>"},
+ {"getattr", jt_getattr, 0, "getattr <id>"},
+ {"setattr", jt_setattr, 0, "setattr <id> <mode>"},
+ {"test_getattr", jt_test_getattr, 0, "test_getattr <count> [verbose]"},
+ {"test_brw", jt_test_brw, 0, "test_brw <count> [write [verbose]]"},
+ {"test_ldlm", jt_test_ldlm, 0, "test lock manager (no args)"},
+
+ /* User interface commands */
+ {"help", Parser_help, 0, "help"},
+ {"exit", Parser_quit, 0, "quit"},
+ {"quit", Parser_quit, 0, "quit"},
+ { 0, 0, 0, NULL }