1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
//! Bind opcodes are interpreted by the dynamic linker to efficiently collect every symbol imported by this binary, and from which library using two-level namespacing
//!
//! Some uses of external symbols do not need to be bound immediately.
//! Instead they can be lazily bound on first use.  The lazy_bind
//! are contains a stream of BIND opcodes to bind all lazy symbols.
//! Normal use is that dyld ignores the lazy_bind section when
//! loading an image.  Instead the static linker arranged for a
//! lazy pointer to initially point to a helper function which
//! pushes the offset into the lazy_bind area for the symbol
//! needing to be bound, then jumps to dyld which simply adds
//! the offset to lazy_bind_off to get the information on what
//! to bind.

pub type Opcode = u8;

// The following are used to encode binding information
pub const BIND_TYPE_POINTER                             : u8     = 1;
pub const BIND_TYPE_TEXT_ABSOLUTE32			: u8     = 2;
pub const BIND_TYPE_TEXT_PCREL32			: u8     = 3;
pub const BIND_SPECIAL_DYLIB_SELF			: u8     = 0;
pub const BIND_SPECIAL_DYLIB_MAIN_EXECUTABLE		: u8     = 0xf; // -1
pub const BIND_SPECIAL_DYLIB_FLAT_LOOKUP		: u8     = 0xe; // -2
pub const BIND_SYMBOL_FLAGS_WEAK_IMPORT	                : u8   = 0x1;
pub const BIND_SYMBOL_FLAGS_NON_WEAK_DEFINITION         : u8    = 0x8;
pub const BIND_OPCODE_MASK				: u8	 = 0xF0;
pub const BIND_IMMEDIATE_MASK                           : u8     = 0x0F;
pub const BIND_OPCODE_DONE				: Opcode = 0x00;
pub const BIND_OPCODE_SET_DYLIB_ORDINAL_IMM		: Opcode = 0x10;
pub const BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB		: Opcode = 0x20;
pub const BIND_OPCODE_SET_DYLIB_SPECIAL_IMM		: Opcode = 0x30;
pub const BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM	: Opcode = 0x40;
pub const BIND_OPCODE_SET_TYPE_IMM			: Opcode = 0x50;
pub const BIND_OPCODE_SET_ADDEND_SLEB                   : Opcode = 0x60;
pub const BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB	: Opcode = 0x70;
pub const BIND_OPCODE_ADD_ADDR_ULEB			: Opcode = 0x80;
pub const BIND_OPCODE_DO_BIND                           : Opcode = 0x90;
pub const BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB		: Opcode = 0xA0;
pub const BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED	: Opcode = 0xB0;
pub const BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB  : Opcode = 0xC0;

pub fn opcode_to_str(opcode: Opcode) -> &'static str {
    match opcode {
        BIND_OPCODE_DONE => "BIND_OPCODE_DONE",
        BIND_OPCODE_SET_DYLIB_ORDINAL_IMM => "BIND_OPCODE_SET_DYLIB_ORDINAL_IMM",
        BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB => "BIND_OPCODE_SET_DYLIB_ORDINAL_ULEB",
        BIND_OPCODE_SET_DYLIB_SPECIAL_IMM => "BIND_OPCODE_SET_DYLIB_SPECIAL_IMM",
        BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM => "BIND_OPCODE_SET_SYMBOL_TRAILING_FLAGS_IMM",
        BIND_OPCODE_SET_TYPE_IMM => "BIND_OPCODE_SET_TYPE_IMM",
        BIND_OPCODE_SET_ADDEND_SLEB => "BIND_OPCODE_SET_ADDEND_SLEB",
        BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB => "BIND_OPCODE_SET_SEGMENT_AND_OFFSET_ULEB",
        BIND_OPCODE_ADD_ADDR_ULEB => "BIND_OPCODE_ADD_ADDR_ULEB",
        BIND_OPCODE_DO_BIND => "BIND_OPCODE_DO_BIND",
        BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB => "BIND_OPCODE_DO_BIND_ADD_ADDR_ULEB",
        BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED => "BIND_OPCODE_DO_BIND_ADD_ADDR_IMM_SCALED",
        BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB => "BIND_OPCODE_DO_BIND_ULEB_TIMES_SKIPPING_ULEB",
        _ => "UNKNOWN OPCODE"
    }
}