ses.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689
  1. /*
  2. * SCSI Enclosure Services
  3. *
  4. * Copyright (C) 2008 James Bottomley <James.Bottomley@HansenPartnership.com>
  5. *
  6. **-----------------------------------------------------------------------------
  7. **
  8. ** This program is free software; you can redistribute it and/or
  9. ** modify it under the terms of the GNU General Public License
  10. ** version 2 as published by the Free Software Foundation.
  11. **
  12. ** This program is distributed in the hope that it will be useful,
  13. ** but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. ** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. ** GNU General Public License for more details.
  16. **
  17. ** You should have received a copy of the GNU General Public License
  18. ** along with this program; if not, write to the Free Software
  19. ** Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  20. **
  21. **-----------------------------------------------------------------------------
  22. */
  23. #include <linux/module.h>
  24. #include <linux/kernel.h>
  25. #include <linux/enclosure.h>
  26. #include <scsi/scsi.h>
  27. #include <scsi/scsi_cmnd.h>
  28. #include <scsi/scsi_dbg.h>
  29. #include <scsi/scsi_device.h>
  30. #include <scsi/scsi_driver.h>
  31. #include <scsi/scsi_host.h>
  32. struct ses_device {
  33. char *page1;
  34. char *page2;
  35. char *page10;
  36. short page1_len;
  37. short page2_len;
  38. short page10_len;
  39. };
  40. struct ses_component {
  41. u64 addr;
  42. unsigned char *desc;
  43. };
  44. static int ses_probe(struct device *dev)
  45. {
  46. struct scsi_device *sdev = to_scsi_device(dev);
  47. int err = -ENODEV;
  48. if (sdev->type != TYPE_ENCLOSURE)
  49. goto out;
  50. err = 0;
  51. sdev_printk(KERN_NOTICE, sdev, "Attached Enclosure device\n");
  52. out:
  53. return err;
  54. }
  55. #define SES_TIMEOUT 30
  56. #define SES_RETRIES 3
  57. static int ses_recv_diag(struct scsi_device *sdev, int page_code,
  58. void *buf, int bufflen)
  59. {
  60. char cmd[] = {
  61. RECEIVE_DIAGNOSTIC,
  62. 1, /* Set PCV bit */
  63. page_code,
  64. bufflen >> 8,
  65. bufflen & 0xff,
  66. 0
  67. };
  68. return scsi_execute_req(sdev, cmd, DMA_FROM_DEVICE, buf, bufflen,
  69. NULL, SES_TIMEOUT, SES_RETRIES);
  70. }
  71. static int ses_send_diag(struct scsi_device *sdev, int page_code,
  72. void *buf, int bufflen)
  73. {
  74. u32 result;
  75. char cmd[] = {
  76. SEND_DIAGNOSTIC,
  77. 0x10, /* Set PF bit */
  78. 0,
  79. bufflen >> 8,
  80. bufflen & 0xff,
  81. 0
  82. };
  83. result = scsi_execute_req(sdev, cmd, DMA_TO_DEVICE, buf, bufflen,
  84. NULL, SES_TIMEOUT, SES_RETRIES);
  85. if (result)
  86. sdev_printk(KERN_ERR, sdev, "SEND DIAGNOSTIC result: %8x\n",
  87. result);
  88. return result;
  89. }
  90. static int ses_set_page2_descriptor(struct enclosure_device *edev,
  91. struct enclosure_component *ecomp,
  92. char *desc)
  93. {
  94. int i, j, count = 0, descriptor = ecomp->number;
  95. struct scsi_device *sdev = to_scsi_device(edev->cdev.dev);
  96. struct ses_device *ses_dev = edev->scratch;
  97. char *type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  98. char *desc_ptr = ses_dev->page2 + 8;
  99. /* Clear everything */
  100. memset(desc_ptr, 0, ses_dev->page2_len - 8);
  101. for (i = 0; i < ses_dev->page1[10]; i++, type_ptr += 4) {
  102. for (j = 0; j < type_ptr[1]; j++) {
  103. desc_ptr += 4;
  104. if (type_ptr[0] != ENCLOSURE_COMPONENT_DEVICE &&
  105. type_ptr[0] != ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  106. continue;
  107. if (count++ == descriptor) {
  108. memcpy(desc_ptr, desc, 4);
  109. /* set select */
  110. desc_ptr[0] |= 0x80;
  111. /* clear reserved, just in case */
  112. desc_ptr[0] &= 0xf0;
  113. }
  114. }
  115. }
  116. return ses_send_diag(sdev, 2, ses_dev->page2, ses_dev->page2_len);
  117. }
  118. static char *ses_get_page2_descriptor(struct enclosure_device *edev,
  119. struct enclosure_component *ecomp)
  120. {
  121. int i, j, count = 0, descriptor = ecomp->number;
  122. struct scsi_device *sdev = to_scsi_device(edev->cdev.dev);
  123. struct ses_device *ses_dev = edev->scratch;
  124. char *type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  125. char *desc_ptr = ses_dev->page2 + 8;
  126. ses_recv_diag(sdev, 2, ses_dev->page2, ses_dev->page2_len);
  127. for (i = 0; i < ses_dev->page1[10]; i++, type_ptr += 4) {
  128. for (j = 0; j < type_ptr[1]; j++) {
  129. desc_ptr += 4;
  130. if (type_ptr[0] != ENCLOSURE_COMPONENT_DEVICE &&
  131. type_ptr[0] != ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  132. continue;
  133. if (count++ == descriptor)
  134. return desc_ptr;
  135. }
  136. }
  137. return NULL;
  138. }
  139. static void ses_get_fault(struct enclosure_device *edev,
  140. struct enclosure_component *ecomp)
  141. {
  142. char *desc;
  143. desc = ses_get_page2_descriptor(edev, ecomp);
  144. ecomp->fault = (desc[3] & 0x60) >> 4;
  145. }
  146. static int ses_set_fault(struct enclosure_device *edev,
  147. struct enclosure_component *ecomp,
  148. enum enclosure_component_setting val)
  149. {
  150. char desc[4] = {0 };
  151. switch (val) {
  152. case ENCLOSURE_SETTING_DISABLED:
  153. /* zero is disabled */
  154. break;
  155. case ENCLOSURE_SETTING_ENABLED:
  156. desc[2] = 0x02;
  157. break;
  158. default:
  159. /* SES doesn't do the SGPIO blink settings */
  160. return -EINVAL;
  161. }
  162. return ses_set_page2_descriptor(edev, ecomp, desc);
  163. }
  164. static void ses_get_status(struct enclosure_device *edev,
  165. struct enclosure_component *ecomp)
  166. {
  167. char *desc;
  168. desc = ses_get_page2_descriptor(edev, ecomp);
  169. ecomp->status = (desc[0] & 0x0f);
  170. }
  171. static void ses_get_locate(struct enclosure_device *edev,
  172. struct enclosure_component *ecomp)
  173. {
  174. char *desc;
  175. desc = ses_get_page2_descriptor(edev, ecomp);
  176. ecomp->locate = (desc[2] & 0x02) ? 1 : 0;
  177. }
  178. static int ses_set_locate(struct enclosure_device *edev,
  179. struct enclosure_component *ecomp,
  180. enum enclosure_component_setting val)
  181. {
  182. char desc[4] = {0 };
  183. switch (val) {
  184. case ENCLOSURE_SETTING_DISABLED:
  185. /* zero is disabled */
  186. break;
  187. case ENCLOSURE_SETTING_ENABLED:
  188. desc[2] = 0x02;
  189. break;
  190. default:
  191. /* SES doesn't do the SGPIO blink settings */
  192. return -EINVAL;
  193. }
  194. return ses_set_page2_descriptor(edev, ecomp, desc);
  195. }
  196. static int ses_set_active(struct enclosure_device *edev,
  197. struct enclosure_component *ecomp,
  198. enum enclosure_component_setting val)
  199. {
  200. char desc[4] = {0 };
  201. switch (val) {
  202. case ENCLOSURE_SETTING_DISABLED:
  203. /* zero is disabled */
  204. ecomp->active = 0;
  205. break;
  206. case ENCLOSURE_SETTING_ENABLED:
  207. desc[2] = 0x80;
  208. ecomp->active = 1;
  209. break;
  210. default:
  211. /* SES doesn't do the SGPIO blink settings */
  212. return -EINVAL;
  213. }
  214. return ses_set_page2_descriptor(edev, ecomp, desc);
  215. }
  216. static struct enclosure_component_callbacks ses_enclosure_callbacks = {
  217. .get_fault = ses_get_fault,
  218. .set_fault = ses_set_fault,
  219. .get_status = ses_get_status,
  220. .get_locate = ses_get_locate,
  221. .set_locate = ses_set_locate,
  222. .set_active = ses_set_active,
  223. };
  224. struct ses_host_edev {
  225. struct Scsi_Host *shost;
  226. struct enclosure_device *edev;
  227. };
  228. int ses_match_host(struct enclosure_device *edev, void *data)
  229. {
  230. struct ses_host_edev *sed = data;
  231. struct scsi_device *sdev;
  232. if (!scsi_is_sdev_device(edev->cdev.dev))
  233. return 0;
  234. sdev = to_scsi_device(edev->cdev.dev);
  235. if (sdev->host != sed->shost)
  236. return 0;
  237. sed->edev = edev;
  238. return 1;
  239. }
  240. static void ses_process_descriptor(struct enclosure_component *ecomp,
  241. unsigned char *desc)
  242. {
  243. int eip = desc[0] & 0x10;
  244. int invalid = desc[0] & 0x80;
  245. enum scsi_protocol proto = desc[0] & 0x0f;
  246. u64 addr = 0;
  247. struct ses_component *scomp = ecomp->scratch;
  248. unsigned char *d;
  249. scomp->desc = desc;
  250. if (invalid)
  251. return;
  252. switch (proto) {
  253. case SCSI_PROTOCOL_SAS:
  254. if (eip)
  255. d = desc + 8;
  256. else
  257. d = desc + 4;
  258. /* only take the phy0 addr */
  259. addr = (u64)d[12] << 56 |
  260. (u64)d[13] << 48 |
  261. (u64)d[14] << 40 |
  262. (u64)d[15] << 32 |
  263. (u64)d[16] << 24 |
  264. (u64)d[17] << 16 |
  265. (u64)d[18] << 8 |
  266. (u64)d[19];
  267. break;
  268. default:
  269. /* FIXME: Need to add more protocols than just SAS */
  270. break;
  271. }
  272. scomp->addr = addr;
  273. }
  274. struct efd {
  275. u64 addr;
  276. struct device *dev;
  277. };
  278. static int ses_enclosure_find_by_addr(struct enclosure_device *edev,
  279. void *data)
  280. {
  281. struct efd *efd = data;
  282. int i;
  283. struct ses_component *scomp;
  284. if (!edev->component[0].scratch)
  285. return 0;
  286. for (i = 0; i < edev->components; i++) {
  287. scomp = edev->component[i].scratch;
  288. if (scomp->addr != efd->addr)
  289. continue;
  290. enclosure_add_device(edev, i, efd->dev);
  291. return 1;
  292. }
  293. return 0;
  294. }
  295. #define VPD_INQUIRY_SIZE 512
  296. static void ses_match_to_enclosure(struct enclosure_device *edev,
  297. struct scsi_device *sdev)
  298. {
  299. unsigned char *buf = kmalloc(VPD_INQUIRY_SIZE, GFP_KERNEL);
  300. unsigned char *desc;
  301. int len;
  302. struct efd efd = {
  303. .addr = 0,
  304. };
  305. unsigned char cmd[] = {
  306. INQUIRY,
  307. 1,
  308. 0x83,
  309. VPD_INQUIRY_SIZE >> 8,
  310. VPD_INQUIRY_SIZE & 0xff,
  311. 0
  312. };
  313. if (!buf)
  314. return;
  315. if (scsi_execute_req(sdev, cmd, DMA_FROM_DEVICE, buf,
  316. VPD_INQUIRY_SIZE, NULL, SES_TIMEOUT, SES_RETRIES))
  317. goto free;
  318. len = (buf[2] << 8) + buf[3];
  319. desc = buf + 4;
  320. while (desc < buf + len) {
  321. enum scsi_protocol proto = desc[0] >> 4;
  322. u8 code_set = desc[0] & 0x0f;
  323. u8 piv = desc[1] & 0x80;
  324. u8 assoc = (desc[1] & 0x30) >> 4;
  325. u8 type = desc[1] & 0x0f;
  326. u8 len = desc[3];
  327. if (piv && code_set == 1 && assoc == 1 && code_set == 1
  328. && proto == SCSI_PROTOCOL_SAS && type == 3 && len == 8)
  329. efd.addr = (u64)desc[4] << 56 |
  330. (u64)desc[5] << 48 |
  331. (u64)desc[6] << 40 |
  332. (u64)desc[7] << 32 |
  333. (u64)desc[8] << 24 |
  334. (u64)desc[9] << 16 |
  335. (u64)desc[10] << 8 |
  336. (u64)desc[11];
  337. desc += len + 4;
  338. }
  339. if (!efd.addr)
  340. goto free;
  341. efd.dev = &sdev->sdev_gendev;
  342. enclosure_for_each_device(ses_enclosure_find_by_addr, &efd);
  343. free:
  344. kfree(buf);
  345. }
  346. #define INIT_ALLOC_SIZE 32
  347. static int ses_intf_add(struct class_device *cdev,
  348. struct class_interface *intf)
  349. {
  350. struct scsi_device *sdev = to_scsi_device(cdev->dev);
  351. struct scsi_device *tmp_sdev;
  352. unsigned char *buf = NULL, *hdr_buf, *type_ptr, *desc_ptr,
  353. *addl_desc_ptr;
  354. struct ses_device *ses_dev;
  355. u32 result;
  356. int i, j, types, len, components = 0;
  357. int err = -ENOMEM;
  358. struct enclosure_device *edev;
  359. struct ses_component *scomp;
  360. if (!scsi_device_enclosure(sdev)) {
  361. /* not an enclosure, but might be in one */
  362. edev = enclosure_find(&sdev->host->shost_gendev);
  363. if (edev) {
  364. ses_match_to_enclosure(edev, sdev);
  365. class_device_put(&edev->cdev);
  366. }
  367. return -ENODEV;
  368. }
  369. /* TYPE_ENCLOSURE prints a message in probe */
  370. if (sdev->type != TYPE_ENCLOSURE)
  371. sdev_printk(KERN_NOTICE, sdev, "Embedded Enclosure Device\n");
  372. ses_dev = kzalloc(sizeof(*ses_dev), GFP_KERNEL);
  373. hdr_buf = kzalloc(INIT_ALLOC_SIZE, GFP_KERNEL);
  374. if (!hdr_buf || !ses_dev)
  375. goto err_init_free;
  376. result = ses_recv_diag(sdev, 1, hdr_buf, INIT_ALLOC_SIZE);
  377. if (result)
  378. goto recv_failed;
  379. if (hdr_buf[1] != 0) {
  380. /* FIXME: need subenclosure support; I've just never
  381. * seen a device with subenclosures and it makes the
  382. * traversal routines more complex */
  383. sdev_printk(KERN_ERR, sdev,
  384. "FIXME driver has no support for subenclosures (%d)\n",
  385. buf[1]);
  386. goto err_free;
  387. }
  388. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  389. buf = kzalloc(len, GFP_KERNEL);
  390. if (!buf)
  391. goto err_free;
  392. ses_dev->page1 = buf;
  393. ses_dev->page1_len = len;
  394. result = ses_recv_diag(sdev, 1, buf, len);
  395. if (result)
  396. goto recv_failed;
  397. types = buf[10];
  398. len = buf[11];
  399. type_ptr = buf + 12 + len;
  400. for (i = 0; i < types; i++, type_ptr += 4) {
  401. if (type_ptr[0] == ENCLOSURE_COMPONENT_DEVICE ||
  402. type_ptr[0] == ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  403. components += type_ptr[1];
  404. }
  405. result = ses_recv_diag(sdev, 2, hdr_buf, INIT_ALLOC_SIZE);
  406. if (result)
  407. goto recv_failed;
  408. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  409. buf = kzalloc(len, GFP_KERNEL);
  410. if (!buf)
  411. goto err_free;
  412. /* make sure getting page 2 actually works */
  413. result = ses_recv_diag(sdev, 2, buf, len);
  414. if (result)
  415. goto recv_failed;
  416. ses_dev->page2 = buf;
  417. ses_dev->page2_len = len;
  418. /* The additional information page --- allows us
  419. * to match up the devices */
  420. result = ses_recv_diag(sdev, 10, hdr_buf, INIT_ALLOC_SIZE);
  421. if (result)
  422. goto no_page10;
  423. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  424. buf = kzalloc(len, GFP_KERNEL);
  425. if (!buf)
  426. goto err_free;
  427. result = ses_recv_diag(sdev, 10, buf, len);
  428. if (result)
  429. goto recv_failed;
  430. ses_dev->page10 = buf;
  431. ses_dev->page10_len = len;
  432. no_page10:
  433. scomp = kmalloc(sizeof(struct ses_component) * components, GFP_KERNEL);
  434. if (!scomp)
  435. goto err_free;
  436. edev = enclosure_register(cdev->dev, sdev->sdev_gendev.bus_id,
  437. components, &ses_enclosure_callbacks);
  438. if (IS_ERR(edev)) {
  439. err = PTR_ERR(edev);
  440. goto err_free;
  441. }
  442. edev->scratch = ses_dev;
  443. for (i = 0; i < components; i++)
  444. edev->component[i].scratch = scomp++;
  445. /* Page 7 for the descriptors is optional */
  446. buf = NULL;
  447. result = ses_recv_diag(sdev, 7, hdr_buf, INIT_ALLOC_SIZE);
  448. if (result)
  449. goto simple_populate;
  450. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  451. /* add 1 for trailing '\0' we'll use */
  452. buf = kzalloc(len + 1, GFP_KERNEL);
  453. result = ses_recv_diag(sdev, 7, buf, len);
  454. if (result) {
  455. simple_populate:
  456. kfree(buf);
  457. buf = NULL;
  458. desc_ptr = NULL;
  459. addl_desc_ptr = NULL;
  460. } else {
  461. desc_ptr = buf + 8;
  462. len = (desc_ptr[2] << 8) + desc_ptr[3];
  463. /* skip past overall descriptor */
  464. desc_ptr += len + 4;
  465. addl_desc_ptr = ses_dev->page10 + 8;
  466. }
  467. type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  468. components = 0;
  469. for (i = 0; i < types; i++, type_ptr += 4) {
  470. for (j = 0; j < type_ptr[1]; j++) {
  471. char *name = NULL;
  472. struct enclosure_component *ecomp;
  473. if (desc_ptr) {
  474. len = (desc_ptr[2] << 8) + desc_ptr[3];
  475. desc_ptr += 4;
  476. /* Add trailing zero - pushes into
  477. * reserved space */
  478. desc_ptr[len] = '\0';
  479. name = desc_ptr;
  480. }
  481. if (type_ptr[0] != ENCLOSURE_COMPONENT_DEVICE &&
  482. type_ptr[0] != ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  483. continue;
  484. ecomp = enclosure_component_register(edev,
  485. components++,
  486. type_ptr[0],
  487. name);
  488. if (desc_ptr) {
  489. desc_ptr += len;
  490. if (!IS_ERR(ecomp))
  491. ses_process_descriptor(ecomp,
  492. addl_desc_ptr);
  493. if (addl_desc_ptr)
  494. addl_desc_ptr += addl_desc_ptr[1] + 2;
  495. }
  496. }
  497. }
  498. kfree(buf);
  499. kfree(hdr_buf);
  500. /* see if there are any devices matching before
  501. * we found the enclosure */
  502. shost_for_each_device(tmp_sdev, sdev->host) {
  503. if (tmp_sdev->lun != 0 || scsi_device_enclosure(tmp_sdev))
  504. continue;
  505. ses_match_to_enclosure(edev, tmp_sdev);
  506. }
  507. return 0;
  508. recv_failed:
  509. sdev_printk(KERN_ERR, sdev, "Failed to get diagnostic page 0x%x\n",
  510. result);
  511. err = -ENODEV;
  512. err_free:
  513. kfree(buf);
  514. kfree(ses_dev->page10);
  515. kfree(ses_dev->page2);
  516. kfree(ses_dev->page1);
  517. err_init_free:
  518. kfree(ses_dev);
  519. kfree(hdr_buf);
  520. sdev_printk(KERN_ERR, sdev, "Failed to bind enclosure %d\n", err);
  521. return err;
  522. }
  523. static int ses_remove(struct device *dev)
  524. {
  525. return 0;
  526. }
  527. static void ses_intf_remove(struct class_device *cdev,
  528. struct class_interface *intf)
  529. {
  530. struct scsi_device *sdev = to_scsi_device(cdev->dev);
  531. struct enclosure_device *edev;
  532. struct ses_device *ses_dev;
  533. if (!scsi_device_enclosure(sdev))
  534. return;
  535. edev = enclosure_find(cdev->dev);
  536. if (!edev)
  537. return;
  538. ses_dev = edev->scratch;
  539. edev->scratch = NULL;
  540. kfree(ses_dev->page1);
  541. kfree(ses_dev->page2);
  542. kfree(ses_dev);
  543. kfree(edev->component[0].scratch);
  544. class_device_put(&edev->cdev);
  545. enclosure_unregister(edev);
  546. }
  547. static struct class_interface ses_interface = {
  548. .add = ses_intf_add,
  549. .remove = ses_intf_remove,
  550. };
  551. static struct scsi_driver ses_template = {
  552. .owner = THIS_MODULE,
  553. .gendrv = {
  554. .name = "ses",
  555. .probe = ses_probe,
  556. .remove = ses_remove,
  557. },
  558. };
  559. static int __init ses_init(void)
  560. {
  561. int err;
  562. err = scsi_register_interface(&ses_interface);
  563. if (err)
  564. return err;
  565. err = scsi_register_driver(&ses_template.gendrv);
  566. if (err)
  567. goto out_unreg;
  568. return 0;
  569. out_unreg:
  570. scsi_unregister_interface(&ses_interface);
  571. return err;
  572. }
  573. static void __exit ses_exit(void)
  574. {
  575. scsi_unregister_driver(&ses_template.gendrv);
  576. scsi_unregister_interface(&ses_interface);
  577. }
  578. module_init(ses_init);
  579. module_exit(ses_exit);
  580. MODULE_ALIAS_SCSI_DEVICE(TYPE_ENCLOSURE);
  581. MODULE_AUTHOR("James Bottomley");
  582. MODULE_DESCRIPTION("SCSI Enclosure Services (ses) driver");
  583. MODULE_LICENSE("GPL v2");