// Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <getopt.h>
#include <string.h>
#include "cgpt.h"
#include "vboot_host.h"
extern const char* progname;
static void Usage(void)
{
printf("\nUsage: %s find [OPTIONS] [DRIVE]\n\n"
"Find a partition by its UUID or label. With no specified DRIVE\n"
"it scans all physical drives.\n\n"
"Options:\n"
" -D NUM Size (in bytes) of the disk where partitions reside\n"
" default 0, meaning partitions and GPT structs are\n"
" both on DRIVE\n"
" -t GUID Search for Partition Type GUID\n"
" -u GUID Search for Partition Unique ID\n"
" -l LABEL Search for Label\n"
" -v Be verbose in displaying matches (repeatable)\n"
" -n Numeric output only\n"
" -1 Fail if more than one match is found\n"
" -M FILE"
" Matching partition data must also contain FILE content\n"
" -O NUM"
" Byte offset into partition to match content (default 0)\n"
"\n", progname);
PrintTypes();
}
// read a file into a buffer, return buffer and update size
static uint8_t *ReadFile(const char *filename, uint64_t *size) {
FILE *f;
uint8_t *buf;
f = fopen(filename, "rb");
if (!f) {
return NULL;
}
fseek(f, 0, SEEK_END);
*size = ftell(f);
rewind(f);
buf = malloc(*size);
if (!buf) {
fclose(f);
return NULL;
}
if(1 != fread(buf, *size, 1, f)) {
fclose(f);
free(buf);
return NULL;
}
fclose(f);
return buf;
}
int cmd_find(int argc, char *argv[]) {
CgptFindParams params;
memset(¶ms, 0, sizeof(params));
int i;
int errorcnt = 0;
char *e = 0;
int c;
opterr = 0; // quiet, you
while ((c=getopt(argc, argv, ":hv1nt:u:l:M:O:D:")) != -1)
{
switch (c)
{
case 'D':
params.drive_size = strtoull(optarg, &e, 0);
if (!*optarg || (e && *e))
{
Error("invalid argument to -%c: \"%s\"\n", c, optarg);
errorcnt++;
}
break;
case 'v':
params.verbose++;
break;
case 'n':
params.numeric = 1;
break;
case '1':
params.oneonly = 1;
break;
case 'l':
params.set_label = 1;
params.label = optarg;
break;
case 't':
params.set_type = 1;
if (CGPT_OK != SupportedType(optarg, ¶ms.type_guid) &&
CGPT_OK != StrToGuid(optarg, ¶ms.type_guid)) {
Error("invalid argument to -%c: %s\n", c, optarg);
errorcnt++;
}
break;
case 'u':
params.set_unique = 1;
if (CGPT_OK != StrToGuid(optarg, ¶ms.unique_guid)) {
Error("invalid argument to -%c: %s\n", c, optarg);
errorcnt++;
}
break;
case 'M':
params.matchbuf = ReadFile(optarg, ¶ms.matchlen);
if (!params.matchbuf || !params.matchlen) {
Error("Unable to read from %s\n", optarg);
errorcnt++;
}
// Go ahead and allocate space for the comparison too
params.comparebuf = (uint8_t *)malloc(params.matchlen);
if (!params.comparebuf) {
Error("Unable to allocate %" PRIu64 "bytes for comparison buffer\n",
params.matchlen);
errorcnt++;
}
break;
case 'O':
params.matchoffset = strtoull(optarg, &e, 0);
if (!*optarg || (e && *e)) {
Error("invalid argument to -%c: \"%s\"\n", c, optarg);
errorcnt++;
}
break;
case 'h':
Usage();
return CGPT_OK;
case '?':
Error("unrecognized option: -%c\n", optopt);
errorcnt++;
break;
case ':':
Error("missing argument to -%c\n", optopt);
errorcnt++;
break;
default:
errorcnt++;
break;
}
}
if (!params.set_unique && !params.set_type && !params.set_label) {
Error("You must specify at least one of -t, -u, or -l\n");
errorcnt++;
}
if (errorcnt)
{
Usage();
return CGPT_FAILED;
}
if (optind < argc) {
for (i=optind; i<argc; i++) {
params.drive_name = argv[i];
CgptFind(¶ms);
}
} else {
CgptFind(¶ms);
}
if (params.oneonly && params.hits != 1) {
return CGPT_FAILED;
}
if (params.match_partnum) {
return CGPT_OK;
}
return CGPT_FAILED;
}