ses.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739
  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. #define INIT_ALLOC_SIZE 32
  301. static void ses_enclosure_data_process(struct enclosure_device *edev,
  302. struct scsi_device *sdev,
  303. int create)
  304. {
  305. u32 result;
  306. unsigned char *buf = NULL, *type_ptr, *desc_ptr, *addl_desc_ptr = NULL;
  307. int i, j, page7_len, len, components;
  308. struct ses_device *ses_dev = edev->scratch;
  309. int types = ses_dev->page1[10];
  310. unsigned char *hdr_buf = kzalloc(INIT_ALLOC_SIZE, GFP_KERNEL);
  311. if (!hdr_buf)
  312. goto simple_populate;
  313. /* re-read page 10 */
  314. if (ses_dev->page10)
  315. ses_recv_diag(sdev, 10, ses_dev->page10, ses_dev->page10_len);
  316. /* Page 7 for the descriptors is optional */
  317. result = ses_recv_diag(sdev, 7, hdr_buf, INIT_ALLOC_SIZE);
  318. if (result)
  319. goto simple_populate;
  320. page7_len = len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  321. /* add 1 for trailing '\0' we'll use */
  322. buf = kzalloc(len + 1, GFP_KERNEL);
  323. if (!buf)
  324. goto simple_populate;
  325. result = ses_recv_diag(sdev, 7, buf, len);
  326. if (result) {
  327. simple_populate:
  328. kfree(buf);
  329. buf = NULL;
  330. desc_ptr = NULL;
  331. len = 0;
  332. page7_len = 0;
  333. } else {
  334. desc_ptr = buf + 8;
  335. len = (desc_ptr[2] << 8) + desc_ptr[3];
  336. /* skip past overall descriptor */
  337. desc_ptr += len + 4;
  338. if (ses_dev->page10)
  339. addl_desc_ptr = ses_dev->page10 + 8;
  340. }
  341. type_ptr = ses_dev->page1 + 12 + ses_dev->page1[11];
  342. components = 0;
  343. for (i = 0; i < types; i++, type_ptr += 4) {
  344. for (j = 0; j < type_ptr[1]; j++) {
  345. char *name = NULL;
  346. struct enclosure_component *ecomp;
  347. if (desc_ptr) {
  348. if (desc_ptr >= buf + page7_len) {
  349. desc_ptr = NULL;
  350. } else {
  351. len = (desc_ptr[2] << 8) + desc_ptr[3];
  352. desc_ptr += 4;
  353. /* Add trailing zero - pushes into
  354. * reserved space */
  355. desc_ptr[len] = '\0';
  356. name = desc_ptr;
  357. }
  358. }
  359. if (type_ptr[0] == ENCLOSURE_COMPONENT_DEVICE ||
  360. type_ptr[0] == ENCLOSURE_COMPONENT_ARRAY_DEVICE) {
  361. if (create)
  362. ecomp = enclosure_component_register(edev,
  363. components++,
  364. type_ptr[0],
  365. name);
  366. else
  367. ecomp = &edev->component[components++];
  368. if (!IS_ERR(ecomp) && addl_desc_ptr)
  369. ses_process_descriptor(ecomp,
  370. addl_desc_ptr);
  371. }
  372. if (desc_ptr)
  373. desc_ptr += len;
  374. if (addl_desc_ptr)
  375. addl_desc_ptr += addl_desc_ptr[1] + 2;
  376. }
  377. }
  378. kfree(buf);
  379. kfree(hdr_buf);
  380. }
  381. static void ses_match_to_enclosure(struct enclosure_device *edev,
  382. struct scsi_device *sdev)
  383. {
  384. unsigned char *buf;
  385. unsigned char *desc;
  386. unsigned int vpd_len;
  387. struct efd efd = {
  388. .addr = 0,
  389. };
  390. buf = scsi_get_vpd_page(sdev, 0x83);
  391. if (!buf)
  392. return;
  393. ses_enclosure_data_process(edev, to_scsi_device(edev->edev.parent), 0);
  394. vpd_len = ((buf[2] << 8) | buf[3]) + 4;
  395. desc = buf + 4;
  396. while (desc < buf + vpd_len) {
  397. enum scsi_protocol proto = desc[0] >> 4;
  398. u8 code_set = desc[0] & 0x0f;
  399. u8 piv = desc[1] & 0x80;
  400. u8 assoc = (desc[1] & 0x30) >> 4;
  401. u8 type = desc[1] & 0x0f;
  402. u8 len = desc[3];
  403. if (piv && code_set == 1 && assoc == 1
  404. && proto == SCSI_PROTOCOL_SAS && type == 3 && len == 8)
  405. efd.addr = (u64)desc[4] << 56 |
  406. (u64)desc[5] << 48 |
  407. (u64)desc[6] << 40 |
  408. (u64)desc[7] << 32 |
  409. (u64)desc[8] << 24 |
  410. (u64)desc[9] << 16 |
  411. (u64)desc[10] << 8 |
  412. (u64)desc[11];
  413. desc += len + 4;
  414. }
  415. if (!efd.addr)
  416. goto free;
  417. efd.dev = &sdev->sdev_gendev;
  418. enclosure_for_each_device(ses_enclosure_find_by_addr, &efd);
  419. free:
  420. kfree(buf);
  421. }
  422. static int ses_intf_add(struct device *cdev,
  423. struct class_interface *intf)
  424. {
  425. struct scsi_device *sdev = to_scsi_device(cdev->parent);
  426. struct scsi_device *tmp_sdev;
  427. unsigned char *buf = NULL, *hdr_buf, *type_ptr;
  428. struct ses_device *ses_dev;
  429. u32 result;
  430. int i, types, len, components = 0;
  431. int err = -ENOMEM;
  432. struct enclosure_device *edev;
  433. struct ses_component *scomp = NULL;
  434. if (!scsi_device_enclosure(sdev)) {
  435. /* not an enclosure, but might be in one */
  436. struct enclosure_device *prev = NULL;
  437. while ((edev = enclosure_find(&sdev->host->shost_gendev, prev)) != NULL) {
  438. ses_match_to_enclosure(edev, sdev);
  439. prev = edev;
  440. }
  441. return -ENODEV;
  442. }
  443. /* TYPE_ENCLOSURE prints a message in probe */
  444. if (sdev->type != TYPE_ENCLOSURE)
  445. sdev_printk(KERN_NOTICE, sdev, "Embedded Enclosure Device\n");
  446. ses_dev = kzalloc(sizeof(*ses_dev), GFP_KERNEL);
  447. hdr_buf = kzalloc(INIT_ALLOC_SIZE, GFP_KERNEL);
  448. if (!hdr_buf || !ses_dev)
  449. goto err_init_free;
  450. result = ses_recv_diag(sdev, 1, hdr_buf, INIT_ALLOC_SIZE);
  451. if (result)
  452. goto recv_failed;
  453. if (hdr_buf[1] != 0) {
  454. /* FIXME: need subenclosure support; I've just never
  455. * seen a device with subenclosures and it makes the
  456. * traversal routines more complex */
  457. sdev_printk(KERN_ERR, sdev,
  458. "FIXME driver has no support for subenclosures (%d)\n",
  459. hdr_buf[1]);
  460. goto err_free;
  461. }
  462. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  463. buf = kzalloc(len, GFP_KERNEL);
  464. if (!buf)
  465. goto err_free;
  466. result = ses_recv_diag(sdev, 1, buf, len);
  467. if (result)
  468. goto recv_failed;
  469. types = buf[10];
  470. type_ptr = buf + 12 + buf[11];
  471. for (i = 0; i < types; i++, type_ptr += 4) {
  472. if (type_ptr[0] == ENCLOSURE_COMPONENT_DEVICE ||
  473. type_ptr[0] == ENCLOSURE_COMPONENT_ARRAY_DEVICE)
  474. components += type_ptr[1];
  475. }
  476. ses_dev->page1 = buf;
  477. ses_dev->page1_len = len;
  478. buf = NULL;
  479. result = ses_recv_diag(sdev, 2, hdr_buf, INIT_ALLOC_SIZE);
  480. if (result)
  481. goto recv_failed;
  482. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  483. buf = kzalloc(len, GFP_KERNEL);
  484. if (!buf)
  485. goto err_free;
  486. /* make sure getting page 2 actually works */
  487. result = ses_recv_diag(sdev, 2, buf, len);
  488. if (result)
  489. goto recv_failed;
  490. ses_dev->page2 = buf;
  491. ses_dev->page2_len = len;
  492. buf = NULL;
  493. /* The additional information page --- allows us
  494. * to match up the devices */
  495. result = ses_recv_diag(sdev, 10, hdr_buf, INIT_ALLOC_SIZE);
  496. if (!result) {
  497. len = (hdr_buf[2] << 8) + hdr_buf[3] + 4;
  498. buf = kzalloc(len, GFP_KERNEL);
  499. if (!buf)
  500. goto err_free;
  501. result = ses_recv_diag(sdev, 10, buf, len);
  502. if (result)
  503. goto recv_failed;
  504. ses_dev->page10 = buf;
  505. ses_dev->page10_len = len;
  506. buf = NULL;
  507. }
  508. kfree(hdr_buf);
  509. scomp = kzalloc(sizeof(struct ses_component) * components, GFP_KERNEL);
  510. if (!scomp)
  511. goto err_free;
  512. edev = enclosure_register(cdev->parent, dev_name(&sdev->sdev_gendev),
  513. components, &ses_enclosure_callbacks);
  514. if (IS_ERR(edev)) {
  515. err = PTR_ERR(edev);
  516. goto err_free;
  517. }
  518. edev->scratch = ses_dev;
  519. for (i = 0; i < components; i++)
  520. edev->component[i].scratch = scomp + i;
  521. ses_enclosure_data_process(edev, sdev, 1);
  522. /* see if there are any devices matching before
  523. * we found the enclosure */
  524. shost_for_each_device(tmp_sdev, sdev->host) {
  525. if (tmp_sdev->lun != 0 || scsi_device_enclosure(tmp_sdev))
  526. continue;
  527. ses_match_to_enclosure(edev, tmp_sdev);
  528. }
  529. return 0;
  530. recv_failed:
  531. sdev_printk(KERN_ERR, sdev, "Failed to get diagnostic page 0x%x\n",
  532. result);
  533. err = -ENODEV;
  534. err_free:
  535. kfree(buf);
  536. kfree(scomp);
  537. kfree(ses_dev->page10);
  538. kfree(ses_dev->page2);
  539. kfree(ses_dev->page1);
  540. err_init_free:
  541. kfree(ses_dev);
  542. kfree(hdr_buf);
  543. sdev_printk(KERN_ERR, sdev, "Failed to bind enclosure %d\n", err);
  544. return err;
  545. }
  546. static int ses_remove(struct device *dev)
  547. {
  548. return 0;
  549. }
  550. static void ses_intf_remove_component(struct scsi_device *sdev)
  551. {
  552. struct enclosure_device *edev, *prev = NULL;
  553. while ((edev = enclosure_find(&sdev->host->shost_gendev, prev)) != NULL) {
  554. prev = edev;
  555. if (!enclosure_remove_device(edev, &sdev->sdev_gendev))
  556. break;
  557. }
  558. if (edev)
  559. put_device(&edev->edev);
  560. }
  561. static void ses_intf_remove_enclosure(struct scsi_device *sdev)
  562. {
  563. struct enclosure_device *edev;
  564. struct ses_device *ses_dev;
  565. /* exact match to this enclosure */
  566. edev = enclosure_find(&sdev->sdev_gendev, NULL);
  567. if (!edev)
  568. return;
  569. ses_dev = edev->scratch;
  570. edev->scratch = NULL;
  571. kfree(ses_dev->page10);
  572. kfree(ses_dev->page1);
  573. kfree(ses_dev->page2);
  574. kfree(ses_dev);
  575. kfree(edev->component[0].scratch);
  576. put_device(&edev->edev);
  577. enclosure_unregister(edev);
  578. }
  579. static void ses_intf_remove(struct device *cdev,
  580. struct class_interface *intf)
  581. {
  582. struct scsi_device *sdev = to_scsi_device(cdev->parent);
  583. if (!scsi_device_enclosure(sdev))
  584. ses_intf_remove_component(sdev);
  585. else
  586. ses_intf_remove_enclosure(sdev);
  587. }
  588. static struct class_interface ses_interface = {
  589. .add_dev = ses_intf_add,
  590. .remove_dev = ses_intf_remove,
  591. };
  592. static struct scsi_driver ses_template = {
  593. .owner = THIS_MODULE,
  594. .gendrv = {
  595. .name = "ses",
  596. .probe = ses_probe,
  597. .remove = ses_remove,
  598. },
  599. };
  600. static int __init ses_init(void)
  601. {
  602. int err;
  603. err = scsi_register_interface(&ses_interface);
  604. if (err)
  605. return err;
  606. err = scsi_register_driver(&ses_template.gendrv);
  607. if (err)
  608. goto out_unreg;
  609. return 0;
  610. out_unreg:
  611. scsi_unregister_interface(&ses_interface);
  612. return err;
  613. }
  614. static void __exit ses_exit(void)
  615. {
  616. scsi_unregister_driver(&ses_template.gendrv);
  617. scsi_unregister_interface(&ses_interface);
  618. }
  619. module_init(ses_init);
  620. module_exit(ses_exit);
  621. MODULE_ALIAS_SCSI_DEVICE(TYPE_ENCLOSURE);
  622. MODULE_AUTHOR("James Bottomley");
  623. MODULE_DESCRIPTION("SCSI Enclosure Services (ses) driver");
  624. MODULE_LICENSE("GPL v2");