ses.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692
  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. unsigned char *page1;
  34. unsigned char *page2;
  35. unsigned 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 * HZ)
  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. unsigned 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, NULL);
  70. }
  71. static int ses_send_diag(struct scsi_device *sdev, int page_code,
  72. void *buf, int bufflen)
  73. {
  74. u32 result;
  75. unsigned 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, NULL);
  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. unsigned char *desc)
  93. {
  94. int i, j, count = 0, descriptor = ecomp->number;
  95. struct scsi_device *sdev = to_scsi_device(edev->edev.parent);
  96. struct ses_device *ses_dev = edev->scratch;
  97. unsigned char *type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  98. unsigned 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 unsigned 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->edev.parent);
  123. struct ses_device *ses_dev = edev->scratch;
  124. unsigned char *type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  125. unsigned 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. unsigned char *desc;
  143. desc = ses_get_page2_descriptor(edev, ecomp);
  144. if (desc)
  145. ecomp->fault = (desc[3] & 0x60) >> 4;
  146. }
  147. static int ses_set_fault(struct enclosure_device *edev,
  148. struct enclosure_component *ecomp,
  149. enum enclosure_component_setting val)
  150. {
  151. unsigned char desc[4] = {0 };
  152. switch (val) {
  153. case ENCLOSURE_SETTING_DISABLED:
  154. /* zero is disabled */
  155. break;
  156. case ENCLOSURE_SETTING_ENABLED:
  157. desc[2] = 0x02;
  158. break;
  159. default:
  160. /* SES doesn't do the SGPIO blink settings */
  161. return -EINVAL;
  162. }
  163. return ses_set_page2_descriptor(edev, ecomp, desc);
  164. }
  165. static void ses_get_status(struct enclosure_device *edev,
  166. struct enclosure_component *ecomp)
  167. {
  168. unsigned char *desc;
  169. desc = ses_get_page2_descriptor(edev, ecomp);
  170. if (desc)
  171. ecomp->status = (desc[0] & 0x0f);
  172. }
  173. static void ses_get_locate(struct enclosure_device *edev,
  174. struct enclosure_component *ecomp)
  175. {
  176. unsigned char *desc;
  177. desc = ses_get_page2_descriptor(edev, ecomp);
  178. if (desc)
  179. ecomp->locate = (desc[2] & 0x02) ? 1 : 0;
  180. }
  181. static int ses_set_locate(struct enclosure_device *edev,
  182. struct enclosure_component *ecomp,
  183. enum enclosure_component_setting val)
  184. {
  185. unsigned char desc[4] = {0 };
  186. switch (val) {
  187. case ENCLOSURE_SETTING_DISABLED:
  188. /* zero is disabled */
  189. break;
  190. case ENCLOSURE_SETTING_ENABLED:
  191. desc[2] = 0x02;
  192. break;
  193. default:
  194. /* SES doesn't do the SGPIO blink settings */
  195. return -EINVAL;
  196. }
  197. return ses_set_page2_descriptor(edev, ecomp, desc);
  198. }
  199. static int ses_set_active(struct enclosure_device *edev,
  200. struct enclosure_component *ecomp,
  201. enum enclosure_component_setting val)
  202. {
  203. unsigned char desc[4] = {0 };
  204. switch (val) {
  205. case ENCLOSURE_SETTING_DISABLED:
  206. /* zero is disabled */
  207. ecomp->active = 0;
  208. break;
  209. case ENCLOSURE_SETTING_ENABLED:
  210. desc[2] = 0x80;
  211. ecomp->active = 1;
  212. break;
  213. default:
  214. /* SES doesn't do the SGPIO blink settings */
  215. return -EINVAL;
  216. }
  217. return ses_set_page2_descriptor(edev, ecomp, desc);
  218. }
  219. static struct enclosure_component_callbacks ses_enclosure_callbacks = {
  220. .get_fault = ses_get_fault,
  221. .set_fault = ses_set_fault,
  222. .get_status = ses_get_status,
  223. .get_locate = ses_get_locate,
  224. .set_locate = ses_set_locate,
  225. .set_active = ses_set_active,
  226. };
  227. struct ses_host_edev {
  228. struct Scsi_Host *shost;
  229. struct enclosure_device *edev;
  230. };
  231. #if 0
  232. int ses_match_host(struct enclosure_device *edev, void *data)
  233. {
  234. struct ses_host_edev *sed = data;
  235. struct scsi_device *sdev;
  236. if (!scsi_is_sdev_device(edev->edev.parent))
  237. return 0;
  238. sdev = to_scsi_device(edev->edev.parent);
  239. if (sdev->host != sed->shost)
  240. return 0;
  241. sed->edev = edev;
  242. return 1;
  243. }
  244. #endif /* 0 */
  245. static void ses_process_descriptor(struct enclosure_component *ecomp,
  246. unsigned char *desc)
  247. {
  248. int eip = desc[0] & 0x10;
  249. int invalid = desc[0] & 0x80;
  250. enum scsi_protocol proto = desc[0] & 0x0f;
  251. u64 addr = 0;
  252. struct ses_component *scomp = ecomp->scratch;
  253. unsigned char *d;
  254. scomp->desc = desc;
  255. if (invalid)
  256. return;
  257. switch (proto) {
  258. case SCSI_PROTOCOL_SAS:
  259. if (eip)
  260. d = desc + 8;
  261. else
  262. d = desc + 4;
  263. /* only take the phy0 addr */
  264. addr = (u64)d[12] << 56 |
  265. (u64)d[13] << 48 |
  266. (u64)d[14] << 40 |
  267. (u64)d[15] << 32 |
  268. (u64)d[16] << 24 |
  269. (u64)d[17] << 16 |
  270. (u64)d[18] << 8 |
  271. (u64)d[19];
  272. break;
  273. default:
  274. /* FIXME: Need to add more protocols than just SAS */
  275. break;
  276. }
  277. scomp->addr = addr;
  278. }
  279. struct efd {
  280. u64 addr;
  281. struct device *dev;
  282. };
  283. static int ses_enclosure_find_by_addr(struct enclosure_device *edev,
  284. void *data)
  285. {
  286. struct efd *efd = data;
  287. int i;
  288. struct ses_component *scomp;
  289. if (!edev->component[0].scratch)
  290. return 0;
  291. for (i = 0; i < edev->components; i++) {
  292. scomp = edev->component[i].scratch;
  293. if (scomp->addr != efd->addr)
  294. continue;
  295. enclosure_add_device(edev, i, efd->dev);
  296. return 1;
  297. }
  298. return 0;
  299. }
  300. static void ses_match_to_enclosure(struct enclosure_device *edev,
  301. struct scsi_device *sdev)
  302. {
  303. unsigned char *buf;
  304. unsigned char *desc;
  305. unsigned int vpd_len;
  306. struct efd efd = {
  307. .addr = 0,
  308. };
  309. buf = scsi_get_vpd_page(sdev, 0x83);
  310. if (!buf)
  311. return;
  312. vpd_len = ((buf[2] << 8) | buf[3]) + 4;
  313. desc = buf + 4;
  314. while (desc < buf + vpd_len) {
  315. enum scsi_protocol proto = desc[0] >> 4;
  316. u8 code_set = desc[0] & 0x0f;
  317. u8 piv = desc[1] & 0x80;
  318. u8 assoc = (desc[1] & 0x30) >> 4;
  319. u8 type = desc[1] & 0x0f;
  320. u8 len = desc[3];
  321. if (piv && code_set == 1 && assoc == 1
  322. && proto == SCSI_PROTOCOL_SAS && type == 3 && len == 8)
  323. efd.addr = (u64)desc[4] << 56 |
  324. (u64)desc[5] << 48 |
  325. (u64)desc[6] << 40 |
  326. (u64)desc[7] << 32 |
  327. (u64)desc[8] << 24 |
  328. (u64)desc[9] << 16 |
  329. (u64)desc[10] << 8 |
  330. (u64)desc[11];
  331. desc += len + 4;
  332. }
  333. if (!efd.addr)
  334. goto free;
  335. efd.dev = &sdev->sdev_gendev;
  336. enclosure_for_each_device(ses_enclosure_find_by_addr, &efd);
  337. free:
  338. kfree(buf);
  339. }
  340. #define INIT_ALLOC_SIZE 32
  341. static int ses_intf_add(struct device *cdev,
  342. struct class_interface *intf)
  343. {
  344. struct scsi_device *sdev = to_scsi_device(cdev->parent);
  345. struct scsi_device *tmp_sdev;
  346. unsigned char *buf = NULL, *hdr_buf, *type_ptr, *desc_ptr = NULL,
  347. *addl_desc_ptr = NULL;
  348. struct ses_device *ses_dev;
  349. u32 result;
  350. int i, j, types, len, page7_len = 0, components = 0;
  351. int err = -ENOMEM;
  352. struct enclosure_device *edev;
  353. struct ses_component *scomp = NULL;
  354. if (!scsi_device_enclosure(sdev)) {
  355. /* not an enclosure, but might be in one */
  356. edev = enclosure_find(&sdev->host->shost_gendev);
  357. if (edev) {
  358. ses_match_to_enclosure(edev, sdev);
  359. put_device(&edev->edev);
  360. }
  361. return -ENODEV;
  362. }
  363. /* TYPE_ENCLOSURE prints a message in probe */
  364. if (sdev->type != TYPE_ENCLOSURE)
  365. sdev_printk(KERN_NOTICE, sdev, "Embedded Enclosure Device\n");
  366. ses_dev = kzalloc(sizeof(*ses_dev), GFP_KERNEL);
  367. hdr_buf = kzalloc(INIT_ALLOC_SIZE, GFP_KERNEL);
  368. if (!hdr_buf || !ses_dev)
  369. goto err_init_free;
  370. result = ses_recv_diag(sdev, 1, hdr_buf, INIT_ALLOC_SIZE);
  371. if (result)
  372. goto recv_failed;
  373. if (hdr_buf[1] != 0) {
  374. /* FIXME: need subenclosure support; I've just never
  375. * seen a device with subenclosures and it makes the
  376. * traversal routines more complex */
  377. sdev_printk(KERN_ERR, sdev,
  378. "FIXME driver has no support for subenclosures (%d)\n",
  379. hdr_buf[1]);
  380. goto err_free;
  381. }
  382. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  383. buf = kzalloc(len, GFP_KERNEL);
  384. if (!buf)
  385. goto err_free;
  386. result = ses_recv_diag(sdev, 1, buf, len);
  387. if (result)
  388. goto recv_failed;
  389. types = buf[10];
  390. type_ptr = buf + 12 + buf[11];
  391. for (i = 0; i < types; i++, type_ptr += 4) {
  392. if (type_ptr[0] == ENCLOSURE_COMPONENT_DEVICE ||
  393. type_ptr[0] == ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  394. components += type_ptr[1];
  395. }
  396. ses_dev->page1 = buf;
  397. ses_dev->page1_len = len;
  398. buf = NULL;
  399. result = ses_recv_diag(sdev, 2, hdr_buf, INIT_ALLOC_SIZE);
  400. if (result)
  401. goto recv_failed;
  402. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  403. buf = kzalloc(len, GFP_KERNEL);
  404. if (!buf)
  405. goto err_free;
  406. /* make sure getting page 2 actually works */
  407. result = ses_recv_diag(sdev, 2, buf, len);
  408. if (result)
  409. goto recv_failed;
  410. ses_dev->page2 = buf;
  411. ses_dev->page2_len = len;
  412. buf = NULL;
  413. /* The additional information page --- allows us
  414. * to match up the devices */
  415. result = ses_recv_diag(sdev, 10, hdr_buf, INIT_ALLOC_SIZE);
  416. if (!result) {
  417. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  418. buf = kzalloc(len, GFP_KERNEL);
  419. if (!buf)
  420. goto err_free;
  421. result = ses_recv_diag(sdev, 10, buf, len);
  422. if (result)
  423. goto recv_failed;
  424. ses_dev->page10 = buf;
  425. ses_dev->page10_len = len;
  426. buf = NULL;
  427. }
  428. scomp = kzalloc(sizeof(struct ses_component) * components, GFP_KERNEL);
  429. if (!scomp)
  430. goto err_free;
  431. edev = enclosure_register(cdev->parent, dev_name(&sdev->sdev_gendev),
  432. components, &ses_enclosure_callbacks);
  433. if (IS_ERR(edev)) {
  434. err = PTR_ERR(edev);
  435. goto err_free;
  436. }
  437. edev->scratch = ses_dev;
  438. for (i = 0; i < components; i++)
  439. edev->component[i].scratch = scomp + i;
  440. /* Page 7 for the descriptors is optional */
  441. result = ses_recv_diag(sdev, 7, hdr_buf, INIT_ALLOC_SIZE);
  442. if (result)
  443. goto simple_populate;
  444. page7_len = len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  445. /* add 1 for trailing '\0' we'll use */
  446. buf = kzalloc(len + 1, GFP_KERNEL);
  447. if (!buf)
  448. goto simple_populate;
  449. result = ses_recv_diag(sdev, 7, buf, len);
  450. if (result) {
  451. simple_populate:
  452. kfree(buf);
  453. buf = NULL;
  454. desc_ptr = NULL;
  455. addl_desc_ptr = NULL;
  456. } else {
  457. desc_ptr = buf + 8;
  458. len = (desc_ptr[2] << 8) + desc_ptr[3];
  459. /* skip past overall descriptor */
  460. desc_ptr += len + 4;
  461. if (ses_dev->page10)
  462. addl_desc_ptr = ses_dev->page10 + 8;
  463. }
  464. type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  465. components = 0;
  466. for (i = 0; i < types; i++, type_ptr += 4) {
  467. for (j = 0; j < type_ptr[1]; j++) {
  468. char *name = NULL;
  469. struct enclosure_component *ecomp;
  470. if (desc_ptr) {
  471. if (desc_ptr >= buf + page7_len) {
  472. desc_ptr = NULL;
  473. } else {
  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. }
  482. if (type_ptr[0] == ENCLOSURE_COMPONENT_DEVICE ||
  483. type_ptr[0] == ENCLOSURE_COMPONENT_ARRAY_DEVICE) {
  484. ecomp = enclosure_component_register(edev,
  485. components++,
  486. type_ptr[0],
  487. name);
  488. if (!IS_ERR(ecomp) && addl_desc_ptr)
  489. ses_process_descriptor(ecomp,
  490. addl_desc_ptr);
  491. }
  492. if (desc_ptr)
  493. desc_ptr += len;
  494. if (addl_desc_ptr)
  495. addl_desc_ptr += addl_desc_ptr[1] + 2;
  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(scomp);
  515. kfree(ses_dev->page10);
  516. kfree(ses_dev->page2);
  517. kfree(ses_dev->page1);
  518. err_init_free:
  519. kfree(ses_dev);
  520. kfree(hdr_buf);
  521. sdev_printk(KERN_ERR, sdev, "Failed to bind enclosure %d\n", err);
  522. return err;
  523. }
  524. static int ses_remove(struct device *dev)
  525. {
  526. return 0;
  527. }
  528. static void ses_intf_remove(struct device *cdev,
  529. struct class_interface *intf)
  530. {
  531. struct scsi_device *sdev = to_scsi_device(cdev->parent);
  532. struct enclosure_device *edev;
  533. struct ses_device *ses_dev;
  534. if (!scsi_device_enclosure(sdev))
  535. return;
  536. edev = enclosure_find(cdev->parent);
  537. if (!edev)
  538. return;
  539. ses_dev = edev->scratch;
  540. edev->scratch = NULL;
  541. kfree(ses_dev->page10);
  542. kfree(ses_dev->page1);
  543. kfree(ses_dev->page2);
  544. kfree(ses_dev);
  545. kfree(edev->component[0].scratch);
  546. put_device(&edev->edev);
  547. enclosure_unregister(edev);
  548. }
  549. static struct class_interface ses_interface = {
  550. .add_dev = ses_intf_add,
  551. .remove_dev = ses_intf_remove,
  552. };
  553. static struct scsi_driver ses_template = {
  554. .owner = THIS_MODULE,
  555. .gendrv = {
  556. .name = "ses",
  557. .probe = ses_probe,
  558. .remove = ses_remove,
  559. },
  560. };
  561. static int __init ses_init(void)
  562. {
  563. int err;
  564. err = scsi_register_interface(&ses_interface);
  565. if (err)
  566. return err;
  567. err = scsi_register_driver(&ses_template.gendrv);
  568. if (err)
  569. goto out_unreg;
  570. return 0;
  571. out_unreg:
  572. scsi_unregister_interface(&ses_interface);
  573. return err;
  574. }
  575. static void __exit ses_exit(void)
  576. {
  577. scsi_unregister_driver(&ses_template.gendrv);
  578. scsi_unregister_interface(&ses_interface);
  579. }
  580. module_init(ses_init);
  581. module_exit(ses_exit);
  582. MODULE_ALIAS_SCSI_DEVICE(TYPE_ENCLOSURE);
  583. MODULE_AUTHOR("James Bottomley");
  584. MODULE_DESCRIPTION("SCSI Enclosure Services (ses) driver");
  585. MODULE_LICENSE("GPL v2");