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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use failure::{bail, ensure, Error, Fail, ResultExt};
use libc::*;
use std::ffi::{CStr, CString};
use std::result;
use std::str::FromStr;
use crate::engine::Engine;
type Result<T> = result::Result<T, Error>;
include!(concat!(env!("OUT_DIR"), "/cvar_array.rs"));
pub const EMPTY_CVAR_T: cvar_t = cvar_t { name: 0 as *const _,
string: 0 as *mut _,
flags: 0,
value: 0f32,
next: 0 as *mut _ };
#[repr(C)]
pub struct cvar_t {
name: *const c_char,
string: *mut c_char,
flags: c_int,
value: c_float,
next: *mut cvar_t,
}
pub struct CVar {
pub engine_cvar: *mut cvar_t,
pub default_value: &'static str,
pub name: &'static str,
}
unsafe impl Send for CVar {}
unsafe impl Sync for CVar {}
impl cvar_t {
#[inline]
pub fn string_is_non_null(&self) -> bool {
!self.string.is_null()
}
}
impl CVar {
#[inline]
pub unsafe fn get_engine_cvar(&self) -> &'static mut cvar_t {
&mut *self.engine_cvar
}
pub fn register(&self, engine: &mut Engine) -> Result<()> {
let ptr = {
let mut engine_cvar = engine.get_engine_cvar(self);
if engine_cvar.name.is_null() {
let name_cstring =
CString::new(self.name).context("could not convert the CVar name to CString")?;
engine_cvar.name = name_cstring.into_raw();
}
let default_value_cstring = CString::new(self.default_value)
.context("could not convert default CVar value to CString")?;
let ptr = default_value_cstring.into_raw();
engine_cvar.string = ptr;
ptr
};
engine.register_variable(self)
.context("could not register the variable")?;
unsafe { CString::from_raw(ptr) };
Ok(())
}
pub fn to_string(&self, engine: &mut Engine) -> Result<String> {
let engine_cvar = engine.get_engine_cvar(self);
ensure!(engine_cvar.string_is_non_null(),
"the CVar string pointer was null");
let string = unsafe { CStr::from_ptr(engine_cvar.string) }
.to_str()
.context("could not convert the CVar string to a Rust string")?;
Ok(string.to_owned())
}
pub fn parse<T>(&self, engine: &mut Engine) -> Result<T>
where T: FromStr,
<T as FromStr>::Err: Fail
{
let string = self.to_string(engine)
.context("could not get this CVar's string value")?;
Ok(string.parse::<T>()
.context("could not convert the CVar string to the desired type")?)
}
}