Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 1 | #------------------------------------------------------------------------------- |
| 2 | # elftools: common/utils.py |
| 3 | # |
| 4 | # Miscellaneous utilities for elftools |
| 5 | # |
| 6 | # Eli Bendersky (eliben@gmail.com) |
| 7 | # This code is in the public domain |
| 8 | #------------------------------------------------------------------------------- |
Eli Bendersky | c207431 | 2011-12-17 14:48:42 +0200 | [diff] [blame] | 9 | from contextlib import contextmanager |
eliben | 033b44f | 2011-09-19 15:48:39 +0300 | [diff] [blame] | 10 | from .exceptions import ELFParseError, ELFError, DWARFError |
Eli Bendersky | 79271e9 | 2012-01-27 10:25:47 +0200 | [diff] [blame] | 11 | from .py3compat import int2byte |
eliben | a7c2547 | 2011-09-18 17:31:10 +0300 | [diff] [blame] | 12 | from ..construct import ConstructError |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 13 | |
| 14 | |
Eli Bendersky | ebe5116 | 2011-10-27 17:34:02 +0200 | [diff] [blame] | 15 | def bytelist2string(bytelist): |
Eli Bendersky | 79271e9 | 2012-01-27 10:25:47 +0200 | [diff] [blame] | 16 | """ Convert a list of byte values (e.g. [0x10 0x20 0x00]) to a bytes object |
| 17 | (e.g. b'\x10\x20\x00'). |
Eli Bendersky | ebe5116 | 2011-10-27 17:34:02 +0200 | [diff] [blame] | 18 | """ |
Eli Bendersky | 79271e9 | 2012-01-27 10:25:47 +0200 | [diff] [blame] | 19 | return b''.join(int2byte(b) for b in bytelist) |
Eli Bendersky | ebe5116 | 2011-10-27 17:34:02 +0200 | [diff] [blame] | 20 | |
| 21 | |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 22 | def struct_parse(struct, stream, stream_pos=None): |
Eli Bendersky | 3f4de3e | 2011-09-14 05:58:06 +0300 | [diff] [blame] | 23 | """ Convenience function for using the given struct to parse a stream. |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 24 | If stream_pos is provided, the stream is seeked to this position before |
Eli Bendersky | 3f4de3e | 2011-09-14 05:58:06 +0300 | [diff] [blame] | 25 | the parsing is done. Otherwise, the current position of the stream is |
| 26 | used. |
| 27 | Wraps the error thrown by construct with ELFParseError. |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 28 | """ |
| 29 | try: |
| 30 | if stream_pos is not None: |
| 31 | stream.seek(stream_pos) |
| 32 | return struct.parse_stream(stream) |
| 33 | except ConstructError as e: |
Martin Panter | 8586ac1 | 2012-12-06 00:00:00 +0000 | [diff] [blame^] | 34 | raise ELFParseError(str(e)) |
Eli Bendersky | 89b280a | 2014-01-18 06:24:23 -0800 | [diff] [blame] | 35 | |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 36 | |
Eli Bendersky | 4a48975 | 2011-11-27 06:34:47 +0200 | [diff] [blame] | 37 | def parse_cstring_from_stream(stream, stream_pos=None): |
| 38 | """ Parse a C-string from the given stream. The string is returned without |
Eli Bendersky | a1d6140 | 2011-11-28 06:25:52 +0200 | [diff] [blame] | 39 | the terminating \x00 byte. If the terminating byte wasn't found, None |
| 40 | is returned (the stream is exhausted). |
Eli Bendersky | 4a48975 | 2011-11-27 06:34:47 +0200 | [diff] [blame] | 41 | If stream_pos is provided, the stream is seeked to this position before |
| 42 | the parsing is done. Otherwise, the current position of the stream is |
| 43 | used. |
Eli Bendersky | ccdb554 | 2013-03-31 14:19:21 -0700 | [diff] [blame] | 44 | Note: a bytes object is returned here, because this is what's read from |
| 45 | the binary file. |
Eli Bendersky | 4a48975 | 2011-11-27 06:34:47 +0200 | [diff] [blame] | 46 | """ |
Eli Bendersky | 4a48975 | 2011-11-27 06:34:47 +0200 | [diff] [blame] | 47 | if stream_pos is not None: |
| 48 | stream.seek(stream_pos) |
Eli Bendersky | a1d6140 | 2011-11-28 06:25:52 +0200 | [diff] [blame] | 49 | CHUNKSIZE = 64 |
| 50 | chunks = [] |
| 51 | found = False |
| 52 | while True: |
| 53 | chunk = stream.read(CHUNKSIZE) |
Eli Bendersky | 79271e9 | 2012-01-27 10:25:47 +0200 | [diff] [blame] | 54 | end_index = chunk.find(b'\x00') |
Eli Bendersky | a1d6140 | 2011-11-28 06:25:52 +0200 | [diff] [blame] | 55 | if end_index >= 0: |
| 56 | chunks.append(chunk[:end_index]) |
| 57 | found = True |
| 58 | break |
| 59 | else: |
| 60 | chunks.append(chunk) |
| 61 | if len(chunk) < CHUNKSIZE: |
| 62 | break |
Eli Bendersky | 79271e9 | 2012-01-27 10:25:47 +0200 | [diff] [blame] | 63 | return b''.join(chunks) if found else None |
Eli Bendersky | 4a48975 | 2011-11-27 06:34:47 +0200 | [diff] [blame] | 64 | |
| 65 | |
eliben | 116899e | 2011-09-08 17:15:53 +0300 | [diff] [blame] | 66 | def elf_assert(cond, msg=''): |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 67 | """ Assert that cond is True, otherwise raise ELFError(msg) |
| 68 | """ |
eliben | 4455651 | 2011-09-19 12:54:32 +0300 | [diff] [blame] | 69 | _assert_with_exception(cond, msg, ELFError) |
Eli Bendersky | e0735d5 | 2011-09-08 20:12:44 +0300 | [diff] [blame] | 70 | |
eliben | 4455651 | 2011-09-19 12:54:32 +0300 | [diff] [blame] | 71 | |
| 72 | def dwarf_assert(cond, msg=''): |
| 73 | """ Assert that cond is True, otherwise raise DWARFError(msg) |
| 74 | """ |
| 75 | _assert_with_exception(cond, msg, DWARFError) |
| 76 | |
| 77 | |
eliben | 3b9ad82 | 2011-09-22 11:46:26 +0300 | [diff] [blame] | 78 | @contextmanager |
| 79 | def preserve_stream_pos(stream): |
| 80 | """ Usage: |
Eli Bendersky | 89b280a | 2014-01-18 06:24:23 -0800 | [diff] [blame] | 81 | # stream has some position FOO (return value of stream.tell()) |
| 82 | with preserve_stream_pos(stream): |
| 83 | # do stuff that manipulates the stream |
| 84 | # stream still has position FOO |
eliben | 3b9ad82 | 2011-09-22 11:46:26 +0300 | [diff] [blame] | 85 | """ |
| 86 | saved_pos = stream.tell() |
| 87 | yield |
| 88 | stream.seek(saved_pos) |
Eli Bendersky | a1d6140 | 2011-11-28 06:25:52 +0200 | [diff] [blame] | 89 | |
Eli Bendersky | c207431 | 2011-12-17 14:48:42 +0200 | [diff] [blame] | 90 | |
| 91 | #------------------------- PRIVATE ------------------------- |
| 92 | |
| 93 | def _assert_with_exception(cond, msg, exception_type): |
| 94 | if not cond: |
| 95 | raise exception_type(msg) |
| 96 | |