part_iso.c 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. /*
  2. * (C) Copyright 2001
  3. * Denis Peter, MPL AG Switzerland, d.peter@mpl.ch.
  4. *
  5. * See file CREDITS for list of people who contributed to this
  6. * project.
  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 as
  10. * published by the Free Software Foundation; either version 2 of
  11. * the License, or (at your option) any later version.
  12. *
  13. * This program is distributed in the hope that it will be useful,
  14. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. * GNU General Public License for more details.
  17. *
  18. * You should have received a copy of the GNU General Public License
  19. * along with this program; if not, write to the Free Software
  20. * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
  21. * MA 02111-1307 USA
  22. */
  23. #include <common.h>
  24. #include <command.h>
  25. #include <cmd_disk.h>
  26. #include "part_iso.h"
  27. #if ((CONFIG_COMMANDS & CFG_CMD_IDE) || (CONFIG_COMMANDS & CFG_CMD_SCSI)) && defined(CONFIG_ISO_PARTITION)
  28. #undef ISO_PART_DEBUG
  29. #ifdef ISO_PART_DEBUG
  30. #define PRINTF(fmt,args...) printf (fmt ,##args)
  31. #else
  32. #define PRINTF(fmt,args...)
  33. #endif
  34. /* enable this if CDs are written with the PowerPC Platform ID */
  35. #undef CHECK_FOR_POWERPC_PLATTFORM
  36. #define CD_SECTSIZE 2048
  37. static unsigned char tmpbuf[CD_SECTSIZE];
  38. /* Convert char[4] in little endian format to the host format integer
  39. */
  40. static inline unsigned long le32_to_int(unsigned char *le32)
  41. {
  42. return ((le32[3] << 24) +
  43. (le32[2] << 16) +
  44. (le32[1] << 8) +
  45. le32[0]
  46. );
  47. }
  48. /* Convert char[2] in little endian format to the host format integer
  49. */
  50. static inline unsigned short le16_to_int(unsigned char *le16)
  51. {
  52. return ((le16[1] << 8) +
  53. le16[0]
  54. );
  55. }
  56. /* only boot records will be listed as valid partitions */
  57. int get_partition_info_iso_verb(block_dev_desc_t * dev_desc, int part_num, disk_partition_t * info, int verb)
  58. {
  59. int i,offset,entry_num;
  60. unsigned short *chksumbuf;
  61. unsigned short chksum;
  62. unsigned long newblkaddr,blkaddr,lastsect,bootaddr;
  63. iso_boot_rec_t *pbr = (iso_boot_rec_t *)tmpbuf; /* boot record */
  64. iso_pri_rec_t *ppr = (iso_pri_rec_t *)tmpbuf; /* primary desc */
  65. iso_val_entry_t *pve = (iso_val_entry_t *)tmpbuf;
  66. iso_init_def_entry_t *pide;
  67. /* the first sector (sector 0x10) must be a primary volume desc */
  68. blkaddr=PVD_OFFSET;
  69. if (dev_desc->block_read (dev_desc->dev, PVD_OFFSET, 1, (ulong *) tmpbuf) != 1)
  70. return (-1);
  71. if(ppr->desctype!=0x01) {
  72. if(verb)
  73. printf ("** First descriptor is NOT a primary desc on %d:%d **\n",
  74. dev_desc->dev, part_num);
  75. return (-1);
  76. }
  77. if(strncmp(ppr->stand_ident,"CD001",5)!=0) {
  78. if(verb)
  79. printf ("** Wrong ISO Ident: %s on %d:%d **\n",
  80. ppr->stand_ident,dev_desc->dev, part_num);
  81. return (-1);
  82. }
  83. lastsect= ((ppr->firstsek_LEpathtab1_LE & 0x000000ff)<<24) +
  84. ((ppr->firstsek_LEpathtab1_LE & 0x0000ff00)<< 8) +
  85. ((ppr->firstsek_LEpathtab1_LE & 0x00ff0000)>> 8) +
  86. ((ppr->firstsek_LEpathtab1_LE & 0xff000000)>>24) ;
  87. info->blksz=ppr->secsize_BE; /* assuming same block size for all entries */
  88. PRINTF(" Lastsect:%08lx\n",lastsect);
  89. for(i=blkaddr;i<lastsect;i++) {
  90. PRINTF("Reading block %d\n", i);
  91. if (dev_desc->block_read (dev_desc->dev, i, 1, (ulong *) tmpbuf) != 1)
  92. return (-1);
  93. if(ppr->desctype==0x00)
  94. break; /* boot entry found */
  95. if(ppr->desctype==0xff) {
  96. if(verb)
  97. printf ("** No valid boot catalog found on %d:%d **\n",
  98. dev_desc->dev, part_num);
  99. return (-1);
  100. }
  101. }
  102. /* boot entry found */
  103. if(strncmp(pbr->ident_str,"EL TORITO SPECIFICATION",23)!=0) {
  104. if(verb)
  105. printf ("** Wrong El Torito ident: %s on %d:%d **\n",
  106. pbr->ident_str,dev_desc->dev, part_num);
  107. return (-1);
  108. }
  109. bootaddr=le32_to_int(pbr->pointer);
  110. PRINTF(" Boot Entry at: %08lX\n",bootaddr);
  111. if (dev_desc->block_read (dev_desc->dev, bootaddr, 1, (ulong *) tmpbuf) != 1) {
  112. if(verb)
  113. printf ("** Can't read Boot Entry at %lX on %d:%d **\n",
  114. bootaddr,dev_desc->dev, part_num);
  115. return (-1);
  116. }
  117. chksum=0;
  118. chksumbuf = (unsigned short *)tmpbuf;
  119. for(i=0;i<0x10;i++)
  120. chksum+=((chksumbuf[i] &0xff)<<8)+((chksumbuf[i] &0xff00)>>8);
  121. if(chksum!=0) {
  122. if(verb)
  123. printf ("** Checksum Error in booting catalog validation entry on %d:%d **\n",
  124. dev_desc->dev, part_num);
  125. return (-1);
  126. }
  127. if((pve->key[0]!=0x55)||(pve->key[1]!=0xAA)) {
  128. if(verb)
  129. printf ("** Key 0x55 0xAA error on %d:%d **\n",
  130. dev_desc->dev, part_num);
  131. return(-1);
  132. }
  133. #ifdef CHECK_FOR_POWERPC_PLATTFORM
  134. if(pve->platform!=0x01) {
  135. if(verb)
  136. printf ("** No PowerPC platform CD on %d:%d **\n",
  137. dev_desc->dev, part_num);
  138. return(-1);
  139. }
  140. #endif
  141. /* the validation entry seems to be ok, now search the "partition" */
  142. entry_num=0;
  143. offset=0x20;
  144. sprintf (info->type, "U-Boot");
  145. switch(dev_desc->if_type) {
  146. case IF_TYPE_IDE:
  147. case IF_TYPE_ATAPI:
  148. sprintf (info->name, "hd%c%d\n", 'a' + dev_desc->dev, part_num);
  149. break;
  150. case IF_TYPE_SCSI:
  151. sprintf (info->name, "sd%c%d\n", 'a' + dev_desc->dev, part_num);
  152. break;
  153. case IF_TYPE_USB:
  154. sprintf (info->name, "usbd%c%d\n", 'a' + dev_desc->dev, part_num);
  155. break;
  156. case IF_TYPE_DOC:
  157. sprintf (info->name, "docd%c%d\n", 'a' + dev_desc->dev, part_num);
  158. break;
  159. default:
  160. sprintf (info->name, "xx%c%d\n", 'a' + dev_desc->dev, part_num);
  161. break;
  162. }
  163. /* the bootcatalog (including validation Entry) is limited to 2048Bytes
  164. * (63 boot entries + validation entry) */
  165. while(offset<2048) {
  166. pide=(iso_init_def_entry_t *)&tmpbuf[offset];
  167. if ((pide->boot_ind==0x88) ||
  168. (pide->boot_ind==0x00)) { /* Header Id for default Sections Entries */
  169. if(entry_num==part_num) { /* part found */
  170. goto found;
  171. }
  172. entry_num++; /* count partitions Entries (boot and non bootables */
  173. offset+=0x20;
  174. continue;
  175. }
  176. if ((pide->boot_ind==0x90) || /* Section Header Entry */
  177. (pide->boot_ind==0x91) || /* Section Header Entry (last) */
  178. (pide->boot_ind==0x44)) { /* Extension Indicator */
  179. offset+=0x20; /* skip unused entries */
  180. }
  181. else {
  182. if(verb)
  183. printf ("** Partition %d not found on device %d **\n",
  184. part_num,dev_desc->dev);
  185. return(-1);
  186. }
  187. }
  188. /* if we reach this point entire sector has been
  189. * searched w/o succsess */
  190. if(verb)
  191. printf ("** Partition %d not found on device %d **\n",
  192. part_num,dev_desc->dev);
  193. return(-1);
  194. found:
  195. if(pide->boot_ind!=0x88) {
  196. if(verb)
  197. printf ("** Partition %d is not bootable on device %d **\n",
  198. part_num,dev_desc->dev);
  199. return (-1);
  200. }
  201. switch(pide->boot_media) {
  202. case 0x00: /* no emulation */
  203. info->size=le16_to_int(pide->sec_cnt)>>2;
  204. break;
  205. case 0x01: info->size=2400>>2; break; /* 1.2MByte Floppy */
  206. case 0x02: info->size=2880>>2; break; /* 1.44MByte Floppy */
  207. case 0x03: info->size=5760>>2; break; /* 2.88MByte Floppy */
  208. case 0x04: info->size=2880>>2; break; /* dummy (HD Emulation) */
  209. default: info->size=0; break;
  210. }
  211. newblkaddr=le32_to_int(pide->rel_block_addr);
  212. info->start=newblkaddr;
  213. PRINTF(" part %d found @ %lx size %lx\n",part_num,newblkaddr,info->size);
  214. return 0;
  215. }
  216. int get_partition_info_iso(block_dev_desc_t * dev_desc, int part_num, disk_partition_t * info)
  217. {
  218. return(get_partition_info_iso_verb(dev_desc, part_num, info, 1));
  219. }
  220. void print_part_iso(block_dev_desc_t * dev_desc)
  221. {
  222. disk_partition_t info;
  223. int i;
  224. if(get_partition_info_iso_verb(dev_desc,0,&info,0)==-1) {
  225. printf("** No boot partition found on device %d **\n",dev_desc->dev);
  226. return;
  227. }
  228. printf("Part Start Sect x Size Type\n");
  229. i=0;
  230. do {
  231. printf (" %2d %8ld %8ld %6ld %.32s\n",
  232. i, info.start, info.size, info.blksz, info.type);
  233. i++;
  234. } while (get_partition_info_iso_verb(dev_desc,i,&info,0)!=-1);
  235. }
  236. int test_part_iso (block_dev_desc_t *dev_desc)
  237. {
  238. disk_partition_t info;
  239. return(get_partition_info_iso_verb(dev_desc,0,&info,0));
  240. }
  241. #endif /* ((CONFIG_COMMANDS & CFG_CMD_IDE) || (CONFIG_COMMANDS & CFG_CMD_SCSI)) && defined(CONFIG_ISO_PARTITION) */