edd.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789
  1. /*
  2. * linux/arch/i386/kernel/edd.c
  3. * Copyright (C) 2002, 2003, 2004 Dell Inc.
  4. * by Matt Domsch <Matt_Domsch@dell.com>
  5. * disk signature by Matt Domsch, Andrew Wilks, and Sandeep K. Shandilya
  6. * legacy CHS by Patrick J. LoPresti <patl@users.sourceforge.net>
  7. *
  8. * BIOS Enhanced Disk Drive Services (EDD)
  9. * conformant to T13 Committee www.t13.org
  10. * projects 1572D, 1484D, 1386D, 1226DT
  11. *
  12. * This code takes information provided by BIOS EDD calls
  13. * fn41 - Check Extensions Present and
  14. * fn48 - Get Device Parametes with EDD extensions
  15. * made in setup.S, copied to safe structures in setup.c,
  16. * and presents it in sysfs.
  17. *
  18. * Please see http://linux.dell.com/edd30/results.html for
  19. * the list of BIOSs which have been reported to implement EDD.
  20. *
  21. * This program is free software; you can redistribute it and/or modify
  22. * it under the terms of the GNU General Public License v2.0 as published by
  23. * the Free Software Foundation
  24. *
  25. * This program is distributed in the hope that it will be useful,
  26. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  27. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  28. * GNU General Public License for more details.
  29. *
  30. */
  31. #include <linux/module.h>
  32. #include <linux/string.h>
  33. #include <linux/types.h>
  34. #include <linux/init.h>
  35. #include <linux/stat.h>
  36. #include <linux/err.h>
  37. #include <linux/ctype.h>
  38. #include <linux/slab.h>
  39. #include <linux/limits.h>
  40. #include <linux/device.h>
  41. #include <linux/pci.h>
  42. #include <linux/blkdev.h>
  43. #include <linux/edd.h>
  44. #define EDD_VERSION "0.16"
  45. #define EDD_DATE "2004-Jun-25"
  46. MODULE_AUTHOR("Matt Domsch <Matt_Domsch@Dell.com>");
  47. MODULE_DESCRIPTION("sysfs interface to BIOS EDD information");
  48. MODULE_LICENSE("GPL");
  49. MODULE_VERSION(EDD_VERSION);
  50. #define left (PAGE_SIZE - (p - buf) - 1)
  51. struct edd_device {
  52. unsigned int index;
  53. unsigned int mbr_signature;
  54. struct edd_info *info;
  55. struct kobject kobj;
  56. };
  57. struct edd_attribute {
  58. struct attribute attr;
  59. ssize_t(*show) (struct edd_device * edev, char *buf);
  60. int (*test) (struct edd_device * edev);
  61. };
  62. /* forward declarations */
  63. static int edd_dev_is_type(struct edd_device *edev, const char *type);
  64. static struct pci_dev *edd_get_pci_dev(struct edd_device *edev);
  65. static struct edd_device *edd_devices[EDD_MBR_SIG_MAX];
  66. #define EDD_DEVICE_ATTR(_name,_mode,_show,_test) \
  67. struct edd_attribute edd_attr_##_name = { \
  68. .attr = {.name = __stringify(_name), .mode = _mode, .owner = THIS_MODULE }, \
  69. .show = _show, \
  70. .test = _test, \
  71. };
  72. static int
  73. edd_has_mbr_signature(struct edd_device *edev)
  74. {
  75. return edev->index < min_t(unsigned char, edd.mbr_signature_nr, EDD_MBR_SIG_MAX);
  76. }
  77. static int
  78. edd_has_edd_info(struct edd_device *edev)
  79. {
  80. return edev->index < min_t(unsigned char, edd.edd_info_nr, EDDMAXNR);
  81. }
  82. static inline struct edd_info *
  83. edd_dev_get_info(struct edd_device *edev)
  84. {
  85. return edev->info;
  86. }
  87. static inline void
  88. edd_dev_set_info(struct edd_device *edev, int i)
  89. {
  90. edev->index = i;
  91. if (edd_has_mbr_signature(edev))
  92. edev->mbr_signature = edd.mbr_signature[i];
  93. if (edd_has_edd_info(edev))
  94. edev->info = &edd.edd_info[i];
  95. }
  96. #define to_edd_attr(_attr) container_of(_attr,struct edd_attribute,attr)
  97. #define to_edd_device(obj) container_of(obj,struct edd_device,kobj)
  98. static ssize_t
  99. edd_attr_show(struct kobject * kobj, struct attribute *attr, char *buf)
  100. {
  101. struct edd_device *dev = to_edd_device(kobj);
  102. struct edd_attribute *edd_attr = to_edd_attr(attr);
  103. ssize_t ret = -EIO;
  104. if (edd_attr->show)
  105. ret = edd_attr->show(dev, buf);
  106. return ret;
  107. }
  108. static struct sysfs_ops edd_attr_ops = {
  109. .show = edd_attr_show,
  110. };
  111. static ssize_t
  112. edd_show_host_bus(struct edd_device *edev, char *buf)
  113. {
  114. struct edd_info *info;
  115. char *p = buf;
  116. int i;
  117. if (!edev)
  118. return -EINVAL;
  119. info = edd_dev_get_info(edev);
  120. if (!info || !buf)
  121. return -EINVAL;
  122. for (i = 0; i < 4; i++) {
  123. if (isprint(info->params.host_bus_type[i])) {
  124. p += scnprintf(p, left, "%c", info->params.host_bus_type[i]);
  125. } else {
  126. p += scnprintf(p, left, " ");
  127. }
  128. }
  129. if (!strncmp(info->params.host_bus_type, "ISA", 3)) {
  130. p += scnprintf(p, left, "\tbase_address: %x\n",
  131. info->params.interface_path.isa.base_address);
  132. } else if (!strncmp(info->params.host_bus_type, "PCIX", 4) ||
  133. !strncmp(info->params.host_bus_type, "PCI", 3)) {
  134. p += scnprintf(p, left,
  135. "\t%02x:%02x.%d channel: %u\n",
  136. info->params.interface_path.pci.bus,
  137. info->params.interface_path.pci.slot,
  138. info->params.interface_path.pci.function,
  139. info->params.interface_path.pci.channel);
  140. } else if (!strncmp(info->params.host_bus_type, "IBND", 4) ||
  141. !strncmp(info->params.host_bus_type, "XPRS", 4) ||
  142. !strncmp(info->params.host_bus_type, "HTPT", 4)) {
  143. p += scnprintf(p, left,
  144. "\tTBD: %llx\n",
  145. info->params.interface_path.ibnd.reserved);
  146. } else {
  147. p += scnprintf(p, left, "\tunknown: %llx\n",
  148. info->params.interface_path.unknown.reserved);
  149. }
  150. return (p - buf);
  151. }
  152. static ssize_t
  153. edd_show_interface(struct edd_device *edev, char *buf)
  154. {
  155. struct edd_info *info;
  156. char *p = buf;
  157. int i;
  158. if (!edev)
  159. return -EINVAL;
  160. info = edd_dev_get_info(edev);
  161. if (!info || !buf)
  162. return -EINVAL;
  163. for (i = 0; i < 8; i++) {
  164. if (isprint(info->params.interface_type[i])) {
  165. p += scnprintf(p, left, "%c", info->params.interface_type[i]);
  166. } else {
  167. p += scnprintf(p, left, " ");
  168. }
  169. }
  170. if (!strncmp(info->params.interface_type, "ATAPI", 5)) {
  171. p += scnprintf(p, left, "\tdevice: %u lun: %u\n",
  172. info->params.device_path.atapi.device,
  173. info->params.device_path.atapi.lun);
  174. } else if (!strncmp(info->params.interface_type, "ATA", 3)) {
  175. p += scnprintf(p, left, "\tdevice: %u\n",
  176. info->params.device_path.ata.device);
  177. } else if (!strncmp(info->params.interface_type, "SCSI", 4)) {
  178. p += scnprintf(p, left, "\tid: %u lun: %llu\n",
  179. info->params.device_path.scsi.id,
  180. info->params.device_path.scsi.lun);
  181. } else if (!strncmp(info->params.interface_type, "USB", 3)) {
  182. p += scnprintf(p, left, "\tserial_number: %llx\n",
  183. info->params.device_path.usb.serial_number);
  184. } else if (!strncmp(info->params.interface_type, "1394", 4)) {
  185. p += scnprintf(p, left, "\teui: %llx\n",
  186. info->params.device_path.i1394.eui);
  187. } else if (!strncmp(info->params.interface_type, "FIBRE", 5)) {
  188. p += scnprintf(p, left, "\twwid: %llx lun: %llx\n",
  189. info->params.device_path.fibre.wwid,
  190. info->params.device_path.fibre.lun);
  191. } else if (!strncmp(info->params.interface_type, "I2O", 3)) {
  192. p += scnprintf(p, left, "\tidentity_tag: %llx\n",
  193. info->params.device_path.i2o.identity_tag);
  194. } else if (!strncmp(info->params.interface_type, "RAID", 4)) {
  195. p += scnprintf(p, left, "\tidentity_tag: %x\n",
  196. info->params.device_path.raid.array_number);
  197. } else if (!strncmp(info->params.interface_type, "SATA", 4)) {
  198. p += scnprintf(p, left, "\tdevice: %u\n",
  199. info->params.device_path.sata.device);
  200. } else {
  201. p += scnprintf(p, left, "\tunknown: %llx %llx\n",
  202. info->params.device_path.unknown.reserved1,
  203. info->params.device_path.unknown.reserved2);
  204. }
  205. return (p - buf);
  206. }
  207. /**
  208. * edd_show_raw_data() - copies raw data to buffer for userspace to parse
  209. *
  210. * Returns: number of bytes written, or -EINVAL on failure
  211. */
  212. static ssize_t
  213. edd_show_raw_data(struct edd_device *edev, char *buf)
  214. {
  215. struct edd_info *info;
  216. ssize_t len = sizeof (info->params);
  217. if (!edev)
  218. return -EINVAL;
  219. info = edd_dev_get_info(edev);
  220. if (!info || !buf)
  221. return -EINVAL;
  222. if (!(info->params.key == 0xBEDD || info->params.key == 0xDDBE))
  223. len = info->params.length;
  224. /* In case of buggy BIOSs */
  225. if (len > (sizeof(info->params)))
  226. len = sizeof(info->params);
  227. memcpy(buf, &info->params, len);
  228. return len;
  229. }
  230. static ssize_t
  231. edd_show_version(struct edd_device *edev, char *buf)
  232. {
  233. struct edd_info *info;
  234. char *p = buf;
  235. if (!edev)
  236. return -EINVAL;
  237. info = edd_dev_get_info(edev);
  238. if (!info || !buf)
  239. return -EINVAL;
  240. p += scnprintf(p, left, "0x%02x\n", info->version);
  241. return (p - buf);
  242. }
  243. static ssize_t
  244. edd_show_mbr_signature(struct edd_device *edev, char *buf)
  245. {
  246. char *p = buf;
  247. p += scnprintf(p, left, "0x%08x\n", edev->mbr_signature);
  248. return (p - buf);
  249. }
  250. static ssize_t
  251. edd_show_extensions(struct edd_device *edev, char *buf)
  252. {
  253. struct edd_info *info;
  254. char *p = buf;
  255. if (!edev)
  256. return -EINVAL;
  257. info = edd_dev_get_info(edev);
  258. if (!info || !buf)
  259. return -EINVAL;
  260. if (info->interface_support & EDD_EXT_FIXED_DISK_ACCESS) {
  261. p += scnprintf(p, left, "Fixed disk access\n");
  262. }
  263. if (info->interface_support & EDD_EXT_DEVICE_LOCKING_AND_EJECTING) {
  264. p += scnprintf(p, left, "Device locking and ejecting\n");
  265. }
  266. if (info->interface_support & EDD_EXT_ENHANCED_DISK_DRIVE_SUPPORT) {
  267. p += scnprintf(p, left, "Enhanced Disk Drive support\n");
  268. }
  269. if (info->interface_support & EDD_EXT_64BIT_EXTENSIONS) {
  270. p += scnprintf(p, left, "64-bit extensions\n");
  271. }
  272. return (p - buf);
  273. }
  274. static ssize_t
  275. edd_show_info_flags(struct edd_device *edev, char *buf)
  276. {
  277. struct edd_info *info;
  278. char *p = buf;
  279. if (!edev)
  280. return -EINVAL;
  281. info = edd_dev_get_info(edev);
  282. if (!info || !buf)
  283. return -EINVAL;
  284. if (info->params.info_flags & EDD_INFO_DMA_BOUNDARY_ERROR_TRANSPARENT)
  285. p += scnprintf(p, left, "DMA boundary error transparent\n");
  286. if (info->params.info_flags & EDD_INFO_GEOMETRY_VALID)
  287. p += scnprintf(p, left, "geometry valid\n");
  288. if (info->params.info_flags & EDD_INFO_REMOVABLE)
  289. p += scnprintf(p, left, "removable\n");
  290. if (info->params.info_flags & EDD_INFO_WRITE_VERIFY)
  291. p += scnprintf(p, left, "write verify\n");
  292. if (info->params.info_flags & EDD_INFO_MEDIA_CHANGE_NOTIFICATION)
  293. p += scnprintf(p, left, "media change notification\n");
  294. if (info->params.info_flags & EDD_INFO_LOCKABLE)
  295. p += scnprintf(p, left, "lockable\n");
  296. if (info->params.info_flags & EDD_INFO_NO_MEDIA_PRESENT)
  297. p += scnprintf(p, left, "no media present\n");
  298. if (info->params.info_flags & EDD_INFO_USE_INT13_FN50)
  299. p += scnprintf(p, left, "use int13 fn50\n");
  300. return (p - buf);
  301. }
  302. static ssize_t
  303. edd_show_legacy_max_cylinder(struct edd_device *edev, char *buf)
  304. {
  305. struct edd_info *info;
  306. char *p = buf;
  307. if (!edev)
  308. return -EINVAL;
  309. info = edd_dev_get_info(edev);
  310. if (!info || !buf)
  311. return -EINVAL;
  312. p += snprintf(p, left, "%u\n", info->legacy_max_cylinder);
  313. return (p - buf);
  314. }
  315. static ssize_t
  316. edd_show_legacy_max_head(struct edd_device *edev, char *buf)
  317. {
  318. struct edd_info *info;
  319. char *p = buf;
  320. if (!edev)
  321. return -EINVAL;
  322. info = edd_dev_get_info(edev);
  323. if (!info || !buf)
  324. return -EINVAL;
  325. p += snprintf(p, left, "%u\n", info->legacy_max_head);
  326. return (p - buf);
  327. }
  328. static ssize_t
  329. edd_show_legacy_sectors_per_track(struct edd_device *edev, char *buf)
  330. {
  331. struct edd_info *info;
  332. char *p = buf;
  333. if (!edev)
  334. return -EINVAL;
  335. info = edd_dev_get_info(edev);
  336. if (!info || !buf)
  337. return -EINVAL;
  338. p += snprintf(p, left, "%u\n", info->legacy_sectors_per_track);
  339. return (p - buf);
  340. }
  341. static ssize_t
  342. edd_show_default_cylinders(struct edd_device *edev, char *buf)
  343. {
  344. struct edd_info *info;
  345. char *p = buf;
  346. if (!edev)
  347. return -EINVAL;
  348. info = edd_dev_get_info(edev);
  349. if (!info || !buf)
  350. return -EINVAL;
  351. p += scnprintf(p, left, "%u\n", info->params.num_default_cylinders);
  352. return (p - buf);
  353. }
  354. static ssize_t
  355. edd_show_default_heads(struct edd_device *edev, char *buf)
  356. {
  357. struct edd_info *info;
  358. char *p = buf;
  359. if (!edev)
  360. return -EINVAL;
  361. info = edd_dev_get_info(edev);
  362. if (!info || !buf)
  363. return -EINVAL;
  364. p += scnprintf(p, left, "%u\n", info->params.num_default_heads);
  365. return (p - buf);
  366. }
  367. static ssize_t
  368. edd_show_default_sectors_per_track(struct edd_device *edev, char *buf)
  369. {
  370. struct edd_info *info;
  371. char *p = buf;
  372. if (!edev)
  373. return -EINVAL;
  374. info = edd_dev_get_info(edev);
  375. if (!info || !buf)
  376. return -EINVAL;
  377. p += scnprintf(p, left, "%u\n", info->params.sectors_per_track);
  378. return (p - buf);
  379. }
  380. static ssize_t
  381. edd_show_sectors(struct edd_device *edev, char *buf)
  382. {
  383. struct edd_info *info;
  384. char *p = buf;
  385. if (!edev)
  386. return -EINVAL;
  387. info = edd_dev_get_info(edev);
  388. if (!info || !buf)
  389. return -EINVAL;
  390. p += scnprintf(p, left, "%llu\n", info->params.number_of_sectors);
  391. return (p - buf);
  392. }
  393. /*
  394. * Some device instances may not have all the above attributes,
  395. * or the attribute values may be meaningless (i.e. if
  396. * the device is < EDD 3.0, it won't have host_bus and interface
  397. * information), so don't bother making files for them. Likewise
  398. * if the default_{cylinders,heads,sectors_per_track} values
  399. * are zero, the BIOS doesn't provide sane values, don't bother
  400. * creating files for them either.
  401. */
  402. static int
  403. edd_has_legacy_max_cylinder(struct edd_device *edev)
  404. {
  405. struct edd_info *info;
  406. if (!edev)
  407. return 0;
  408. info = edd_dev_get_info(edev);
  409. if (!info)
  410. return 0;
  411. return info->legacy_max_cylinder > 0;
  412. }
  413. static int
  414. edd_has_legacy_max_head(struct edd_device *edev)
  415. {
  416. struct edd_info *info;
  417. if (!edev)
  418. return 0;
  419. info = edd_dev_get_info(edev);
  420. if (!info)
  421. return 0;
  422. return info->legacy_max_head > 0;
  423. }
  424. static int
  425. edd_has_legacy_sectors_per_track(struct edd_device *edev)
  426. {
  427. struct edd_info *info;
  428. if (!edev)
  429. return 0;
  430. info = edd_dev_get_info(edev);
  431. if (!info)
  432. return 0;
  433. return info->legacy_sectors_per_track > 0;
  434. }
  435. static int
  436. edd_has_default_cylinders(struct edd_device *edev)
  437. {
  438. struct edd_info *info;
  439. if (!edev)
  440. return 0;
  441. info = edd_dev_get_info(edev);
  442. if (!info)
  443. return 0;
  444. return info->params.num_default_cylinders > 0;
  445. }
  446. static int
  447. edd_has_default_heads(struct edd_device *edev)
  448. {
  449. struct edd_info *info;
  450. if (!edev)
  451. return 0;
  452. info = edd_dev_get_info(edev);
  453. if (!info)
  454. return 0;
  455. return info->params.num_default_heads > 0;
  456. }
  457. static int
  458. edd_has_default_sectors_per_track(struct edd_device *edev)
  459. {
  460. struct edd_info *info;
  461. if (!edev)
  462. return 0;
  463. info = edd_dev_get_info(edev);
  464. if (!info)
  465. return 0;
  466. return info->params.sectors_per_track > 0;
  467. }
  468. static int
  469. edd_has_edd30(struct edd_device *edev)
  470. {
  471. struct edd_info *info;
  472. int i, nonzero_path = 0;
  473. char c;
  474. if (!edev)
  475. return 0;
  476. info = edd_dev_get_info(edev);
  477. if (!info)
  478. return 0;
  479. if (!(info->params.key == 0xBEDD || info->params.key == 0xDDBE)) {
  480. return 0;
  481. }
  482. for (i = 30; i <= 73; i++) {
  483. c = *(((uint8_t *) info) + i + 4);
  484. if (c) {
  485. nonzero_path++;
  486. break;
  487. }
  488. }
  489. if (!nonzero_path) {
  490. return 0;
  491. }
  492. return 1;
  493. }
  494. static EDD_DEVICE_ATTR(raw_data, 0444, edd_show_raw_data, edd_has_edd_info);
  495. static EDD_DEVICE_ATTR(version, 0444, edd_show_version, edd_has_edd_info);
  496. static EDD_DEVICE_ATTR(extensions, 0444, edd_show_extensions, edd_has_edd_info);
  497. static EDD_DEVICE_ATTR(info_flags, 0444, edd_show_info_flags, edd_has_edd_info);
  498. static EDD_DEVICE_ATTR(sectors, 0444, edd_show_sectors, edd_has_edd_info);
  499. static EDD_DEVICE_ATTR(legacy_max_cylinder, 0444,
  500. edd_show_legacy_max_cylinder,
  501. edd_has_legacy_max_cylinder);
  502. static EDD_DEVICE_ATTR(legacy_max_head, 0444, edd_show_legacy_max_head,
  503. edd_has_legacy_max_head);
  504. static EDD_DEVICE_ATTR(legacy_sectors_per_track, 0444,
  505. edd_show_legacy_sectors_per_track,
  506. edd_has_legacy_sectors_per_track);
  507. static EDD_DEVICE_ATTR(default_cylinders, 0444, edd_show_default_cylinders,
  508. edd_has_default_cylinders);
  509. static EDD_DEVICE_ATTR(default_heads, 0444, edd_show_default_heads,
  510. edd_has_default_heads);
  511. static EDD_DEVICE_ATTR(default_sectors_per_track, 0444,
  512. edd_show_default_sectors_per_track,
  513. edd_has_default_sectors_per_track);
  514. static EDD_DEVICE_ATTR(interface, 0444, edd_show_interface, edd_has_edd30);
  515. static EDD_DEVICE_ATTR(host_bus, 0444, edd_show_host_bus, edd_has_edd30);
  516. static EDD_DEVICE_ATTR(mbr_signature, 0444, edd_show_mbr_signature, edd_has_mbr_signature);
  517. /* These are default attributes that are added for every edd
  518. * device discovered. There are none.
  519. */
  520. static struct attribute * def_attrs[] = {
  521. NULL,
  522. };
  523. /* These attributes are conditional and only added for some devices. */
  524. static struct edd_attribute * edd_attrs[] = {
  525. &edd_attr_raw_data,
  526. &edd_attr_version,
  527. &edd_attr_extensions,
  528. &edd_attr_info_flags,
  529. &edd_attr_sectors,
  530. &edd_attr_legacy_max_cylinder,
  531. &edd_attr_legacy_max_head,
  532. &edd_attr_legacy_sectors_per_track,
  533. &edd_attr_default_cylinders,
  534. &edd_attr_default_heads,
  535. &edd_attr_default_sectors_per_track,
  536. &edd_attr_interface,
  537. &edd_attr_host_bus,
  538. &edd_attr_mbr_signature,
  539. NULL,
  540. };
  541. /**
  542. * edd_release - free edd structure
  543. * @kobj: kobject of edd structure
  544. *
  545. * This is called when the refcount of the edd structure
  546. * reaches 0. This should happen right after we unregister,
  547. * but just in case, we use the release callback anyway.
  548. */
  549. static void edd_release(struct kobject * kobj)
  550. {
  551. struct edd_device * dev = to_edd_device(kobj);
  552. kfree(dev);
  553. }
  554. static struct kobj_type ktype_edd = {
  555. .release = edd_release,
  556. .sysfs_ops = &edd_attr_ops,
  557. .default_attrs = def_attrs,
  558. };
  559. static decl_subsys(edd,&ktype_edd,NULL);
  560. /**
  561. * edd_dev_is_type() - is this EDD device a 'type' device?
  562. * @edev
  563. * @type - a host bus or interface identifier string per the EDD spec
  564. *
  565. * Returns 1 (TRUE) if it is a 'type' device, 0 otherwise.
  566. */
  567. static int
  568. edd_dev_is_type(struct edd_device *edev, const char *type)
  569. {
  570. struct edd_info *info;
  571. if (!edev)
  572. return 0;
  573. info = edd_dev_get_info(edev);
  574. if (type && info) {
  575. if (!strncmp(info->params.host_bus_type, type, strlen(type)) ||
  576. !strncmp(info->params.interface_type, type, strlen(type)))
  577. return 1;
  578. }
  579. return 0;
  580. }
  581. /**
  582. * edd_get_pci_dev() - finds pci_dev that matches edev
  583. * @edev - edd_device
  584. *
  585. * Returns pci_dev if found, or NULL
  586. */
  587. static struct pci_dev *
  588. edd_get_pci_dev(struct edd_device *edev)
  589. {
  590. struct edd_info *info = edd_dev_get_info(edev);
  591. if (edd_dev_is_type(edev, "PCI")) {
  592. return pci_find_slot(info->params.interface_path.pci.bus,
  593. PCI_DEVFN(info->params.interface_path.pci.slot,
  594. info->params.interface_path.pci.
  595. function));
  596. }
  597. return NULL;
  598. }
  599. static int
  600. edd_create_symlink_to_pcidev(struct edd_device *edev)
  601. {
  602. struct pci_dev *pci_dev = edd_get_pci_dev(edev);
  603. if (!pci_dev)
  604. return 1;
  605. return sysfs_create_link(&edev->kobj,&pci_dev->dev.kobj,"pci_dev");
  606. }
  607. static inline void
  608. edd_device_unregister(struct edd_device *edev)
  609. {
  610. kobject_unregister(&edev->kobj);
  611. }
  612. static void edd_populate_dir(struct edd_device * edev)
  613. {
  614. struct edd_attribute * attr;
  615. int error = 0;
  616. int i;
  617. for (i = 0; (attr = edd_attrs[i]) && !error; i++) {
  618. if (!attr->test ||
  619. (attr->test && attr->test(edev)))
  620. error = sysfs_create_file(&edev->kobj,&attr->attr);
  621. }
  622. if (!error) {
  623. edd_create_symlink_to_pcidev(edev);
  624. }
  625. }
  626. static int
  627. edd_device_register(struct edd_device *edev, int i)
  628. {
  629. int error;
  630. if (!edev)
  631. return 1;
  632. edd_dev_set_info(edev, i);
  633. kobject_set_name(&edev->kobj, "int13_dev%02x",
  634. 0x80 + i);
  635. kobj_set_kset_s(edev,edd_subsys);
  636. error = kobject_register(&edev->kobj);
  637. if (!error)
  638. edd_populate_dir(edev);
  639. return error;
  640. }
  641. static inline int edd_num_devices(void)
  642. {
  643. return max_t(unsigned char,
  644. min_t(unsigned char, EDD_MBR_SIG_MAX, edd.mbr_signature_nr),
  645. min_t(unsigned char, EDDMAXNR, edd.edd_info_nr));
  646. }
  647. /**
  648. * edd_init() - creates sysfs tree of EDD data
  649. */
  650. static int __init
  651. edd_init(void)
  652. {
  653. unsigned int i;
  654. int rc=0;
  655. struct edd_device *edev;
  656. printk(KERN_INFO "BIOS EDD facility v%s %s, %d devices found\n",
  657. EDD_VERSION, EDD_DATE, edd_num_devices());
  658. if (!edd_num_devices()) {
  659. printk(KERN_INFO "EDD information not available.\n");
  660. return 1;
  661. }
  662. rc = firmware_register(&edd_subsys);
  663. if (rc)
  664. return rc;
  665. for (i = 0; i < edd_num_devices() && !rc; i++) {
  666. edev = kzalloc(sizeof (*edev), GFP_KERNEL);
  667. if (!edev)
  668. return -ENOMEM;
  669. rc = edd_device_register(edev, i);
  670. if (rc) {
  671. kfree(edev);
  672. break;
  673. }
  674. edd_devices[i] = edev;
  675. }
  676. if (rc)
  677. firmware_unregister(&edd_subsys);
  678. return rc;
  679. }
  680. static void __exit
  681. edd_exit(void)
  682. {
  683. int i;
  684. struct edd_device *edev;
  685. for (i = 0; i < edd_num_devices(); i++) {
  686. if ((edev = edd_devices[i]))
  687. edd_device_unregister(edev);
  688. }
  689. firmware_unregister(&edd_subsys);
  690. }
  691. late_initcall(edd_init);
  692. module_exit(edd_exit);