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