-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathconstraint.rs
81 lines (71 loc) · 2.45 KB
/
constraint.rs
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
use crate::scip::ScipPtr;
use crate::{ffi, Row};
use std::rc::Rc;
/// A constraint in an optimization problem.
#[derive(Debug, Clone)]
#[allow(dead_code)]
pub struct Constraint {
/// A pointer to the underlying `SCIP_CONS` C struct.
pub(crate) raw: *mut ffi::SCIP_CONS,
/// A reference to the SCIP instance that owns this constraint (to prevent freeing the model while the constraint is live).
pub(crate) scip: Rc<ScipPtr>,
}
impl Constraint {
/// Returns a pointer to the underlying `SCIP_CONS` C struct.
pub fn inner(&self) -> *mut ffi::SCIP_CONS {
self.raw
}
/// Returns the name of the constraint.
pub fn name(&self) -> String {
unsafe {
let name = ffi::SCIPconsGetName(self.raw);
String::from(std::ffi::CStr::from_ptr(name).to_str().unwrap())
}
}
/// Returns the row associated with the constraint.
pub fn row(&self) -> Option<Row> {
let row_ptr = unsafe { ffi::SCIPconsGetRow(self.scip.raw, self.raw) };
if row_ptr.is_null() {
None
} else {
Some(Row {
raw: row_ptr,
scip: Rc::clone(&self.scip),
})
}
}
/// Returns the dual solution of the linear constraint in the current LP.
/// Returns `None` if the constraint is not a linear constraint.
pub fn dual_sol(&self) -> Option<f64> {
let cons_handler = unsafe { ffi::SCIPconsGetHdlr(self.raw) };
if cons_handler.is_null() {
return None;
}
let cons_handler_name = unsafe { ffi::SCIPconshdlrGetName(cons_handler) };
if cons_handler_name.is_null() {
return None;
}
let cons_handler_name = unsafe { std::ffi::CStr::from_ptr(cons_handler_name) };
if cons_handler_name.to_str().unwrap() != "linear" {
return None;
}
Some(unsafe { ffi::SCIPgetDualsolLinear(self.scip.raw, self.raw) })
}
}
#[cfg(test)]
mod tests {
use crate::prelude::*;
#[test]
fn test_constraint_mem_safety() {
// Create model
let mut model = Model::new()
.hide_output()
.include_default_plugins()
.create_prob("test")
.set_obj_sense(ObjSense::Maximize);
let x1 = model.add_var(0., f64::INFINITY, 3., "x1", VarType::Integer);
let cons = model.add_cons(vec![&x1], &[1.], 4., 4., "cons");
drop(model);
assert_eq!(cons.name(), "cons");
}
}