mbr.c 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. /*-
  2. * Copyright (c) 2014 Juniper Networks, Inc.
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions
  7. * are met:
  8. * 1. Redistributions of source code must retain the above copyright
  9. * notice, this list of conditions and the following disclaimer.
  10. * 2. Redistributions in binary form must reproduce the above copyright
  11. * notice, this list of conditions and the following disclaimer in the
  12. * documentation and/or other materials provided with the distribution.
  13. *
  14. * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
  15. * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  16. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  17. * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
  18. * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  19. * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  20. * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  21. * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  22. * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  23. * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  24. * SUCH DAMAGE.
  25. */
  26. #include <sys/cdefs.h>
  27. #include <sys/errno.h>
  28. #include <stdint.h>
  29. #include <stdlib.h>
  30. #include <string.h>
  31. #include <mbr.h>
  32. #include "endian.h"
  33. #include "image.h"
  34. #include "mkimg.h"
  35. #include "scheme.h"
  36. static struct mkimg_alias mbr_aliases[] = {
  37. { ALIAS_EBR, ALIAS_INT2TYPE(DOSPTYP_EXT) },
  38. { ALIAS_EFI, ALIAS_INT2TYPE(DOSPTYP_EFI) },
  39. { ALIAS_FAT16B, ALIAS_INT2TYPE(DOSPTYP_FAT16) },
  40. { ALIAS_FAT32, ALIAS_INT2TYPE(DOSPTYP_FAT32) },
  41. { ALIAS_FAT32LBA, ALIAS_INT2TYPE(DOSPTYP_FAT32LBA) },
  42. { ALIAS_FREEBSD, ALIAS_INT2TYPE(DOSPTYP_386BSD) },
  43. { ALIAS_NTFS, ALIAS_INT2TYPE(DOSPTYP_NTFS) },
  44. { ALIAS_PPCBOOT, ALIAS_INT2TYPE(DOSPTYP_PPCBOOT) },
  45. { ALIAS_NONE, 0 } /* Keep last! */
  46. };
  47. static lba_t
  48. mbr_metadata(u_int where, lba_t blk)
  49. {
  50. blk += (where == SCHEME_META_IMG_START) ? 1 : 0;
  51. return (round_track(blk));
  52. }
  53. static void
  54. mbr_chs(u_char *cylp, u_char *hdp, u_char *secp, lba_t lba)
  55. {
  56. u_int cyl, hd, sec;
  57. mkimg_chs(lba, 1023, &cyl, &hd, &sec);
  58. *cylp = cyl;
  59. *hdp = hd;
  60. *secp = (sec & 0x3f) | ((cyl >> 2) & 0xc0);
  61. }
  62. static int
  63. mbr_write(lba_t imgsz __unused, void *bootcode)
  64. {
  65. u_char *mbr;
  66. struct dos_partition *dpbase, *dp;
  67. struct part *part;
  68. lba_t size;
  69. int error;
  70. mbr = malloc(secsz);
  71. if (mbr == NULL)
  72. return (ENOMEM);
  73. if (bootcode != NULL) {
  74. memcpy(mbr, bootcode, DOSPARTOFF);
  75. memset(mbr + DOSPARTOFF, 0, secsz - DOSPARTOFF);
  76. } else
  77. memset(mbr, 0, secsz);
  78. le16enc(mbr + DOSMAGICOFFSET, DOSMAGIC);
  79. dpbase = (void *)(mbr + DOSPARTOFF);
  80. TAILQ_FOREACH(part, &partlist, link) {
  81. size = round_track(part->size);
  82. dp = dpbase + part->index;
  83. if (active_partition != 0)
  84. dp->dp_flag =
  85. (part->index + 1 == active_partition) ? 0x80 : 0;
  86. else
  87. dp->dp_flag =
  88. (part->index == 0 && bootcode != NULL) ? 0x80 : 0;
  89. mbr_chs(&dp->dp_scyl, &dp->dp_shd, &dp->dp_ssect,
  90. part->block);
  91. dp->dp_typ = ALIAS_TYPE2INT(part->type);
  92. mbr_chs(&dp->dp_ecyl, &dp->dp_ehd, &dp->dp_esect,
  93. part->block + size - 1);
  94. le32enc(&dp->dp_start, part->block);
  95. le32enc(&dp->dp_size, size);
  96. }
  97. error = image_write(0, mbr, 1);
  98. free(mbr);
  99. return (error);
  100. }
  101. static struct mkimg_scheme mbr_scheme = {
  102. .name = "mbr",
  103. .description = "Master Boot Record",
  104. .aliases = mbr_aliases,
  105. .metadata = mbr_metadata,
  106. .write = mbr_write,
  107. .bootcode = 512,
  108. .nparts = NDOSPART,
  109. .maxsecsz = 4096
  110. };
  111. SCHEME_DEFINE(mbr_scheme);