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
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
#![warn(missing_docs)]

//! A library to handle Quake 3 virtual machines.

// Both `nom` and `error_chain` can recurse deeply
#![recursion_limit="1024"]

#[macro_use]
extern crate error_chain;
#[macro_use]
extern crate nom;

pub mod errors;
pub mod bytecode;
pub mod opcodes;
pub mod parser;

pub use bytecode::Instruction;

use errors::*;

const VM_MAGIC: [u8; 4] = [0x44, 0x14, 0x72, 0x12];

/// A Quake 3 virtual machine image.
///
/// A VM consists of instructions and data, where data is separated into
///
/// * word-sized data
/// * byte-sized data (LIT)
/// * uninitialized data (BSS)
#[derive(Debug,PartialEq)]
pub struct QVM {
    code: Vec<Instruction>,
    data: Vec<u32>,
    lit: Vec<u8>,
    bss_length: u32,
}

impl QVM {
    // TODO: Validate instructions; addresses might be out of bounds etc.
    /// Creates a new VM instance.
    ///
    /// # Errors
    /// lorem ipsum
    pub fn new(code: Vec<Instruction>,
               data: Vec<u32>,
               lit: Vec<u8>,
               bss_length: u32)
               -> Result<QVM> {
        Ok(QVM {
               code: code,
               data: data,
               lit: lit,
               bss_length: bss_length,
           })
    }

    /// Returns the instructions of the code segment.
    pub fn instructions(&self) -> &Vec<Instruction> {
        &self.code
    }

    /// Returns the word-sized data of the data segment.
    pub fn data(&self) -> &Vec<u32> {
        &self.data
    }

    /// Returns the byte-sized data of the LIT segment.
    pub fn lit(&self) -> &Vec<u8> {
        &self.lit
    }

    /// Returns the length of the uninitialized BSS segment.
    pub fn bss_length(&self) -> u32 {
        self.bss_length
    }
}

/// The different segments/sections in a QVM file.
///
/// See ioquake3's `segmentName_t` in `tools/asm/q3asm.c`
// These should match the names in ioquake3
#[allow(non_camel_case_types)]
#[derive(Debug, Clone, Copy)]
pub enum Segment {
    /// The code segment, consisting of instructions.
    CODE,
    /// The data segment, consisting of word-sized data.
    DATA,
    /// The LIT segment, consisting of byte-sized data.
    LIT,
    /// The BSS pseudo-segment, consisting of uninitialized data.
    BSS,
    /// The jump table targets pseudo-segment, consisting of jump label addresses.
    JTRG,
}