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

SCSI/ASPI

开发平台:

MultiPlatform

  1. /* @(#)scsi-hpux.c 1.17 00/01/11 Copyright 1997 J. Schilling */
  2. #ifndef lint
  3. static char __sccsid[] =
  4. "@(#)scsi-hpux.c 1.17 00/01/11 Copyright 1997 J. Schilling";
  5. #endif
  6. /*
  7.  * Interface for the HP-UX generic SCSI implementation.
  8.  *
  9.  * Warning: you may change this source, but if you do that
  10.  * you need to change the _scg_version and _scg_auth* string below.
  11.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  12.  * Choose your name instead of "schily" and make clear that the version
  13.  * string is related to a modified source.
  14.  *
  15.  * Copyright (c) 1997 J. Schilling
  16.  */
  17. /*
  18.  * This program is free software; you can redistribute it and/or modify
  19.  * it under the terms of the GNU General Public License as published by
  20.  * the Free Software Foundation; either version 2, or (at your option)
  21.  * any later version.
  22.  *
  23.  * This program is distributed in the hope that it will be useful,
  24.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  25.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  26.  * GNU General Public License for more details.
  27.  *
  28.  * You should have received a copy of the GNU General Public License
  29.  * along with this program; see the file COPYING.  If not, write to
  30.  * the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
  31.  */
  32. #undef sense
  33. #include <sys/scsi.h>
  34. /*
  35.  * Warning: you may change this source, but if you do that
  36.  * you need to change the _scg_version and _scg_auth* string below.
  37.  * You may not return "schily" for an SCG_AUTHOR request anymore.
  38.  * Choose your name instead of "schily" and make clear that the version
  39.  * string is related to a modified source.
  40.  */
  41. LOCAL char _scg_trans_version[] = "scsi-hpux.c-1.17"; /* The version for this transport*/
  42. #define MAX_SCG 16 /* Max # of SCSI controllers */
  43. #define MAX_TGT 16
  44. #define MAX_LUN 8
  45. struct scg_local {
  46. short scgfiles[MAX_SCG][MAX_TGT][MAX_LUN];
  47. };
  48. #define scglocal(p) ((struct scg_local *)((p)->local)) 
  49. #ifdef SCSI_MAXPHYS
  50. # define MAX_DMA_HP SCSI_MAXPHYS
  51. #else
  52. # define MAX_DMA_HP (63*1024) /* Check if this is not too big */
  53. #endif
  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[64];
  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 ((device != NULL && *device != '') || (busno == -2 && tgt == -2)) {
  103. errno = EINVAL;
  104. if (scgp->errstr)
  105. js_snprintf(scgp->errstr, SCSI_ERRSTR_SIZE,
  106. "Open by 'devname' not supported on this OS");
  107. return (-1);
  108. }
  109. if (scgp->local == NULL) {
  110. scgp->local = malloc(sizeof(struct scg_local));
  111. if (scgp->local == NULL)
  112. return (0);
  113. for (b=0; b < MAX_SCG; b++) {
  114. for (t=0; t < MAX_TGT; t++) {
  115. for (l=0; l < MAX_LUN ; l++)
  116. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  117. }
  118. }
  119. }
  120. if (busno >= 0 && tgt >= 0 && tlun >= 0) {
  121. sprintf(devname, "/dev/rscsi/c%xt%xl%x", busno, tgt, tlun);
  122. f = open(devname, O_RDWR);
  123. if (f < 0)
  124. return(-1);
  125. scglocal(scgp)->scgfiles[busno][tgt][tlun] = f;
  126. return(1);
  127. } else {
  128. for (b=0; b < MAX_SCG; b++) {
  129. for (t=0; t < MAX_TGT; t++) {
  130. /* for (l=0; l < MAX_LUN ; l++) {*/
  131. for (l=0; l < 1 ; l++) {
  132. sprintf(devname, "/dev/rscsi/c%xt%xl%x", b, t, l);
  133. /*error("name: '%s'n", devname);*/
  134. f = open(devname, O_RDWR);
  135. if (f >= 0) {
  136. scglocal(scgp)->scgfiles[b][t][l] = (short)f;
  137. nopen++;
  138. } else if (scgp->debug) {
  139. errmsg("open '%s'n", devname);
  140. }
  141. }
  142. }
  143. }
  144. }
  145. return (nopen);
  146. }
  147. EXPORT int
  148. scsi_close(scgp)
  149. SCSI *scgp;
  150. {
  151. register int f;
  152. register int b;
  153. register int t;
  154. register int l;
  155. if (scgp->local == NULL)
  156. return (-1);
  157. for (b=0; b < MAX_SCG; b++) {
  158. for (t=0; t < MAX_TGT; t++) {
  159. for (l=0; l < MAX_LUN ; l++)
  160. f = scglocal(scgp)->scgfiles[b][t][l];
  161. if (f >= 0)
  162. close(f);
  163. scglocal(scgp)->scgfiles[b][t][l] = (short)-1;
  164. }
  165. }
  166. return (0);
  167. }
  168. LOCAL long
  169. scsi_maxdma(scgp)
  170. SCSI *scgp;
  171. {
  172. return (MAX_DMA_HP);
  173. }
  174. EXPORT void *
  175. scsi_getbuf(scgp, amt)
  176. SCSI *scgp;
  177. long amt;
  178. {
  179. if (amt <= 0 || amt > scsi_maxdma(scgp))
  180. return ((void *)0);
  181. if (scgp->debug)
  182. printf("scsi_getbuf: %ld bytesn", amt);
  183. scgp->bufbase = valloc((size_t)(amt));
  184. return (scgp->bufbase);
  185. }
  186. EXPORT void
  187. scsi_freebuf(scgp)
  188. SCSI *scgp;
  189. {
  190. if (scgp->bufbase)
  191. free(scgp->bufbase);
  192. scgp->bufbase = NULL;
  193. }
  194. EXPORT
  195. BOOL scsi_havebus(scgp, busno)
  196. SCSI *scgp;
  197. int busno;
  198. {
  199. register int t;
  200. register int l;
  201. if (busno < 0 || busno >= MAX_SCG)
  202. return (FALSE);
  203. if (scgp->local == NULL)
  204. return (FALSE);
  205. for (t=0; t < MAX_TGT; t++) {
  206. for (l=0; l < MAX_LUN ; l++)
  207. if (scglocal(scgp)->scgfiles[busno][t][l] >= 0)
  208. return (TRUE);
  209. }
  210. return (FALSE);
  211. }
  212. EXPORT
  213. int scsi_fileno(scgp, busno, tgt, tlun)
  214. SCSI *scgp;
  215. int busno;
  216. int tgt;
  217. int tlun;
  218. {
  219. if (busno < 0 || busno >= MAX_SCG ||
  220.     tgt < 0 || tgt >= MAX_TGT ||
  221.     tlun < 0 || tlun >= MAX_LUN)
  222. return (-1);
  223. if (scgp->local == NULL)
  224. return (-1);
  225. return ((int)scglocal(scgp)->scgfiles[busno][tgt][tlun]);
  226. }
  227. EXPORT int
  228. scsi_initiator_id(scgp)
  229. SCSI *scgp;
  230. {
  231. return (-1);
  232. }
  233. EXPORT
  234. int scsi_isatapi(scgp)
  235. SCSI *scgp;
  236. {
  237. return (FALSE);
  238. }
  239. EXPORT
  240. int scsireset(scgp)
  241. SCSI *scgp;
  242. {
  243. int f = scsi_fileno(scgp, scgp->scsibus, scgp->target, scgp->lun);
  244. return (ioctl(f, SIOC_RESET_BUS, 0));
  245. }
  246. LOCAL int
  247. scsi_send(scgp, f, sp)
  248. SCSI *scgp;
  249. int f;
  250. struct scg_cmd *sp;
  251. {
  252. int ret;
  253. int flags;
  254. struct sctl_io sctl_io;
  255. if ((f < 0) || (sp->cdb_len > sizeof(sctl_io.cdb))) {
  256. sp->error = SCG_FATAL;
  257. return (0);
  258. }
  259. fillbytes((caddr_t)&sctl_io, sizeof(sctl_io), '');
  260. flags = 0;
  261. /* flags = SCTL_INIT_WDTR|SCTL_INIT_SDTR;*/
  262. if (sp->flags & SCG_RECV_DATA)
  263. flags |= SCTL_READ;
  264. if ((sp->flags & SCG_DISRE_ENA) == 0)
  265. flags |= SCTL_NO_ATN;
  266. sctl_io.flags = flags;
  267. movebytes(&sp->cdb, sctl_io.cdb, sp->cdb_len);
  268. sctl_io.cdb_length = sp->cdb_len;
  269. sctl_io.data_length = sp->size;
  270. sctl_io.data = sp->addr;
  271. if (sp->timeout == 0)
  272. sctl_io.max_msecs = 0;
  273. else
  274. sctl_io.max_msecs = (sp->timeout * 1000) + 500;
  275. errno = 0;
  276. sp->error = SCG_NO_ERROR;
  277. sp->sense_count = 0;
  278. sp->u_scb.cmd_scb[0] = 0;
  279. sp->resid = 0;
  280. ret = ioctl(f, SIOC_IO, &sctl_io);
  281. if (ret < 0) {
  282. sp->error = SCG_FATAL;
  283. sp->ux_errno = errno;
  284. return (ret);
  285. }
  286. if (scgp->debug)
  287. error("cdb_status: %X, size: %d xfer: %dn", sctl_io.cdb_status, sctl_io.data_length, sctl_io.data_xfer);
  288. if (sctl_io.cdb_status == 0 || sctl_io.cdb_status == 0x02)
  289. sp->resid = sp->size - sctl_io.data_xfer;
  290. if (sctl_io.cdb_status & SCTL_SELECT_TIMEOUT ||
  291. sctl_io.cdb_status & SCTL_INVALID_REQUEST) {
  292. sp->error = SCG_FATAL;
  293. } else if (sctl_io.cdb_status & SCTL_INCOMPLETE) {
  294. sp->error = SCG_TIMEOUT;
  295. } else if (sctl_io.cdb_status > 0xFF) {
  296. errmsgno(EX_BAD, "SCSI problems: cdb_status: %Xn", sctl_io.cdb_status);
  297. } else if ((sctl_io.cdb_status & 0xFF) != 0) {
  298. sp->error = SCG_RETRYABLE;
  299. sp->ux_errno = EIO;
  300. sp->u_scb.cmd_scb[0] = sctl_io.cdb_status & 0xFF;
  301. sp->sense_count = sctl_io.sense_xfer;
  302. if (sp->sense_count > SCG_MAX_SENSE)
  303. sp->sense_count = SCG_MAX_SENSE;
  304. if (sctl_io.sense_status != S_GOOD) {
  305. sp->sense_count = 0;
  306. } else {
  307. movebytes(sctl_io.sense, sp->u_sense.cmd_sense, sp->sense_count);
  308. }
  309. }
  310. return (ret);
  311. }
  312. #define sense u_sense.Sense