1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.compatibility.common.util;
18 
19 import java.io.File;
20 import java.io.IOException;
21 import java.io.RandomAccessFile;
22 import java.util.HashMap;
23 import java.util.Map;
24 
25 /**
26  * A poor man's implementation of the readelf command. This program is designed
27  * to parse ELF (Executable and Linkable Format) files.
28  */
29 public class ReadElf implements AutoCloseable {
30     /** The magic values for the ELF identification. */
31     private static final byte[] ELFMAG = {
32             (byte) 0x7F, (byte) 'E', (byte) 'L', (byte) 'F', };
33 
34     private static final int EI_NIDENT = 16;
35 
36     private static final int EI_CLASS = 4;
37     private static final int EI_DATA = 5;
38 
39     private static final int EM_386 = 3;
40     private static final int EM_ARM = 40;
41     private static final int EM_X86_64 = 62;
42     // http://en.wikipedia.org/wiki/Qualcomm_Hexagon
43     private static final int EM_QDSP6 = 164;
44     private static final int EM_AARCH64 = 183;
45     private static final int EM_RISCV = 243;
46 
47     private static final int ELFCLASS32 = 1;
48     private static final int ELFCLASS64 = 2;
49 
50     private static final int ELFDATA2LSB = 1;
51     private static final int ELFDATA2MSB = 2;
52 
53     private static final int EV_CURRENT = 1;
54 
55     private static final long PT_LOAD = 1;
56 
57     private static final int SHT_SYMTAB = 2;
58     private static final int SHT_STRTAB = 3;
59     private static final int SHT_DYNAMIC = 6;
60     private static final int SHT_DYNSYM = 11;
61 
62     public static class Symbol {
63         public static final int STB_LOCAL = 0;
64         public static final int STB_GLOBAL = 1;
65         public static final int STB_WEAK = 2;
66         public static final int STB_LOPROC = 13;
67         public static final int STB_HIPROC = 15;
68 
69         public static final int STT_NOTYPE = 0;
70         public static final int STT_OBJECT = 1;
71         public static final int STT_FUNC = 2;
72         public static final int STT_SECTION = 3;
73         public static final int STT_FILE = 4;
74         public static final int STT_COMMON = 5;
75         public static final int STT_TLS = 6;
76 
77         public final String name;
78         public final int bind;
79         public final int type;
80 
Symbol(String name, int st_info)81         Symbol(String name, int st_info) {
82             this.name = name;
83             this.bind = (st_info >> 4) & 0x0F;
84             this.type = st_info & 0x0F;
85         }
86 
87         @Override
toString()88         public String toString() {
89             return "Symbol[" + name + "," + toBind() + "," + toType() + "]";
90         }
91 
toBind()92         private String toBind() {
93             switch (bind) {
94                 case STB_LOCAL:
95                     return "LOCAL";
96                 case STB_GLOBAL:
97                     return "GLOBAL";
98                 case STB_WEAK:
99                     return "WEAK";
100             }
101             return "STB_??? (" + bind + ")";
102         }
103 
toType()104         private String toType() {
105             switch (type) {
106                 case STT_NOTYPE:
107                     return "NOTYPE";
108                 case STT_OBJECT:
109                     return "OBJECT";
110                 case STT_FUNC:
111                     return "FUNC";
112                 case STT_SECTION:
113                     return "SECTION";
114                 case STT_FILE:
115                     return "FILE";
116                 case STT_COMMON:
117                     return "COMMON";
118                 case STT_TLS:
119                     return "TLS";
120             }
121             return "STT_??? (" + type + ")";
122         }
123     }
124 
125     private final String mPath;
126     private final RandomAccessFile mFile;
127     private final byte[] mBuffer = new byte[512];
128     private int mEndian;
129     private boolean mIsDynamic;
130     private boolean mIsPIE;
131     private int mType;
132     private int mAddrSize;
133 
134     /** Symbol Table offset */
135     private long mSymTabOffset;
136 
137     /** Symbol Table size */
138     private long mSymTabSize;
139 
140     /** Dynamic Symbol Table offset */
141     private long mDynSymOffset;
142 
143     /** Dynamic Symbol Table size */
144     private long mDynSymSize;
145 
146     /** Section Header String Table offset */
147     private long mShStrTabOffset;
148 
149     /** Section Header String Table size */
150     private long mShStrTabSize;
151 
152     /** String Table offset */
153     private long mStrTabOffset;
154 
155     /** String Table size */
156     private long mStrTabSize;
157 
158     /** Dynamic String Table offset */
159     private long mDynStrOffset;
160 
161     /** Dynamic String Table size */
162     private long mDynStrSize;
163 
164     /** Symbol Table symbol names */
165     private Map<String, Symbol> mSymbols;
166 
167     /** Dynamic Symbol Table symbol names */
168     private Map<String, Symbol> mDynamicSymbols;
169 
read(File file)170     public static ReadElf read(File file) throws IOException {
171         return new ReadElf(file);
172     }
173 
main(String[] args)174     public static void main(String[] args) throws IOException {
175         for (String arg : args) {
176             ReadElf re = new ReadElf(new File(arg));
177             re.getSymbol("x");
178             re.getDynamicSymbol("x");
179             re.close();
180         }
181     }
182 
isDynamic()183     public boolean isDynamic() {
184         return mIsDynamic;
185     }
186 
getType()187     public int getType() {
188         return mType;
189     }
190 
isPIE()191     public boolean isPIE() {
192         return mIsPIE;
193     }
194 
ReadElf(File file)195     private ReadElf(File file) throws IOException {
196         mPath = file.getPath();
197         mFile = new RandomAccessFile(file, "r");
198 
199         if (mFile.length() < EI_NIDENT) {
200             throw new IllegalArgumentException("Too small to be an ELF file: " + file);
201         }
202 
203         readHeader();
204     }
205 
206     @Override
close()207     public void close() {
208         try {
209             mFile.close();
210         } catch (IOException ignored) {
211         }
212     }
213 
214     @Override
finalize()215     protected void finalize() throws Throwable {
216         try {
217             close();
218         } finally {
219             super.finalize();
220         }
221     }
222 
readHeader()223     private void readHeader() throws IOException {
224         mFile.seek(0);
225         mFile.readFully(mBuffer, 0, EI_NIDENT);
226 
227         if (mBuffer[0] != ELFMAG[0] || mBuffer[1] != ELFMAG[1] ||
228                 mBuffer[2] != ELFMAG[2] || mBuffer[3] != ELFMAG[3]) {
229             throw new IllegalArgumentException("Invalid ELF file: " + mPath);
230         }
231 
232         int elfClass = mBuffer[EI_CLASS];
233         if (elfClass == ELFCLASS32) {
234             mAddrSize = 4;
235         } else if (elfClass == ELFCLASS64) {
236             mAddrSize = 8;
237         } else {
238             throw new IOException("Invalid ELF EI_CLASS: " + elfClass + ": " + mPath);
239         }
240 
241         mEndian = mBuffer[EI_DATA];
242         if (mEndian == ELFDATA2LSB) {
243         } else if (mEndian == ELFDATA2MSB) {
244             throw new IOException("Unsupported ELFDATA2MSB file: " + mPath);
245         } else {
246             throw new IOException("Invalid ELF EI_DATA: " + mEndian + ": " + mPath);
247         }
248 
249         mType = readHalf();
250 
251         int e_machine = readHalf();
252         if (e_machine != EM_386 && e_machine != EM_X86_64 &&
253                 e_machine != EM_AARCH64 && e_machine != EM_ARM &&
254                 e_machine != EM_RISCV && e_machine != EM_QDSP6) {
255             throw new IOException("Invalid ELF e_machine: " + e_machine + ": " + mPath);
256         }
257 
258         // AbiTest relies on us rejecting any unsupported combinations.
259         if ((e_machine == EM_386 && elfClass != ELFCLASS32) ||
260                 (e_machine == EM_AARCH64 && elfClass != ELFCLASS64) ||
261                 (e_machine == EM_ARM && elfClass != ELFCLASS32) ||
262                 (e_machine == EM_QDSP6 && elfClass != ELFCLASS32) ||
263                 (e_machine == EM_RISCV && elfClass != ELFCLASS64) ||
264                 (e_machine == EM_X86_64 && elfClass != ELFCLASS64)) {
265             throw new IOException("Invalid e_machine/EI_CLASS ELF combination: " +
266                     e_machine + "/" + elfClass + ": " + mPath);
267         }
268 
269         long e_version = readWord();
270         if (e_version != EV_CURRENT) {
271             throw new IOException("Invalid e_version: " + e_version + ": " + mPath);
272         }
273 
274         long e_entry = readAddr();
275 
276         long ph_off = readOff();
277         long sh_off = readOff();
278 
279         long e_flags = readWord();
280         int e_ehsize = readHalf();
281         int e_phentsize = readHalf();
282         int e_phnum = readHalf();
283         int e_shentsize = readHalf();
284         int e_shnum = readHalf();
285         int e_shstrndx = readHalf();
286 
287         readSectionHeaders(sh_off, e_shnum, e_shentsize, e_shstrndx);
288         readProgramHeaders(ph_off, e_phnum, e_phentsize);
289     }
290 
readSectionHeaders(long sh_off, int e_shnum, int e_shentsize, int e_shstrndx)291     private void readSectionHeaders(long sh_off, int e_shnum, int e_shentsize, int e_shstrndx)
292             throws IOException {
293         // Read the Section Header String Table offset first.
294         {
295             mFile.seek(sh_off + e_shstrndx * e_shentsize);
296 
297             long sh_name = readWord();
298             long sh_type = readWord();
299             long sh_flags = readX(mAddrSize);
300             long sh_addr = readAddr();
301             long sh_offset = readOff();
302             long sh_size = readX(mAddrSize);
303             // ...
304 
305             if (sh_type == SHT_STRTAB) {
306                 mShStrTabOffset = sh_offset;
307                 mShStrTabSize = sh_size;
308             }
309         }
310 
311         for (int i = 0; i < e_shnum; ++i) {
312             // Don't bother to re-read the Section Header StrTab.
313             if (i == e_shstrndx) {
314                 continue;
315             }
316 
317             mFile.seek(sh_off + i * e_shentsize);
318 
319             long sh_name = readWord();
320             long sh_type = readWord();
321             long sh_flags = readX(mAddrSize);
322             long sh_addr = readAddr();
323             long sh_offset = readOff();
324             long sh_size = readX(mAddrSize);
325 
326             if (sh_type == SHT_SYMTAB || sh_type == SHT_DYNSYM) {
327                 final String symTabName = readShStrTabEntry(sh_name);
328                 if (".symtab".equals(symTabName)) {
329                     mSymTabOffset = sh_offset;
330                     mSymTabSize = sh_size;
331                 } else if (".dynsym".equals(symTabName)) {
332                     mDynSymOffset = sh_offset;
333                     mDynSymSize = sh_size;
334                 }
335             } else if (sh_type == SHT_STRTAB) {
336                 final String strTabName = readShStrTabEntry(sh_name);
337                 if (".strtab".equals(strTabName)) {
338                     mStrTabOffset = sh_offset;
339                     mStrTabSize = sh_size;
340                 } else if (".dynstr".equals(strTabName)) {
341                     mDynStrOffset = sh_offset;
342                     mDynStrSize = sh_size;
343                 }
344             } else if (sh_type == SHT_DYNAMIC) {
345                 mIsDynamic = true;
346             }
347         }
348     }
349 
readProgramHeaders(long ph_off, int e_phnum, int e_phentsize)350     private void readProgramHeaders(long ph_off, int e_phnum, int e_phentsize) throws IOException {
351         for (int i = 0; i < e_phnum; ++i) {
352             mFile.seek(ph_off + i * e_phentsize);
353 
354             long p_type = readWord();
355             if (p_type == PT_LOAD) {
356                 if (mAddrSize == 8) {
357                     // Only in Elf64_phdr; in Elf32_phdr p_flags is at the end.
358                     long p_flags = readWord();
359                 }
360                 long p_offset = readOff();
361                 long p_vaddr = readAddr();
362                 // ...
363 
364                 if (p_vaddr == 0) {
365                     mIsPIE = true;
366                 }
367             }
368         }
369     }
370 
readSymbolTable(long symStrOffset, long symStrSize, long tableOffset, long tableSize)371     private HashMap<String, Symbol> readSymbolTable(long symStrOffset, long symStrSize,
372             long tableOffset, long tableSize) throws IOException {
373         HashMap<String, Symbol> result = new HashMap<String, Symbol>();
374         mFile.seek(tableOffset);
375         while (mFile.getFilePointer() < tableOffset + tableSize) {
376             long st_name = readWord();
377             int st_info;
378             if (mAddrSize == 8) {
379                 st_info = readByte();
380                 int st_other = readByte();
381                 int st_shndx = readHalf();
382                 long st_value = readAddr();
383                 long st_size = readX(mAddrSize);
384             } else {
385                 long st_value = readAddr();
386                 long st_size = readWord();
387                 st_info = readByte();
388                 int st_other = readByte();
389                 int st_shndx = readHalf();
390             }
391             if (st_name == 0) {
392                 continue;
393             }
394 
395             final String symName = readStrTabEntry(symStrOffset, symStrSize, st_name);
396             if (symName != null) {
397                 Symbol s = new Symbol(symName, st_info);
398                 result.put(symName, s);
399             }
400         }
401         return result;
402     }
403 
readShStrTabEntry(long strOffset)404     private String readShStrTabEntry(long strOffset) throws IOException {
405         if (mShStrTabOffset == 0 || strOffset < 0 || strOffset >= mShStrTabSize) {
406             return null;
407         }
408         return readString(mShStrTabOffset + strOffset);
409     }
410 
readStrTabEntry(long tableOffset, long tableSize, long strOffset)411     private String readStrTabEntry(long tableOffset, long tableSize, long strOffset)
412             throws IOException {
413         if (tableOffset == 0 || strOffset < 0 || strOffset >= tableSize) {
414             return null;
415         }
416         return readString(tableOffset + strOffset);
417     }
418 
readHalf()419     private int readHalf() throws IOException {
420         return (int) readX(2);
421     }
422 
readWord()423     private long readWord() throws IOException {
424         return readX(4);
425     }
426 
readOff()427     private long readOff() throws IOException {
428         return readX(mAddrSize);
429     }
430 
readAddr()431     private long readAddr() throws IOException {
432         return readX(mAddrSize);
433     }
434 
readX(int byteCount)435     private long readX(int byteCount) throws IOException {
436         mFile.readFully(mBuffer, 0, byteCount);
437 
438         int answer = 0;
439         if (mEndian == ELFDATA2LSB) {
440             for (int i = byteCount - 1; i >= 0; i--) {
441                 answer = (answer << 8) | (mBuffer[i] & 0xff);
442             }
443         } else {
444             final int N = byteCount - 1;
445             for (int i = 0; i <= N; ++i) {
446                 answer = (answer << 8) | (mBuffer[i] & 0xff);
447             }
448         }
449 
450         return answer;
451     }
452 
readString(long offset)453     private String readString(long offset) throws IOException {
454         long originalOffset = mFile.getFilePointer();
455         mFile.seek(offset);
456         mFile.readFully(mBuffer, 0, (int) Math.min(mBuffer.length, mFile.length() - offset));
457         mFile.seek(originalOffset);
458 
459         for (int i = 0; i < mBuffer.length; ++i) {
460             if (mBuffer[i] == 0) {
461                 return new String(mBuffer, 0, i);
462             }
463         }
464 
465         return null;
466     }
467 
readByte()468     private int readByte() throws IOException {
469         return mFile.read() & 0xff;
470     }
471 
getSymbol(String name)472     public Symbol getSymbol(String name) {
473         if (mSymbols == null) {
474             try {
475                 mSymbols = readSymbolTable(mStrTabOffset, mStrTabSize, mSymTabOffset, mSymTabSize);
476             } catch (IOException e) {
477                 return null;
478             }
479         }
480         return mSymbols.get(name);
481     }
482 
getDynamicSymbol(String name)483     public Symbol getDynamicSymbol(String name) {
484         if (mDynamicSymbols == null) {
485             try {
486                 mDynamicSymbols = readSymbolTable(
487                         mDynStrOffset, mDynStrSize, mDynSymOffset, mDynSymSize);
488             } catch (IOException e) {
489                 return null;
490             }
491         }
492         return mDynamicSymbols.get(name);
493     }
494 }
495