Home | History | Annotate | Download | only in minijail
      1 /* Copyright (c) 2014 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 
      6 #include <elf.h>
      7 #include <endian.h>
      8 #include <stdint.h>
      9 #include <stdio.h>
     10 #include <stdlib.h>
     11 #include <unistd.h>
     12 
     13 #include "elfparse.h"
     14 
     15 int is_elf_magic (const uint8_t *buf)
     16 {
     17 	return (buf[EI_MAG0] == ELFMAG0) &&
     18 	       (buf[EI_MAG1] == ELFMAG1) &&
     19 	       (buf[EI_MAG2] == ELFMAG2) &&
     20 	       (buf[EI_MAG3] == ELFMAG3);
     21 }
     22 
     23 #define parseElftemplate(bit)                                                \
     24 ElfType parseElf ## bit(FILE *elf_file, uint8_t *pHead, int little_endian)   \
     25 {                                                                            \
     26 	ElfType                      ret          = ELFSTATIC;               \
     27 	Minijail_Elf ## bit ## _Ehdr *pHeader     = NULL;                    \
     28 	Minijail_Elf ## bit ## _Phdr pheader;                                \
     29 	uint32_t                     i            = 0;                       \
     30 	                                                                     \
     31 	if (!elf_file || !pHead)                                             \
     32 		return ELFERROR;                                             \
     33 	                                                                     \
     34 	pHeader = (Minijail_Elf ## bit ## _Ehdr *)pHead;                     \
     35 	if (little_endian) {                                                 \
     36 		pHeader->e_phoff = le ## bit ## toh(pHeader->e_phoff);       \
     37 		pHeader->e_phentsize = le16toh(pHeader->e_phentsize);        \
     38 		pHeader->e_phnum = le16toh(pHeader->e_phnum);                \
     39 	} else {                                                             \
     40 		pHeader->e_phoff = be ## bit ## toh(pHeader->e_phoff);       \
     41 		pHeader->e_phentsize = be16toh(pHeader->e_phentsize);        \
     42 		pHeader->e_phnum = be16toh(pHeader->e_phnum);                \
     43 	}                                                                    \
     44 	if (pHeader->e_phentsize != sizeof(Minijail_Elf ## bit ## _Phdr))    \
     45 		return ELFERROR;                                             \
     46 	                                                                     \
     47 	if (fseek(elf_file, pHeader->e_phoff, SEEK_SET) != 0)                \
     48 		return ELFERROR;                                             \
     49 	                                                                     \
     50 	for (i = 0; i < pHeader->e_phnum; i++) {                             \
     51 		if (fread(&pheader, sizeof(pheader), 1, elf_file) == 1) {    \
     52 			if (pheader.p_type == PT_INTERP) {                   \
     53 				ret = ELFDYNAMIC;                            \
     54 				break;                                       \
     55 			}                                                    \
     56 		} else {                                                     \
     57 			ret = ELFERROR;                                      \
     58 			break;                                               \
     59 		}                                                            \
     60 	}                                                                    \
     61 	return ret;                                                          \
     62 }
     63 parseElftemplate(64)
     64 parseElftemplate(32)
     65 
     66 /* Public function to determine the linkage of an ELF. */
     67 ElfType get_elf_linkage(const char *path)
     68 {
     69 	ElfType ret = ELFERROR;
     70 	FILE *elf_file = NULL;
     71 	uint8_t pHeader[HEADERSIZE] = "";
     72 
     73 	elf_file = fopen(path, "r");
     74 	if (elf_file) {
     75 		if (fread(pHeader, 1, HEADERSIZE, elf_file) == HEADERSIZE) {
     76 			if (is_elf_magic(pHeader)) {
     77 				if ((pHeader[EI_DATA] == ELFDATA2LSB) &&
     78 				    (pHeader[EI_CLASS] == ELFCLASS64)) {
     79 					/* 64-bit little endian. */
     80 					ret = parseElf64(elf_file, pHeader, 1);
     81 				} else if ((pHeader[EI_DATA] == ELFDATA2MSB) &&
     82 					  (pHeader[EI_CLASS] == ELFCLASS64)) {
     83 					/* 64-bit big endian. */
     84 					ret = parseElf64(elf_file, pHeader, 0);
     85 				} else if ((pHeader[EI_DATA] == ELFDATA2LSB) &&
     86 					  (pHeader[EI_CLASS] == ELFCLASS32)) {
     87 					/* 32-bit little endian. */
     88 					ret = parseElf32(elf_file, pHeader, 1);
     89 				} else if ((pHeader[EI_DATA] == ELFDATA2MSB) &&
     90 					  (pHeader[EI_CLASS] == ELFCLASS32)) {
     91 					/* 32-bit big endian. */
     92 					ret = parseElf32(elf_file, pHeader, 0);
     93 				}
     94 			} else {
     95 				/*
     96 				 * The binary is not an ELF. We assume it's a
     97 				 * script. We should parse the #! line and
     98 				 * check the interpreter to guard against
     99 				 * static interpreters escaping the sandbox.
    100 				 * As Minijail is only called from the rootfs
    101 				 * it was deemed not necessary to check this.
    102 				 * So we will just let execve(2) decide if this
    103 				 * is valid.
    104 				 */
    105 				ret = ELFDYNAMIC;
    106 			}
    107 		} else {
    108 			/*
    109 			 * The file is smaller than |HEADERSIZE| bytes.
    110 			 * We assume it's a short script. See above for
    111 			 * reasoning on scripts.
    112 			 */
    113 			ret = ELFDYNAMIC;
    114 		}
    115 		fclose(elf_file);
    116 	}
    117 	return ret;
    118 }
    119