hexdump.c 3.31 KB
Newer Older
1 2 3
/* hexdump.c - command to dump the contents of a file or memory */
/*
 *  GRUB  --  GRand Unified Bootloader
4
 *  Copyright (C) 2007,2008,2009  Free Software Foundation, Inc.
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 *
 *  GRUB is free software: you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation, either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  GRUB is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with GRUB.  If not, see <http://www.gnu.org/licenses/>.
 */

#include <grub/dl.h>
#include <grub/file.h>
#include <grub/disk.h>
#include <grub/misc.h>
24
#include <grub/lib/hexdump.h>
25
#include <grub/extcmd.h>
26
#include <grub/i18n.h>
27

28 29
GRUB_MOD_LICENSE ("GPLv3+");

30
static const struct grub_arg_option options[] = {
31
  {"skip", 's', 0, N_("Skip offset bytes from the beginning of file."), 0,
32
   ARG_TYPE_INT},
33
  {"length", 'n', 0, N_("Read only LENGTH bytes."), 0, ARG_TYPE_INT},
34 35 36 37
  {0, 0, 0, 0, 0, 0}
};

static grub_err_t
38
grub_cmd_hexdump (grub_extcmd_context_t ctxt, int argc, char **args)
39
{
40
  struct grub_arg_list *state = ctxt->state;
41
  char buf[GRUB_DISK_SECTOR_SIZE * 4];
42
  grub_ssize_t size, length;
43
  grub_disk_addr_t skip;
44
  int namelen;
45 46

  if (argc != 1)
47
    return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected"));
48

49
  namelen = grub_strlen (args[0]);
50
  skip = (state[0].set) ? grub_strtoull (state[0].arg, 0, 0) : 0;
51
  length = (state[1].set) ? grub_strtoul (state[1].arg, 0, 0) : 256;
52

53
  if (!grub_strcmp (args[0], "(mem)"))
54
    hexdump (skip, (char *) (grub_addr_t) skip, length);
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
  else if ((args[0][0] == '(') && (args[0][namelen - 1] == ')'))
    {
      grub_disk_t disk;
      grub_disk_addr_t sector;
      grub_size_t ofs;

      args[0][namelen - 1] = 0;
      disk = grub_disk_open (&args[0][1]);
      if (! disk)
        return 0;

      sector = (skip >> (GRUB_DISK_SECTOR_BITS + 2)) * 4;
      ofs = skip & (GRUB_DISK_SECTOR_SIZE * 4 - 1);
      while (length)
        {
70
          grub_size_t len;
71 72

          len = length;
73 74
          if (len > sizeof (buf))
            len = sizeof (buf);
75

76
          if (grub_disk_read (disk, sector, ofs, len, buf))
77 78
            break;

79
          hexdump (skip, buf, len);
80 81 82 83 84 85 86 87 88 89

          ofs = 0;
          skip += len;
          length -= len;
          sector += 4;
        }

      grub_disk_close (disk);
    }
  else
90
    {
91 92
      grub_file_t file;

93
      file = grub_file_open (args[0]);
94
      if (! file)
95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119
	return 0;

      file->offset = skip;

      while ((size = grub_file_read (file, buf, sizeof (buf))) > 0)
	{
	  unsigned long len;

	  len = ((length) && (size > length)) ? length : size;
	  hexdump (skip, buf, len);
	  skip += len;
	  if (length)
	    {
	      length -= len;
	      if (!length)
		break;
	    }
	}

      grub_file_close (file);
    }

  return 0;
}

120
static grub_extcmd_t cmd;
121 122 123

GRUB_MOD_INIT (hexdump)
{
124
  cmd = grub_register_extcmd ("hexdump", grub_cmd_hexdump, 0,
125
			      N_("[OPTIONS] FILE_OR_DEVICE"),
126
			      N_("Show raw contents of a file or memory."),
127
			      options);
128 129 130 131
}

GRUB_MOD_FINI (hexdump)
{
132
  grub_unregister_extcmd (cmd);
133
}