1 // Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include <errno.h>
6 #include <fcntl.h>
7 #include <string.h>
8 #include <unistd.h>
9 
10 #include "cgpt.h"
11 #include "cgpt_params.h"
12 #include "cgptlib_internal.h"
13 #include "vboot_host.h"
14 
15 int CgptGetBootPartitionNumber(CgptBootParams *params) {
16   struct drive drive;
17   int gpt_retval= 0;
18   int retval;
19 
20   if (params == NULL)
21     return CGPT_FAILED;
22 
23   if (CGPT_OK != DriveOpen(params->drive_name, &drive, O_RDONLY,
24                            params->drive_size))
25     return CGPT_FAILED;
26 
27   if (GPT_SUCCESS != (gpt_retval = GptSanityCheck(&drive.gpt))) {
28     Error("GptSanityCheck() returned %d: %s\n",
29           gpt_retval, GptError(gpt_retval));
30     retval = CGPT_FAILED;
31     goto done;
32   }
33 
34   if (CGPT_OK != ReadPMBR(&drive)) {
35     Error("Unable to read PMBR\n");
36     retval = CGPT_FAILED;
37     goto done;
38   }
39 
40   char buf[GUID_STRLEN];
41   GuidToStr(&drive.pmbr.boot_guid, buf, sizeof(buf));
42 
43   int numEntries = GetNumberOfEntries(&drive);
44   int i;
45   for(i = 0; i < numEntries; i++) {
46       GptEntry *entry = GetEntry(&drive.gpt, ANY_VALID, i);
47 
48       if (GuidEqual(&entry->unique, &drive.pmbr.boot_guid)) {
49         params->partition = i + 1;
50         retval = CGPT_OK;
51         goto done;
52       }
53   }
54 
55   Error("Didn't find any boot partition\n");
56   params->partition = 0;
57   retval = CGPT_FAILED;
58 
59 done:
60   (void) DriveClose(&drive, 1);
61   return retval;
62 }
63 
64 
65 int CgptBoot(CgptBootParams *params) {
66   struct drive drive;
67   int retval = 1;
68   int gpt_retval= 0;
69   int mode = O_RDONLY;
70 
71   if (params == NULL)
72     return CGPT_FAILED;
73 
74   if (params->create_pmbr || params->partition || params->bootfile)
75     mode = O_RDWR;
76 
77   if (CGPT_OK != DriveOpen(params->drive_name, &drive, mode,
78                            params->drive_size)) {
79     return CGPT_FAILED;
80   }
81 
82   if (CGPT_OK != ReadPMBR(&drive)) {
83     Error("Unable to read PMBR\n");
84     goto done;
85   }
86 
87   if (params->create_pmbr) {
88     drive.pmbr.magic[0] = 0x1d;
89     drive.pmbr.magic[1] = 0x9a;
90     drive.pmbr.sig[0] = 0x55;
91     drive.pmbr.sig[1] = 0xaa;
92     memset(&drive.pmbr.part, 0, sizeof(drive.pmbr.part));
93     drive.pmbr.part[0].f_head = 0x00;
94     drive.pmbr.part[0].f_sect = 0x02;
95     drive.pmbr.part[0].f_cyl = 0x00;
96     drive.pmbr.part[0].type = 0xee;
97     drive.pmbr.part[0].l_head = 0xff;
98     drive.pmbr.part[0].l_sect = 0xff;
99     drive.pmbr.part[0].l_cyl = 0xff;
100     drive.pmbr.part[0].f_lba = htole32(1);
101     uint32_t max = 0xffffffff;
102     if (drive.gpt.streaming_drive_sectors < 0xffffffff)
103       max = drive.gpt.streaming_drive_sectors - 1;
104     drive.pmbr.part[0].num_sect = htole32(max);
105   }
106 
107   if (params->partition) {
108     if (GPT_SUCCESS != (gpt_retval = GptSanityCheck(&drive.gpt))) {
109       Error("GptSanityCheck() returned %d: %s\n",
110             gpt_retval, GptError(gpt_retval));
111       goto done;
112     }
113 
114     if (params->partition > GetNumberOfEntries(&drive)) {
115       Error("invalid partition number: %d\n", params->partition);
116       goto done;
117     }
118 
119     uint32_t index = params->partition - 1;
120     GptEntry *entry = GetEntry(&drive.gpt, ANY_VALID, index);
121     memcpy(&drive.pmbr.boot_guid, &entry->unique, sizeof(Guid));
122   }
123 
124   if (params->bootfile) {
125     int fd = open(params->bootfile, O_RDONLY);
126     if (fd < 0) {
127       Error("Can't read %s: %s\n", params->bootfile, strerror(errno));
128       goto done;
129     }
130 
131     int n = read(fd, drive.pmbr.bootcode, sizeof(drive.pmbr.bootcode));
132     if (n < 1) {
133       Error("problem reading %s: %s\n", params->bootfile, strerror(errno));
134       close(fd);
135       goto done;
136     }
137 
138     close(fd);
139   }
140 
141   char buf[GUID_STRLEN];
142   GuidToStr(&drive.pmbr.boot_guid, buf, sizeof(buf));
143   printf("%s\n", buf);
144 
145   // Write it all out, if needed.
146   if (mode == O_RDONLY || CGPT_OK == WritePMBR(&drive))
147     retval = 0;
148 
149 done:
150   (void) DriveClose(&drive, 1);
151   return retval;
152 }
153