machoXX.c 9.8 KB
Newer Older
phcoder's avatar
phcoder committed
1 2 3 4

#include <grub/file.h>
#include <grub/mm.h>
#include <grub/misc.h>
5
#include <grub/i18n.h>
phcoder's avatar
phcoder committed
6 7 8

#define min(a,b) (((a) < (b)) ? (a) : (b))

9
static int
phcoder's avatar
phcoder committed
10 11 12 13 14 15
SUFFIX (grub_macho_contains_macho) (grub_macho_t macho)
{
  return macho->offsetXX != -1;
}

void
16
SUFFIX (grub_macho_parse) (grub_macho_t macho, const char *filename)
phcoder's avatar
phcoder committed
17
{
18 19 20 21
  union {
    struct grub_macho_lzss_header lzss;
    grub_macho_header_t macho;
  } head;
phcoder's avatar
phcoder committed
22 23 24 25 26

  /* Is there any candidate at all? */
  if (macho->offsetXX == -1)
    return;

27
  /* Read header and check magic.  */
phcoder's avatar
phcoder committed
28 29
  if (grub_file_seek (macho->file, macho->offsetXX) == (grub_off_t) -1
      || grub_file_read (macho->file, &head, sizeof (head))
30
      != sizeof (head))
phcoder's avatar
phcoder committed
31
    {
32 33 34
      if (!grub_errno)
	grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
		    filename);
phcoder's avatar
phcoder committed
35 36 37
      macho->offsetXX = -1;
      return;
    }
38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56
  if (grub_memcmp (head.lzss.magic, GRUB_MACHO_LZSS_MAGIC,
		   sizeof (head.lzss.magic)) == 0)
    {
      macho->compressed_sizeXX = grub_be_to_cpu32 (head.lzss.compressed_size);
      macho->uncompressed_sizeXX
	= grub_be_to_cpu32 (head.lzss.uncompressed_size);
      if (macho->uncompressed_sizeXX < sizeof (head.macho))
	{
	  grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
		      filename);
	  macho->offsetXX = -1;
	  return;
	}
      /* Skip header check.  */
      macho->compressedXX = 1;
      return;
    }

  if (head.macho.magic != GRUB_MACHO_MAGIC)
phcoder's avatar
phcoder committed
57
    {
58
      grub_error (GRUB_ERR_BAD_OS, "invalid Mach-O  header");
phcoder's avatar
phcoder committed
59 60 61 62 63
      macho->offsetXX = -1;
      return;
    }

  /* Read commands. */
64 65 66
  macho->ncmdsXX = head.macho.ncmds;
  macho->cmdsizeXX = head.macho.sizeofcmds;
  macho->cmdsXX = grub_malloc (macho->cmdsizeXX);
phcoder's avatar
phcoder committed
67
  if (! macho->cmdsXX)
68
    return;
69 70 71
  if (grub_file_seek (macho->file, macho->offsetXX
		      + sizeof (grub_macho_header_t)) == (grub_off_t) -1
      || grub_file_read (macho->file, macho->cmdsXX,
phcoder's avatar
phcoder committed
72 73 74
		      (grub_size_t) macho->cmdsizeXX)
      != (grub_ssize_t) macho->cmdsizeXX)
    {
75 76 77
      if (!grub_errno)
	grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
		    filename);
phcoder's avatar
phcoder committed
78 79 80 81
      macho->offsetXX = -1;
    }
}

82
typedef int (*grub_macho_iter_hook_t)
phcoder's avatar
phcoder committed
83 84 85 86 87 88
(grub_macho_t , struct grub_macho_cmd *,
	       void *);

static grub_err_t
grub_macho_cmds_iterate (grub_macho_t macho,
			 grub_macho_iter_hook_t hook,
89 90
			 void *hook_arg,
			 const char *filename)
phcoder's avatar
phcoder committed
91
{
92
  grub_uint8_t *hdrs;
phcoder's avatar
phcoder committed
93
  int i;
94 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 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155

  if (macho->compressedXX && !macho->uncompressedXX)
    {
      grub_uint8_t *tmp;
      grub_macho_header_t *head;
      macho->uncompressedXX = grub_malloc (macho->uncompressed_sizeXX);
      if (!macho->uncompressedXX)
	return grub_errno;
      tmp = grub_malloc (macho->compressed_sizeXX);
      if (!tmp)
	{
	  grub_free (macho->uncompressedXX);
	  macho->uncompressedXX = 0;
	  return grub_errno;
	}
      if (grub_file_seek (macho->file, macho->offsetXX
			  + GRUB_MACHO_LZSS_OFFSET) == (grub_off_t) -1
	  || grub_file_read (macho->file, tmp,
			     (grub_size_t) macho->compressed_sizeXX)
	  != (grub_ssize_t) macho->compressed_sizeXX)
	{
	  if (!grub_errno)
	    grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
			filename);
	  grub_free (tmp);
	  grub_free (macho->uncompressedXX);
	  macho->uncompressedXX = 0;
	  macho->offsetXX = -1;
	  return grub_errno;
	}
      if (grub_decompress_lzss (macho->uncompressedXX,
				macho->uncompressedXX
				+ macho->uncompressed_sizeXX,
				tmp, tmp + macho->compressed_sizeXX)
	  != macho->uncompressed_sizeXX)
	{
	  if (!grub_errno)
	    grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
			filename);
	  grub_free (tmp);
	  grub_free (macho->uncompressedXX);
	  macho->uncompressedXX = 0;
	  macho->offsetXX = -1;
	  return grub_errno;
	}
      grub_free (tmp);
      head = (grub_macho_header_t *) macho->uncompressedXX;
      macho->ncmdsXX = head->ncmds;
      macho->cmdsizeXX = head->sizeofcmds;
      macho->cmdsXX = macho->uncompressedXX + sizeof (grub_macho_header_t);
      if (sizeof (grub_macho_header_t) + macho->cmdsizeXX
	  >= macho->uncompressed_sizeXX)
	{
	  grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
		      filename);
	  grub_free (macho->uncompressedXX);
	  macho->uncompressedXX = 0;
	  macho->offsetXX = -1;
	  return grub_errno;
	}
    }

phcoder's avatar
phcoder committed
156
  if (! macho->cmdsXX)
157
    return grub_error (GRUB_ERR_BAD_OS, "couldn't find Mach-O commands");
158
  hdrs = macho->cmdsXX;
phcoder's avatar
phcoder committed
159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178
  for (i = 0; i < macho->ncmdsXX; i++)
    {
      struct grub_macho_cmd *hdr = (struct grub_macho_cmd *) hdrs;
      if (hook (macho, hdr, hook_arg))
	break;
      hdrs += hdr->cmdsize;
    }

  return grub_errno;
}

grub_size_t
SUFFIX (grub_macho_filesize) (grub_macho_t macho)
{
  if (SUFFIX (grub_macho_contains_macho) (macho))
    return macho->endXX - macho->offsetXX;
  return 0;
}

grub_err_t
179 180 181
SUFFIX (grub_macho_readfile) (grub_macho_t macho,
			      const char *filename,
			      void *dest)
phcoder's avatar
phcoder committed
182 183 184 185
{
  grub_ssize_t read;
  if (! SUFFIX (grub_macho_contains_macho) (macho))
    return grub_error (GRUB_ERR_BAD_OS,
186
		       "couldn't read architecture-specific part");
phcoder's avatar
phcoder committed
187

188
  if (grub_file_seek (macho->file, macho->offsetXX) == (grub_off_t) -1)
189
    return grub_errno;
phcoder's avatar
phcoder committed
190 191

  read = grub_file_read (macho->file, dest,
192 193
			 macho->endXX - macho->offsetXX);
  if (read != (grub_ssize_t) (macho->endXX - macho->offsetXX))
phcoder's avatar
phcoder committed
194
    {
195 196 197 198
      if (!grub_errno)
	grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
		    filename);
      return grub_errno;
phcoder's avatar
phcoder committed
199 200 201 202
    }
  return GRUB_ERR_NONE;
}

203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236
struct calcsize_ctx
{
  int flags;
  int nr_phdrs;
  grub_macho_addr_t *segments_start;
  grub_macho_addr_t *segments_end;
};

/* Run through the program headers to calculate the total memory size we
   should claim.  */
static int
calcsize (grub_macho_t _macho __attribute__ ((unused)),
	  struct grub_macho_cmd *hdr0,
	  void *_arg)
{
  grub_macho_segment_t *hdr = (grub_macho_segment_t *) hdr0;
  struct calcsize_ctx *ctx = _arg;
  if (hdr->cmd != GRUB_MACHO_CMD_SEGMENT)
    return 0;

  if (! hdr->vmsize)
    return 0;

  if (! hdr->filesize && (ctx->flags & GRUB_MACHO_NOBSS))
    return 0;

  ctx->nr_phdrs++;
  if (hdr->vmaddr < *ctx->segments_start)
    *ctx->segments_start = hdr->vmaddr;
  if (hdr->vmaddr + hdr->vmsize > *ctx->segments_end)
    *ctx->segments_end = hdr->vmaddr + hdr->vmsize;
  return 0;
}

phcoder's avatar
phcoder committed
237 238 239
/* Calculate the amount of memory spanned by the segments. */
grub_err_t
SUFFIX (grub_macho_size) (grub_macho_t macho, grub_macho_addr_t *segments_start,
240 241
			  grub_macho_addr_t *segments_end, int flags,
			  const char *filename)
phcoder's avatar
phcoder committed
242
{
243 244 245 246 247 248
  struct calcsize_ctx ctx = {
    .flags = flags,
    .nr_phdrs = 0,
    .segments_start = segments_start,
    .segments_end = segments_end,
  };
phcoder's avatar
phcoder committed
249 250 251 252

  *segments_start = (grub_macho_addr_t) -1;
  *segments_end = 0;

253
  grub_macho_cmds_iterate (macho, calcsize, &ctx, filename);
phcoder's avatar
phcoder committed
254

255
  if (ctx.nr_phdrs == 0)
256
    return grub_error (GRUB_ERR_BAD_OS, "no program headers present");
phcoder's avatar
phcoder committed
257 258 259

  if (*segments_end < *segments_start)
    /* Very bad addresses.  */
260
    return grub_error (GRUB_ERR_BAD_OS, "bad program header load addresses");
phcoder's avatar
phcoder committed
261 262 263 264

  return GRUB_ERR_NONE;
}

265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346
struct do_load_ctx
{
  int flags;
  char *offset;
  const char *filename;
  int *darwin_version;
};

static int
do_load(grub_macho_t _macho,
	struct grub_macho_cmd *hdr0,
	void *_arg)
{
  grub_macho_segment_t *hdr = (grub_macho_segment_t *) hdr0;
  struct do_load_ctx *ctx = _arg;

  if (hdr->cmd != GRUB_MACHO_CMD_SEGMENT)
    return 0;

  if (! hdr->filesize && (ctx->flags & GRUB_MACHO_NOBSS))
    return 0;
  if (! hdr->vmsize)
    return 0;

  if (hdr->filesize)
    {
      grub_ssize_t read, toread = min (hdr->filesize, hdr->vmsize);
      if (_macho->uncompressedXX)
	{
	  if (hdr->fileoff + (grub_size_t) toread
	      > _macho->uncompressed_sizeXX)
	    read = -1;
	  else
	    {
	      read = toread;
	      grub_memcpy (ctx->offset + hdr->vmaddr,
			   _macho->uncompressedXX + hdr->fileoff, read);
	    }
	}
      else
	{
	  if (grub_file_seek (_macho->file, hdr->fileoff
			      + _macho->offsetXX) == (grub_off_t) -1)
	    return 1;
	  read = grub_file_read (_macho->file, ctx->offset + hdr->vmaddr,
				 toread);
	}

      if (read != toread)
	{
	  /* XXX How can we free memory from `load_hook'? */
	  if (!grub_errno)
	    grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
			ctx->filename);

	  return 1;
	}
      if (ctx->darwin_version)
	{
	  const char *ptr = ctx->offset + hdr->vmaddr;
	  const char *end = ptr + min (hdr->filesize, hdr->vmsize)
	    - (sizeof ("Darwin Kernel Version ") - 1);
	  for (; ptr < end; ptr++)
	    if (grub_memcmp (ptr, "Darwin Kernel Version ",
			     sizeof ("Darwin Kernel Version ") - 1) == 0)
	      {
		ptr += sizeof ("Darwin Kernel Version ") - 1;
		*ctx->darwin_version = 0;
		end += (sizeof ("Darwin Kernel Version ") - 1);
		while (ptr < end && grub_isdigit (*ptr))
		  *ctx->darwin_version = (*ptr++ - '0') + *ctx->darwin_version * 10;
		break;
	      }
	}
    }

  if (hdr->filesize < hdr->vmsize)
    grub_memset (ctx->offset + hdr->vmaddr + hdr->filesize,
		 0, hdr->vmsize - hdr->filesize);
  return 0;
}

phcoder's avatar
phcoder committed
347 348
/* Load every loadable segment into memory specified by `_load_hook'.  */
grub_err_t
349
SUFFIX (grub_macho_load) (grub_macho_t macho, const char *filename,
350
			  char *offset, int flags, int *darwin_version)
phcoder's avatar
phcoder committed
351
{
352 353 354 355 356 357
  struct do_load_ctx ctx = {
    .flags = flags,
    .offset = offset,
    .filename = filename,
    .darwin_version = darwin_version
  };
phcoder's avatar
phcoder committed
358

359 360 361
  if (darwin_version)
    *darwin_version = 0;

362
  grub_macho_cmds_iterate (macho, do_load, &ctx, filename);
phcoder's avatar
phcoder committed
363

364
  return grub_errno;
phcoder's avatar
phcoder committed
365 366
}

367 368 369 370 371 372 373 374 375 376 377
static int
find_entry_point (grub_macho_t _macho __attribute__ ((unused)),
			    struct grub_macho_cmd *hdr,
			    void *_arg)
{
  grub_macho_addr_t *entry_point = _arg;
  if (hdr->cmd == GRUB_MACHO_CMD_THREAD)
    *entry_point = ((grub_macho_thread_t *) hdr)->entry_point;
  return 0;
}

phcoder's avatar
phcoder committed
378
grub_macho_addr_t
379
SUFFIX (grub_macho_get_entry_point) (grub_macho_t macho, const char *filename)
phcoder's avatar
phcoder committed
380 381
{
  grub_macho_addr_t entry_point = 0;
382
  grub_macho_cmds_iterate (macho, find_entry_point, &entry_point, filename);
phcoder's avatar
phcoder committed
383 384
  return entry_point;
}