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
//! The [PostScript information][1].
//!
//! [1]: https://www.microsoft.com/typography/otspec/post.htm

use {Result, Tape, Value, q32};

/// PostScript information.
#[derive(Clone, Debug)]
pub enum PostScript {
    /// Version 1.0.
    Version1(PostScript1),
    /// Version 2.0.
    Version2(PostScript2),
    /// Version 3.0.
    Version3(PostScript3),
}

table! {
    #[doc = "PostScript information of version 1.0."]
    #[derive(Copy)]
    pub PostScript1 {
        version             (q32), // version
        italic_angle        (q32), // italicAngle
        underline_position  (i16), // underlinePosition
        underline_thickness (i16), // underlineThickness
        is_fixed_pitch      (u32), // isFixedPitch
        min_memory_type42   (u32), // minMemType42
        max_memory_type42   (u32), // maxMemType42
        min_memory_type1    (u32), // minMemType1
        max_memory_type1    (u32), // maxMemType1
    }
}

table! {
    #[doc = "PostScript information of version 2.0."]
    pub PostScript2 {
        version             (q32), // version
        italic_angle        (q32), // italicAngle
        underline_position  (i16), // underlinePosition
        underline_thickness (i16), // underlineThickness
        is_fixed_pitch      (u32), // isFixedPitch
        min_memory_type42   (u32), // minMemType42
        max_memory_type42   (u32), // maxMemType42
        min_memory_type1    (u32), // minMemType1
        max_memory_type1    (u32), // maxMemType1
        glyph_count         (u16), // numberOfGlyphs

        glyph_name_indices (Vec<u16>) |this, tape| { // glyphNameIndex
            tape.take_given(this.glyph_count as usize)
        },

        glyph_names (Vec<String>) |this, tape| { // names
            read_pascal_strings(tape, &this.glyph_name_indices)
        },
    }
}

/// PostScript information of version 3.0.
pub type PostScript3 = PostScript1;

impl Value for PostScript {
    fn read<T: Tape>(tape: &mut T) -> Result<Self> {
        Ok(match try!(tape.peek::<q32>()) {
            q32(0x00010000) => PostScript::Version1(try!(tape.take())),
            q32(0x00020000) => PostScript::Version2(try!(tape.take())),
            q32(0x00030000) => PostScript::Version3(try!(tape.take())),
            _ => raise!("found an unknown format of the PostScript information"),
        })
    }
}

fn read_pascal_strings<T: Tape>(tape: &mut T, indices: &[u16]) -> Result<Vec<String>> {
    let count = indices.iter().fold(0, |n, &i| if 258 <= i && i <= 32767 { n + 1 } else { n });
    let mut names = Vec::with_capacity(count);
    for _ in 0..count {
        let length = try!(tape.take::<u8>()) as usize;
        match String::from_utf8(try!(tape.take_bytes(length))) {
            Ok(name) => names.push(name),
            _ => names.push("<malformed>".into()),
        }
    }
    Ok(names)
}