]> CyberLeo.Net >> Repos - FreeBSD/releng/9.0.git/blob - sbin/iscontrol/iscontrol.c
Copy stable/9 to releng/9.0 as part of the FreeBSD 9.0-RELEASE release
[FreeBSD/releng/9.0.git] / sbin / iscontrol / iscontrol.c
1 /*-
2  * Copyright (c) 2005-2010 Daniel Braniss <danny@cs.huji.ac.il>
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  *
26  */
27 /*
28  | $Id: iscontrol.c,v 2.2 2006/12/01 09:11:56 danny Exp danny $
29  */
30 /*
31  | the user level initiator (client)
32  */
33
34 #include <sys/cdefs.h>
35 __FBSDID("$FreeBSD$");
36
37 #include <sys/param.h>
38 #include <sys/types.h>
39 #include <sys/socket.h>
40 #include <sys/sysctl.h>
41
42 #include <netinet/in.h>
43 #include <netinet/tcp.h>
44 #include <arpa/inet.h>
45 #include <sys/ioctl.h>
46 #include <netdb.h>
47 #include <stdlib.h>
48 #include <unistd.h>
49 #include <stdio.h>
50 #include <string.h>
51 #include <errno.h>
52 #include <fcntl.h>
53 #include <time.h>
54 #include <camlib.h>
55
56 #include <dev/iscsi/initiator/iscsi.h>
57 #include "iscontrol.h"
58
59 #define USAGE "[-v] [-d] [-c config] [-n name] [-t target] [-p pidfile]"
60 #define OPTIONS "vdc:t:n:p:"
61
62 token_t AuthMethods[] = {
63      {"None",   NONE},
64      {"KRB5",   KRB5},
65      {"SPKM1",  SPKM1},
66      {"SPKM2",  SPKM2},
67      {"SRP",    SRP},
68      {"CHAP",   CHAP},
69      {0, 0}
70 };
71
72 token_t DigestMethods[] = {
73      {"None",   0},
74      {"CRC32",  1},
75      {"CRC32C", 1},
76      {0, 0}
77 };
78
79 u_char  isid[6 + 6];
80 /*
81  | Default values
82  */
83 isc_opt_t opvals = {
84      .port                      = 3260,
85      .sockbufsize               = 128,
86      .iqn                       = "iqn.2005-01.il.ac.huji.cs:",
87
88      .sessionType               = "Normal",
89      .targetAddress             = 0,
90      .targetName                = 0,
91      .initiatorName             = 0,
92      .authMethod                = "None",
93      .headerDigest              = "None,CRC32C",
94      .dataDigest                = "None,CRC32C",
95      .maxConnections            = 1,
96      .maxRecvDataSegmentLength  = 64 * 1024,
97      .maxXmitDataSegmentLength  = 8 * 1024, // 64 * 1024,
98      .maxBurstLength            = 128 * 1024,
99      .firstBurstLength          = 64 * 1024, // must be less than maxBurstLength
100      .defaultTime2Wait          = 0,
101      .defaultTime2Retain        = 0,
102      .maxOutstandingR2T         = 1,
103      .errorRecoveryLevel        = 0,
104
105      .dataPDUInOrder            = TRUE,
106      .dataSequenceInOrder       = TRUE,
107
108      .initialR2T                = TRUE,
109      .immediateData             = TRUE,
110 };
111
112 int
113 lookup(token_t *tbl, char *m)
114 {
115      token_t    *tp;
116
117      for(tp = tbl; tp->name != NULL; tp++)
118           if(strcasecmp(tp->name, m) == 0)
119                return tp->val;
120      return 0;
121 }
122
123 int
124 main(int cc, char **vv)
125 {
126      int        ch, disco;
127      char       *pname, *pidfile, *p, *q, *ta, *kw;
128      isc_opt_t  *op;
129      FILE       *fd;
130
131      op = &opvals;
132      iscsidev = "/dev/"ISCSIDEV;
133      fd = NULL;
134      pname = vv[0];
135      if((p = strrchr(pname, '/')) != NULL)
136           pname = p + 1;
137
138      kw = ta = 0;
139      disco = 0;
140      pidfile = NULL;
141
142      while((ch = getopt(cc, vv, OPTIONS)) != -1) {
143           switch(ch) {
144           case 'v':
145                vflag++;
146                break;
147           case 'c':
148                fd = fopen(optarg, "r");
149                if(fd == NULL) {
150                     perror(optarg);
151                     exit(1);
152                }
153                break;
154           case 'd':
155                disco = 1;
156                break;
157           case 't':
158                ta = optarg;
159                break;
160           case 'n':
161                kw = optarg;
162                break;
163           case 'p':
164                pidfile = optarg;
165                break;
166           default:
167           badu:
168                fprintf(stderr, "Usage: %s %s\n", pname, USAGE);
169                exit(1);
170           }
171      }
172      if(fd == NULL)
173           fd = fopen("/etc/iscsi.conf", "r");
174
175      if(fd != NULL) {
176           parseConfig(fd, kw, op);
177           fclose(fd);
178      }
179      cc -= optind;
180      vv += optind;
181      if(cc > 0) {
182           if(vflag)
183                printf("adding '%s'\n", *vv);
184           parseArgs(cc, vv, op);
185      }
186      if(ta)
187           op->targetAddress = ta;
188
189      if(op->targetAddress == NULL) {
190           fprintf(stderr, "No target!\n");
191           goto badu;
192      }
193      q = op->targetAddress;
194      if(*q == '[' && (q = strchr(q, ']')) != NULL) {
195           *q++ = '\0';
196           op->targetAddress++;
197      } else
198           q = op->targetAddress;
199      if((p = strchr(q, ':')) != NULL) {
200           *p++ = 0;
201           op->port = atoi(p);
202           p = strchr(p, ',');
203      }
204      if(p || ((p = strchr(q, ',')) != NULL)) {
205           *p++ = 0;
206           op->targetPortalGroupTag = atoi(p);
207      }
208      if(op->initiatorName == 0) {
209           char  hostname[256];
210
211           if(op->iqn) {
212                if(gethostname(hostname, sizeof(hostname)) == 0)
213                     asprintf(&op->initiatorName, "%s:%s", op->iqn, hostname);
214                else
215                     asprintf(&op->initiatorName, "%s:%d", op->iqn, (int)time(0) & 0xff); // XXX:
216           }
217           else {
218                if(gethostname(hostname, sizeof(hostname)) == 0)
219                     asprintf(&op->initiatorName, "%s", hostname);
220                else
221                     asprintf(&op->initiatorName, "%d", (int)time(0) & 0xff); // XXX:
222           }
223      }
224      if(disco) {
225           op->sessionType = "Discovery";
226           op->targetName = 0;
227      }
228      op->pidfile = pidfile;
229      fsm(op);
230
231      exit(0);
232 }