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

SCSI/ASPI

开发平台:

MultiPlatform

  1. /* @(#)scsi-sgi.c 1.19 99/09/17 Copyright 1997 J. Schilling */
  2. #ifndef lint
  3. static char __sccsid[] =
  4. "@(#)scsi-sgi.c 1.19 99/09/17 Copyright 1997 J. Schilling";
  5. #endif
  6. /*
  7.  * Interface for the SGI generic SCSI implementation.
  8.  *
  9.  * First Hacky implementation
  10.  * (needed libds, did not support bus scanning and had no error checking)
  11.  * from "Frank van Beek" <frank@neogeo.nl>
  12.  *
  13.  * Actual implementation supports all scg features.
  14.  *
  15.  * Warning: you may change this source, but if you do that
  16.  * you need to change the _scg_version and _scg_auth* string below.
  17.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  18.  * Choose your name instead of "schily" and make clear that the version
  19.  * string is related to a modified source.
  20.  *
  21.  * Copyright (c) 1997 J. Schilling
  22.  */
  23. /*
  24.  * This program is free software; you can redistribute it and/or modify
  25.  * it under the terms of the GNU General Public License as published by
  26.  * the Free Software Foundation; either version 2, or (at your option)
  27.  * any later version.
  28.  *
  29.  * This program is distributed in the hope that it will be useful,
  30.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  31.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  32.  * GNU General Public License for more details.
  33.  *
  34.  * You should have received a copy of the GNU General Public License
  35.  * along with this program; see the file COPYING.  If not, write to
  36.  * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  37.  */
  38. #include <dslib.h>
  39. /*
  40.  * Warning: you may change this source, but if you do that
  41.  * you need to change the _scg_version and _scg_auth* string below.
  42.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  43.  * Choose your name instead of "schily" and make clear that the version
  44.  * string is related to a modified source.
  45.  */
  46. LOCAL char _scg_trans_version[] = "scsi-sgi.c-1.19"; /* The version for this transport*/
  47. #ifdef USE_DSLIB
  48. struct dsreq * dsp = 0;
  49. #define MAX_SCG 1 /* Max # of SCSI controllers */
  50. #else
  51. #define MAX_SCG 16 /* Max # of SCSI controllers */
  52. #define MAX_TGT 16
  53. #define MAX_LUN 8
  54. struct scg_local {
  55. short scgfiles[MAX_SCG][MAX_TGT][MAX_LUN];
  56. };
  57. #define scglocal(p) ((struct scg_local *)((p)->local)) 
  58. #endif
  59. #define MAX_DMA_SGI (256*1024) /* Check if this is not too big */
  60. #ifndef USE_DSLIB
  61. LOCAL int scsi_sendreq __PR((SCSI *scgp, int f, struct scg_cmd *sp, struct dsreq *dsp));
  62. #endif
  63. /*
  64.  * Return version information for the low level SCSI transport code.
  65.  * This has been introduced to make it easier to trace down problems
  66.  * in applications.
  67.  */
  68. EXPORT char *
  69. scg__version(scgp, what)
  70. SCSI *scgp;
  71. int what;
  72. {
  73. if (scgp != (SCSI *)0) {
  74. switch (what) {
  75. case SCG_VERSION:
  76. return (_scg_trans_version);
  77. /*
  78.  * If you changed this source, you are not allowed to
  79.  * return "schily" for the SCG_AUTHOR request.
  80.  */
  81. case SCG_AUTHOR:
  82. return (_scg_auth_schily);
  83. case SCG_SCCS_ID:
  84. return (__sccsid);
  85. }
  86. }
  87. return ((char *)0);
  88. }
  89. EXPORT int
  90. scsi_open(scgp, device, busno, tgt, tlun)
  91. SCSI *scgp;
  92. char *device;
  93. int busno;
  94. int tgt;
  95. int tlun;
  96. {
  97. register int f;
  98. register int b;
  99. register int t;
  100. register int l;
  101. register int nopen = 0;
  102. char devname[64];
  103. if (busno >= MAX_SCG || tgt >= MAX_TGT || tlun >= MAX_LUN) {
  104. errno = EINVAL;
  105. if (scgp->errstr)
  106. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  107. "Illegal value for busno, target or lun '%d,%d,%d'",
  108. busno, tgt, tlun);
  109. return (-1);
  110. }
  111. if ((device != NULL && *device != '') || (busno == -2 && tgt == -2)) {
  112. errno = EINVAL;
  113. if (scgp->errstr)
  114. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  115. "Open by 'devname' not supported on this OS");
  116. return (-1);
  117. }
  118. if (scgp->local == NULL) {
  119. scgp->local = malloc(sizeof(struct scg_local));
  120. if (scgp->local == NULL)
  121. return (0);
  122. for (b=0; b < MAX_SCG; b++) {
  123. for (t=0; t < MAX_TGT; t++) {
  124. for (l=0; l < MAX_LUN ; l++)
  125. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  126. }
  127. }
  128. }
  129. if (busno >= 0 && tgt >= 0 && tlun >= 0) {
  130. sprintf(devname, "/dev/scsi/sc%dd%dl%d", busno, tgt, tlun);
  131. #ifdef USE_DSLIB
  132. dsp = dsopen(devname, O_RDWR);
  133. if (dsp == 0)
  134. return(-1);
  135. #else
  136. f = open(devname, O_RDWR);
  137. if (f < 0)
  138. return(-1);
  139. scglocal(scgp)->scgfiles[busno][tgt][tlun] = f;
  140. #endif
  141. return(1);
  142. } else {
  143. #ifdef USE_DSLIB
  144. return(-1);
  145. #else
  146. for (b=0; b < MAX_SCG; b++) {
  147. for (t=0; t < MAX_TGT; t++) {
  148. /* for (l=0; l < MAX_LUN ; l++) {*/
  149. for (l=0; l < 1 ; l++) {
  150. sprintf(devname, "/dev/scsi/sc%dd%dl%d", b, t, l);
  151. f = open(devname, O_RDWR);
  152. if (f >= 0) {
  153. scglocal(scgp)->scgfiles[b][t][l] = (short)f;
  154. nopen++;
  155. }
  156. }
  157. }
  158. }
  159. #endif
  160. }
  161. return (nopen);
  162. }
  163. EXPORT int
  164. scsi_close(scgp)
  165. SCSI *scgp;
  166. {
  167. #ifndef USE_DSLIB
  168. register int f;
  169. register int b;
  170. register int t;
  171. register int l;
  172. if (scgp->local == NULL)
  173. return (-1);
  174. for (b=0; b < MAX_SCG; b++) {
  175. for (t=0; t < MAX_TGT; t++) {
  176. for (l=0; l < MAX_LUN ; l++)
  177. f = scglocal(scgp)->scgfiles[b][t][l];
  178. if (f >= 0)
  179. close(f);
  180. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  181. }
  182. }
  183. #else
  184. dsclose(dsp);
  185. #endif
  186. return (0);
  187. }
  188. LOCAL long
  189. scsi_maxdma(scgp)
  190. SCSI *scgp;
  191. {
  192. return (MAX_DMA_SGI);
  193. }
  194. EXPORT void *
  195. scsi_getbuf(scgp, amt)
  196. SCSI *scgp;
  197. long amt;
  198. {
  199. if (amt <= 0 || amt > scsi_maxdma(scgp))
  200. return ((void *)0);
  201. if (scgp->debug)
  202. printf("scsi_getbuf: %ld bytesn", amt);
  203. scgp->bufbase = valloc((size_t)(amt));
  204. return (scgp->bufbase);
  205. }
  206. EXPORT void
  207. scsi_freebuf(scgp)
  208. SCSI *scgp;
  209. {
  210. if (scgp->bufbase)
  211. free(scgp->bufbase);
  212. scgp->bufbase = NULL;
  213. }
  214. EXPORT
  215. BOOL scsi_havebus(scgp, busno)
  216. SCSI *scgp;
  217. int busno;
  218. {
  219. register int t;
  220. register int l;
  221. if (busno < 0 || busno >= MAX_SCG)
  222. return (FALSE);
  223. if (scgp->local == NULL)
  224. return (FALSE);
  225. for (t=0; t < MAX_TGT; t++) {
  226. for (l=0; l < MAX_LUN ; l++)
  227. if (scglocal(scgp)->scgfiles[busno][t][l] >= 0)
  228. return (TRUE);
  229. }
  230. return (FALSE);
  231. }
  232. EXPORT
  233. int scsi_fileno(scgp, busno, tgt, tlun)
  234. SCSI *scgp;
  235. int busno;
  236. int tgt;
  237. int tlun;
  238. {
  239. #ifdef USE_DSLIB
  240. if (dsp == NULL)
  241. return (-1);
  242. return (getfd(dsp));
  243. #else
  244. if (busno < 0 || busno >= MAX_SCG ||
  245.     tgt < 0 || tgt >= MAX_TGT ||
  246.     tlun < 0 || tlun >= MAX_LUN)
  247. return (-1);
  248. if (scgp->local == NULL)
  249. return (-1);
  250. return ((int)scglocal(scgp)->scgfiles[busno][tgt][tlun]);
  251. #endif
  252. }
  253. EXPORT int
  254. scsi_initiator_id(scgp)
  255. SCSI *scgp;
  256. {
  257. return (-1);
  258. }
  259. EXPORT
  260. int scsi_isatapi(scgp)
  261. SCSI *scgp;
  262. {
  263. return (FALSE);
  264. }
  265. EXPORT
  266. int scsireset(scgp)
  267. SCSI *scgp;
  268. {
  269. /*
  270.  * Do we have a SCSI reset on SGI?
  271.  */
  272. #ifdef DS_RESET
  273. int f = scsi_fileno(scgp, scgp->scsibus, scgp->target, scgp->lun);
  274. return (ioctl(f, DS_RESET, 0));
  275. #else
  276. return (-1);
  277. #endif
  278. }
  279. #ifndef USE_DSLIB
  280. LOCAL int
  281. scsi_sendreq(scgp, f, sp, dsp)
  282. SCSI *scgp;
  283. int f;
  284. struct scg_cmd *sp;
  285. struct dsreq *dsp;
  286. {
  287. int ret;
  288. int retries = 4;
  289. u_char status;
  290. /* if ((sp->flags & SCG_CMD_RETRY) == 0)*/
  291. /* retries = 0;*/
  292. while (--retries > 0) {
  293. ret = ioctl(f, DS_ENTER, dsp);
  294. if (ret < 0)  {
  295. RET(dsp) = DSRT_DEVSCSI;
  296. return (-1);
  297. }
  298. /*
  299.  * SGI implementattion botch!!!
  300.  * If a target does not select the bus,
  301.  * the return code is DSRT_TIMEOUT
  302.  */
  303. if (RET(dsp) == DSRT_TIMEOUT) {
  304. struct timeval to;
  305. to.tv_sec = TIME(dsp)/1000;
  306. to.tv_usec = TIME(dsp)%1000;
  307. scsitimes(scgp);
  308. if (sp->cdb.g0_cdb.cmd == SC_TEST_UNIT_READY &&
  309.     scgp->cmdstop->tv_sec < to.tv_sec ||
  310.     (scgp->cmdstop->tv_sec == to.tv_sec &&
  311. scgp->cmdstop->tv_usec < to.tv_usec)) {
  312. RET(dsp) = DSRT_NOSEL;
  313. return (-1);
  314. }
  315. }
  316. if (RET(dsp) == DSRT_NOSEL)
  317. continue; /* retry noselect 3X */
  318. status = STATUS(dsp);
  319. switch (status) {
  320. case 0x08: /*  BUSY */
  321. case 0x18: /*  RESERV CONFLICT */
  322. if (retries > 0)
  323. sleep(2);
  324. continue;
  325. case 0x00: /*  GOOD */
  326. case 0x02: /*  CHECK CONDITION */
  327. case 0x10: /*  INTERM/GOOD */
  328. default:
  329. return (status);
  330. }
  331. }
  332. return (-1); /* failed retry limit */
  333. }
  334. #endif
  335. LOCAL int
  336. scsi_send(scgp, f, sp)
  337. SCSI *scgp;
  338. int f;
  339. struct scg_cmd *sp;
  340. {
  341. int ret;
  342. int i;
  343. int amt = sp->cdb_len;
  344. int flags;
  345. #ifndef USE_DSLIB
  346. struct dsreq ds;
  347. struct dsreq *dsp = &ds;
  348. dsp->ds_iovbuf = 0;
  349. dsp->ds_iovlen = 0;
  350. #endif
  351. if (f < 0) {
  352. sp->error = SCG_FATAL;
  353. return (0);
  354. }
  355. flags = DSRQ_SENSE;
  356. if (sp->flags & SCG_RECV_DATA)
  357. flags |= DSRQ_READ;
  358. else if (sp->size > 0)
  359. flags |= DSRQ_WRITE;
  360. dsp->ds_flags = flags;
  361. dsp->ds_link = 0;
  362. dsp->ds_synch = 0;
  363. dsp->ds_ret   = 0;
  364. DATABUF(dsp)  = sp->addr;
  365. DATALEN(dsp) = sp->size;
  366. CMDBUF(dsp) = (void *) &sp->cdb;
  367. CMDLEN(dsp) = sp->cdb_len;
  368. SENSEBUF(dsp) = sp->u_sense.cmd_sense;
  369. SENSELEN(dsp) = sizeof (sp->u_sense.cmd_sense);
  370. TIME(dsp) = (sp->timeout * 1000) + 100;
  371. errno = 0;
  372. sp->ux_errno = 0;
  373. sp->sense_count = 0;
  374. #ifdef USE_DSLIB
  375. ret = doscsireq(f, dsp);
  376. #else
  377. ret = scsi_sendreq(scgp, f, sp, dsp);
  378. #endif
  379. if (RET(dsp) != DSRT_DEVSCSI)
  380. ret = 0;
  381. if (RET(dsp)) {
  382. if (RET(dsp) == DSRT_SHORT) {
  383. sp->resid = DATALEN(dsp)- DATASENT(dsp);
  384. } else if (errno) {
  385. sp->ux_errno = errno;
  386. } else {
  387. sp->ux_errno = EIO;
  388. }
  389. sp->u_scb.cmd_scb[0] = STATUS(dsp);
  390. sp->sense_count = SENSESENT(dsp);
  391. if (sp->sense_count > SCG_MAX_SENSE)
  392. sp->sense_count = SCG_MAX_SENSE;
  393. }
  394. switch(RET(dsp)) {
  395. default:
  396. sp->error = SCG_RETRYABLE; break;
  397. case 0: /* OK         */
  398. case DSRT_SHORT: /* not implemented        */
  399. sp->error = SCG_NO_ERROR; break;
  400. case DSRT_UNIMPL: /* not implemented        */
  401. case DSRT_REVCODE: /* software obsolete must recompile   */
  402. case DSRT_NOSEL:
  403. sp->u_scb.cmd_scb[0] = 0;
  404. sp->error = SCG_FATAL; break;
  405. case DSRT_TIMEOUT:
  406. sp->u_scb.cmd_scb[0] = 0;
  407. sp->error = SCG_TIMEOUT; break;
  408. }
  409. return (ret);
  410. }