ibm.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. /*
  2. * File...........: linux/fs/partitions/ibm.c
  3. * Author(s)......: Holger Smolinski <Holger.Smolinski@de.ibm.com>
  4. * Volker Sameske <sameske@de.ibm.com>
  5. * Bugreports.to..: <Linux390@de.ibm.com>
  6. * (C) IBM Corporation, IBM Deutschland Entwicklung GmbH, 1999,2000
  7. * History of changes (starts July 2000)
  8. * 07/10/00 Fixed detection of CMS formatted disks
  9. * 02/13/00 VTOC partition support added
  10. * 12/27/01 fixed PL030593 (CMS reserved minidisk not detected on 64 bit)
  11. * 07/24/03 no longer using contents of freed page for CMS label recognition (BZ3611)
  12. */
  13. #include <linux/config.h>
  14. #include <linux/buffer_head.h>
  15. #include <linux/hdreg.h>
  16. #include <linux/slab.h>
  17. #include <asm/dasd.h>
  18. #include <asm/ebcdic.h>
  19. #include <asm/uaccess.h>
  20. #include <asm/vtoc.h>
  21. #include "check.h"
  22. #include "ibm.h"
  23. /*
  24. * compute the block number from a
  25. * cyl-cyl-head-head structure
  26. */
  27. static inline int
  28. cchh2blk (struct vtoc_cchh *ptr, struct hd_geometry *geo) {
  29. return ptr->cc * geo->heads * geo->sectors +
  30. ptr->hh * geo->sectors;
  31. }
  32. /*
  33. * compute the block number from a
  34. * cyl-cyl-head-head-block structure
  35. */
  36. static inline int
  37. cchhb2blk (struct vtoc_cchhb *ptr, struct hd_geometry *geo) {
  38. return ptr->cc * geo->heads * geo->sectors +
  39. ptr->hh * geo->sectors +
  40. ptr->b;
  41. }
  42. /*
  43. */
  44. int
  45. ibm_partition(struct parsed_partitions *state, struct block_device *bdev)
  46. {
  47. int blocksize, offset, size;
  48. dasd_information_t *info;
  49. struct hd_geometry *geo;
  50. char type[5] = {0,};
  51. char name[7] = {0,};
  52. struct vtoc_volume_label *vlabel;
  53. unsigned char *data;
  54. Sector sect;
  55. if ((info = kmalloc(sizeof(dasd_information_t), GFP_KERNEL)) == NULL)
  56. goto out_noinfo;
  57. if ((geo = kmalloc(sizeof(struct hd_geometry), GFP_KERNEL)) == NULL)
  58. goto out_nogeo;
  59. if ((vlabel = kmalloc(sizeof(struct vtoc_volume_label),
  60. GFP_KERNEL)) == NULL)
  61. goto out_novlab;
  62. if (ioctl_by_bdev(bdev, BIODASDINFO, (unsigned long)info) != 0 ||
  63. ioctl_by_bdev(bdev, HDIO_GETGEO, (unsigned long)geo) != 0)
  64. goto out_noioctl;
  65. if ((blocksize = bdev_hardsect_size(bdev)) <= 0)
  66. goto out_badsect;
  67. /*
  68. * Get volume label, extract name and type.
  69. */
  70. data = read_dev_sector(bdev, info->label_block*(blocksize/512), &sect);
  71. if (data == NULL)
  72. goto out_readerr;
  73. strncpy (type, data, 4);
  74. if ((!info->FBA_layout) && (!strcmp(info->type, "ECKD")))
  75. strncpy(name, data + 8, 6);
  76. else
  77. strncpy(name, data + 4, 6);
  78. memcpy (vlabel, data, sizeof(struct vtoc_volume_label));
  79. put_dev_sector(sect);
  80. EBCASC(type, 4);
  81. EBCASC(name, 6);
  82. /*
  83. * Three different types: CMS1, VOL1 and LNX1/unlabeled
  84. */
  85. if (strncmp(type, "CMS1", 4) == 0) {
  86. /*
  87. * VM style CMS1 labeled disk
  88. */
  89. int *label = (int *) vlabel;
  90. if (label[13] != 0) {
  91. printk("CMS1/%8s(MDSK):", name);
  92. /* disk is reserved minidisk */
  93. blocksize = label[3];
  94. offset = label[13];
  95. size = (label[7] - 1)*(blocksize >> 9);
  96. } else {
  97. printk("CMS1/%8s:", name);
  98. offset = (info->label_block + 1);
  99. size = bdev->bd_inode->i_size >> 9;
  100. }
  101. put_partition(state, 1, offset*(blocksize >> 9),
  102. size-offset*(blocksize >> 9));
  103. } else if ((strncmp(type, "VOL1", 4) == 0) &&
  104. (!info->FBA_layout) && (!strcmp(info->type, "ECKD"))) {
  105. /*
  106. * New style VOL1 labeled disk
  107. */
  108. unsigned int blk;
  109. int counter;
  110. printk("VOL1/%8s:", name);
  111. /* get block number and read then go through format1 labels */
  112. blk = cchhb2blk(&vlabel->vtoc, geo) + 1;
  113. counter = 0;
  114. while ((data = read_dev_sector(bdev, blk*(blocksize/512),
  115. &sect)) != NULL) {
  116. struct vtoc_format1_label f1;
  117. memcpy(&f1, data, sizeof(struct vtoc_format1_label));
  118. put_dev_sector(sect);
  119. /* skip FMT4 / FMT5 / FMT7 labels */
  120. if (f1.DS1FMTID == _ascebc['4']
  121. || f1.DS1FMTID == _ascebc['5']
  122. || f1.DS1FMTID == _ascebc['7']) {
  123. blk++;
  124. continue;
  125. }
  126. /* only FMT1 valid at this point */
  127. if (f1.DS1FMTID != _ascebc['1'])
  128. break;
  129. /* OK, we got valid partition data */
  130. offset = cchh2blk(&f1.DS1EXT1.llimit, geo);
  131. size = cchh2blk(&f1.DS1EXT1.ulimit, geo) -
  132. offset + geo->sectors;
  133. if (counter >= state->limit)
  134. break;
  135. put_partition(state, counter + 1,
  136. offset * (blocksize >> 9),
  137. size * (blocksize >> 9));
  138. counter++;
  139. blk++;
  140. }
  141. } else {
  142. /*
  143. * Old style LNX1 or unlabeled disk
  144. */
  145. if (strncmp(type, "LNX1", 4) == 0)
  146. printk ("LNX1/%8s:", name);
  147. else
  148. printk("(nonl)/%8s:", name);
  149. offset = (info->label_block + 1);
  150. size = (bdev->bd_inode->i_size >> 9);
  151. put_partition(state, 1, offset*(blocksize >> 9),
  152. size-offset*(blocksize >> 9));
  153. }
  154. printk("\n");
  155. kfree(vlabel);
  156. kfree(geo);
  157. kfree(info);
  158. return 1;
  159. out_readerr:
  160. out_badsect:
  161. out_noioctl:
  162. kfree(vlabel);
  163. out_novlab:
  164. kfree(geo);
  165. out_nogeo:
  166. kfree(info);
  167. out_noinfo:
  168. return 0;
  169. }