scsi-aix.c
上传用户:xiejiait
上传日期:2007-01-06
资源大小:881k
文件大小:10k
源码类别:

SCSI/ASPI

开发平台:

MultiPlatform

  1. /* @(#)scsi-aix.c 1.19 99/10/07 Copyright 1997 J. Schilling */
  2. #ifndef lint
  3. static char __sccsid[] =
  4. "@(#)scsi-aix.c 1.19 99/10/07 Copyright 1997 J. Schilling";
  5. #endif
  6. /*
  7.  * Interface for the AIX generic SCSI implementation.
  8.  *
  9.  * This is a hack, that tries to emulate the functionality
  10.  * of the scg driver.
  11.  *
  12.  * Warning: you may change this source, but if you do that
  13.  * you need to change the _scg_version and _scg_auth* string below.
  14.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  15.  * Choose your name instead of "schily" and make clear that the version
  16.  * string is related to a modified source.
  17.  *
  18.  * Copyright (c) 1997 J. Schilling
  19.  */
  20. /*
  21.  * This program is free software; you can redistribute it and/or modify
  22.  * it under the terms of the GNU General Public License as published by
  23.  * the Free Software Foundation; either version 2, or (at your option)
  24.  * any later version.
  25.  *
  26.  * This program is distributed in the hope that it will be useful,
  27.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  28.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  29.  * GNU General Public License for more details.
  30.  *
  31.  * You should have received a copy of the GNU General Public License
  32.  * along with this program; see the file COPYING.  If not, write to
  33.  * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  34.  */
  35. #include <sys/scdisk.h>
  36. /*
  37.  * Warning: you may change this source, but if you do that
  38.  * you need to change the _scg_version and _scg_auth* string below.
  39.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  40.  * Choose your name instead of "schily" and make clear that the version
  41.  * string is related to a modified source.
  42.  */
  43. LOCAL char _scg_trans_version[] = "scsi-aix.c-1.19"; /* The version for this transport*/
  44. #define MAX_SCG 16 /* Max # of SCSI controllers */
  45. #define MAX_TGT 16
  46. #define MAX_LUN 8
  47. struct scg_local {
  48. short scgfiles[MAX_SCG][MAX_TGT][MAX_LUN];
  49. };
  50. #define scglocal(p) ((struct scg_local *)((p)->local)) 
  51. #define MAX_DMA_AIX (64*1024)
  52. LOCAL int do_scsi_cmd __PR((SCSI *scgp, int f, struct scg_cmd *sp));
  53. LOCAL int do_scsi_sense __PR((SCSI *scgp, int f, struct scg_cmd *sp));
  54. /*
  55.  * Return version information for the low level SCSI transport code.
  56.  * This has been introduced to make it easier to trace down problems
  57.  * in applications.
  58.  */
  59. EXPORT char *
  60. scg__version(scgp, what)
  61. SCSI *scgp;
  62. int what;
  63. {
  64. if (scgp != (SCSI *)0) {
  65. switch (what) {
  66. case SCG_VERSION:
  67. return (_scg_trans_version);
  68. /*
  69.  * If you changed this source, you are not allowed to
  70.  * return "schily" for the SCG_AUTHOR request.
  71.  */
  72. case SCG_AUTHOR:
  73. return (_scg_auth_schily);
  74. case SCG_SCCS_ID:
  75. return (__sccsid);
  76. }
  77. }
  78. return ((char *)0);
  79. }
  80. EXPORT int
  81. scsi_open(scgp, device, busno, tgt, tlun)
  82. SCSI *scgp;
  83. char *device;
  84. int busno;
  85. int tgt;
  86. int tlun;
  87. {
  88. register int f;
  89. register int b;
  90. register int t;
  91. register int l;
  92. register int nopen = 0;
  93. char devname[32];
  94. if (busno >= MAX_SCG || tgt >= MAX_TGT || tlun >= MAX_LUN) {
  95. errno = EINVAL;
  96. if (scgp->errstr)
  97. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  98. "Illegal value for busno, target or lun '%d,%d,%d'",
  99. busno, tgt, tlun);
  100. return (-1);
  101. }
  102. if (scgp->local == NULL) {
  103. scgp->local = malloc(sizeof(struct scg_local));
  104. if (scgp->local == NULL)
  105. return (0);
  106. for (b=0; b < MAX_SCG; b++) {
  107. for (t=0; t < MAX_TGT; t++) {
  108. for (l=0; l < MAX_LUN ; l++)
  109. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  110. }
  111. }
  112. }
  113. if ((device != NULL && *device != '') || (busno == -2 && tgt == -2))
  114. goto openbydev;
  115. if (busno >= 0 && tgt >= 0 && tlun >= 0) {
  116. sprintf(devname, "/dev/rcd%d", tgt);
  117. f = openx(devname, 0, 0, SC_DIAGNOSTIC);
  118. if (f < 0) {
  119. if (scgp->errstr)
  120. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  121. "Cannot open '%s'. Specify device number (1 for cd1) as target (1,0)",
  122. devname);
  123. return (0);
  124. }
  125. scglocal(scgp)->scgfiles[busno][tgt][tlun] = f;
  126. return 1;
  127. } else {
  128. if (scgp->errstr)
  129. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  130. "Unable to scan on AIX");
  131. return (0);
  132. }
  133. openbydev:
  134. if (device != NULL && *device != '' && busno >= 0 && tgt >= 0 && tlun >= 0) {
  135. f = openx(device, 0, 0, SC_DIAGNOSTIC);
  136. if (f < 0) {
  137. if (scgp->errstr)
  138. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  139. "Cannot open '%s'",
  140. devname);
  141. return (0);
  142. }
  143. scgp->scsibus = busno;
  144. scgp->target = tgt;
  145. scgp->lun = tlun;
  146. scglocal(scgp)->scgfiles[busno][tgt][tlun] = f;
  147. return (++nopen);
  148. }
  149. return (nopen);
  150. }
  151. EXPORT int
  152. scsi_close(scgp)
  153. SCSI *scgp;
  154. {
  155. register int f;
  156. register int b;
  157. register int t;
  158. register int l;
  159. if (scgp->local == NULL)
  160. return (-1);
  161. for (b=0; b < MAX_SCG; b++) {
  162. for (t=0; t < MAX_TGT; t++) {
  163. for (l=0; l < MAX_LUN ; l++)
  164. f = scglocal(scgp)->scgfiles[b][t][l];
  165. if (f >= 0)
  166. close(f);
  167. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  168. }
  169. }
  170. return (0);
  171. }
  172. LOCAL long
  173. scsi_maxdma(scgp)
  174. SCSI *scgp;
  175. {
  176. return (MAX_DMA_AIX);
  177. }
  178. #define palign(x, a) (((char *)(x)) + ((a) - 1 - (((unsigned)((x)-1))%(a))))
  179. EXPORT void *
  180. scsi_getbuf(scgp, amt)
  181. SCSI *scgp;
  182. long amt;
  183. {
  184. void *ret;
  185. int pagesize;
  186. #ifdef _SC_PAGESIZE
  187. pagesize = sysconf(_SC_PAGESIZE);
  188. #else
  189. pagesize = getpagesize();
  190. #endif
  191. if (amt <= 0 || amt > scsi_maxdma(scgp))
  192. return ((void *)0);
  193. if (scgp->debug)
  194. printf("scsi_getbuf: %ld bytesn", amt);
  195. /*
  196.  * Damn AIX is a paged system but has no valloc()
  197.  */
  198. scgp->bufbase = ret = malloc((size_t)(amt+pagesize));
  199. if (ret == NULL)
  200. return (ret);
  201. ret = palign(ret, pagesize);
  202. return (ret);
  203. }
  204. EXPORT void
  205. scsi_freebuf(scgp)
  206. SCSI *scgp;
  207. {
  208. if (scgp->bufbase)
  209. free(scgp->bufbase);
  210. scgp->bufbase = NULL;
  211. }
  212. EXPORT
  213. BOOL scsi_havebus(scgp, busno)
  214. SCSI *scgp;
  215. int busno;
  216. {
  217. register int t;
  218. register int l;
  219. if (busno < 0 || busno >= MAX_SCG)
  220. return (FALSE);
  221. if (scgp->local == NULL)
  222. return (FALSE);
  223. for (t=0; t < MAX_TGT; t++) {
  224. for (l=0; l < MAX_LUN ; l++)
  225. if (scglocal(scgp)->scgfiles[busno][t][l] >= 0)
  226. return (TRUE);
  227. }
  228. return (FALSE);
  229. }
  230. EXPORT
  231. int scsi_fileno(scgp, busno, tgt, tlun)
  232. SCSI *scgp;
  233. int busno;
  234. int tgt;
  235. int tlun;
  236. {
  237. if (busno < 0 || busno >= MAX_SCG ||
  238.     tgt < 0 || tgt >= MAX_TGT ||
  239.     tlun < 0 || tlun >= MAX_LUN)
  240. return (-1);
  241. if (scgp->local == NULL)
  242. return (-1);
  243. return ((int)scglocal(scgp)->scgfiles[busno][tgt][tlun]);
  244. }
  245. EXPORT int
  246. scsi_initiator_id(scgp)
  247. SCSI *scgp;
  248. {
  249. return (-1);
  250. }
  251. EXPORT
  252. int scsi_isatapi(scgp)
  253. SCSI *scgp;
  254. {
  255. return (FALSE);
  256. }
  257. EXPORT
  258. int scsireset(scgp)
  259. SCSI *scgp;
  260. {
  261. int f = scsi_fileno(scgp, scgp->scsibus, scgp->target, scgp->lun);
  262. return (ioctl(f, SCIORESET, IDLUN(scgp->target, scgp->lun)));
  263. }
  264. LOCAL int
  265. do_scsi_cmd(scgp, f, sp)
  266. SCSI *scgp;
  267. int f;
  268. struct scg_cmd *sp;
  269. {
  270. struct sc_iocmd req;
  271. int ret;
  272. if (sp->cdb_len > 12)
  273. comerrno(EX_BAD, "Can't do %d byte command.n", sp->cdb_len);
  274. fillbytes(&req, sizeof(req), '');
  275. req.flags = SC_ASYNC;
  276. if (sp->flags & SCG_RECV_DATA) {
  277. req.flags |= B_READ;
  278. } else if (sp->size > 0) {
  279. req.flags |= B_WRITE;
  280. }
  281. req.data_length = sp->size;
  282. req.buffer = sp->addr;
  283. req.timeout_value = sp->timeout;
  284. req.command_length = sp->cdb_len;
  285. movebytes(&sp->cdb, req.scsi_cdb, 12);
  286. errno = 0;
  287. ret = ioctl(f, DKIOCMD, &req);
  288. if (scgp->debug) {
  289. printf("ret: %d errno: %d (%s)n", ret, errno, errmsgstr(errno));
  290. printf("data_length:     %dn", req.data_length);
  291. printf("buffer:          0x%Xn", req.buffer);
  292. printf("timeout_value:   %dn", req.timeout_value);
  293. printf("status_validity: %dn", req.status_validity);
  294. printf("scsi_bus_status: 0x%Xn", req.scsi_bus_status);
  295. printf("adapter_status:  0x%Xn", req.adapter_status);
  296. printf("adap_q_status:   0x%Xn", req.adap_q_status);
  297. printf("q_tag_msg:       0x%Xn", req.q_tag_msg);
  298. printf("flags:           0X%Xn", req.flags);
  299. }
  300. if (ret < 0) {
  301. sp->ux_errno = geterrno();
  302. /*
  303.  * Check if SCSI command cound not be send at all.
  304.  */
  305. if (sp->ux_errno == ENOTTY || sp->ux_errno == ENXIO ||
  306.     sp->ux_errno == EINVAL || sp->ux_errno == EACCES) {
  307. return (-1);
  308. }
  309. } else {
  310. sp->ux_errno = 0;
  311. }
  312. ret = 0;
  313. sp->sense_count = 0;
  314. sp->resid = 0; /* AIX is the same rubbish as Linux here */
  315. fillbytes(&sp->scb, sizeof(sp->scb), '');
  316. fillbytes(&sp->u_sense.cmd_sense, sizeof(sp->u_sense.cmd_sense), '');
  317. if (req.status_validity == 0) {
  318. sp->error = SCG_NO_ERROR;
  319. return (0);
  320. }
  321. if (req.status_validity & 1) {
  322. sp->u_scb.cmd_scb[0] = req.scsi_bus_status;
  323. sp->error = SCG_RETRYABLE;
  324. }
  325. if (req.status_validity & 2) {
  326. if (req.adapter_status & SC_NO_DEVICE_RESPONSE) {
  327. sp->error = SCG_FATAL;
  328. } else if (req.adapter_status & SC_CMD_TIMEOUT) {
  329. sp->error = SCG_TIMEOUT;
  330. } else if (req.adapter_status != 0) {
  331. sp->error = SCG_RETRYABLE;
  332. }
  333. }
  334. return (ret);
  335. }
  336. LOCAL int
  337. do_scsi_sense(scgp, f, sp)
  338. SCSI *scgp;
  339. int f;
  340. struct scg_cmd *sp;
  341. {
  342. int ret;
  343. struct scg_cmd s_cmd;
  344. if (sp->sense_len > SCG_MAX_SENSE)
  345. sp->sense_len = SCG_MAX_SENSE;
  346. fillbytes((caddr_t)&s_cmd, sizeof(s_cmd), '');
  347. s_cmd.addr = sp->u_sense.cmd_sense;
  348. s_cmd.size = sp->sense_len;
  349. s_cmd.flags = SCG_RECV_DATA|SCG_DISRE_ENA;
  350. s_cmd.cdb_len = SC_G0_CDBLEN;
  351. s_cmd.sense_len = CCS_SENSE_LEN;
  352. s_cmd.target = scgp->target;
  353. s_cmd.cdb.g0_cdb.cmd = SC_REQUEST_SENSE;
  354. s_cmd.cdb.g0_cdb.lun = sp->cdb.g0_cdb.lun;
  355. s_cmd.cdb.g0_cdb.count = sp->sense_len;
  356. ret = do_scsi_cmd(scgp, f, &s_cmd);
  357. if (ret < 0)
  358. return (ret);
  359. if (s_cmd.u_scb.cmd_scb[0] & 02) {
  360. /* XXX ??? Check condition on request Sense ??? */
  361. }
  362. sp->sense_count = sp->sense_len - s_cmd.resid;
  363. return (ret);
  364. }
  365. LOCAL int
  366. scsi_send(scgp, f, sp)
  367. SCSI *scgp;
  368. int f;
  369. struct scg_cmd *sp;
  370. {
  371. int ret;
  372. if (f < 0) {
  373. sp->error = SCG_FATAL;
  374. return (0);
  375. }
  376. ret = do_scsi_cmd(scgp, f, sp);
  377. if (ret < 0)
  378. return (ret);
  379. if (sp->u_scb.cmd_scb[0] & 02)
  380. ret = do_scsi_sense(scgp, f, sp);
  381. return (ret);
  382. }